You can not select more than 25 topics
Topics must start with a letter or number, can include dashes ('-') and can be up to 35 characters long.
494 lines
15 KiB
494 lines
15 KiB
/* |
|
* Copyright (c) 2025 Arduino SA |
|
* |
|
* SPDX-License-Identifier: Apache-2.0 |
|
*/ |
|
|
|
#define DT_DRV_COMPAT sitronix_st7701 |
|
|
|
#include <zephyr/kernel.h> |
|
#include <zephyr/drivers/display.h> |
|
#include <zephyr/drivers/mipi_dsi.h> |
|
#include <zephyr/drivers/gpio.h> |
|
#include <zephyr/sys/byteorder.h> |
|
#include <zephyr/logging/log.h> |
|
|
|
LOG_MODULE_REGISTER(st7701, CONFIG_DISPLAY_LOG_LEVEL); |
|
|
|
/* Command2 BKx selection command */ |
|
#define DSI_CMD2BKX_SEL 0xFF |
|
#define DSI_CMD2BK1_SEL 0x11 |
|
#define DSI_CMD2BK0_SEL 0x10 |
|
#define DSI_CMD2BKX_SEL_NONE 0x00 |
|
|
|
/* Command2, BK0 commands */ |
|
#define DSI_CMD2_BK0_PVGAMCTRL 0xB0 /* Positive Voltage Gamma Control */ |
|
#define DSI_CMD2_BK0_NVGAMCTRL 0xB1 /* Negative Voltage Gamma Control */ |
|
#define DSI_CMD2_BK0_LNESET 0xC0 /* Display Line setting */ |
|
#define DSI_CMD2_BK0_PORCTRL 0xC1 /* Porch control */ |
|
#define DSI_CMD2_BK0_INVSEL 0xC2 /* Inversion selection, Frame Rate Control */ |
|
|
|
/* Command2, BK1 commands */ |
|
#define DSI_CMD2_BK1_VRHS 0xB0 /* Vop amplitude setting */ |
|
#define DSI_CMD2_BK1_VCOM 0xB1 /* VCOM amplitude setting */ |
|
#define DSI_CMD2_BK1_VGHSS 0xB2 /* VGH Voltage setting */ |
|
#define DSI_CMD2_BK1_TESTCMD 0xB3 /* TEST Command Setting */ |
|
#define DSI_CMD2_BK1_VGLS 0xB5 /* VGL Voltage setting */ |
|
#define DSI_CMD2_BK1_PWCTLR1 0xB7 /* Power Control 1 */ |
|
#define DSI_CMD2_BK1_PWCTLR2 0xB8 /* Power Control 2 */ |
|
#define DSI_CMD2_BK1_SPD1 0xC1 /* Source pre_drive timing set1 */ |
|
#define DSI_CMD2_BK1_SPD2 0xC2 /* Source EQ2 Setting */ |
|
#define DSI_CMD2_BK1_MIPISET1 0xD0 /* MIPI Setting 1 */ |
|
|
|
#define ST7701_CMD_ID1 0xDA |
|
#define ST7701_ID 0xFF |
|
|
|
/* Write Control Display: brightness control. */ |
|
#define ST7701_WRCTRLD_BCTRL BIT(5) |
|
/* Write Control Display: display dimming. */ |
|
#define ST7701_WRCTRLD_DD BIT(3) |
|
/* Write Control Display: backlight. */ |
|
#define ST7701_WRCTRLD_BL BIT(2) |
|
|
|
/* Adaptive Brightness Control: off. */ |
|
#define ST7701_WRCABC_OFF 0x00U |
|
/* Adaptive Brightness Control: user interface. */ |
|
#define ST7701_WRCABC_UI 0x01U |
|
/* Adaptive Brightness Control: still picture. */ |
|
#define ST7701_WRCABC_ST 0x02U |
|
/* Adaptive Brightness Control: moving image. */ |
|
#define ST7701_WRCABC_MV 0x03U |
|
|
|
struct st7701_config { |
|
const struct device *mipi_dsi; |
|
const struct gpio_dt_spec reset; |
|
const struct gpio_dt_spec backlight; |
|
uint8_t data_lanes; |
|
uint16_t width; |
|
uint16_t height; |
|
uint8_t channel; |
|
uint16_t rotation; |
|
uint32_t hbp; |
|
uint32_t hsync; |
|
uint32_t hfp; |
|
uint32_t vbp; |
|
uint32_t vsync; |
|
uint32_t vfp; |
|
uint8_t gip_e0[4]; |
|
uint8_t gip_e1[12]; |
|
uint8_t gip_e2[14]; |
|
uint8_t gip_e3[5]; |
|
uint8_t gip_e4[3]; |
|
uint8_t gip_e5[17]; |
|
uint8_t gip_e6[5]; |
|
uint8_t gip_e7[3]; |
|
uint8_t gip_e8[17]; |
|
uint8_t gip_eb[8]; |
|
uint8_t gip_ec[3]; |
|
uint8_t gip_ed[17]; |
|
uint8_t pvgamctrl[17]; |
|
uint8_t nvgamctrl[17]; |
|
}; |
|
|
|
struct st7701_data { |
|
uint16_t xres; |
|
uint16_t yres; |
|
uint8_t dsi_pixel_format; |
|
enum display_pixel_format pixel_format; |
|
enum display_orientation orientation; |
|
}; |
|
|
|
static inline int st7701_dcs_write(const struct device *dev, uint8_t cmd, const void *buf, |
|
size_t len) |
|
{ |
|
const struct st7701_config *cfg = dev->config; |
|
int ret; |
|
|
|
ret = mipi_dsi_dcs_write(cfg->mipi_dsi, cfg->channel, cmd, buf, len); |
|
if (ret < 0) { |
|
LOG_ERR("DCS 0x%x write failed! (%d)", cmd, ret); |
|
return ret; |
|
} |
|
|
|
return 0; |
|
} |
|
|
|
static int st7701_short_write_1p(const struct device *dev, uint8_t cmd, uint8_t val) |
|
{ |
|
const struct st7701_config *cfg = dev->config; |
|
int ret; |
|
uint8_t buf[] = {cmd, val}; |
|
|
|
ret = mipi_dsi_generic_write(cfg->mipi_dsi, cfg->channel, buf, sizeof(val)); |
|
if (ret < 0) { |
|
LOG_ERR("Short write failed! (%d)", ret); |
|
return ret; |
|
} |
|
|
|
return 0; |
|
} |
|
|
|
static int st7701_generic_write(const struct device *dev, const void *buf, size_t len) |
|
{ |
|
const struct st7701_config *cfg = dev->config; |
|
int ret; |
|
|
|
ret = mipi_dsi_generic_write(cfg->mipi_dsi, cfg->channel, buf, len); |
|
if (ret < 0) { |
|
LOG_ERR("Generic write failed! (%d)", ret); |
|
return ret; |
|
} |
|
|
|
return 0; |
|
} |
|
|
|
static int st7701_check_id(const struct device *dev) |
|
{ |
|
const struct st7701_config *cfg = dev->config; |
|
uint32_t id = 0; |
|
int ret; |
|
|
|
ret = mipi_dsi_dcs_read(cfg->mipi_dsi, cfg->channel, ST7701_CMD_ID1, &id, sizeof(id)); |
|
if (ret != sizeof(id)) { |
|
LOG_ERR("Read panel ID failed! (%d)", ret); |
|
return -EIO; |
|
} |
|
|
|
if (id != ST7701_ID) { |
|
LOG_ERR("ID 0x%x (should 0x%x)", id, ST7701_ID); |
|
return -EINVAL; |
|
} |
|
|
|
return 0; |
|
} |
|
|
|
static int st7701_configure(const struct device *dev) |
|
{ |
|
struct st7701_data *data = dev->data; |
|
const struct st7701_config *cfg = dev->config; |
|
uint8_t buf[4]; |
|
int ret; |
|
|
|
const uint8_t ff1[] = {DSI_CMD2BKX_SEL, 0x77, 0x01, 0x00, 0x00, DSI_CMD2BK1_SEL}; |
|
const uint8_t ff2[] = {DSI_CMD2BKX_SEL, 0x77, 0x01, 0x00, 0x00, DSI_CMD2BKX_SEL_NONE}; |
|
|
|
const uint8_t control0[] = {DSI_CMD2BKX_SEL, 0x77, 0x01, 0x00, 0x00, DSI_CMD2BK0_SEL}; |
|
const uint8_t control1[] = {0xC0, 0x63, 0x00}; |
|
const uint8_t control2[] = {0xC1, 0x11, 0x02}; |
|
const uint8_t control3[] = {0xC2, 0x01, 0x08}; |
|
const uint8_t control4[] = {0xCC, 0x18}; |
|
|
|
st7701_generic_write(dev, control0, sizeof(control0)); |
|
st7701_generic_write(dev, control1, sizeof(control1)); |
|
st7701_generic_write(dev, control2, sizeof(control2)); |
|
st7701_generic_write(dev, control3, sizeof(control3)); |
|
st7701_generic_write(dev, control4, sizeof(control4)); |
|
|
|
/* Gamma Cluster Setting */ |
|
st7701_generic_write(dev, cfg->pvgamctrl, sizeof(cfg->pvgamctrl)); |
|
st7701_generic_write(dev, cfg->nvgamctrl, sizeof(cfg->nvgamctrl)); |
|
|
|
/* Initial power control registers */ |
|
st7701_generic_write(dev, ff1, sizeof(ff1)); |
|
|
|
st7701_short_write_1p(dev, DSI_CMD2_BK1_VRHS, 0x65); |
|
st7701_short_write_1p(dev, DSI_CMD2_BK1_VCOM, 0x34); |
|
st7701_short_write_1p(dev, DSI_CMD2_BK1_VGHSS, 0x87); |
|
st7701_short_write_1p(dev, DSI_CMD2_BK1_TESTCMD, 0x80); |
|
|
|
st7701_short_write_1p(dev, DSI_CMD2_BK1_VGLS, 0x49); |
|
st7701_short_write_1p(dev, DSI_CMD2_BK1_PWCTLR1, 0x85); |
|
|
|
st7701_short_write_1p(dev, DSI_CMD2_BK1_PWCTLR2, 0x20); |
|
st7701_short_write_1p(dev, 0xB9, 0x10); |
|
st7701_short_write_1p(dev, DSI_CMD2_BK1_SPD1, 0x78); |
|
st7701_short_write_1p(dev, DSI_CMD2_BK1_SPD2, 0x78); |
|
st7701_short_write_1p(dev, DSI_CMD2_BK1_MIPISET1, 0x88); |
|
k_msleep(100); |
|
|
|
/* GIP Setting */ |
|
st7701_generic_write(dev, cfg->gip_e0, sizeof(cfg->gip_e0)); |
|
st7701_generic_write(dev, cfg->gip_e1, sizeof(cfg->gip_e1)); |
|
st7701_generic_write(dev, cfg->gip_e2, sizeof(cfg->gip_e2)); |
|
st7701_generic_write(dev, cfg->gip_e3, sizeof(cfg->gip_e3)); |
|
st7701_generic_write(dev, cfg->gip_e4, sizeof(cfg->gip_e4)); |
|
st7701_generic_write(dev, cfg->gip_e5, sizeof(cfg->gip_e5)); |
|
st7701_generic_write(dev, cfg->gip_e6, sizeof(cfg->gip_e6)); |
|
st7701_generic_write(dev, cfg->gip_e7, sizeof(cfg->gip_e7)); |
|
st7701_generic_write(dev, cfg->gip_e8, sizeof(cfg->gip_e8)); |
|
st7701_generic_write(dev, cfg->gip_eb, sizeof(cfg->gip_eb)); |
|
st7701_generic_write(dev, cfg->gip_ec, sizeof(cfg->gip_ec)); |
|
st7701_generic_write(dev, cfg->gip_ed, sizeof(cfg->gip_ed)); |
|
|
|
/* Bank1 setting */ |
|
st7701_generic_write(dev, ff2, sizeof(ff2)); |
|
|
|
/* Exit sleep mode */ |
|
ret = st7701_dcs_write(dev, MIPI_DCS_EXIT_SLEEP_MODE, NULL, 0); |
|
if (ret < 0) { |
|
return ret; |
|
} |
|
|
|
k_msleep(50); |
|
|
|
/* Set pixel color format */ |
|
switch (data->dsi_pixel_format) { |
|
case MIPI_DSI_PIXFMT_RGB565: |
|
buf[0] = MIPI_DCS_PIXEL_FORMAT_16BIT; |
|
break; |
|
case MIPI_DSI_PIXFMT_RGB888: |
|
buf[0] = MIPI_DCS_PIXEL_FORMAT_24BIT; |
|
break; |
|
default: |
|
LOG_ERR("Unsupported pixel format 0x%x!", data->dsi_pixel_format); |
|
return -ENOTSUP; |
|
} |
|
|
|
ret = st7701_dcs_write(dev, MIPI_DCS_SET_PIXEL_FORMAT, buf, 1); |
|
if (ret < 0) { |
|
return ret; |
|
} |
|
|
|
buf[0] = 0x00; |
|
buf[1] = 0x00; |
|
sys_put_be16(data->xres, (uint8_t *)&buf[2]); |
|
ret = st7701_dcs_write(dev, MIPI_DCS_SET_COLUMN_ADDRESS, buf, 4); |
|
if (ret < 0) { |
|
return ret; |
|
} |
|
|
|
buf[0] = 0x00; |
|
buf[1] = 0x00; |
|
sys_put_be16(data->yres, (uint8_t *)&buf[2]); |
|
ret = st7701_dcs_write(dev, MIPI_DCS_SET_PAGE_ADDRESS, buf, 4); |
|
if (ret < 0) { |
|
return ret; |
|
} |
|
|
|
/* Backlight control */ |
|
buf[0] = ST7701_WRCTRLD_BCTRL | ST7701_WRCTRLD_DD | ST7701_WRCTRLD_BL; |
|
ret = st7701_dcs_write(dev, MIPI_DCS_WRITE_CONTROL_DISPLAY, buf, 1); |
|
if (ret < 0) { |
|
return ret; |
|
} |
|
|
|
/* Adaptive brightness control */ |
|
buf[0] = ST7701_WRCABC_UI; |
|
ret = st7701_dcs_write(dev, MIPI_DCS_WRITE_POWER_SAVE, buf, 1); |
|
if (ret < 0) { |
|
return ret; |
|
} |
|
|
|
/* Adaptive brightness control minimum brightness */ |
|
buf[0] = 0xFF; |
|
ret = st7701_dcs_write(dev, MIPI_DCS_SET_CABC_MIN_BRIGHTNESS, buf, 1); |
|
if (ret < 0) { |
|
return ret; |
|
} |
|
|
|
/* Brightness */ |
|
buf[0] = 0xFF; |
|
ret = st7701_dcs_write(dev, MIPI_DCS_SET_DISPLAY_BRIGHTNESS, buf, 1); |
|
if (ret < 0) { |
|
return ret; |
|
} |
|
|
|
/* Display On */ |
|
ret = st7701_dcs_write(dev, MIPI_DCS_SET_DISPLAY_ON, NULL, 0); |
|
if (ret < 0) { |
|
return ret; |
|
} |
|
|
|
return 0; |
|
} |
|
|
|
static int st7701_blanking_on(const struct device *dev) |
|
{ |
|
const struct st7701_config *cfg = dev->config; |
|
int ret; |
|
|
|
if (cfg->backlight.port != NULL) { |
|
ret = gpio_pin_set_dt(&cfg->backlight, 0); |
|
if (ret) { |
|
LOG_ERR("Disable backlight failed! (%d)", ret); |
|
return ret; |
|
} |
|
} |
|
|
|
return st7701_dcs_write(dev, MIPI_DCS_SET_DISPLAY_OFF, NULL, 0); |
|
} |
|
|
|
static int st7701_blanking_off(const struct device *dev) |
|
{ |
|
const struct st7701_config *cfg = dev->config; |
|
int ret; |
|
|
|
if (cfg->backlight.port != NULL) { |
|
ret = gpio_pin_set_dt(&cfg->backlight, 1); |
|
if (ret) { |
|
LOG_ERR("Enable backlight failed! (%d)", ret); |
|
return ret; |
|
} |
|
} |
|
|
|
return st7701_dcs_write(dev, MIPI_DCS_SET_DISPLAY_ON, NULL, 0); |
|
} |
|
|
|
static int st7701_set_brightness(const struct device *dev, uint8_t brightness) |
|
{ |
|
return st7701_dcs_write(dev, MIPI_DCS_SET_DISPLAY_BRIGHTNESS, &brightness, 1); |
|
} |
|
|
|
static void st7701_get_capabilities(const struct device *dev, |
|
struct display_capabilities *capabilities) |
|
{ |
|
const struct st7701_config *cfg = dev->config; |
|
struct st7701_data *data = dev->data; |
|
|
|
if (!capabilities) { |
|
return; |
|
} |
|
|
|
memset(capabilities, 0, sizeof(struct display_capabilities)); |
|
capabilities->x_resolution = cfg->width; |
|
capabilities->y_resolution = cfg->height; |
|
capabilities->supported_pixel_formats = data->pixel_format; |
|
capabilities->current_pixel_format = data->pixel_format; |
|
capabilities->current_orientation = data->orientation; |
|
} |
|
|
|
static DEVICE_API(display, st7701_api) = { |
|
.blanking_on = st7701_blanking_on, |
|
.blanking_off = st7701_blanking_off, |
|
.set_brightness = st7701_set_brightness, |
|
.get_capabilities = st7701_get_capabilities, |
|
}; |
|
|
|
static int st7701_init(const struct device *dev) |
|
{ |
|
const struct st7701_config *cfg = dev->config; |
|
struct st7701_data *data = dev->data; |
|
struct mipi_dsi_device mdev; |
|
int ret; |
|
|
|
if (cfg->reset.port) { |
|
if (!gpio_is_ready_dt(&cfg->reset)) { |
|
LOG_ERR("Reset GPIO device is not ready!"); |
|
return -ENODEV; |
|
} |
|
|
|
ret = gpio_pin_configure_dt(&cfg->reset, GPIO_OUTPUT_INACTIVE); |
|
if (ret < 0) { |
|
LOG_ERR("Reset display failed! (%d)", ret); |
|
return ret; |
|
} |
|
|
|
k_msleep(10); |
|
ret = gpio_pin_set_dt(&cfg->reset, 1); |
|
if (ret < 0) { |
|
LOG_ERR("Enable display failed! (%d)", ret); |
|
return ret; |
|
} |
|
|
|
k_msleep(100); |
|
} |
|
|
|
/* store x/y resolution & rotation */ |
|
if (cfg->rotation == 0) { |
|
data->xres = cfg->width; |
|
data->yres = cfg->height; |
|
data->orientation = DISPLAY_ORIENTATION_NORMAL; |
|
} else if (cfg->rotation == 90) { |
|
data->xres = cfg->height; |
|
data->yres = cfg->width; |
|
data->orientation = DISPLAY_ORIENTATION_ROTATED_90; |
|
} else if (cfg->rotation == 180) { |
|
data->xres = cfg->width; |
|
data->yres = cfg->height; |
|
data->orientation = DISPLAY_ORIENTATION_ROTATED_180; |
|
} else if (cfg->rotation == 270) { |
|
data->xres = cfg->height; |
|
data->yres = cfg->width; |
|
data->orientation = DISPLAY_ORIENTATION_ROTATED_270; |
|
} |
|
|
|
/* attach to MIPI-DSI host */ |
|
mdev.data_lanes = cfg->data_lanes; |
|
mdev.pixfmt = data->dsi_pixel_format; |
|
mdev.mode_flags = MIPI_DSI_MODE_VIDEO | MIPI_DSI_MODE_VIDEO_BURST | MIPI_DSI_MODE_LPM; |
|
|
|
mdev.timings.hactive = cfg->width; |
|
mdev.timings.hbp = cfg->hbp; |
|
mdev.timings.hsync = cfg->hsync; |
|
mdev.timings.hfp = cfg->hfp; |
|
mdev.timings.vactive = cfg->height; |
|
mdev.timings.vbp = cfg->vbp; |
|
mdev.timings.vsync = cfg->vsync; |
|
mdev.timings.vfp = cfg->vfp; |
|
|
|
ret = mipi_dsi_attach(cfg->mipi_dsi, cfg->channel, &mdev); |
|
if (ret < 0) { |
|
LOG_ERR("MIPI-DSI attach failed! (%d)", ret); |
|
return ret; |
|
} |
|
|
|
ret = st7701_check_id(dev); |
|
if (ret) { |
|
LOG_ERR("Panel ID check failed! (%d)", ret); |
|
return ret; |
|
} |
|
|
|
ret = st7701_configure(dev); |
|
if (ret) { |
|
LOG_ERR("DSI init sequence failed! (%d)", ret); |
|
return ret; |
|
} |
|
|
|
ret = st7701_blanking_off(dev); |
|
if (ret) { |
|
LOG_ERR("Display blanking off failed! (%d)", ret); |
|
return ret; |
|
} |
|
|
|
return 0; |
|
} |
|
|
|
#define ST7701_DEVICE(inst) \ |
|
static const struct st7701_config st7701_config_##inst = { \ |
|
.mipi_dsi = DEVICE_DT_GET(DT_INST_BUS(inst)), \ |
|
.reset = GPIO_DT_SPEC_INST_GET_OR(inst, reset_gpios, {0}), \ |
|
.backlight = GPIO_DT_SPEC_INST_GET_OR(inst, bl_gpios, {0}), \ |
|
.data_lanes = DT_INST_PROP_BY_IDX(inst, data_lanes, 0), \ |
|
.width = DT_INST_PROP(inst, width), \ |
|
.height = DT_INST_PROP(inst, height), \ |
|
.channel = DT_INST_REG_ADDR(inst), \ |
|
.rotation = DT_INST_PROP(inst, rotation), \ |
|
.hbp = DT_PROP(DT_INST_CHILD(inst, display_timings), hback_porch), \ |
|
.hsync = DT_PROP(DT_INST_CHILD(inst, display_timings), hsync_len), \ |
|
.hfp = DT_PROP(DT_INST_CHILD(inst, display_timings), hfront_porch), \ |
|
.vbp = DT_PROP(DT_INST_CHILD(inst, display_timings), vback_porch), \ |
|
.vsync = DT_PROP(DT_INST_CHILD(inst, display_timings), vsync_len), \ |
|
.vfp = DT_PROP(DT_INST_CHILD(inst, display_timings), vfront_porch), \ |
|
.gip_e0 = DT_INST_PROP_OR(inst, gip_e0, {}), \ |
|
.gip_e1 = DT_INST_PROP_OR(inst, gip_e1, {}), \ |
|
.gip_e2 = DT_INST_PROP_OR(inst, gip_e2, {}), \ |
|
.gip_e3 = DT_INST_PROP_OR(inst, gip_e3, {}), \ |
|
.gip_e4 = DT_INST_PROP_OR(inst, gip_e4, {}), \ |
|
.gip_e5 = DT_INST_PROP_OR(inst, gip_e5, {}), \ |
|
.gip_e6 = DT_INST_PROP_OR(inst, gip_e6, {}), \ |
|
.gip_e7 = DT_INST_PROP_OR(inst, gip_e7, {}), \ |
|
.gip_e8 = DT_INST_PROP_OR(inst, gip_e8, {}), \ |
|
.gip_eb = DT_INST_PROP_OR(inst, gip_eb, {}), \ |
|
.gip_ec = DT_INST_PROP_OR(inst, gip_ec, {}), \ |
|
.gip_ed = DT_INST_PROP_OR(inst, gip_ed, {}), \ |
|
.pvgamctrl = DT_INST_PROP_OR(inst, pvgamctrl, {}), \ |
|
.nvgamctrl = DT_INST_PROP_OR(inst, nvgamctrl, {}), \ |
|
}; \ |
|
static struct st7701_data st7701_data_##inst = { \ |
|
.dsi_pixel_format = DT_INST_PROP(inst, pixel_format), \ |
|
}; \ |
|
DEVICE_DT_INST_DEFINE(inst, &st7701_init, NULL, &st7701_data_##inst, \ |
|
&st7701_config_##inst, POST_KERNEL, \ |
|
CONFIG_DISPLAY_INIT_PRIORITY, &st7701_api); |
|
|
|
DT_INST_FOREACH_STATUS_OKAY(ST7701_DEVICE)
|
|
|