Linux kernel mirror (for testing) git.kernel.org/pub/scm/linux/kernel/git/torvalds/linux.git
kernel os linux

backlight: otm3225a: Add support for ORISE OTM3225A LCD SoC

This patch adds a LCD driver supporting the OTM3225A LCD SoC
from ORISE Technology. This device can drive TFT LC panels having a
resolution of 240x320 pixels. After initializing the OTM3225A using
it's SPI interface it switches to use 16-bib RGB as external
display interface.

Signed-off-by: Felix Brack <fb@ltec.ch>
Acked-by: Daniel Thompson <daniel.thompson@linaro.org>
Signed-off-by: Lee Jones <lee.jones@linaro.org>

authored by

Felix Brack and committed by
Lee Jones
e943da47 bb084c0f

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