2018-09-17 10:26:31 +02:00
|
|
|
/*
|
|
|
|
* Copyright (C) 2018 Koen Zandberg
|
2023-07-09 11:57:59 +02:00
|
|
|
* 2023 Gunar Schorcht
|
2018-09-17 10:26:31 +02:00
|
|
|
*
|
|
|
|
* This file is subject to the terms and conditions of the GNU Lesser
|
|
|
|
* General Public License v2.1. See the file LICENSE in the top level
|
|
|
|
* directory for more details.
|
|
|
|
*/
|
|
|
|
|
|
|
|
/**
|
|
|
|
* @ingroup drivers_ili9341
|
|
|
|
* @{
|
|
|
|
*
|
|
|
|
* @file
|
|
|
|
* @brief Device driver implementation for the ili9341 display controller
|
|
|
|
*
|
|
|
|
* @author Koen Zandberg <koen@bergzand.net>
|
2023-07-09 11:57:59 +02:00
|
|
|
* @author Gunar Schorcht <gunar@schorcht.net>
|
2018-09-17 10:26:31 +02:00
|
|
|
*
|
|
|
|
* @}
|
|
|
|
*/
|
|
|
|
|
2020-01-15 15:33:32 +01:00
|
|
|
#include <assert.h>
|
2018-09-17 10:26:31 +02:00
|
|
|
#include <string.h>
|
|
|
|
#include "byteorder.h"
|
|
|
|
#include "periph/spi.h"
|
2020-12-02 11:44:14 +01:00
|
|
|
#include "kernel_defines.h"
|
2023-07-09 11:57:59 +02:00
|
|
|
#include "ztimer.h"
|
2020-12-02 11:44:14 +01:00
|
|
|
|
2018-09-17 10:26:31 +02:00
|
|
|
#include "ili9341.h"
|
|
|
|
#include "ili9341_internal.h"
|
2021-03-15 15:13:23 +01:00
|
|
|
#include "lcd.h"
|
|
|
|
#include "lcd_internal.h"
|
2018-09-17 10:26:31 +02:00
|
|
|
|
2020-10-22 11:34:31 +02:00
|
|
|
#define ENABLE_DEBUG 0
|
2018-09-17 10:26:31 +02:00
|
|
|
#include "debug.h"
|
|
|
|
|
|
|
|
/* datasheet page 178, table converted to equation.
|
|
|
|
* gvdd in 1mv increments: 4850 = 4.85V */
|
|
|
|
static uint8_t _ili9341_calc_pwrctl1(uint16_t gvdd)
|
|
|
|
{
|
|
|
|
return (gvdd - 2850) / 50;
|
|
|
|
}
|
|
|
|
|
|
|
|
static uint8_t _ili9341_calc_vmh(uint16_t vcomh)
|
|
|
|
{
|
|
|
|
return (vcomh - 2700) / 25;
|
|
|
|
}
|
|
|
|
|
|
|
|
static uint8_t _ili9341_calc_vml(int16_t vcoml)
|
|
|
|
{
|
|
|
|
return (vcoml + 2500) / 25;
|
|
|
|
}
|
|
|
|
|
|
|
|
|
2021-03-15 15:13:23 +01:00
|
|
|
static int _init(lcd_t *dev, const lcd_params_t *params)
|
2018-09-17 10:26:31 +02:00
|
|
|
{
|
2020-01-15 15:33:32 +01:00
|
|
|
assert(params->lines >= 16 && params->lines <= 320 && !(params->lines & 0x7));
|
2018-09-17 10:26:31 +02:00
|
|
|
|
2023-07-09 11:57:59 +02:00
|
|
|
uint8_t command_params[4] = { 0 };
|
2018-09-17 10:26:31 +02:00
|
|
|
|
|
|
|
/* Acquire once at release at the end */
|
2023-07-09 11:57:59 +02:00
|
|
|
lcd_ll_acquire(dev);
|
2021-03-15 15:13:23 +01:00
|
|
|
|
2018-09-17 10:26:31 +02:00
|
|
|
/* Soft Reset */
|
2023-07-09 11:57:59 +02:00
|
|
|
lcd_ll_write_cmd(dev, LCD_CMD_SWRESET, NULL, 0);
|
2021-10-21 18:24:07 +02:00
|
|
|
ztimer_sleep(ZTIMER_MSEC, 120);
|
2018-09-17 10:26:31 +02:00
|
|
|
|
|
|
|
/* Display off */
|
2023-07-09 11:57:59 +02:00
|
|
|
lcd_ll_write_cmd(dev, LCD_CMD_DISPOFF, NULL, 0);
|
2018-09-17 10:26:31 +02:00
|
|
|
|
|
|
|
/* PWRCTL1/2 */
|
2020-11-30 16:33:32 +01:00
|
|
|
command_params[0] = _ili9341_calc_pwrctl1(CONFIG_ILI9341_GVDD);
|
2023-07-09 11:57:59 +02:00
|
|
|
lcd_ll_write_cmd(dev, LCD_CMD_PWCTRL1, command_params, 1);
|
2018-09-17 10:26:31 +02:00
|
|
|
|
|
|
|
command_params[0] = 0x10; /* PWRCTL 0 0 0 */
|
2023-07-09 11:57:59 +02:00
|
|
|
lcd_ll_write_cmd(dev, LCD_CMD_PWCTRL2, command_params, 1);
|
2018-09-17 10:26:31 +02:00
|
|
|
|
|
|
|
/* VCOMCTL */
|
2020-11-30 16:35:43 +01:00
|
|
|
command_params[0] = _ili9341_calc_vmh(CONFIG_ILI9341_VCOMH);
|
2020-11-30 16:37:44 +01:00
|
|
|
command_params[1] = _ili9341_calc_vml(CONFIG_ILI9341_VCOML);
|
2023-07-09 11:57:59 +02:00
|
|
|
lcd_ll_write_cmd(dev, LCD_CMD_VMCTRL1, command_params, 2);
|
2018-09-17 10:26:31 +02:00
|
|
|
|
|
|
|
command_params[0] = 0x86;
|
2023-07-09 11:57:59 +02:00
|
|
|
lcd_ll_write_cmd(dev, LCD_CMD_VMCTRL2, command_params, 1);
|
2018-09-17 10:26:31 +02:00
|
|
|
|
|
|
|
/* Memory access CTL */
|
2021-08-24 11:04:12 +02:00
|
|
|
command_params[0] = dev->params->rotation;
|
2021-03-15 15:13:23 +01:00
|
|
|
command_params[0] |= dev->params->rgb ? 0 : LCD_MADCTL_BGR;
|
2023-07-09 11:57:59 +02:00
|
|
|
lcd_ll_write_cmd(dev, LCD_CMD_MADCTL, command_params, 1);
|
2018-09-17 10:26:31 +02:00
|
|
|
|
|
|
|
/* Frame control */
|
|
|
|
command_params[0] = 0x00;
|
|
|
|
command_params[1] = 0x18;
|
2023-07-09 11:57:59 +02:00
|
|
|
lcd_ll_write_cmd(dev, LCD_CMD_FRAMECTL1, command_params, 2);
|
2018-09-17 10:26:31 +02:00
|
|
|
|
|
|
|
/* Display function control */
|
|
|
|
command_params[0] = 0x08;
|
|
|
|
command_params[1] = 0x82;
|
2020-01-15 15:33:32 +01:00
|
|
|
/* number of lines, see datasheet p. 166 (DISCTRL::NL) */
|
|
|
|
command_params[2] = (params->lines >> 3) - 1;
|
2023-07-09 11:57:59 +02:00
|
|
|
lcd_ll_write_cmd(dev, LCD_CMD_DFUNC, command_params, 3);
|
2018-09-17 10:26:31 +02:00
|
|
|
|
|
|
|
/* Pixel format */
|
|
|
|
command_params[0] = 0x55; /* 16 bit mode */
|
2023-07-09 11:57:59 +02:00
|
|
|
lcd_ll_write_cmd(dev, LCD_CMD_PIXSET, command_params, 1);
|
2018-09-17 10:26:31 +02:00
|
|
|
|
|
|
|
command_params[0] = 0x01;
|
2023-07-09 11:57:59 +02:00
|
|
|
lcd_ll_write_cmd(dev, LCD_CMD_GAMSET, command_params, 1);
|
2018-09-17 10:26:31 +02:00
|
|
|
|
|
|
|
/* Gamma correction */
|
|
|
|
{
|
|
|
|
static const uint8_t gamma_pos[] = {
|
2021-03-15 15:13:23 +01:00
|
|
|
0x0F, 0x31, 0x2B, 0x0C, 0x0E, 0x08, 0x4E, 0xF1,
|
|
|
|
0x37, 0x07, 0x10, 0x03, 0x0E, 0x09, 0x00
|
2018-09-17 10:26:31 +02:00
|
|
|
};
|
2023-07-09 11:57:59 +02:00
|
|
|
lcd_ll_write_cmd(dev, LCD_CMD_PGAMCTRL, gamma_pos,
|
|
|
|
sizeof(gamma_pos));
|
2018-09-17 10:26:31 +02:00
|
|
|
}
|
|
|
|
{
|
|
|
|
static const uint8_t gamma_neg[] = {
|
2021-03-15 15:13:23 +01:00
|
|
|
0x00, 0x0E, 0x14, 0x03, 0x11, 0x07, 0x31, 0xC1,
|
|
|
|
0x48, 0x08, 0x0F, 0x0C, 0x31, 0x36, 0x0F
|
2018-09-17 10:26:31 +02:00
|
|
|
};
|
2023-07-09 11:57:59 +02:00
|
|
|
lcd_ll_write_cmd(dev, LCD_CMD_NGAMCTRL, gamma_neg,
|
|
|
|
sizeof(gamma_neg));
|
2018-09-17 10:26:31 +02:00
|
|
|
|
|
|
|
}
|
2020-01-11 23:46:02 +01:00
|
|
|
|
|
|
|
if (dev->params->inverted) {
|
2023-07-09 11:57:59 +02:00
|
|
|
lcd_ll_write_cmd(dev, LCD_CMD_DINVON, NULL, 0);
|
2020-01-11 23:46:02 +01:00
|
|
|
}
|
2018-09-17 10:26:31 +02:00
|
|
|
/* Sleep out (turn off sleep mode) */
|
2023-07-09 11:57:59 +02:00
|
|
|
lcd_ll_write_cmd(dev, LCD_CMD_SLPOUT, NULL, 0);
|
2018-09-17 10:26:31 +02:00
|
|
|
/* Display on */
|
2023-07-09 11:57:59 +02:00
|
|
|
lcd_ll_write_cmd(dev, LCD_CMD_DISPON, NULL, 0);
|
|
|
|
|
|
|
|
/* Finally release the device */
|
|
|
|
lcd_ll_release(dev);
|
|
|
|
|
2018-09-17 10:26:31 +02:00
|
|
|
return 0;
|
|
|
|
}
|
|
|
|
|
2021-03-15 15:13:23 +01:00
|
|
|
static void _set_area(const lcd_t *dev, uint16_t x1, uint16_t x2,
|
|
|
|
uint16_t y1, uint16_t y2)
|
2018-09-17 10:26:31 +02:00
|
|
|
{
|
2021-03-15 15:13:23 +01:00
|
|
|
be_uint16_t params[2];
|
2018-09-17 10:26:31 +02:00
|
|
|
|
2022-04-13 10:34:37 +02:00
|
|
|
x1 += dev->params->offset_x;
|
|
|
|
x2 += dev->params->offset_x;
|
|
|
|
y1 += dev->params->offset_y;
|
|
|
|
y2 += dev->params->offset_y;
|
|
|
|
|
2021-03-15 15:13:23 +01:00
|
|
|
params[0] = byteorder_htons(x1);
|
|
|
|
params[1] = byteorder_htons(x2);
|
2020-01-11 23:46:02 +01:00
|
|
|
|
2023-07-09 11:57:59 +02:00
|
|
|
/* Function is called by a high level function of the LCD driver where
|
|
|
|
* the device is already acquired. So we don't must acquire it here.
|
|
|
|
* Therefore the low level write command function is called. */
|
|
|
|
lcd_ll_write_cmd(dev, LCD_CMD_CASET, (uint8_t *)params,
|
|
|
|
sizeof(params));
|
2021-03-15 15:13:23 +01:00
|
|
|
params[0] = byteorder_htons(y1);
|
|
|
|
params[1] = byteorder_htons(y2);
|
2023-07-09 11:57:59 +02:00
|
|
|
lcd_ll_write_cmd(dev, LCD_CMD_PASET, (uint8_t *)params,
|
|
|
|
sizeof(params));
|
2018-09-17 10:26:31 +02:00
|
|
|
}
|
|
|
|
|
2021-03-15 15:13:23 +01:00
|
|
|
const lcd_driver_t lcd_ili9341_driver = {
|
|
|
|
.init = _init,
|
|
|
|
.set_area = _set_area,
|
|
|
|
};
|