s5pc110: fb: updated framebuffer driver.
[kernel/u-boot.git] / drivers / video / s5p-fb.c
1 /*
2  * S5PC100 and S5PC110 LCD Controller driver.
3  *
4  * Author: InKi Dae <inki.dae@samsung.com>
5  *
6  * This program is free software; you can redistribute it and/or
7  * modify it under the terms of the GNU General Public License as
8  * published by the Free Software Foundation; either version 2 of
9  * the License, or (at your option) any later version.
10  *
11  * This program is distributed in the hope that it will be useful,
12  * but WITHOUT ANY WARRANTY; without even the implied warranty of
13  * MERCHANTABILITY or FITNESS FOR A PARTICULAR PURPOSE.  See the
14  * GNU General Public License for more details.
15  *
16  * You should have received a copy of the GNU General Public License
17  * along with this program; if not, write to the Free Software
18  * Foundation, Inc., 59 Temple Place, Suite 330, Boston,
19  * MA 02111-1307 USA
20  */
21
22 #include <config.h>
23 #include <common.h>
24 #include <version.h>
25 #include <stdarg.h>
26 #include <linux/types.h>
27 #include <asm/io.h>
28 #include <asm/arch/cpu.h>
29 #include <lcd.h>
30
31 #include "s5p-fb.h"
32 #include "logo.h"
33 /*
34 #include "logo_rgb24_wvga_portrait.h"
35 #include "opening_logo_rgb24_143_44.h"
36 */
37
38 extern void tl2796_panel_power_on(void);
39 extern void tl2796_panel_enable(void);
40 extern void tl2796_panel_init(void);
41
42 int lcd_line_length;
43 int lcd_color_fg;
44 int lcd_color_bg;
45
46 void *lcd_base;
47 void *lcd_console_address;
48
49 short console_col;
50 short console_row;
51
52 static unsigned int panel_width, panel_height;
53
54 static unsigned short makepixel565(char r, char g, char b)
55 {
56     return (unsigned short)(((r >> 3) << 11) | ((g >> 2) << 5) | (b >> 3));
57 }
58
59 static unsigned int makepixel8888(char a, char r, char g, char b)
60 {
61         return (unsigned int)((a << 24) | (r << 16) | (g << 8)  | b);
62 }
63
64 static void read_image16(char* pImg, int x1pos, int y1pos, int x2pos,
65                 int y2pos, unsigned short pixel)
66 {
67         unsigned short *pDst = (unsigned short *)pImg;
68         unsigned int offset_s;
69         int i, j;
70
71         for(i = y1pos; i < y2pos; i++) {
72                 for(j = x1pos; j < x2pos; j++) {
73                         offset_s = i * panel_width + j;
74                         *(pDst + offset_s) = pixel;
75                 }
76         }
77 }
78
79 static void read_image32(char* pImg, int x1pos, int y1pos, int x2pos,
80                 int y2pos, unsigned int pixel)
81 {
82         unsigned int *pDst = (unsigned int *)pImg;
83         unsigned int offset_s;
84         int i, j;
85
86         for (i = y1pos; i < y2pos; i++) {
87                 for (j = x1pos; j < x2pos; j++) {
88                         offset_s = i * panel_width + j;
89                         *(pDst+offset_s) = pixel;
90                 }
91         }
92 }
93
94 /* LCD Panel data */
95 vidinfo_t panel_info;
96
97 static void s5pc_lcd_init_mem(void *lcdbase, vidinfo_t *vid)
98 {
99         unsigned long palette_size, palette_mem_size;
100         unsigned int fb_size;
101
102         fb_size = vid->vl_row * vid->vl_col * (vid->vl_bpix >> 3);
103
104         lcd_base = lcdbase;
105
106         palette_size = NBITS(vid->vl_bpix) == 8 ? 256 : 16;
107         palette_mem_size = palette_size * sizeof(u32);
108
109         s5pc_fimd_lcd_init_mem((unsigned long)lcd_base, (unsigned long)fb_size, palette_size);
110
111         udebug("fb_size=%d, screen_base=%x, palette_size=%d, palettle_mem_size=%d\n",
112                         fb_size, (unsigned int)lcd_base, (int)palette_size, (int)palette_mem_size);
113 }
114
115 static void s5pc_lcd_init(vidinfo_t *vid)
116 {
117         s5pc_fimd_lcd_init(vid);
118 }
119
120 static void lcd_test(void)
121 {
122         /* red */
123         read_image32((char *)lcd_base, 0, 0, 1024, 200,
124                         makepixel8888(0, 255, 0, 0));
125         /* green */
126         read_image32((char *)lcd_base, 0, 200, 1024, 400,
127                         makepixel8888(0, 0, 255, 0));
128         /* blue */
129         read_image32((char *)lcd_base, 0, 400, 1024, 600,
130                         makepixel8888(0, 0, 0, 255));
131         /* write */
132         //read_image32((char *)lcd_base, 0, 600, 1024, 800,
133         //              makepixel8888(0, 255, 255, 255));
134 }
135
136 int conv_rgb565_to_rgb888(unsigned short rgb565, unsigned int sw)
137 {
138         char red, green, blue;
139         unsigned int threshold = 150;
140
141         red = (rgb565 & 0xF800) >> 11;
142         green = (rgb565 & 0x7E0) >> 5;
143         blue = (rgb565 & 0x1F);
144
145         red = red << 3;
146         green = green << 2;
147         blue = blue << 3;
148
149         /* correct error pixels of samsung logo. */
150         if (sw) {
151                 if (red > threshold)
152                         red = 255;
153                 if (green > threshold)
154                         green = 255;
155                 if (blue > threshold)
156                         blue = 255;
157         }
158
159         return (red << 16 | green << 8 | blue);
160 }
161
162 void draw_bitmap(void *lcdbase, int x, int y, int w, int h, unsigned long *bmp)
163 {
164         int i, j;
165         short k = 0;
166         unsigned long *fb = (unsigned  long*)lcdbase;
167
168         for (j = y; j < (y + h); j++) {
169                 for (i = x; i < (x + w); i++)
170                         *(fb + (j * panel_width) + i) = *(bmp + k++);
171         }
172 }
173
174 void _draw_samsung_logo(void *lcdbase, int x, int y, int w, int h, unsigned short *bmp)
175 {
176         int i, j, error_range = 40;
177         short k = 0;
178         unsigned int pixel;
179         unsigned long *fb = (unsigned  long*)lcdbase;
180
181         for (j = y; j < (y + h); j++) {
182                 for (i = x; i < (x + w); i++) {
183                         pixel = (*(bmp + k++));
184
185                         /* 40 lines under samsung logo image are error range. */
186                         if (j > h + y - error_range)
187                                 *(fb + (j * panel_width) + i) =
188                                         conv_rgb565_to_rgb888(pixel, 1);
189                         else
190                                 *(fb + (j * panel_width) + i) =
191                                         conv_rgb565_to_rgb888(pixel, 0);
192                 }
193         }
194 }
195
196 static void draw_samsung_logo(void* lcdbase)
197 {
198         int x, y;
199
200         x = ((panel_width - 298) >> 1);
201         y = ((panel_height - 78) >> 1) - 5;
202
203         _draw_samsung_logo(lcdbase, x, y, 298, 78, (unsigned short *) logo);
204 }
205
206 static void lcd_panel_on(vidinfo_t *vid)
207 {
208         udelay(vid->init_delay);
209
210         if (vid->cfg_gpio)
211                 vid->cfg_gpio();
212
213         if (vid->lcd_power_on) {
214                 vid->lcd_power_on(1);
215         }
216
217         udelay(vid->power_on_delay);
218
219         if (vid->reset_lcd) {
220                 vid->reset_lcd();
221         }
222
223         udelay(vid->reset_delay);
224
225         if (vid->backlight_on) {
226                 vid->backlight_on(1);
227         }
228
229         if (vid->cfg_ldo) {
230                 vid->cfg_ldo();
231         }
232
233         if (vid->enable_ldo) {
234                 vid->enable_ldo(1);
235         }
236 }
237
238 /* extern void init_onenand_ext2(void); */
239 extern void init_panel_info(vidinfo_t *vid);
240
241 void lcd_ctrl_init(void *lcdbase)
242 {
243         char *option;
244
245         s5pc_lcd_init_mem(lcdbase, &panel_info);
246
247         /* initialize parameters which is specific to panel. */
248         init_panel_info(&panel_info);
249
250         panel_width = panel_info.vl_width;
251         panel_height = panel_info.vl_height;
252
253         option = getenv("lcd");
254
255         /*
256         if (strcmp(option, "test") == 0) {
257                 memset(lcdbase, 0, PANEL_WIDTH*PANEL_HEIGHT*S5P_LCD_BPP >> 3);
258                 lcd_test();
259         } else if (strcmp(option, "image") == 0)
260                 memcpy(lcdbase, LOGO_RGB24, PANEL_WIDTH*PANEL_HEIGHT*S5P_LCD_BPP >> 3);
261         else {
262                 memset(lcdbase, 0, PANEL_WIDTH*PANEL_HEIGHT*S5P_LCD_BPP >> 3);
263                 draw_samsung_logo(lcdbase);
264         }
265         */
266
267         memset(lcdbase, 0, panel_width * panel_height * (32 >> 3));
268         draw_samsung_logo(lcdbase);
269
270         s5pc_lcd_init(&panel_info);
271
272         /* font test */
273         /*
274         init_font();
275         set_font_color(FONT_WHITE);
276         fb_printf("Test\n");
277         exit_font();
278         */
279
280         /* init_onenand_ext2(); */
281 }
282
283
284 void lcd_setcolreg(ushort regno, ushort red, ushort green, ushort blud)
285 {
286         return;
287 }
288
289 void lcd_enable(void)
290 {
291         lcd_panel_on(&panel_info);
292 }
293
294 ulong calc_fbsize(void)
295 {
296         return s5pc_fimd_calc_fbsize();
297 }
298
299 void s5pc1xxfb_test(void *lcdbase)
300 {
301         lcd_ctrl_init(lcdbase);
302         lcd_enable();
303 }