1*aaf9128aSKuninori Morimoto // SPDX-License-Identifier: GPL-2.0
29f26e659SMagnus Damm /*
39f26e659SMagnus Damm * KFR2R09 LCD panel support
49f26e659SMagnus Damm *
59f26e659SMagnus Damm * Copyright (C) 2009 Magnus Damm
69f26e659SMagnus Damm *
79f26e659SMagnus Damm * Register settings based on the out-of-tree t33fb.c driver
89f26e659SMagnus Damm * Copyright (C) 2008 Lineo Solutions, Inc.
99f26e659SMagnus Damm */
109f26e659SMagnus Damm
119f26e659SMagnus Damm #include <linux/delay.h>
129f26e659SMagnus Damm #include <linux/err.h>
139f26e659SMagnus Damm #include <linux/fb.h>
149f26e659SMagnus Damm #include <linux/init.h>
159f26e659SMagnus Damm #include <linux/kernel.h>
169f26e659SMagnus Damm #include <linux/module.h>
179f26e659SMagnus Damm #include <linux/gpio.h>
189f26e659SMagnus Damm #include <video/sh_mobile_lcdc.h>
199f26e659SMagnus Damm #include <mach/kfr2r09.h>
209f26e659SMagnus Damm #include <cpu/sh7724.h>
219f26e659SMagnus Damm
229f26e659SMagnus Damm /* The on-board LCD module is a Hitachi TX07D34VM0AAA. This module is made
239f26e659SMagnus Damm * up of a 240x400 LCD hooked up to a R61517 driver IC. The driver IC is
249f26e659SMagnus Damm * communicating with the main port of the LCDC using an 18-bit SYS interface.
259f26e659SMagnus Damm *
269f26e659SMagnus Damm * The device code for this LCD module is 0x01221517.
279f26e659SMagnus Damm */
289f26e659SMagnus Damm
299f26e659SMagnus Damm static const unsigned char data_frame_if[] = {
309f26e659SMagnus Damm 0x02, /* WEMODE: 1=cont, 0=one-shot */
319f26e659SMagnus Damm 0x00, 0x00,
329f26e659SMagnus Damm 0x00, /* EPF, DFM */
339f26e659SMagnus Damm 0x02, /* RIM[1] : 1 (18bpp) */
349f26e659SMagnus Damm };
359f26e659SMagnus Damm
369f26e659SMagnus Damm static const unsigned char data_panel[] = {
379f26e659SMagnus Damm 0x0b,
389f26e659SMagnus Damm 0x63, /* 400 lines */
399f26e659SMagnus Damm 0x04, 0x00, 0x00, 0x04, 0x11, 0x00, 0x00,
409f26e659SMagnus Damm };
419f26e659SMagnus Damm
429f26e659SMagnus Damm static const unsigned char data_timing[] = {
439f26e659SMagnus Damm 0x00, 0x00, 0x13, 0x08, 0x08,
449f26e659SMagnus Damm };
459f26e659SMagnus Damm
469f26e659SMagnus Damm static const unsigned char data_timing_src[] = {
479f26e659SMagnus Damm 0x11, 0x01, 0x00, 0x01,
489f26e659SMagnus Damm };
499f26e659SMagnus Damm
509f26e659SMagnus Damm static const unsigned char data_gamma[] = {
519f26e659SMagnus Damm 0x01, 0x02, 0x08, 0x23, 0x03, 0x0c, 0x00, 0x06, 0x00, 0x00,
529f26e659SMagnus Damm 0x01, 0x00, 0x0c, 0x23, 0x03, 0x08, 0x02, 0x06, 0x00, 0x00,
539f26e659SMagnus Damm };
549f26e659SMagnus Damm
559f26e659SMagnus Damm static const unsigned char data_power[] = {
569f26e659SMagnus Damm 0x07, 0xc5, 0xdc, 0x02, 0x33, 0x0a,
579f26e659SMagnus Damm };
589f26e659SMagnus Damm
read_reg(void * sohandle,struct sh_mobile_lcdc_sys_bus_ops * so)599f26e659SMagnus Damm static unsigned long read_reg(void *sohandle,
609f26e659SMagnus Damm struct sh_mobile_lcdc_sys_bus_ops *so)
619f26e659SMagnus Damm {
629f26e659SMagnus Damm return so->read_data(sohandle);
639f26e659SMagnus Damm }
649f26e659SMagnus Damm
write_reg(void * sohandle,struct sh_mobile_lcdc_sys_bus_ops * so,int i,unsigned long v)659f26e659SMagnus Damm static void write_reg(void *sohandle,
669f26e659SMagnus Damm struct sh_mobile_lcdc_sys_bus_ops *so,
679f26e659SMagnus Damm int i, unsigned long v)
689f26e659SMagnus Damm {
699f26e659SMagnus Damm if (i)
709f26e659SMagnus Damm so->write_data(sohandle, v); /* PTH4/LCDRS High [param, 17:0] */
719f26e659SMagnus Damm else
729f26e659SMagnus Damm so->write_index(sohandle, v); /* PTH4/LCDRS Low [cmd, 7:0] */
739f26e659SMagnus Damm }
749f26e659SMagnus Damm
write_data(void * sohandle,struct sh_mobile_lcdc_sys_bus_ops * so,unsigned char const * data,int no_data)759f26e659SMagnus Damm static void write_data(void *sohandle,
769f26e659SMagnus Damm struct sh_mobile_lcdc_sys_bus_ops *so,
779f26e659SMagnus Damm unsigned char const *data, int no_data)
789f26e659SMagnus Damm {
799f26e659SMagnus Damm int i;
809f26e659SMagnus Damm
819f26e659SMagnus Damm for (i = 0; i < no_data; i++)
829f26e659SMagnus Damm write_reg(sohandle, so, 1, data[i]);
839f26e659SMagnus Damm }
849f26e659SMagnus Damm
read_device_code(void * sohandle,struct sh_mobile_lcdc_sys_bus_ops * so)859f26e659SMagnus Damm static unsigned long read_device_code(void *sohandle,
869f26e659SMagnus Damm struct sh_mobile_lcdc_sys_bus_ops *so)
879f26e659SMagnus Damm {
889f26e659SMagnus Damm unsigned long device_code;
899f26e659SMagnus Damm
909f26e659SMagnus Damm /* access protect OFF */
919f26e659SMagnus Damm write_reg(sohandle, so, 0, 0xb0);
929f26e659SMagnus Damm write_reg(sohandle, so, 1, 0x00);
939f26e659SMagnus Damm
949f26e659SMagnus Damm /* deep standby OFF */
959f26e659SMagnus Damm write_reg(sohandle, so, 0, 0xb1);
969f26e659SMagnus Damm write_reg(sohandle, so, 1, 0x00);
979f26e659SMagnus Damm
989f26e659SMagnus Damm /* device code command */
999f26e659SMagnus Damm write_reg(sohandle, so, 0, 0xbf);
1009f26e659SMagnus Damm mdelay(50);
1019f26e659SMagnus Damm
1029f26e659SMagnus Damm /* dummy read */
1039f26e659SMagnus Damm read_reg(sohandle, so);
1049f26e659SMagnus Damm
1059f26e659SMagnus Damm /* read device code */
1069f26e659SMagnus Damm device_code = ((read_reg(sohandle, so) & 0xff) << 24);
1079f26e659SMagnus Damm device_code |= ((read_reg(sohandle, so) & 0xff) << 16);
1089f26e659SMagnus Damm device_code |= ((read_reg(sohandle, so) & 0xff) << 8);
1099f26e659SMagnus Damm device_code |= (read_reg(sohandle, so) & 0xff);
1109f26e659SMagnus Damm
1119f26e659SMagnus Damm return device_code;
1129f26e659SMagnus Damm }
1139f26e659SMagnus Damm
write_memory_start(void * sohandle,struct sh_mobile_lcdc_sys_bus_ops * so)1149f26e659SMagnus Damm static void write_memory_start(void *sohandle,
1159f26e659SMagnus Damm struct sh_mobile_lcdc_sys_bus_ops *so)
1169f26e659SMagnus Damm {
1179f26e659SMagnus Damm write_reg(sohandle, so, 0, 0x2c);
1189f26e659SMagnus Damm }
1199f26e659SMagnus Damm
clear_memory(void * sohandle,struct sh_mobile_lcdc_sys_bus_ops * so)1209f26e659SMagnus Damm static void clear_memory(void *sohandle,
1219f26e659SMagnus Damm struct sh_mobile_lcdc_sys_bus_ops *so)
1229f26e659SMagnus Damm {
1239f26e659SMagnus Damm int i;
1249f26e659SMagnus Damm
1259f26e659SMagnus Damm /* write start */
1269f26e659SMagnus Damm write_memory_start(sohandle, so);
1279f26e659SMagnus Damm
1289f26e659SMagnus Damm /* paint it black */
1299f26e659SMagnus Damm for (i = 0; i < (240 * 400); i++)
1309f26e659SMagnus Damm write_reg(sohandle, so, 1, 0x00);
1319f26e659SMagnus Damm }
1329f26e659SMagnus Damm
display_on(void * sohandle,struct sh_mobile_lcdc_sys_bus_ops * so)1339f26e659SMagnus Damm static void display_on(void *sohandle,
1349f26e659SMagnus Damm struct sh_mobile_lcdc_sys_bus_ops *so)
1359f26e659SMagnus Damm {
1369f26e659SMagnus Damm /* access protect off */
1379f26e659SMagnus Damm write_reg(sohandle, so, 0, 0xb0);
1389f26e659SMagnus Damm write_reg(sohandle, so, 1, 0x00);
1399f26e659SMagnus Damm
1409f26e659SMagnus Damm /* exit deep standby mode */
1419f26e659SMagnus Damm write_reg(sohandle, so, 0, 0xb1);
1429f26e659SMagnus Damm write_reg(sohandle, so, 1, 0x00);
1439f26e659SMagnus Damm
1449f26e659SMagnus Damm /* frame memory I/F */
1459f26e659SMagnus Damm write_reg(sohandle, so, 0, 0xb3);
1469f26e659SMagnus Damm write_data(sohandle, so, data_frame_if, ARRAY_SIZE(data_frame_if));
1479f26e659SMagnus Damm
1489f26e659SMagnus Damm /* display mode and frame memory write mode */
1499f26e659SMagnus Damm write_reg(sohandle, so, 0, 0xb4);
1509f26e659SMagnus Damm write_reg(sohandle, so, 1, 0x00); /* DBI, internal clock */
1519f26e659SMagnus Damm
1529f26e659SMagnus Damm /* panel */
1539f26e659SMagnus Damm write_reg(sohandle, so, 0, 0xc0);
1549f26e659SMagnus Damm write_data(sohandle, so, data_panel, ARRAY_SIZE(data_panel));
1559f26e659SMagnus Damm
1569f26e659SMagnus Damm /* timing (normal) */
1579f26e659SMagnus Damm write_reg(sohandle, so, 0, 0xc1);
1589f26e659SMagnus Damm write_data(sohandle, so, data_timing, ARRAY_SIZE(data_timing));
1599f26e659SMagnus Damm
1609f26e659SMagnus Damm /* timing (partial) */
1619f26e659SMagnus Damm write_reg(sohandle, so, 0, 0xc2);
1629f26e659SMagnus Damm write_data(sohandle, so, data_timing, ARRAY_SIZE(data_timing));
1639f26e659SMagnus Damm
1649f26e659SMagnus Damm /* timing (idle) */
1659f26e659SMagnus Damm write_reg(sohandle, so, 0, 0xc3);
1669f26e659SMagnus Damm write_data(sohandle, so, data_timing, ARRAY_SIZE(data_timing));
1679f26e659SMagnus Damm
1689f26e659SMagnus Damm /* timing (source/VCOM/gate driving) */
1699f26e659SMagnus Damm write_reg(sohandle, so, 0, 0xc4);
1709f26e659SMagnus Damm write_data(sohandle, so, data_timing_src, ARRAY_SIZE(data_timing_src));
1719f26e659SMagnus Damm
1729f26e659SMagnus Damm /* gamma (red) */
1739f26e659SMagnus Damm write_reg(sohandle, so, 0, 0xc8);
1749f26e659SMagnus Damm write_data(sohandle, so, data_gamma, ARRAY_SIZE(data_gamma));
1759f26e659SMagnus Damm
1769f26e659SMagnus Damm /* gamma (green) */
1779f26e659SMagnus Damm write_reg(sohandle, so, 0, 0xc9);
1789f26e659SMagnus Damm write_data(sohandle, so, data_gamma, ARRAY_SIZE(data_gamma));
1799f26e659SMagnus Damm
1809f26e659SMagnus Damm /* gamma (blue) */
1819f26e659SMagnus Damm write_reg(sohandle, so, 0, 0xca);
1829f26e659SMagnus Damm write_data(sohandle, so, data_gamma, ARRAY_SIZE(data_gamma));
1839f26e659SMagnus Damm
1849f26e659SMagnus Damm /* power (common) */
1859f26e659SMagnus Damm write_reg(sohandle, so, 0, 0xd0);
1869f26e659SMagnus Damm write_data(sohandle, so, data_power, ARRAY_SIZE(data_power));
1879f26e659SMagnus Damm
1889f26e659SMagnus Damm /* VCOM */
1899f26e659SMagnus Damm write_reg(sohandle, so, 0, 0xd1);
1909f26e659SMagnus Damm write_reg(sohandle, so, 1, 0x00);
1919f26e659SMagnus Damm write_reg(sohandle, so, 1, 0x0f);
1929f26e659SMagnus Damm write_reg(sohandle, so, 1, 0x02);
1939f26e659SMagnus Damm
1949f26e659SMagnus Damm /* power (normal) */
1959f26e659SMagnus Damm write_reg(sohandle, so, 0, 0xd2);
1969f26e659SMagnus Damm write_reg(sohandle, so, 1, 0x63);
1979f26e659SMagnus Damm write_reg(sohandle, so, 1, 0x24);
1989f26e659SMagnus Damm
1999f26e659SMagnus Damm /* power (partial) */
2009f26e659SMagnus Damm write_reg(sohandle, so, 0, 0xd3);
2019f26e659SMagnus Damm write_reg(sohandle, so, 1, 0x63);
2029f26e659SMagnus Damm write_reg(sohandle, so, 1, 0x24);
2039f26e659SMagnus Damm
2049f26e659SMagnus Damm /* power (idle) */
2059f26e659SMagnus Damm write_reg(sohandle, so, 0, 0xd4);
2069f26e659SMagnus Damm write_reg(sohandle, so, 1, 0x63);
2079f26e659SMagnus Damm write_reg(sohandle, so, 1, 0x24);
2089f26e659SMagnus Damm
2099f26e659SMagnus Damm write_reg(sohandle, so, 0, 0xd8);
2109f26e659SMagnus Damm write_reg(sohandle, so, 1, 0x77);
2119f26e659SMagnus Damm write_reg(sohandle, so, 1, 0x77);
2129f26e659SMagnus Damm
2139f26e659SMagnus Damm /* TE signal */
2149f26e659SMagnus Damm write_reg(sohandle, so, 0, 0x35);
2159f26e659SMagnus Damm write_reg(sohandle, so, 1, 0x00);
2169f26e659SMagnus Damm
2179f26e659SMagnus Damm /* TE signal line */
2189f26e659SMagnus Damm write_reg(sohandle, so, 0, 0x44);
2199f26e659SMagnus Damm write_reg(sohandle, so, 1, 0x00);
2209f26e659SMagnus Damm write_reg(sohandle, so, 1, 0x00);
2219f26e659SMagnus Damm
2229f26e659SMagnus Damm /* column address */
2239f26e659SMagnus Damm write_reg(sohandle, so, 0, 0x2a);
2249f26e659SMagnus Damm write_reg(sohandle, so, 1, 0x00);
2259f26e659SMagnus Damm write_reg(sohandle, so, 1, 0x00);
2269f26e659SMagnus Damm write_reg(sohandle, so, 1, 0x00);
2279f26e659SMagnus Damm write_reg(sohandle, so, 1, 0xef);
2289f26e659SMagnus Damm
2299f26e659SMagnus Damm /* page address */
2309f26e659SMagnus Damm write_reg(sohandle, so, 0, 0x2b);
2319f26e659SMagnus Damm write_reg(sohandle, so, 1, 0x00);
2329f26e659SMagnus Damm write_reg(sohandle, so, 1, 0x00);
2339f26e659SMagnus Damm write_reg(sohandle, so, 1, 0x01);
2349f26e659SMagnus Damm write_reg(sohandle, so, 1, 0x8f);
2359f26e659SMagnus Damm
2369f26e659SMagnus Damm /* exit sleep mode */
2379f26e659SMagnus Damm write_reg(sohandle, so, 0, 0x11);
2389f26e659SMagnus Damm
2399f26e659SMagnus Damm mdelay(120);
2409f26e659SMagnus Damm
2419f26e659SMagnus Damm /* clear vram */
2429f26e659SMagnus Damm clear_memory(sohandle, so);
2439f26e659SMagnus Damm
2449f26e659SMagnus Damm /* display ON */
2459f26e659SMagnus Damm write_reg(sohandle, so, 0, 0x29);
2469f26e659SMagnus Damm mdelay(1);
2479f26e659SMagnus Damm
2489f26e659SMagnus Damm write_memory_start(sohandle, so);
2499f26e659SMagnus Damm }
2509f26e659SMagnus Damm
kfr2r09_lcd_setup(void * sohandle,struct sh_mobile_lcdc_sys_bus_ops * so)251018882aaSLaurent Pinchart int kfr2r09_lcd_setup(void *sohandle, struct sh_mobile_lcdc_sys_bus_ops *so)
2529f26e659SMagnus Damm {
2539f26e659SMagnus Damm /* power on */
2549f26e659SMagnus Damm gpio_set_value(GPIO_PTF4, 0); /* PROTECT/ -> L */
2559f26e659SMagnus Damm gpio_set_value(GPIO_PTE4, 0); /* LCD_RST/ -> L */
2569f26e659SMagnus Damm gpio_set_value(GPIO_PTF4, 1); /* PROTECT/ -> H */
2579f26e659SMagnus Damm udelay(1100);
2589f26e659SMagnus Damm gpio_set_value(GPIO_PTE4, 1); /* LCD_RST/ -> H */
2599f26e659SMagnus Damm udelay(10);
2609f26e659SMagnus Damm gpio_set_value(GPIO_PTF4, 0); /* PROTECT/ -> L */
2619f26e659SMagnus Damm mdelay(20);
2629f26e659SMagnus Damm
2639f26e659SMagnus Damm if (read_device_code(sohandle, so) != 0x01221517)
2649f26e659SMagnus Damm return -ENODEV;
2659f26e659SMagnus Damm
2669f26e659SMagnus Damm pr_info("KFR2R09 WQVGA LCD Module detected.\n");
2679f26e659SMagnus Damm
2689f26e659SMagnus Damm display_on(sohandle, so);
2699f26e659SMagnus Damm return 0;
2709f26e659SMagnus Damm }
2719f26e659SMagnus Damm
kfr2r09_lcd_start(void * sohandle,struct sh_mobile_lcdc_sys_bus_ops * so)272018882aaSLaurent Pinchart void kfr2r09_lcd_start(void *sohandle, struct sh_mobile_lcdc_sys_bus_ops *so)
273657bf0bdSMagnus Damm {
274657bf0bdSMagnus Damm write_memory_start(sohandle, so);
275657bf0bdSMagnus Damm }
276