summaryrefslogtreecommitdiff
path: root/firmware/target/arm/iriver/h10/lcd-h10_5gb.c
diff options
context:
space:
mode:
Diffstat (limited to 'firmware/target/arm/iriver/h10/lcd-h10_5gb.c')
-rw-r--r--firmware/target/arm/iriver/h10/lcd-h10_5gb.c114
1 files changed, 75 insertions, 39 deletions
diff --git a/firmware/target/arm/iriver/h10/lcd-h10_5gb.c b/firmware/target/arm/iriver/h10/lcd-h10_5gb.c
index 680d2c47a5..8972fd1e9c 100644
--- a/firmware/target/arm/iriver/h10/lcd-h10_5gb.c
+++ b/firmware/target/arm/iriver/h10/lcd-h10_5gb.c
@@ -108,7 +108,6 @@ void lcd_init_device(void)
108{ 108{
109 CLCD_CLOCK_SRC |= 0xc0000000; /* Set LCD interface clock to PLL */ 109 CLCD_CLOCK_SRC |= 0xc0000000; /* Set LCD interface clock to PLL */
110 /* H10 LCD is initialised by the bootloader */ 110 /* H10 LCD is initialised by the bootloader */
111 lcd_write_reg(R_ENTRY_MODE, 0x1030); /* BGR =1, ID1 = 1, ID0 = 1 */
112} 111}
113 112
114/*** update functions ***/ 113/*** update functions ***/
@@ -290,51 +289,88 @@ void lcd_yuv_blit(unsigned char * const src[3],
290 289
291 290
292/* Update a fraction of the display. */ 291/* Update a fraction of the display. */
293void lcd_update_rect(int x, int y, int width, int height) 292void lcd_update_rect(int x0, int y0, int width, int height)
294{ 293{
295 const fb_data *addr; 294 int x1, y1;
296 int bytes_to_write; 295 int newx,newwidth;
297 296 unsigned long *addr;
298 if (x + width >= LCD_WIDTH)
299 width = LCD_WIDTH - x;
300 if (y + height >= LCD_HEIGHT)
301 height = LCD_HEIGHT - y;
302
303 if ((width <= 0) || (height <= 0))
304 return; /* Nothing left to do. 0 would hang the transfer. */
305
306 /* Ensure x and width are both even, so we can read
307 * 32-bit aligned data from the framebuffer */
308 width = (width + (x & 1) + 1) & ~1;
309 x &= ~1;
310
311 lcd_write_reg(R_VERT_RAM_ADDR_POS, (LCD_HEIGHT-1) << 8);
312 lcd_write_reg(R_HORIZ_RAM_ADDR_POS, ((x + width - 1) << 8) | x);
313 lcd_write_reg(R_RAM_ADDR_SET, (y << 8) | x);
314 297
298 /* Ensure x and width are both even - so we can read 32-bit aligned
299 data from lcd_framebuffer */
300 newx=x0&~1;
301 newwidth=width&~1;
302 if (newx+newwidth < x0+width) { newwidth+=2; }
303 x0=newx; width=newwidth;
304
305 /* calculate the drawing region */
306 y1 = (y0 + height) - 1; /* max vert */
307 x1 = (x0 + width) - 1; /* max horiz */
308
309
310 /* swap max horiz < start horiz */
311 if (y1 < y0) {
312 int t;
313 t = y0;
314 y0 = y1;
315 y1 = t;
316 }
317
318 /* swap max vert < start vert */
319 if (x1 < x0) {
320 int t;
321 t = x0;
322 x0 = x1;
323 x1 = t;
324 }
325
326 /* max horiz << 8 | start horiz */
327 lcd_write_reg(R_HORIZ_RAM_ADDR_POS, (x1 << 8) | x0);
328
329 /* max vert << 8 | start vert */
330 lcd_write_reg(R_VERT_RAM_ADDR_POS, (y1 << 8) | y0);
331
332 /* start vert << 8 | start horiz */
333 lcd_write_reg(R_RAM_ADDR_SET, (y0 << 8) | x0);
334
335 /* start drawing */
315 lcd_send_cmd(R_WRITE_DATA_2_GRAM); 336 lcd_send_cmd(R_WRITE_DATA_2_GRAM);
316 337
317 addr = &lcd_framebuffer[y][x]; 338 addr = (unsigned long*)&lcd_framebuffer[y0][x0];
318 bytes_to_write = width * height * sizeof(fb_data);
319 /* must be <= 0x10000, but that's guaranteed on H10. */
320 339
321 LCD2_BLOCK_CTRL = 0x10000080; 340 while (height > 0) {
322 LCD2_BLOCK_CONFIG = 0xc0010000 | (bytes_to_write - 1); 341 int c, r;
323 LCD2_BLOCK_CTRL = 0x34000000; 342 int h, pixels_to_write;
324 343
325 do 344 pixels_to_write = (width * height) * 2;
326 { 345 h = height;
327 int w = width >> 1; 346
328 do 347 /* calculate how much we can do in one go */
329 { 348 if (pixels_to_write > 0x10000) {
330 while (!(LCD2_BLOCK_CTRL & LCD2_BLOCK_TXOK)); 349 h = (0x10000/2) / width;
331 LCD2_BLOCK_DATA = *(unsigned long*)addr; /* output 2 pixels */ 350 pixels_to_write = (width * h) * 2;
332 addr += 2;
333 } 351 }
334 while (--w > 0); 352
335 addr += LCD_WIDTH - width; 353 LCD2_BLOCK_CTRL = 0x10000080;
354 LCD2_BLOCK_CONFIG = 0xc0010000 | (pixels_to_write - 1);
355 LCD2_BLOCK_CTRL = 0x34000000;
356
357 /* for each row */
358 for (r = 0; r < h; r++) {
359 /* for each column */
360 for (c = 0; c < width; c += 2) {
361 while (!(LCD2_BLOCK_CTRL & LCD2_BLOCK_TXOK));
362
363 /* output 2 pixels */
364 LCD2_BLOCK_DATA = *addr++;
365 }
366 addr += (LCD_WIDTH - width)/2;
367 }
368
369 while (!(LCD2_BLOCK_CTRL & LCD2_BLOCK_READY));
370 LCD2_BLOCK_CONFIG = 0;
371
372 height -= h;
336 } 373 }
337 while (--height > 0);
338} 374}
339 375
340/* Update the display. 376/* Update the display.