|  | // SPDX-License-Identifier: GPL-2.0 | 
|  | /* | 
|  | * Copyright (C) 2020 Marek Vasut <marex@denx.de> | 
|  | * | 
|  | * Based on tc358764.c by | 
|  | *  Andrzej Hajda <a.hajda@samsung.com> | 
|  | *  Maciej Purski <m.purski@samsung.com> | 
|  | * | 
|  | * Based on rpi_touchscreen.c by | 
|  | *  Eric Anholt <eric@anholt.net> | 
|  | */ | 
|  |  | 
|  | #include <linux/delay.h> | 
|  | #include <linux/module.h> | 
|  | #include <linux/of_graph.h> | 
|  | #include <linux/regulator/consumer.h> | 
|  |  | 
|  | #include <video/mipi_display.h> | 
|  |  | 
|  | #include <drm/drm_atomic_helper.h> | 
|  | #include <drm/drm_crtc.h> | 
|  | #include <drm/drm_fb_helper.h> | 
|  | #include <drm/drm_mipi_dsi.h> | 
|  | #include <drm/drm_of.h> | 
|  | #include <drm/drm_panel.h> | 
|  | #include <drm/drm_print.h> | 
|  | #include <drm/drm_probe_helper.h> | 
|  |  | 
|  | /* PPI layer registers */ | 
|  | #define PPI_STARTPPI		0x0104 /* START control bit */ | 
|  | #define PPI_LPTXTIMECNT		0x0114 /* LPTX timing signal */ | 
|  | #define PPI_D0S_ATMR		0x0144 | 
|  | #define PPI_D1S_ATMR		0x0148 | 
|  | #define PPI_D0S_CLRSIPOCOUNT	0x0164 /* Assertion timer for Lane 0 */ | 
|  | #define PPI_D1S_CLRSIPOCOUNT	0x0168 /* Assertion timer for Lane 1 */ | 
|  | #define PPI_START_FUNCTION	1 | 
|  |  | 
|  | /* DSI layer registers */ | 
|  | #define DSI_STARTDSI		0x0204 /* START control bit of DSI-TX */ | 
|  | #define DSI_LANEENABLE		0x0210 /* Enables each lane */ | 
|  | #define DSI_RX_START		1 | 
|  |  | 
|  | /* LCDC/DPI Host Registers */ | 
|  | #define LCDCTRL			0x0420 | 
|  |  | 
|  | /* SPI Master Registers */ | 
|  | #define SPICMR			0x0450 | 
|  | #define SPITCR			0x0454 | 
|  |  | 
|  | /* System Controller Registers */ | 
|  | #define SYSCTRL			0x0464 | 
|  |  | 
|  | /* System registers */ | 
|  | #define LPX_PERIOD		3 | 
|  |  | 
|  | /* Lane enable PPI and DSI register bits */ | 
|  | #define LANEENABLE_CLEN		BIT(0) | 
|  | #define LANEENABLE_L0EN		BIT(1) | 
|  | #define LANEENABLE_L1EN		BIT(2) | 
|  |  | 
|  | struct tc358762 { | 
|  | struct device *dev; | 
|  | struct drm_bridge bridge; | 
|  | struct drm_connector connector; | 
|  | struct regulator *regulator; | 
|  | struct drm_bridge *panel_bridge; | 
|  | bool pre_enabled; | 
|  | int error; | 
|  | }; | 
|  |  | 
|  | static int tc358762_clear_error(struct tc358762 *ctx) | 
|  | { | 
|  | int ret = ctx->error; | 
|  |  | 
|  | ctx->error = 0; | 
|  | return ret; | 
|  | } | 
|  |  | 
|  | static void tc358762_write(struct tc358762 *ctx, u16 addr, u32 val) | 
|  | { | 
|  | struct mipi_dsi_device *dsi = to_mipi_dsi_device(ctx->dev); | 
|  | ssize_t ret; | 
|  | u8 data[6]; | 
|  |  | 
|  | if (ctx->error) | 
|  | return; | 
|  |  | 
|  | data[0] = addr; | 
|  | data[1] = addr >> 8; | 
|  | data[2] = val; | 
|  | data[3] = val >> 8; | 
|  | data[4] = val >> 16; | 
|  | data[5] = val >> 24; | 
|  |  | 
|  | ret = mipi_dsi_generic_write(dsi, data, sizeof(data)); | 
|  | if (ret < 0) | 
|  | ctx->error = ret; | 
|  | } | 
|  |  | 
|  | static inline struct tc358762 *bridge_to_tc358762(struct drm_bridge *bridge) | 
|  | { | 
|  | return container_of(bridge, struct tc358762, bridge); | 
|  | } | 
|  |  | 
|  | static int tc358762_init(struct tc358762 *ctx) | 
|  | { | 
|  | tc358762_write(ctx, DSI_LANEENABLE, | 
|  | LANEENABLE_L0EN | LANEENABLE_CLEN); | 
|  | tc358762_write(ctx, PPI_D0S_CLRSIPOCOUNT, 5); | 
|  | tc358762_write(ctx, PPI_D1S_CLRSIPOCOUNT, 5); | 
|  | tc358762_write(ctx, PPI_D0S_ATMR, 0); | 
|  | tc358762_write(ctx, PPI_D1S_ATMR, 0); | 
|  | tc358762_write(ctx, PPI_LPTXTIMECNT, LPX_PERIOD); | 
|  |  | 
|  | tc358762_write(ctx, SPICMR, 0x00); | 
|  | tc358762_write(ctx, LCDCTRL, 0x00100150); | 
|  | tc358762_write(ctx, SYSCTRL, 0x040f); | 
|  | msleep(100); | 
|  |  | 
|  | tc358762_write(ctx, PPI_STARTPPI, PPI_START_FUNCTION); | 
|  | tc358762_write(ctx, DSI_STARTDSI, DSI_RX_START); | 
|  |  | 
|  | msleep(100); | 
|  |  | 
|  | return tc358762_clear_error(ctx); | 
|  | } | 
|  |  | 
|  | static void tc358762_post_disable(struct drm_bridge *bridge) | 
|  | { | 
|  | struct tc358762 *ctx = bridge_to_tc358762(bridge); | 
|  | int ret; | 
|  |  | 
|  | /* | 
|  | * The post_disable hook might be called multiple times. | 
|  | * We want to avoid regulator imbalance below. | 
|  | */ | 
|  | if (!ctx->pre_enabled) | 
|  | return; | 
|  |  | 
|  | ctx->pre_enabled = false; | 
|  |  | 
|  | ret = regulator_disable(ctx->regulator); | 
|  | if (ret < 0) | 
|  | dev_err(ctx->dev, "error disabling regulators (%d)\n", ret); | 
|  | } | 
|  |  | 
|  | static void tc358762_pre_enable(struct drm_bridge *bridge) | 
|  | { | 
|  | struct tc358762 *ctx = bridge_to_tc358762(bridge); | 
|  | int ret; | 
|  |  | 
|  | ret = regulator_enable(ctx->regulator); | 
|  | if (ret < 0) | 
|  | dev_err(ctx->dev, "error enabling regulators (%d)\n", ret); | 
|  |  | 
|  | ret = tc358762_init(ctx); | 
|  | if (ret < 0) | 
|  | dev_err(ctx->dev, "error initializing bridge (%d)\n", ret); | 
|  |  | 
|  | ctx->pre_enabled = true; | 
|  | } | 
|  |  | 
|  | static int tc358762_attach(struct drm_bridge *bridge, | 
|  | enum drm_bridge_attach_flags flags) | 
|  | { | 
|  | struct tc358762 *ctx = bridge_to_tc358762(bridge); | 
|  |  | 
|  | return drm_bridge_attach(bridge->encoder, ctx->panel_bridge, | 
|  | bridge, flags); | 
|  | } | 
|  |  | 
|  | static const struct drm_bridge_funcs tc358762_bridge_funcs = { | 
|  | .post_disable = tc358762_post_disable, | 
|  | .pre_enable = tc358762_pre_enable, | 
|  | .attach = tc358762_attach, | 
|  | }; | 
|  |  | 
|  | static int tc358762_parse_dt(struct tc358762 *ctx) | 
|  | { | 
|  | struct drm_bridge *panel_bridge; | 
|  | struct device *dev = ctx->dev; | 
|  | struct drm_panel *panel; | 
|  | int ret; | 
|  |  | 
|  | ret = drm_of_find_panel_or_bridge(dev->of_node, 1, 0, &panel, NULL); | 
|  | if (ret) | 
|  | return ret; | 
|  |  | 
|  | panel_bridge = devm_drm_panel_bridge_add(dev, panel); | 
|  |  | 
|  | if (IS_ERR(panel_bridge)) | 
|  | return PTR_ERR(panel_bridge); | 
|  |  | 
|  | ctx->panel_bridge = panel_bridge; | 
|  |  | 
|  | return 0; | 
|  | } | 
|  |  | 
|  | static int tc358762_configure_regulators(struct tc358762 *ctx) | 
|  | { | 
|  | ctx->regulator = devm_regulator_get(ctx->dev, "vddc"); | 
|  | if (IS_ERR(ctx->regulator)) | 
|  | return PTR_ERR(ctx->regulator); | 
|  |  | 
|  | return 0; | 
|  | } | 
|  |  | 
|  | static int tc358762_probe(struct mipi_dsi_device *dsi) | 
|  | { | 
|  | struct device *dev = &dsi->dev; | 
|  | struct tc358762 *ctx; | 
|  | int ret; | 
|  |  | 
|  | ctx = devm_kzalloc(dev, sizeof(struct tc358762), GFP_KERNEL); | 
|  | if (!ctx) | 
|  | return -ENOMEM; | 
|  |  | 
|  | mipi_dsi_set_drvdata(dsi, ctx); | 
|  |  | 
|  | ctx->dev = dev; | 
|  | ctx->pre_enabled = false; | 
|  |  | 
|  | /* TODO: Find out how to get dual-lane mode working */ | 
|  | dsi->lanes = 1; | 
|  | dsi->format = MIPI_DSI_FMT_RGB888; | 
|  | dsi->mode_flags = MIPI_DSI_MODE_VIDEO | MIPI_DSI_MODE_VIDEO_SYNC_PULSE | | 
|  | MIPI_DSI_MODE_LPM | MIPI_DSI_MODE_VIDEO_HSE; | 
|  |  | 
|  | ret = tc358762_parse_dt(ctx); | 
|  | if (ret < 0) | 
|  | return ret; | 
|  |  | 
|  | ret = tc358762_configure_regulators(ctx); | 
|  | if (ret < 0) | 
|  | return ret; | 
|  |  | 
|  | ctx->bridge.funcs = &tc358762_bridge_funcs; | 
|  | ctx->bridge.type = DRM_MODE_CONNECTOR_DPI; | 
|  | ctx->bridge.of_node = dev->of_node; | 
|  |  | 
|  | drm_bridge_add(&ctx->bridge); | 
|  |  | 
|  | ret = mipi_dsi_attach(dsi); | 
|  | if (ret < 0) { | 
|  | drm_bridge_remove(&ctx->bridge); | 
|  | dev_err(dev, "failed to attach dsi\n"); | 
|  | } | 
|  |  | 
|  | return ret; | 
|  | } | 
|  |  | 
|  | static int tc358762_remove(struct mipi_dsi_device *dsi) | 
|  | { | 
|  | struct tc358762 *ctx = mipi_dsi_get_drvdata(dsi); | 
|  |  | 
|  | mipi_dsi_detach(dsi); | 
|  | drm_bridge_remove(&ctx->bridge); | 
|  |  | 
|  | return 0; | 
|  | } | 
|  |  | 
|  | static const struct of_device_id tc358762_of_match[] = { | 
|  | { .compatible = "toshiba,tc358762" }, | 
|  | { } | 
|  | }; | 
|  | MODULE_DEVICE_TABLE(of, tc358762_of_match); | 
|  |  | 
|  | static struct mipi_dsi_driver tc358762_driver = { | 
|  | .probe = tc358762_probe, | 
|  | .remove = tc358762_remove, | 
|  | .driver = { | 
|  | .name = "tc358762", | 
|  | .of_match_table = tc358762_of_match, | 
|  | }, | 
|  | }; | 
|  | module_mipi_dsi_driver(tc358762_driver); | 
|  |  | 
|  | MODULE_AUTHOR("Marek Vasut <marex@denx.de>"); | 
|  | MODULE_DESCRIPTION("MIPI-DSI based Driver for TC358762 DSI/DPI Bridge"); | 
|  | MODULE_LICENSE("GPL v2"); |