aboutsummaryrefslogtreecommitdiffstats
diff options
context:
space:
mode:
-rw-r--r--drivers/video/sm501fb.c53
1 files changed, 42 insertions, 11 deletions
diff --git a/drivers/video/sm501fb.c b/drivers/video/sm501fb.c
index b473cf665d83..f94ae84a58cd 100644
--- a/drivers/video/sm501fb.c
+++ b/drivers/video/sm501fb.c
@@ -48,10 +48,15 @@ enum sm501_controller {
48 HEAD_PANEL = 1, 48 HEAD_PANEL = 1,
49}; 49};
50 50
51/* SM501 memory address */ 51/* SM501 memory address.
52 *
53 * This structure is used to track memory usage within the SM501 framebuffer
54 * allocation. The sm_addr field is stored as an offset as it is often used
55 * against both the physical and mapped addresses.
56 */
52struct sm501_mem { 57struct sm501_mem {
53 unsigned long size; 58 unsigned long size;
54 unsigned long sm_addr; 59 unsigned long sm_addr; /* offset from base of sm501 fb. */
55 void __iomem *k_addr; 60 void __iomem *k_addr;
56}; 61};
57 62
@@ -142,37 +147,63 @@ static inline void sm501fb_sync_regs(struct sm501fb_info *info)
142static int sm501_alloc_mem(struct sm501fb_info *inf, struct sm501_mem *mem, 147static int sm501_alloc_mem(struct sm501fb_info *inf, struct sm501_mem *mem,
143 unsigned int why, size_t size) 148 unsigned int why, size_t size)
144{ 149{
145 unsigned int ptr = 0; 150 struct sm501fb_par *par;
146 unsigned int end;
147 struct fb_info *fbi; 151 struct fb_info *fbi;
152 unsigned int ptr;
153 unsigned int end;
148 154
149 switch (why) { 155 switch (why) {
150 case SM501_MEMF_CURSOR: 156 case SM501_MEMF_CURSOR:
151 ptr = inf->fbmem_len - size; 157 ptr = inf->fbmem_len - size;
152 inf->fbmem_len = ptr; 158 inf->fbmem_len = ptr; /* adjust available memory. */
153 break; 159 break;
154 160
155 case SM501_MEMF_PANEL: 161 case SM501_MEMF_PANEL:
156 ptr = inf->fbmem_len - size; 162 ptr = inf->fbmem_len - size;
157 fbi = inf->fb[0]; 163 fbi = inf->fb[HEAD_CRT];
164
165 /* round down, some programs such as directfb do not draw
166 * 0,0 correctly unless the start is aligned to a page start.
167 */
168
169 if (ptr > 0)
170 ptr &= ~(PAGE_SIZE - 1);
158 171
159 if (fbi && ptr < fbi->fix.smem_len) 172 if (fbi && ptr < fbi->fix.smem_len)
160 return -ENOMEM; 173 return -ENOMEM;
161 174
175 if (ptr < 0)
176 return -ENOMEM;
177
162 break; 178 break;
163 179
164 case SM501_MEMF_CRT: 180 case SM501_MEMF_CRT:
165 ptr = 0; 181 ptr = 0;
182
183 /* check to see if we have panel memory allocated
184 * which would put an limit on available memory. */
185
186 fbi = inf->fb[HEAD_PANEL];
187 if (fbi) {
188 par = fbi->par;
189 end = par->screen.k_addr ? par->screen.sm_addr : inf->fbmem_len;
190 } else
191 end = inf->fbmem_len;
192
193 if ((ptr + size) > end)
194 return -ENOMEM;
195
166 break; 196 break;
167 197
168 case SM501_MEMF_ACCEL: 198 case SM501_MEMF_ACCEL:
169 fbi = inf->fb[0]; 199 fbi = inf->fb[HEAD_CRT];
170 ptr = fbi ? fbi->fix.smem_len : 0; 200 ptr = fbi ? fbi->fix.smem_len : 0;
171 201
172 fbi = inf->fb[1]; 202 fbi = inf->fb[HEAD_PANEL];
173 if (fbi) 203 if (fbi) {
174 end = (fbi->fix.smem_start - inf->fbmem_res->start); 204 par = fbi->par;
175 else 205 end = par->screen.sm_addr;
206 } else
176 end = inf->fbmem_len; 207 end = inf->fbmem_len;
177 208
178 if ((ptr + size) > end) 209 if ((ptr + size) > end)