1*e943da47SFelix Brack // SPDX-License-Identifier: GPL-2.0 2*e943da47SFelix Brack /* Driver for ORISE Technology OTM3225A SOC for TFT LCD 3*e943da47SFelix Brack * Copyright (C) 2017, EETS GmbH, Felix Brack <fb@ltec.ch> 4*e943da47SFelix Brack * 5*e943da47SFelix Brack * This driver implements a lcd device for the ORISE OTM3225A display 6*e943da47SFelix Brack * controller. The control interface to the display is SPI and the display's 7*e943da47SFelix Brack * memory is updated over the 16-bit RGB interface. 8*e943da47SFelix Brack * The main source of information for writing this driver was provided by the 9*e943da47SFelix Brack * OTM3225A datasheet from ORISE Technology. Some information arise from the 10*e943da47SFelix Brack * ILI9328 datasheet from ILITEK as well as from the datasheets and sample code 11*e943da47SFelix Brack * provided by Crystalfontz America Inc. who sells the CFAF240320A-032T, a 3.2" 12*e943da47SFelix Brack * TFT LC display using the OTM3225A controller. 13*e943da47SFelix Brack */ 14*e943da47SFelix Brack 15*e943da47SFelix Brack #include <linux/delay.h> 16*e943da47SFelix Brack #include <linux/device.h> 17*e943da47SFelix Brack #include <linux/kernel.h> 18*e943da47SFelix Brack #include <linux/lcd.h> 19*e943da47SFelix Brack #include <linux/module.h> 20*e943da47SFelix Brack #include <linux/spi/spi.h> 21*e943da47SFelix Brack 22*e943da47SFelix Brack #define OTM3225A_INDEX_REG 0x70 23*e943da47SFelix Brack #define OTM3225A_DATA_REG 0x72 24*e943da47SFelix Brack 25*e943da47SFelix Brack /* instruction register list */ 26*e943da47SFelix Brack #define DRIVER_OUTPUT_CTRL_1 0x01 27*e943da47SFelix Brack #define DRIVER_WAVEFORM_CTRL 0x02 28*e943da47SFelix Brack #define ENTRY_MODE 0x03 29*e943da47SFelix Brack #define SCALING_CTRL 0x04 30*e943da47SFelix Brack #define DISPLAY_CTRL_1 0x07 31*e943da47SFelix Brack #define DISPLAY_CTRL_2 0x08 32*e943da47SFelix Brack #define DISPLAY_CTRL_3 0x09 33*e943da47SFelix Brack #define FRAME_CYCLE_CTRL 0x0A 34*e943da47SFelix Brack #define EXT_DISP_IFACE_CTRL_1 0x0C 35*e943da47SFelix Brack #define FRAME_MAKER_POS 0x0D 36*e943da47SFelix Brack #define EXT_DISP_IFACE_CTRL_2 0x0F 37*e943da47SFelix Brack #define POWER_CTRL_1 0x10 38*e943da47SFelix Brack #define POWER_CTRL_2 0x11 39*e943da47SFelix Brack #define POWER_CTRL_3 0x12 40*e943da47SFelix Brack #define POWER_CTRL_4 0x13 41*e943da47SFelix Brack #define GRAM_ADDR_HORIZ_SET 0x20 42*e943da47SFelix Brack #define GRAM_ADDR_VERT_SET 0x21 43*e943da47SFelix Brack #define GRAM_READ_WRITE 0x22 44*e943da47SFelix Brack #define POWER_CTRL_7 0x29 45*e943da47SFelix Brack #define FRAME_RATE_CTRL 0x2B 46*e943da47SFelix Brack #define GAMMA_CTRL_1 0x30 47*e943da47SFelix Brack #define GAMMA_CTRL_2 0x31 48*e943da47SFelix Brack #define GAMMA_CTRL_3 0x32 49*e943da47SFelix Brack #define GAMMA_CTRL_4 0x35 50*e943da47SFelix Brack #define GAMMA_CTRL_5 0x36 51*e943da47SFelix Brack #define GAMMA_CTRL_6 0x37 52*e943da47SFelix Brack #define GAMMA_CTRL_7 0x38 53*e943da47SFelix Brack #define GAMMA_CTRL_8 0x39 54*e943da47SFelix Brack #define GAMMA_CTRL_9 0x3C 55*e943da47SFelix Brack #define GAMMA_CTRL_10 0x3D 56*e943da47SFelix Brack #define WINDOW_HORIZ_RAM_START 0x50 57*e943da47SFelix Brack #define WINDOW_HORIZ_RAM_END 0x51 58*e943da47SFelix Brack #define WINDOW_VERT_RAM_START 0x52 59*e943da47SFelix Brack #define WINDOW_VERT_RAM_END 0x53 60*e943da47SFelix Brack #define DRIVER_OUTPUT_CTRL_2 0x60 61*e943da47SFelix Brack #define BASE_IMG_DISPLAY_CTRL 0x61 62*e943da47SFelix Brack #define VERT_SCROLL_CTRL 0x6A 63*e943da47SFelix Brack #define PD1_DISPLAY_POS 0x80 64*e943da47SFelix Brack #define PD1_RAM_START 0x81 65*e943da47SFelix Brack #define PD1_RAM_END 0x82 66*e943da47SFelix Brack #define PD2_DISPLAY_POS 0x83 67*e943da47SFelix Brack #define PD2_RAM_START 0x84 68*e943da47SFelix Brack #define PD2_RAM_END 0x85 69*e943da47SFelix Brack #define PANEL_IFACE_CTRL_1 0x90 70*e943da47SFelix Brack #define PANEL_IFACE_CTRL_2 0x92 71*e943da47SFelix Brack #define PANEL_IFACE_CTRL_4 0x95 72*e943da47SFelix Brack #define PANEL_IFACE_CTRL_5 0x97 73*e943da47SFelix Brack 74*e943da47SFelix Brack struct otm3225a_data { 75*e943da47SFelix Brack struct spi_device *spi; 76*e943da47SFelix Brack struct lcd_device *ld; 77*e943da47SFelix Brack int power; 78*e943da47SFelix Brack }; 79*e943da47SFelix Brack 80*e943da47SFelix Brack struct otm3225a_spi_instruction { 81*e943da47SFelix Brack unsigned char reg; /* register to write */ 82*e943da47SFelix Brack unsigned short value; /* data to write to 'reg' */ 83*e943da47SFelix Brack unsigned short delay; /* delay in ms after write */ 84*e943da47SFelix Brack }; 85*e943da47SFelix Brack 86*e943da47SFelix Brack static struct otm3225a_spi_instruction display_init[] = { 87*e943da47SFelix Brack { DRIVER_OUTPUT_CTRL_1, 0x0000, 0 }, 88*e943da47SFelix Brack { DRIVER_WAVEFORM_CTRL, 0x0700, 0 }, 89*e943da47SFelix Brack { ENTRY_MODE, 0x50A0, 0 }, 90*e943da47SFelix Brack { SCALING_CTRL, 0x0000, 0 }, 91*e943da47SFelix Brack { DISPLAY_CTRL_2, 0x0606, 0 }, 92*e943da47SFelix Brack { DISPLAY_CTRL_3, 0x0000, 0 }, 93*e943da47SFelix Brack { FRAME_CYCLE_CTRL, 0x0000, 0 }, 94*e943da47SFelix Brack { EXT_DISP_IFACE_CTRL_1, 0x0000, 0 }, 95*e943da47SFelix Brack { FRAME_MAKER_POS, 0x0000, 0 }, 96*e943da47SFelix Brack { EXT_DISP_IFACE_CTRL_2, 0x0002, 0 }, 97*e943da47SFelix Brack { POWER_CTRL_2, 0x0007, 0 }, 98*e943da47SFelix Brack { POWER_CTRL_3, 0x0000, 0 }, 99*e943da47SFelix Brack { POWER_CTRL_4, 0x0000, 200 }, 100*e943da47SFelix Brack { DISPLAY_CTRL_1, 0x0101, 0 }, 101*e943da47SFelix Brack { POWER_CTRL_1, 0x12B0, 0 }, 102*e943da47SFelix Brack { POWER_CTRL_2, 0x0007, 0 }, 103*e943da47SFelix Brack { POWER_CTRL_3, 0x01BB, 50 }, 104*e943da47SFelix Brack { POWER_CTRL_4, 0x0013, 0 }, 105*e943da47SFelix Brack { POWER_CTRL_7, 0x0010, 50 }, 106*e943da47SFelix Brack { GAMMA_CTRL_1, 0x000A, 0 }, 107*e943da47SFelix Brack { GAMMA_CTRL_2, 0x1326, 0 }, 108*e943da47SFelix Brack { GAMMA_CTRL_3, 0x0A29, 0 }, 109*e943da47SFelix Brack { GAMMA_CTRL_4, 0x0A0A, 0 }, 110*e943da47SFelix Brack { GAMMA_CTRL_5, 0x1E03, 0 }, 111*e943da47SFelix Brack { GAMMA_CTRL_6, 0x031E, 0 }, 112*e943da47SFelix Brack { GAMMA_CTRL_7, 0x0706, 0 }, 113*e943da47SFelix Brack { GAMMA_CTRL_8, 0x0303, 0 }, 114*e943da47SFelix Brack { GAMMA_CTRL_9, 0x010E, 0 }, 115*e943da47SFelix Brack { GAMMA_CTRL_10, 0x040E, 0 }, 116*e943da47SFelix Brack { WINDOW_HORIZ_RAM_START, 0x0000, 0 }, 117*e943da47SFelix Brack { WINDOW_HORIZ_RAM_END, 0x00EF, 0 }, 118*e943da47SFelix Brack { WINDOW_VERT_RAM_START, 0x0000, 0 }, 119*e943da47SFelix Brack { WINDOW_VERT_RAM_END, 0x013F, 0 }, 120*e943da47SFelix Brack { DRIVER_OUTPUT_CTRL_2, 0x2700, 0 }, 121*e943da47SFelix Brack { BASE_IMG_DISPLAY_CTRL, 0x0001, 0 }, 122*e943da47SFelix Brack { VERT_SCROLL_CTRL, 0x0000, 0 }, 123*e943da47SFelix Brack { PD1_DISPLAY_POS, 0x0000, 0 }, 124*e943da47SFelix Brack { PD1_RAM_START, 0x0000, 0 }, 125*e943da47SFelix Brack { PD1_RAM_END, 0x0000, 0 }, 126*e943da47SFelix Brack { PD2_DISPLAY_POS, 0x0000, 0 }, 127*e943da47SFelix Brack { PD2_RAM_START, 0x0000, 0 }, 128*e943da47SFelix Brack { PD2_RAM_END, 0x0000, 0 }, 129*e943da47SFelix Brack { PANEL_IFACE_CTRL_1, 0x0010, 0 }, 130*e943da47SFelix Brack { PANEL_IFACE_CTRL_2, 0x0000, 0 }, 131*e943da47SFelix Brack { PANEL_IFACE_CTRL_4, 0x0210, 0 }, 132*e943da47SFelix Brack { PANEL_IFACE_CTRL_5, 0x0000, 0 }, 133*e943da47SFelix Brack { DISPLAY_CTRL_1, 0x0133, 0 }, 134*e943da47SFelix Brack }; 135*e943da47SFelix Brack 136*e943da47SFelix Brack static struct otm3225a_spi_instruction display_enable_rgb_interface[] = { 137*e943da47SFelix Brack { ENTRY_MODE, 0x1080, 0 }, 138*e943da47SFelix Brack { GRAM_ADDR_HORIZ_SET, 0x0000, 0 }, 139*e943da47SFelix Brack { GRAM_ADDR_VERT_SET, 0x0000, 0 }, 140*e943da47SFelix Brack { EXT_DISP_IFACE_CTRL_1, 0x0111, 500 }, 141*e943da47SFelix Brack }; 142*e943da47SFelix Brack 143*e943da47SFelix Brack static struct otm3225a_spi_instruction display_off[] = { 144*e943da47SFelix Brack { DISPLAY_CTRL_1, 0x0131, 100 }, 145*e943da47SFelix Brack { DISPLAY_CTRL_1, 0x0130, 100 }, 146*e943da47SFelix Brack { DISPLAY_CTRL_1, 0x0100, 0 }, 147*e943da47SFelix Brack { POWER_CTRL_1, 0x0280, 0 }, 148*e943da47SFelix Brack { POWER_CTRL_3, 0x018B, 0 }, 149*e943da47SFelix Brack }; 150*e943da47SFelix Brack 151*e943da47SFelix Brack static struct otm3225a_spi_instruction display_on[] = { 152*e943da47SFelix Brack { POWER_CTRL_1, 0x1280, 0 }, 153*e943da47SFelix Brack { DISPLAY_CTRL_1, 0x0101, 100 }, 154*e943da47SFelix Brack { DISPLAY_CTRL_1, 0x0121, 0 }, 155*e943da47SFelix Brack { DISPLAY_CTRL_1, 0x0123, 100 }, 156*e943da47SFelix Brack { DISPLAY_CTRL_1, 0x0133, 10 }, 157*e943da47SFelix Brack }; 158*e943da47SFelix Brack 159*e943da47SFelix Brack static void otm3225a_write(struct spi_device *spi, 160*e943da47SFelix Brack struct otm3225a_spi_instruction *instruction, 161*e943da47SFelix Brack unsigned int count) 162*e943da47SFelix Brack { 163*e943da47SFelix Brack unsigned char buf[3]; 164*e943da47SFelix Brack 165*e943da47SFelix Brack while (count--) { 166*e943da47SFelix Brack /* address register using index register */ 167*e943da47SFelix Brack buf[0] = OTM3225A_INDEX_REG; 168*e943da47SFelix Brack buf[1] = 0x00; 169*e943da47SFelix Brack buf[2] = instruction->reg; 170*e943da47SFelix Brack spi_write(spi, buf, 3); 171*e943da47SFelix Brack 172*e943da47SFelix Brack /* write data to addressed register */ 173*e943da47SFelix Brack buf[0] = OTM3225A_DATA_REG; 174*e943da47SFelix Brack buf[1] = (instruction->value >> 8) & 0xff; 175*e943da47SFelix Brack buf[2] = instruction->value & 0xff; 176*e943da47SFelix Brack spi_write(spi, buf, 3); 177*e943da47SFelix Brack 178*e943da47SFelix Brack /* execute delay if any */ 179*e943da47SFelix Brack if (instruction->delay) 180*e943da47SFelix Brack msleep(instruction->delay); 181*e943da47SFelix Brack instruction++; 182*e943da47SFelix Brack } 183*e943da47SFelix Brack } 184*e943da47SFelix Brack 185*e943da47SFelix Brack static int otm3225a_set_power(struct lcd_device *ld, int power) 186*e943da47SFelix Brack { 187*e943da47SFelix Brack struct otm3225a_data *dd = lcd_get_data(ld); 188*e943da47SFelix Brack 189*e943da47SFelix Brack if (power == dd->power) 190*e943da47SFelix Brack return 0; 191*e943da47SFelix Brack 192*e943da47SFelix Brack if (power > FB_BLANK_UNBLANK) 193*e943da47SFelix Brack otm3225a_write(dd->spi, display_off, ARRAY_SIZE(display_off)); 194*e943da47SFelix Brack else 195*e943da47SFelix Brack otm3225a_write(dd->spi, display_on, ARRAY_SIZE(display_on)); 196*e943da47SFelix Brack dd->power = power; 197*e943da47SFelix Brack 198*e943da47SFelix Brack return 0; 199*e943da47SFelix Brack } 200*e943da47SFelix Brack 201*e943da47SFelix Brack static int otm3225a_get_power(struct lcd_device *ld) 202*e943da47SFelix Brack { 203*e943da47SFelix Brack struct otm3225a_data *dd = lcd_get_data(ld); 204*e943da47SFelix Brack 205*e943da47SFelix Brack return dd->power; 206*e943da47SFelix Brack } 207*e943da47SFelix Brack 208*e943da47SFelix Brack static struct lcd_ops otm3225a_ops = { 209*e943da47SFelix Brack .set_power = otm3225a_set_power, 210*e943da47SFelix Brack .get_power = otm3225a_get_power, 211*e943da47SFelix Brack }; 212*e943da47SFelix Brack 213*e943da47SFelix Brack static int otm3225a_probe(struct spi_device *spi) 214*e943da47SFelix Brack { 215*e943da47SFelix Brack struct otm3225a_data *dd; 216*e943da47SFelix Brack struct lcd_device *ld; 217*e943da47SFelix Brack struct device *dev = &spi->dev; 218*e943da47SFelix Brack 219*e943da47SFelix Brack dd = devm_kzalloc(dev, sizeof(struct otm3225a_data), GFP_KERNEL); 220*e943da47SFelix Brack if (dd == NULL) 221*e943da47SFelix Brack return -ENOMEM; 222*e943da47SFelix Brack 223*e943da47SFelix Brack ld = devm_lcd_device_register(dev, dev_name(dev), dev, dd, 224*e943da47SFelix Brack &otm3225a_ops); 225*e943da47SFelix Brack if (IS_ERR(ld)) 226*e943da47SFelix Brack return PTR_ERR(ld); 227*e943da47SFelix Brack 228*e943da47SFelix Brack dd->spi = spi; 229*e943da47SFelix Brack dd->ld = ld; 230*e943da47SFelix Brack dev_set_drvdata(dev, dd); 231*e943da47SFelix Brack 232*e943da47SFelix Brack dev_info(dev, "Initializing and switching to RGB interface"); 233*e943da47SFelix Brack otm3225a_write(spi, display_init, ARRAY_SIZE(display_init)); 234*e943da47SFelix Brack otm3225a_write(spi, display_enable_rgb_interface, 235*e943da47SFelix Brack ARRAY_SIZE(display_enable_rgb_interface)); 236*e943da47SFelix Brack return 0; 237*e943da47SFelix Brack } 238*e943da47SFelix Brack 239*e943da47SFelix Brack static struct spi_driver otm3225a_driver = { 240*e943da47SFelix Brack .driver = { 241*e943da47SFelix Brack .name = "otm3225a", 242*e943da47SFelix Brack .owner = THIS_MODULE, 243*e943da47SFelix Brack }, 244*e943da47SFelix Brack .probe = otm3225a_probe, 245*e943da47SFelix Brack }; 246*e943da47SFelix Brack 247*e943da47SFelix Brack module_spi_driver(otm3225a_driver); 248*e943da47SFelix Brack 249*e943da47SFelix Brack MODULE_AUTHOR("Felix Brack <fb@ltec.ch>"); 250*e943da47SFelix Brack MODULE_DESCRIPTION("OTM3225A TFT LCD driver"); 251*e943da47SFelix Brack MODULE_VERSION("1.0.0"); 252*e943da47SFelix Brack MODULE_LICENSE("GPL v2"); 253