2 * KFR2R09 LCD panel support
4 * Copyright (C) 2009 Magnus Damm
6 * Register settings based on the out-of-tree t33fb.c driver
7 * Copyright (C) 2008 Lineo Solutions, Inc.
9 * This file is subject to the terms and conditions of the GNU General Public
10 * License. See the file COPYING in the main directory of this archive for
14 #include <linux/delay.h>
15 #include <linux/err.h>
17 #include <linux/init.h>
18 #include <linux/kernel.h>
19 #include <linux/module.h>
20 #include <linux/gpio.h>
21 #include <video/sh_mobile_lcdc.h>
22 #include <mach/kfr2r09.h>
23 #include <cpu/sh7724.h>
25 /* The on-board LCD module is a Hitachi TX07D34VM0AAA. This module is made
26 * up of a 240x400 LCD hooked up to a R61517 driver IC. The driver IC is
27 * communicating with the main port of the LCDC using an 18-bit SYS interface.
29 * The device code for this LCD module is 0x01221517.
32 static const unsigned char data_frame_if
[] = {
33 0x02, /* WEMODE: 1=cont, 0=one-shot */
36 0x02, /* RIM[1] : 1 (18bpp) */
39 static const unsigned char data_panel
[] = {
42 0x04, 0x00, 0x00, 0x04, 0x11, 0x00, 0x00,
45 static const unsigned char data_timing
[] = {
46 0x00, 0x00, 0x13, 0x08, 0x08,
49 static const unsigned char data_timing_src
[] = {
50 0x11, 0x01, 0x00, 0x01,
53 static const unsigned char data_gamma
[] = {
54 0x01, 0x02, 0x08, 0x23, 0x03, 0x0c, 0x00, 0x06, 0x00, 0x00,
55 0x01, 0x00, 0x0c, 0x23, 0x03, 0x08, 0x02, 0x06, 0x00, 0x00,
58 static const unsigned char data_power
[] = {
59 0x07, 0xc5, 0xdc, 0x02, 0x33, 0x0a,
62 static unsigned long read_reg(void *sohandle
,
63 struct sh_mobile_lcdc_sys_bus_ops
*so
)
65 return so
->read_data(sohandle
);
68 static void write_reg(void *sohandle
,
69 struct sh_mobile_lcdc_sys_bus_ops
*so
,
70 int i
, unsigned long v
)
73 so
->write_data(sohandle
, v
); /* PTH4/LCDRS High [param, 17:0] */
75 so
->write_index(sohandle
, v
); /* PTH4/LCDRS Low [cmd, 7:0] */
78 static void write_data(void *sohandle
,
79 struct sh_mobile_lcdc_sys_bus_ops
*so
,
80 unsigned char const *data
, int no_data
)
84 for (i
= 0; i
< no_data
; i
++)
85 write_reg(sohandle
, so
, 1, data
[i
]);
88 static unsigned long read_device_code(void *sohandle
,
89 struct sh_mobile_lcdc_sys_bus_ops
*so
)
91 unsigned long device_code
;
93 /* access protect OFF */
94 write_reg(sohandle
, so
, 0, 0xb0);
95 write_reg(sohandle
, so
, 1, 0x00);
97 /* deep standby OFF */
98 write_reg(sohandle
, so
, 0, 0xb1);
99 write_reg(sohandle
, so
, 1, 0x00);
101 /* device code command */
102 write_reg(sohandle
, so
, 0, 0xbf);
106 read_reg(sohandle
, so
);
108 /* read device code */
109 device_code
= ((read_reg(sohandle
, so
) & 0xff) << 24);
110 device_code
|= ((read_reg(sohandle
, so
) & 0xff) << 16);
111 device_code
|= ((read_reg(sohandle
, so
) & 0xff) << 8);
112 device_code
|= (read_reg(sohandle
, so
) & 0xff);
117 static void write_memory_start(void *sohandle
,
118 struct sh_mobile_lcdc_sys_bus_ops
*so
)
120 write_reg(sohandle
, so
, 0, 0x2c);
123 static void clear_memory(void *sohandle
,
124 struct sh_mobile_lcdc_sys_bus_ops
*so
)
129 write_memory_start(sohandle
, so
);
132 for (i
= 0; i
< (240 * 400); i
++)
133 write_reg(sohandle
, so
, 1, 0x00);
136 static void display_on(void *sohandle
,
137 struct sh_mobile_lcdc_sys_bus_ops
*so
)
139 /* access protect off */
140 write_reg(sohandle
, so
, 0, 0xb0);
141 write_reg(sohandle
, so
, 1, 0x00);
143 /* exit deep standby mode */
144 write_reg(sohandle
, so
, 0, 0xb1);
145 write_reg(sohandle
, so
, 1, 0x00);
147 /* frame memory I/F */
148 write_reg(sohandle
, so
, 0, 0xb3);
149 write_data(sohandle
, so
, data_frame_if
, ARRAY_SIZE(data_frame_if
));
151 /* display mode and frame memory write mode */
152 write_reg(sohandle
, so
, 0, 0xb4);
153 write_reg(sohandle
, so
, 1, 0x00); /* DBI, internal clock */
156 write_reg(sohandle
, so
, 0, 0xc0);
157 write_data(sohandle
, so
, data_panel
, ARRAY_SIZE(data_panel
));
159 /* timing (normal) */
160 write_reg(sohandle
, so
, 0, 0xc1);
161 write_data(sohandle
, so
, data_timing
, ARRAY_SIZE(data_timing
));
163 /* timing (partial) */
164 write_reg(sohandle
, so
, 0, 0xc2);
165 write_data(sohandle
, so
, data_timing
, ARRAY_SIZE(data_timing
));
168 write_reg(sohandle
, so
, 0, 0xc3);
169 write_data(sohandle
, so
, data_timing
, ARRAY_SIZE(data_timing
));
171 /* timing (source/VCOM/gate driving) */
172 write_reg(sohandle
, so
, 0, 0xc4);
173 write_data(sohandle
, so
, data_timing_src
, ARRAY_SIZE(data_timing_src
));
176 write_reg(sohandle
, so
, 0, 0xc8);
177 write_data(sohandle
, so
, data_gamma
, ARRAY_SIZE(data_gamma
));
180 write_reg(sohandle
, so
, 0, 0xc9);
181 write_data(sohandle
, so
, data_gamma
, ARRAY_SIZE(data_gamma
));
184 write_reg(sohandle
, so
, 0, 0xca);
185 write_data(sohandle
, so
, data_gamma
, ARRAY_SIZE(data_gamma
));
188 write_reg(sohandle
, so
, 0, 0xd0);
189 write_data(sohandle
, so
, data_power
, ARRAY_SIZE(data_power
));
192 write_reg(sohandle
, so
, 0, 0xd1);
193 write_reg(sohandle
, so
, 1, 0x00);
194 write_reg(sohandle
, so
, 1, 0x0f);
195 write_reg(sohandle
, so
, 1, 0x02);
198 write_reg(sohandle
, so
, 0, 0xd2);
199 write_reg(sohandle
, so
, 1, 0x63);
200 write_reg(sohandle
, so
, 1, 0x24);
202 /* power (partial) */
203 write_reg(sohandle
, so
, 0, 0xd3);
204 write_reg(sohandle
, so
, 1, 0x63);
205 write_reg(sohandle
, so
, 1, 0x24);
208 write_reg(sohandle
, so
, 0, 0xd4);
209 write_reg(sohandle
, so
, 1, 0x63);
210 write_reg(sohandle
, so
, 1, 0x24);
212 write_reg(sohandle
, so
, 0, 0xd8);
213 write_reg(sohandle
, so
, 1, 0x77);
214 write_reg(sohandle
, so
, 1, 0x77);
217 write_reg(sohandle
, so
, 0, 0x35);
218 write_reg(sohandle
, so
, 1, 0x00);
221 write_reg(sohandle
, so
, 0, 0x44);
222 write_reg(sohandle
, so
, 1, 0x00);
223 write_reg(sohandle
, so
, 1, 0x00);
226 write_reg(sohandle
, so
, 0, 0x2a);
227 write_reg(sohandle
, so
, 1, 0x00);
228 write_reg(sohandle
, so
, 1, 0x00);
229 write_reg(sohandle
, so
, 1, 0x00);
230 write_reg(sohandle
, so
, 1, 0xef);
233 write_reg(sohandle
, so
, 0, 0x2b);
234 write_reg(sohandle
, so
, 1, 0x00);
235 write_reg(sohandle
, so
, 1, 0x00);
236 write_reg(sohandle
, so
, 1, 0x01);
237 write_reg(sohandle
, so
, 1, 0x8f);
239 /* exit sleep mode */
240 write_reg(sohandle
, so
, 0, 0x11);
245 clear_memory(sohandle
, so
);
248 write_reg(sohandle
, so
, 0, 0x29);
251 write_memory_start(sohandle
, so
);
254 int kfr2r09_lcd_setup(void *board_data
, void *sohandle
,
255 struct sh_mobile_lcdc_sys_bus_ops
*so
)
258 gpio_set_value(GPIO_PTF4
, 0); /* PROTECT/ -> L */
259 gpio_set_value(GPIO_PTE4
, 0); /* LCD_RST/ -> L */
260 gpio_set_value(GPIO_PTF4
, 1); /* PROTECT/ -> H */
262 gpio_set_value(GPIO_PTE4
, 1); /* LCD_RST/ -> H */
264 gpio_set_value(GPIO_PTF4
, 0); /* PROTECT/ -> L */
267 if (read_device_code(sohandle
, so
) != 0x01221517)
270 pr_info("KFR2R09 WQVGA LCD Module detected.\n");
272 display_on(sohandle
, so
);
276 void kfr2r09_lcd_start(void *board_data
, void *sohandle
,
277 struct sh_mobile_lcdc_sys_bus_ops
*so
)
279 write_memory_start(sohandle
, so
);
282 #define CTRL_CKSW 0x10
283 #define CTRL_C10 0x20
284 #define CTRL_CPSW 0x80
285 #define MAIN_MLED4 0x40
286 #define MAIN_MSW 0x80
288 static int kfr2r09_lcd_backlight(int on
)
290 struct i2c_adapter
*a
;
292 unsigned char buf
[2];
295 a
= i2c_get_adapter(0);
301 buf
[1] = CTRL_CPSW
| CTRL_C10
| CTRL_CKSW
;
309 ret
= i2c_transfer(a
, &msg
, 1);
315 buf
[1] = MAIN_MSW
| MAIN_MLED4
| 0x0c;
323 ret
= i2c_transfer(a
, &msg
, 1);
330 void kfr2r09_lcd_on(void *board_data
)
332 kfr2r09_lcd_backlight(1);
335 void kfr2r09_lcd_off(void *board_data
)
337 kfr2r09_lcd_backlight(0);