Path: blob/master/drivers/gpu/drm/bridge/chipone-icn6211.c
26494 views
// SPDX-License-Identifier: GPL-2.0+1/*2* Copyright (C) 2020 Amarula Solutions(India)3* Author: Jagan Teki <[email protected]>4*/56#include <drm/drm_atomic_helper.h>7#include <drm/drm_of.h>8#include <drm/drm_print.h>9#include <drm/drm_mipi_dsi.h>1011#include <linux/bitfield.h>12#include <linux/bits.h>13#include <linux/clk.h>14#include <linux/delay.h>15#include <linux/gpio/consumer.h>16#include <linux/i2c.h>17#include <linux/media-bus-format.h>18#include <linux/module.h>19#include <linux/of.h>20#include <linux/regmap.h>21#include <linux/regulator/consumer.h>2223#define VENDOR_ID 0x0024#define DEVICE_ID_H 0x0125#define DEVICE_ID_L 0x0226#define VERSION_ID 0x0327#define FIRMWARE_VERSION 0x0828#define CONFIG_FINISH 0x0929#define PD_CTRL(n) (0x0a + ((n) & 0x3)) /* 0..3 */30#define RST_CTRL(n) (0x0e + ((n) & 0x1)) /* 0..1 */31#define SYS_CTRL(n) (0x10 + ((n) & 0x7)) /* 0..4 */32#define SYS_CTRL_1_CLK_PHASE_MSK GENMASK(5, 4)33#define CLK_PHASE_0 034#define CLK_PHASE_1_4 135#define CLK_PHASE_1_2 236#define CLK_PHASE_3_4 337#define RGB_DRV(n) (0x18 + ((n) & 0x3)) /* 0..3 */38#define RGB_DLY(n) (0x1c + ((n) & 0x1)) /* 0..1 */39#define RGB_TEST_CTRL 0x1e40#define ATE_PLL_EN 0x1f41#define HACTIVE_LI 0x2042#define VACTIVE_LI 0x2143#define VACTIVE_HACTIVE_HI 0x2244#define HFP_LI 0x2345#define HSYNC_LI 0x2446#define HBP_LI 0x2547#define HFP_HSW_HBP_HI 0x2648#define HFP_HSW_HBP_HI_HFP(n) (((n) & 0x300) >> 4)49#define HFP_HSW_HBP_HI_HS(n) (((n) & 0x300) >> 6)50#define HFP_HSW_HBP_HI_HBP(n) (((n) & 0x300) >> 8)51#define VFP 0x2752#define VSYNC 0x2853#define VBP 0x2954#define BIST_POL 0x2a55#define BIST_POL_BIST_MODE(n) (((n) & 0xf) << 4)56#define BIST_POL_BIST_GEN BIT(3)57#define BIST_POL_HSYNC_POL BIT(2)58#define BIST_POL_VSYNC_POL BIT(1)59#define BIST_POL_DE_POL BIT(0)60#define BIST_RED 0x2b61#define BIST_GREEN 0x2c62#define BIST_BLUE 0x2d63#define BIST_CHESS_X 0x2e64#define BIST_CHESS_Y 0x2f65#define BIST_CHESS_XY_H 0x3066#define BIST_FRAME_TIME_L 0x3167#define BIST_FRAME_TIME_H 0x3268#define FIFO_MAX_ADDR_LOW 0x3369#define SYNC_EVENT_DLY 0x3470#define HSW_MIN 0x3571#define HFP_MIN 0x3672#define LOGIC_RST_NUM 0x3773#define OSC_CTRL(n) (0x48 + ((n) & 0x7)) /* 0..5 */74#define BG_CTRL 0x4e75#define LDO_PLL 0x4f76#define PLL_CTRL(n) (0x50 + ((n) & 0xf)) /* 0..15 */77#define PLL_CTRL_6_EXTERNAL 0x9078#define PLL_CTRL_6_MIPI_CLK 0x9279#define PLL_CTRL_6_INTERNAL 0x9380#define PLL_REM(n) (0x60 + ((n) & 0x3)) /* 0..2 */81#define PLL_DIV(n) (0x63 + ((n) & 0x3)) /* 0..2 */82#define PLL_FRAC(n) (0x66 + ((n) & 0x3)) /* 0..2 */83#define PLL_INT(n) (0x69 + ((n) & 0x1)) /* 0..1 */84#define PLL_REF_DIV 0x6b85#define PLL_REF_DIV_P(n) ((n) & 0xf)86#define PLL_REF_DIV_Pe BIT(4)87#define PLL_REF_DIV_S(n) (((n) & 0x7) << 5)88#define PLL_SSC_P(n) (0x6c + ((n) & 0x3)) /* 0..2 */89#define PLL_SSC_STEP(n) (0x6f + ((n) & 0x3)) /* 0..2 */90#define PLL_SSC_OFFSET(n) (0x72 + ((n) & 0x3)) /* 0..3 */91#define GPIO_OEN 0x7992#define MIPI_CFG_PW 0x7a93#define MIPI_CFG_PW_CONFIG_DSI 0xc194#define MIPI_CFG_PW_CONFIG_I2C 0x3e95#define GPIO_SEL(n) (0x7b + ((n) & 0x1)) /* 0..1 */96#define IRQ_SEL 0x7d97#define DBG_SEL 0x7e98#define DBG_SIGNAL 0x7f99#define MIPI_ERR_VECTOR_L 0x80100#define MIPI_ERR_VECTOR_H 0x81101#define MIPI_ERR_VECTOR_EN_L 0x82102#define MIPI_ERR_VECTOR_EN_H 0x83103#define MIPI_MAX_SIZE_L 0x84104#define MIPI_MAX_SIZE_H 0x85105#define DSI_CTRL 0x86106#define DSI_CTRL_UNKNOWN 0x28107#define DSI_CTRL_DSI_LANES(n) ((n) & 0x3)108#define MIPI_PN_SWAP 0x87109#define MIPI_PN_SWAP_CLK BIT(4)110#define MIPI_PN_SWAP_D(n) BIT((n) & 0x3)111#define MIPI_SOT_SYNC_BIT(n) (0x88 + ((n) & 0x1)) /* 0..1 */112#define MIPI_ULPS_CTRL 0x8a113#define MIPI_CLK_CHK_VAR 0x8e114#define MIPI_CLK_CHK_INI 0x8f115#define MIPI_T_TERM_EN 0x90116#define MIPI_T_HS_SETTLE 0x91117#define MIPI_T_TA_SURE_PRE 0x92118#define MIPI_T_LPX_SET 0x94119#define MIPI_T_CLK_MISS 0x95120#define MIPI_INIT_TIME_L 0x96121#define MIPI_INIT_TIME_H 0x97122#define MIPI_T_CLK_TERM_EN 0x99123#define MIPI_T_CLK_SETTLE 0x9a124#define MIPI_TO_HS_RX_L 0x9e125#define MIPI_TO_HS_RX_H 0x9f126#define MIPI_PHY(n) (0xa0 + ((n) & 0x7)) /* 0..5 */127#define MIPI_PD_RX 0xb0128#define MIPI_PD_TERM 0xb1129#define MIPI_PD_HSRX 0xb2130#define MIPI_PD_LPTX 0xb3131#define MIPI_PD_LPRX 0xb4132#define MIPI_PD_CK_LANE 0xb5133#define MIPI_FORCE_0 0xb6134#define MIPI_RST_CTRL 0xb7135#define MIPI_RST_NUM 0xb8136#define MIPI_DBG_SET(n) (0xc0 + ((n) & 0xf)) /* 0..9 */137#define MIPI_DBG_SEL 0xe0138#define MIPI_DBG_DATA 0xe1139#define MIPI_ATE_TEST_SEL 0xe2140#define MIPI_ATE_STATUS(n) (0xe3 + ((n) & 0x1)) /* 0..1 */141142struct chipone {143struct device *dev;144struct regmap *regmap;145struct i2c_client *client;146struct drm_bridge bridge;147struct drm_display_mode mode;148struct drm_bridge *panel_bridge;149struct mipi_dsi_device *dsi;150struct gpio_desc *enable_gpio;151struct regulator *vdd1;152struct regulator *vdd2;153struct regulator *vdd3;154struct clk *refclk;155unsigned long refclk_rate;156bool interface_i2c;157};158159static const struct regmap_range chipone_dsi_readable_ranges[] = {160regmap_reg_range(VENDOR_ID, VERSION_ID),161regmap_reg_range(FIRMWARE_VERSION, PLL_SSC_OFFSET(3)),162regmap_reg_range(GPIO_OEN, MIPI_ULPS_CTRL),163regmap_reg_range(MIPI_CLK_CHK_VAR, MIPI_T_TA_SURE_PRE),164regmap_reg_range(MIPI_T_LPX_SET, MIPI_INIT_TIME_H),165regmap_reg_range(MIPI_T_CLK_TERM_EN, MIPI_T_CLK_SETTLE),166regmap_reg_range(MIPI_TO_HS_RX_L, MIPI_PHY(5)),167regmap_reg_range(MIPI_PD_RX, MIPI_RST_NUM),168regmap_reg_range(MIPI_DBG_SET(0), MIPI_DBG_SET(9)),169regmap_reg_range(MIPI_DBG_SEL, MIPI_ATE_STATUS(1)),170};171172static const struct regmap_access_table chipone_dsi_readable_table = {173.yes_ranges = chipone_dsi_readable_ranges,174.n_yes_ranges = ARRAY_SIZE(chipone_dsi_readable_ranges),175};176177static const struct regmap_range chipone_dsi_writeable_ranges[] = {178regmap_reg_range(CONFIG_FINISH, PLL_SSC_OFFSET(3)),179regmap_reg_range(GPIO_OEN, MIPI_ULPS_CTRL),180regmap_reg_range(MIPI_CLK_CHK_VAR, MIPI_T_TA_SURE_PRE),181regmap_reg_range(MIPI_T_LPX_SET, MIPI_INIT_TIME_H),182regmap_reg_range(MIPI_T_CLK_TERM_EN, MIPI_T_CLK_SETTLE),183regmap_reg_range(MIPI_TO_HS_RX_L, MIPI_PHY(5)),184regmap_reg_range(MIPI_PD_RX, MIPI_RST_NUM),185regmap_reg_range(MIPI_DBG_SET(0), MIPI_DBG_SET(9)),186regmap_reg_range(MIPI_DBG_SEL, MIPI_ATE_STATUS(1)),187};188189static const struct regmap_access_table chipone_dsi_writeable_table = {190.yes_ranges = chipone_dsi_writeable_ranges,191.n_yes_ranges = ARRAY_SIZE(chipone_dsi_writeable_ranges),192};193194static const struct regmap_config chipone_regmap_config = {195.reg_bits = 8,196.val_bits = 8,197.rd_table = &chipone_dsi_readable_table,198.wr_table = &chipone_dsi_writeable_table,199.cache_type = REGCACHE_MAPLE,200.max_register = MIPI_ATE_STATUS(1),201};202203static int chipone_dsi_read(void *context,204const void *reg, size_t reg_size,205void *val, size_t val_size)206{207struct mipi_dsi_device *dsi = context;208const u16 reg16 = (val_size << 8) | *(u8 *)reg;209int ret;210211ret = mipi_dsi_generic_read(dsi, ®16, 2, val, val_size);212213return ret == val_size ? 0 : -EINVAL;214}215216static int chipone_dsi_write(void *context, const void *data, size_t count)217{218struct mipi_dsi_device *dsi = context;219220return mipi_dsi_generic_write(dsi, data, 2);221}222223static const struct regmap_bus chipone_dsi_regmap_bus = {224.read = chipone_dsi_read,225.write = chipone_dsi_write,226.reg_format_endian_default = REGMAP_ENDIAN_NATIVE,227.val_format_endian_default = REGMAP_ENDIAN_NATIVE,228};229230static inline struct chipone *bridge_to_chipone(struct drm_bridge *bridge)231{232return container_of(bridge, struct chipone, bridge);233}234235static void chipone_readb(struct chipone *icn, u8 reg, u8 *val)236{237int ret, pval;238239ret = regmap_read(icn->regmap, reg, &pval);240241*val = ret ? 0 : pval & 0xff;242}243244static int chipone_writeb(struct chipone *icn, u8 reg, u8 val)245{246return regmap_write(icn->regmap, reg, val);247}248249static void chipone_configure_pll(struct chipone *icn,250const struct drm_display_mode *mode)251{252unsigned int best_p = 0, best_m = 0, best_s = 0;253unsigned int mode_clock = mode->clock * 1000;254unsigned int delta, min_delta = 0xffffffff;255unsigned int freq_p, freq_s, freq_out;256unsigned int p_min, p_max;257unsigned int p, m, s;258unsigned int fin;259bool best_p_pot;260u8 ref_div;261262/*263* DSI byte clock frequency (input into PLL) is calculated as:264* DSI_CLK = HS clock / 4265*266* DPI pixel clock frequency (output from PLL) is mode clock.267*268* The chip contains fractional PLL which works as follows:269* DPI_CLK = ((DSI_CLK / P) * M) / S270* P is pre-divider, register PLL_REF_DIV[3:0] is 1:n divider271* register PLL_REF_DIV[4] is extra 1:2 divider272* M is integer multiplier, register PLL_INT(0) is multiplier273* S is post-divider, register PLL_REF_DIV[7:5] is 2^(n+1) divider274*275* It seems the PLL input clock after applying P pre-divider have276* to be lower than 20 MHz.277*/278if (icn->refclk)279fin = icn->refclk_rate;280else281fin = icn->dsi->hs_rate / 4; /* in Hz */282283/* Minimum value of P predivider for PLL input in 5..20 MHz */284p_min = clamp(DIV_ROUND_UP(fin, 20000000), 1U, 31U);285p_max = clamp(fin / 5000000, 1U, 31U);286287for (p = p_min; p < p_max; p++) { /* PLL_REF_DIV[4,3:0] */288if (p > 16 && p & 1) /* P > 16 uses extra /2 */289continue;290freq_p = fin / p;291if (freq_p == 0) /* Divider too high */292break;293294for (s = 0; s < 0x7; s++) { /* PLL_REF_DIV[7:5] */295freq_s = freq_p / BIT(s + 1);296if (freq_s == 0) /* Divider too high */297break;298299m = mode_clock / freq_s;300301/* Multiplier is 8 bit */302if (m > 0xff)303continue;304305/* Limit PLL VCO frequency to 1 GHz */306freq_out = (fin * m) / p;307if (freq_out > 1000000000)308continue;309310/* Apply post-divider */311freq_out /= BIT(s + 1);312313delta = abs(mode_clock - freq_out);314if (delta < min_delta) {315best_p = p;316best_m = m;317best_s = s;318min_delta = delta;319}320}321}322323best_p_pot = !(best_p & 1);324325dev_dbg(icn->dev,326"PLL: P[3:0]=%d P[4]=2*%d M=%d S[7:5]=2^%d delta=%d => DSI f_in(%s)=%d Hz ; DPI f_out=%d Hz\n",327best_p >> best_p_pot, best_p_pot, best_m, best_s + 1,328min_delta, icn->refclk ? "EXT" : "DSI", fin,329(fin * best_m) / (best_p << (best_s + 1)));330331ref_div = PLL_REF_DIV_P(best_p >> best_p_pot) | PLL_REF_DIV_S(best_s);332if (best_p_pot) /* Prefer /2 pre-divider */333ref_div |= PLL_REF_DIV_Pe;334335/* Clock source selection either external clock or MIPI DSI clock lane */336chipone_writeb(icn, PLL_CTRL(6),337icn->refclk ? PLL_CTRL_6_EXTERNAL : PLL_CTRL_6_MIPI_CLK);338chipone_writeb(icn, PLL_REF_DIV, ref_div);339chipone_writeb(icn, PLL_INT(0), best_m);340}341342static void chipone_atomic_enable(struct drm_bridge *bridge,343struct drm_atomic_state *state)344{345struct chipone *icn = bridge_to_chipone(bridge);346struct drm_display_mode *mode = &icn->mode;347const struct drm_bridge_state *bridge_state;348u16 hfp, hbp, hsync;349u32 bus_flags;350u8 pol, sys_ctrl_1, id[4];351352chipone_readb(icn, VENDOR_ID, id);353chipone_readb(icn, DEVICE_ID_H, id + 1);354chipone_readb(icn, DEVICE_ID_L, id + 2);355chipone_readb(icn, VERSION_ID, id + 3);356357dev_dbg(icn->dev,358"Chip IDs: Vendor=0x%02x Device=0x%02x:0x%02x Version=0x%02x\n",359id[0], id[1], id[2], id[3]);360361if (id[0] != 0xc1 || id[1] != 0x62 || id[2] != 0x11) {362dev_dbg(icn->dev, "Invalid Chip IDs, aborting configuration\n");363return;364}365366/* Get the DPI flags from the bridge state. */367bridge_state = drm_atomic_get_new_bridge_state(state, bridge);368bus_flags = bridge_state->output_bus_cfg.flags;369370if (icn->interface_i2c)371chipone_writeb(icn, MIPI_CFG_PW, MIPI_CFG_PW_CONFIG_I2C);372else373chipone_writeb(icn, MIPI_CFG_PW, MIPI_CFG_PW_CONFIG_DSI);374375chipone_writeb(icn, HACTIVE_LI, mode->hdisplay & 0xff);376377chipone_writeb(icn, VACTIVE_LI, mode->vdisplay & 0xff);378379/*380* lsb nibble: 2nd nibble of hdisplay381* msb nibble: 2nd nibble of vdisplay382*/383chipone_writeb(icn, VACTIVE_HACTIVE_HI,384((mode->hdisplay >> 8) & 0xf) |385(((mode->vdisplay >> 8) & 0xf) << 4));386387hfp = mode->hsync_start - mode->hdisplay;388hsync = mode->hsync_end - mode->hsync_start;389hbp = mode->htotal - mode->hsync_end;390391chipone_writeb(icn, HFP_LI, hfp & 0xff);392chipone_writeb(icn, HSYNC_LI, hsync & 0xff);393chipone_writeb(icn, HBP_LI, hbp & 0xff);394/* Top two bits of Horizontal Front porch/Sync/Back porch */395chipone_writeb(icn, HFP_HSW_HBP_HI,396HFP_HSW_HBP_HI_HFP(hfp) |397HFP_HSW_HBP_HI_HS(hsync) |398HFP_HSW_HBP_HI_HBP(hbp));399400chipone_writeb(icn, VFP, mode->vsync_start - mode->vdisplay);401402chipone_writeb(icn, VSYNC, mode->vsync_end - mode->vsync_start);403404chipone_writeb(icn, VBP, mode->vtotal - mode->vsync_end);405406/* dsi specific sequence */407chipone_writeb(icn, SYNC_EVENT_DLY, 0x80);408chipone_writeb(icn, HFP_MIN, hfp & 0xff);409410/* DSI data lane count */411chipone_writeb(icn, DSI_CTRL,412DSI_CTRL_UNKNOWN | DSI_CTRL_DSI_LANES(icn->dsi->lanes - 1));413414chipone_writeb(icn, MIPI_PD_CK_LANE, 0xa0);415chipone_writeb(icn, PLL_CTRL(12), 0xff);416chipone_writeb(icn, MIPI_PN_SWAP, 0x00);417418/* DPI HS/VS/DE polarity */419pol = ((mode->flags & DRM_MODE_FLAG_PHSYNC) ? BIST_POL_HSYNC_POL : 0) |420((mode->flags & DRM_MODE_FLAG_PVSYNC) ? BIST_POL_VSYNC_POL : 0) |421((bus_flags & DRM_BUS_FLAG_DE_HIGH) ? BIST_POL_DE_POL : 0);422chipone_writeb(icn, BIST_POL, pol);423424/* Configure PLL settings */425chipone_configure_pll(icn, mode);426427chipone_writeb(icn, SYS_CTRL(0), 0x40);428sys_ctrl_1 = 0x88;429430if (bus_flags & DRM_BUS_FLAG_PIXDATA_DRIVE_POSEDGE)431sys_ctrl_1 |= FIELD_PREP(SYS_CTRL_1_CLK_PHASE_MSK, CLK_PHASE_0);432else433sys_ctrl_1 |= FIELD_PREP(SYS_CTRL_1_CLK_PHASE_MSK, CLK_PHASE_1_2);434435chipone_writeb(icn, SYS_CTRL(1), sys_ctrl_1);436437/* icn6211 specific sequence */438chipone_writeb(icn, MIPI_FORCE_0, 0x20);439chipone_writeb(icn, PLL_CTRL(1), 0x20);440chipone_writeb(icn, CONFIG_FINISH, 0x10);441442usleep_range(10000, 11000);443}444445static void chipone_atomic_pre_enable(struct drm_bridge *bridge,446struct drm_atomic_state *state)447{448struct chipone *icn = bridge_to_chipone(bridge);449int ret;450451if (icn->vdd1) {452ret = regulator_enable(icn->vdd1);453if (ret)454DRM_DEV_ERROR(icn->dev,455"failed to enable VDD1 regulator: %d\n", ret);456}457458if (icn->vdd2) {459ret = regulator_enable(icn->vdd2);460if (ret)461DRM_DEV_ERROR(icn->dev,462"failed to enable VDD2 regulator: %d\n", ret);463}464465if (icn->vdd3) {466ret = regulator_enable(icn->vdd3);467if (ret)468DRM_DEV_ERROR(icn->dev,469"failed to enable VDD3 regulator: %d\n", ret);470}471472ret = clk_prepare_enable(icn->refclk);473if (ret)474DRM_DEV_ERROR(icn->dev,475"failed to enable RECLK clock: %d\n", ret);476477gpiod_set_value(icn->enable_gpio, 1);478479usleep_range(10000, 11000);480}481482static void chipone_atomic_post_disable(struct drm_bridge *bridge,483struct drm_atomic_state *state)484{485struct chipone *icn = bridge_to_chipone(bridge);486487clk_disable_unprepare(icn->refclk);488489if (icn->vdd1)490regulator_disable(icn->vdd1);491492if (icn->vdd2)493regulator_disable(icn->vdd2);494495if (icn->vdd3)496regulator_disable(icn->vdd3);497498gpiod_set_value(icn->enable_gpio, 0);499}500501static void chipone_mode_set(struct drm_bridge *bridge,502const struct drm_display_mode *mode,503const struct drm_display_mode *adjusted_mode)504{505struct chipone *icn = bridge_to_chipone(bridge);506507drm_mode_copy(&icn->mode, adjusted_mode);508};509510static int chipone_dsi_attach(struct chipone *icn)511{512struct mipi_dsi_device *dsi = icn->dsi;513struct device *dev = icn->dev;514int dsi_lanes, ret;515516dsi_lanes = drm_of_get_data_lanes_count_ep(dev->of_node, 0, 0, 1, 4);517518/*519* If the 'data-lanes' property does not exist in DT or is invalid,520* default to previously hard-coded behavior, which was 4 data lanes.521*/522if (dsi_lanes < 0)523icn->dsi->lanes = 4;524else525icn->dsi->lanes = dsi_lanes;526527dsi->format = MIPI_DSI_FMT_RGB888;528dsi->mode_flags = MIPI_DSI_MODE_VIDEO | MIPI_DSI_MODE_VIDEO_BURST |529MIPI_DSI_MODE_LPM | MIPI_DSI_MODE_NO_EOT_PACKET;530dsi->hs_rate = 500000000;531dsi->lp_rate = 16000000;532533ret = mipi_dsi_attach(dsi);534if (ret < 0)535dev_err(icn->dev, "failed to attach dsi\n");536537return ret;538}539540static int chipone_dsi_host_attach(struct chipone *icn)541{542struct device *dev = icn->dev;543struct device_node *host_node;544struct device_node *endpoint;545struct mipi_dsi_device *dsi;546struct mipi_dsi_host *host;547int ret = 0;548549const struct mipi_dsi_device_info info = {550.type = "chipone",551.channel = 0,552.node = NULL,553};554555endpoint = of_graph_get_endpoint_by_regs(dev->of_node, 0, 0);556host_node = of_graph_get_remote_port_parent(endpoint);557of_node_put(endpoint);558559if (!host_node)560return -EINVAL;561562host = of_find_mipi_dsi_host_by_node(host_node);563of_node_put(host_node);564if (!host)565return dev_err_probe(dev, -EPROBE_DEFER, "failed to find dsi host\n");566567dsi = mipi_dsi_device_register_full(host, &info);568if (IS_ERR(dsi)) {569return dev_err_probe(dev, PTR_ERR(dsi),570"failed to create dsi device\n");571}572573icn->dsi = dsi;574575ret = chipone_dsi_attach(icn);576if (ret < 0)577mipi_dsi_device_unregister(dsi);578579return ret;580}581582static int chipone_attach(struct drm_bridge *bridge,583struct drm_encoder *encoder,584enum drm_bridge_attach_flags flags)585{586struct chipone *icn = bridge_to_chipone(bridge);587588return drm_bridge_attach(encoder, icn->panel_bridge, bridge, flags);589}590591#define MAX_INPUT_SEL_FORMATS 1592593static u32 *594chipone_atomic_get_input_bus_fmts(struct drm_bridge *bridge,595struct drm_bridge_state *bridge_state,596struct drm_crtc_state *crtc_state,597struct drm_connector_state *conn_state,598u32 output_fmt,599unsigned int *num_input_fmts)600{601u32 *input_fmts;602603*num_input_fmts = 0;604605input_fmts = kcalloc(MAX_INPUT_SEL_FORMATS, sizeof(*input_fmts),606GFP_KERNEL);607if (!input_fmts)608return NULL;609610/* This is the DSI-end bus format */611input_fmts[0] = MEDIA_BUS_FMT_RGB888_1X24;612*num_input_fmts = 1;613614return input_fmts;615}616617static const struct drm_bridge_funcs chipone_bridge_funcs = {618.atomic_duplicate_state = drm_atomic_helper_bridge_duplicate_state,619.atomic_destroy_state = drm_atomic_helper_bridge_destroy_state,620.atomic_reset = drm_atomic_helper_bridge_reset,621.atomic_pre_enable = chipone_atomic_pre_enable,622.atomic_enable = chipone_atomic_enable,623.atomic_post_disable = chipone_atomic_post_disable,624.mode_set = chipone_mode_set,625.attach = chipone_attach,626.atomic_get_input_bus_fmts = chipone_atomic_get_input_bus_fmts,627};628629static int chipone_parse_dt(struct chipone *icn)630{631struct device *dev = icn->dev;632int ret;633634icn->refclk = devm_clk_get_optional(dev, "refclk");635if (IS_ERR(icn->refclk)) {636ret = PTR_ERR(icn->refclk);637DRM_DEV_ERROR(dev, "failed to get REFCLK clock: %d\n", ret);638return ret;639} else if (icn->refclk) {640icn->refclk_rate = clk_get_rate(icn->refclk);641if (icn->refclk_rate < 10000000 || icn->refclk_rate > 154000000) {642DRM_DEV_ERROR(dev, "REFCLK out of range: %ld Hz\n",643icn->refclk_rate);644return -EINVAL;645}646}647648icn->vdd1 = devm_regulator_get_optional(dev, "vdd1");649if (IS_ERR(icn->vdd1)) {650ret = PTR_ERR(icn->vdd1);651if (ret == -EPROBE_DEFER)652return -EPROBE_DEFER;653icn->vdd1 = NULL;654DRM_DEV_DEBUG(dev, "failed to get VDD1 regulator: %d\n", ret);655}656657icn->vdd2 = devm_regulator_get_optional(dev, "vdd2");658if (IS_ERR(icn->vdd2)) {659ret = PTR_ERR(icn->vdd2);660if (ret == -EPROBE_DEFER)661return -EPROBE_DEFER;662icn->vdd2 = NULL;663DRM_DEV_DEBUG(dev, "failed to get VDD2 regulator: %d\n", ret);664}665666icn->vdd3 = devm_regulator_get_optional(dev, "vdd3");667if (IS_ERR(icn->vdd3)) {668ret = PTR_ERR(icn->vdd3);669if (ret == -EPROBE_DEFER)670return -EPROBE_DEFER;671icn->vdd3 = NULL;672DRM_DEV_DEBUG(dev, "failed to get VDD3 regulator: %d\n", ret);673}674675icn->enable_gpio = devm_gpiod_get(dev, "enable", GPIOD_OUT_LOW);676if (IS_ERR(icn->enable_gpio)) {677DRM_DEV_ERROR(dev, "failed to get enable GPIO\n");678return PTR_ERR(icn->enable_gpio);679}680681icn->panel_bridge = devm_drm_of_get_bridge(dev, dev->of_node, 1, 0);682if (IS_ERR(icn->panel_bridge))683return PTR_ERR(icn->panel_bridge);684685return 0;686}687688static int chipone_common_probe(struct device *dev, struct chipone **icnr)689{690struct chipone *icn;691int ret;692693icn = devm_drm_bridge_alloc(dev, struct chipone, bridge,694&chipone_bridge_funcs);695if (IS_ERR(icn))696return PTR_ERR(icn);697698icn->dev = dev;699700ret = chipone_parse_dt(icn);701if (ret)702return ret;703704icn->bridge.type = DRM_MODE_CONNECTOR_DPI;705icn->bridge.of_node = dev->of_node;706707*icnr = icn;708709return ret;710}711712static int chipone_dsi_probe(struct mipi_dsi_device *dsi)713{714struct device *dev = &dsi->dev;715struct chipone *icn;716int ret;717718ret = chipone_common_probe(dev, &icn);719if (ret)720return ret;721722icn->regmap = devm_regmap_init(dev, &chipone_dsi_regmap_bus,723dsi, &chipone_regmap_config);724if (IS_ERR(icn->regmap))725return PTR_ERR(icn->regmap);726727icn->interface_i2c = false;728icn->dsi = dsi;729730mipi_dsi_set_drvdata(dsi, icn);731732drm_bridge_add(&icn->bridge);733734ret = chipone_dsi_attach(icn);735if (ret)736drm_bridge_remove(&icn->bridge);737738return ret;739}740741static int chipone_i2c_probe(struct i2c_client *client)742{743struct device *dev = &client->dev;744struct chipone *icn;745int ret;746747ret = chipone_common_probe(dev, &icn);748if (ret)749return ret;750751icn->regmap = devm_regmap_init_i2c(client, &chipone_regmap_config);752if (IS_ERR(icn->regmap))753return PTR_ERR(icn->regmap);754755icn->interface_i2c = true;756icn->client = client;757dev_set_drvdata(dev, icn);758i2c_set_clientdata(client, icn);759760drm_bridge_add(&icn->bridge);761762return chipone_dsi_host_attach(icn);763}764765static void chipone_dsi_remove(struct mipi_dsi_device *dsi)766{767struct chipone *icn = mipi_dsi_get_drvdata(dsi);768769mipi_dsi_detach(dsi);770drm_bridge_remove(&icn->bridge);771}772773static const struct of_device_id chipone_of_match[] = {774{ .compatible = "chipone,icn6211", },775{ /* sentinel */ }776};777MODULE_DEVICE_TABLE(of, chipone_of_match);778779static struct mipi_dsi_driver chipone_dsi_driver = {780.probe = chipone_dsi_probe,781.remove = chipone_dsi_remove,782.driver = {783.name = "chipone-icn6211",784.of_match_table = chipone_of_match,785},786};787788static const struct i2c_device_id chipone_i2c_id[] = {789{ "chipone,icn6211" },790{},791};792MODULE_DEVICE_TABLE(i2c, chipone_i2c_id);793794static struct i2c_driver chipone_i2c_driver = {795.probe = chipone_i2c_probe,796.id_table = chipone_i2c_id,797.driver = {798.name = "chipone-icn6211-i2c",799.of_match_table = chipone_of_match,800},801};802803static int __init chipone_init(void)804{805if (IS_ENABLED(CONFIG_DRM_MIPI_DSI))806mipi_dsi_driver_register(&chipone_dsi_driver);807808return i2c_add_driver(&chipone_i2c_driver);809}810module_init(chipone_init);811812static void __exit chipone_exit(void)813{814i2c_del_driver(&chipone_i2c_driver);815816if (IS_ENABLED(CONFIG_DRM_MIPI_DSI))817mipi_dsi_driver_unregister(&chipone_dsi_driver);818}819module_exit(chipone_exit);820821MODULE_AUTHOR("Jagan Teki <[email protected]>");822MODULE_DESCRIPTION("Chipone ICN6211 MIPI-DSI to RGB Converter Bridge");823MODULE_LICENSE("GPL");824825826