cachepc-linux

Fork of AMDESE/linux with modifications for CachePC side-channel attack
git clone https://git.sinitax.com/sinitax/cachepc-linux
Log | Files | Refs | README | LICENSE | sfeed.txt

fb_ili9341.c (4628B)


      1// SPDX-License-Identifier: GPL-2.0+
      2/*
      3 * FB driver for the ILI9341 LCD display controller
      4 *
      5 * This display uses 9-bit SPI: Data/Command bit + 8 data bits
      6 * For platforms that doesn't support 9-bit, the driver is capable
      7 * of emulating this using 8-bit transfer.
      8 * This is done by transferring eight 9-bit words in 9 bytes.
      9 *
     10 * Copyright (C) 2013 Christian Vogelgsang
     11 * Based on adafruit22fb.c by Noralf Tronnes
     12 */
     13
     14#include <linux/module.h>
     15#include <linux/kernel.h>
     16#include <linux/init.h>
     17#include <linux/delay.h>
     18#include <video/mipi_display.h>
     19
     20#include "fbtft.h"
     21
     22#define DRVNAME		"fb_ili9341"
     23#define WIDTH		240
     24#define HEIGHT		320
     25#define TXBUFLEN	(4 * PAGE_SIZE)
     26#define DEFAULT_GAMMA	"1F 1A 18 0A 0F 06 45 87 32 0A 07 02 07 05 00\n" \
     27			"00 25 27 05 10 09 3A 78 4D 05 18 0D 38 3A 1F"
     28
     29static int init_display(struct fbtft_par *par)
     30{
     31	par->fbtftops.reset(par);
     32
     33	/* startup sequence for MI0283QT-9A */
     34	write_reg(par, MIPI_DCS_SOFT_RESET);
     35	mdelay(5);
     36	write_reg(par, MIPI_DCS_SET_DISPLAY_OFF);
     37	/* --------------------------------------------------------- */
     38	write_reg(par, 0xCF, 0x00, 0x83, 0x30);
     39	write_reg(par, 0xED, 0x64, 0x03, 0x12, 0x81);
     40	write_reg(par, 0xE8, 0x85, 0x01, 0x79);
     41	write_reg(par, 0xCB, 0x39, 0X2C, 0x00, 0x34, 0x02);
     42	write_reg(par, 0xF7, 0x20);
     43	write_reg(par, 0xEA, 0x00, 0x00);
     44	/* ------------power control-------------------------------- */
     45	write_reg(par, 0xC0, 0x26);
     46	write_reg(par, 0xC1, 0x11);
     47	/* ------------VCOM --------- */
     48	write_reg(par, 0xC5, 0x35, 0x3E);
     49	write_reg(par, 0xC7, 0xBE);
     50	/* ------------memory access control------------------------ */
     51	write_reg(par, MIPI_DCS_SET_PIXEL_FORMAT, 0x55); /* 16bit pixel */
     52	/* ------------frame rate----------------------------------- */
     53	write_reg(par, 0xB1, 0x00, 0x1B);
     54	/* ------------Gamma---------------------------------------- */
     55	/* write_reg(par, 0xF2, 0x08); */ /* Gamma Function Disable */
     56	write_reg(par, MIPI_DCS_SET_GAMMA_CURVE, 0x01);
     57	/* ------------display-------------------------------------- */
     58	write_reg(par, 0xB7, 0x07); /* entry mode set */
     59	write_reg(par, 0xB6, 0x0A, 0x82, 0x27, 0x00);
     60	write_reg(par, MIPI_DCS_EXIT_SLEEP_MODE);
     61	mdelay(100);
     62	write_reg(par, MIPI_DCS_SET_DISPLAY_ON);
     63	mdelay(20);
     64
     65	return 0;
     66}
     67
     68static void set_addr_win(struct fbtft_par *par, int xs, int ys, int xe, int ye)
     69{
     70	write_reg(par, MIPI_DCS_SET_COLUMN_ADDRESS,
     71		  (xs >> 8) & 0xFF, xs & 0xFF, (xe >> 8) & 0xFF, xe & 0xFF);
     72
     73	write_reg(par, MIPI_DCS_SET_PAGE_ADDRESS,
     74		  (ys >> 8) & 0xFF, ys & 0xFF, (ye >> 8) & 0xFF, ye & 0xFF);
     75
     76	write_reg(par, MIPI_DCS_WRITE_MEMORY_START);
     77}
     78
     79#define MEM_Y   BIT(7) /* MY row address order */
     80#define MEM_X   BIT(6) /* MX column address order */
     81#define MEM_V   BIT(5) /* MV row / column exchange */
     82#define MEM_L   BIT(4) /* ML vertical refresh order */
     83#define MEM_H   BIT(2) /* MH horizontal refresh order */
     84#define MEM_BGR (3) /* RGB-BGR Order */
     85static int set_var(struct fbtft_par *par)
     86{
     87	switch (par->info->var.rotate) {
     88	case 0:
     89		write_reg(par, MIPI_DCS_SET_ADDRESS_MODE,
     90			  MEM_X | (par->bgr << MEM_BGR));
     91		break;
     92	case 270:
     93		write_reg(par, MIPI_DCS_SET_ADDRESS_MODE,
     94			  MEM_V | MEM_L | (par->bgr << MEM_BGR));
     95		break;
     96	case 180:
     97		write_reg(par, MIPI_DCS_SET_ADDRESS_MODE,
     98			  MEM_Y | (par->bgr << MEM_BGR));
     99		break;
    100	case 90:
    101		write_reg(par, MIPI_DCS_SET_ADDRESS_MODE,
    102			  MEM_Y | MEM_X | MEM_V | (par->bgr << MEM_BGR));
    103		break;
    104	}
    105
    106	return 0;
    107}
    108
    109/*
    110 * Gamma string format:
    111 *  Positive: Par1 Par2 [...] Par15
    112 *  Negative: Par1 Par2 [...] Par15
    113 */
    114#define CURVE(num, idx)  curves[(num) * par->gamma.num_values + (idx)]
    115static int set_gamma(struct fbtft_par *par, u32 *curves)
    116{
    117	int i;
    118
    119	for (i = 0; i < par->gamma.num_curves; i++)
    120		write_reg(par, 0xE0 + i,
    121			  CURVE(i, 0), CURVE(i, 1), CURVE(i, 2),
    122			  CURVE(i, 3), CURVE(i, 4), CURVE(i, 5),
    123			  CURVE(i, 6), CURVE(i, 7), CURVE(i, 8),
    124			  CURVE(i, 9), CURVE(i, 10), CURVE(i, 11),
    125			  CURVE(i, 12), CURVE(i, 13), CURVE(i, 14));
    126
    127	return 0;
    128}
    129
    130#undef CURVE
    131
    132static struct fbtft_display display = {
    133	.regwidth = 8,
    134	.width = WIDTH,
    135	.height = HEIGHT,
    136	.txbuflen = TXBUFLEN,
    137	.gamma_num = 2,
    138	.gamma_len = 15,
    139	.gamma = DEFAULT_GAMMA,
    140	.fbtftops = {
    141		.init_display = init_display,
    142		.set_addr_win = set_addr_win,
    143		.set_var = set_var,
    144		.set_gamma = set_gamma,
    145	},
    146};
    147
    148FBTFT_REGISTER_DRIVER(DRVNAME, "ilitek,ili9341", &display);
    149
    150MODULE_ALIAS("spi:" DRVNAME);
    151MODULE_ALIAS("platform:" DRVNAME);
    152MODULE_ALIAS("spi:ili9341");
    153MODULE_ALIAS("platform:ili9341");
    154
    155MODULE_DESCRIPTION("FB driver for the ILI9341 LCD display controller");
    156MODULE_AUTHOR("Christian Vogelgsang");
    157MODULE_LICENSE("GPL");