1 // SPDX-License-Identifier: GPL-2.0-or-later
3 * Copyright (C) 2021 BayLibre, SAS
5 * Copyright (C) 2015 Amlogic, Inc. All rights reserved.
9 #include <linux/kernel.h>
10 #include <linux/module.h>
11 #include <linux/of_device.h>
12 #include <linux/of_graph.h>
13 #include <linux/reset.h>
14 #include <linux/phy/phy.h>
15 #include <linux/bitfield.h>
17 #include <video/mipi_display.h>
19 #include <drm/bridge/dw_mipi_dsi.h>
20 #include <drm/drm_mipi_dsi.h>
22 #include <drm/drm_atomic_helper.h>
23 #include <drm/drm_device.h>
24 #include <drm/drm_probe_helper.h>
25 #include <drm/drm_print.h>
27 #include "meson_drv.h"
28 #include "meson_dw_mipi_dsi.h"
29 #include "meson_registers.h"
30 #include "meson_venc.h"
32 #define DRIVER_NAME "meson-dw-mipi-dsi"
33 #define DRIVER_DESC "Amlogic Meson MIPI-DSI DRM driver"
35 struct meson_dw_mipi_dsi {
36 struct meson_drm *priv;
40 union phy_configure_opts phy_opts;
41 struct dw_mipi_dsi *dmd;
42 struct dw_mipi_dsi_plat_data pdata;
43 struct mipi_dsi_device *dsi_device;
44 const struct drm_display_mode *mode;
47 struct reset_control *top_rst;
50 #define encoder_to_meson_dw_mipi_dsi(x) \
51 container_of(x, struct meson_dw_mipi_dsi, encoder)
53 static void meson_dw_mipi_dsi_hw_init(struct meson_dw_mipi_dsi *mipi_dsi)
56 writel_bits_relaxed(MIPI_DSI_TOP_SW_RESET_DWC | MIPI_DSI_TOP_SW_RESET_INTR |
57 MIPI_DSI_TOP_SW_RESET_DPI | MIPI_DSI_TOP_SW_RESET_TIMING,
58 MIPI_DSI_TOP_SW_RESET_DWC | MIPI_DSI_TOP_SW_RESET_INTR |
59 MIPI_DSI_TOP_SW_RESET_DPI | MIPI_DSI_TOP_SW_RESET_TIMING,
60 mipi_dsi->base + MIPI_DSI_TOP_SW_RESET);
61 writel_bits_relaxed(MIPI_DSI_TOP_SW_RESET_DWC | MIPI_DSI_TOP_SW_RESET_INTR |
62 MIPI_DSI_TOP_SW_RESET_DPI | MIPI_DSI_TOP_SW_RESET_TIMING,
63 0, mipi_dsi->base + MIPI_DSI_TOP_SW_RESET);
66 writel_bits_relaxed(MIPI_DSI_TOP_CLK_SYSCLK_EN | MIPI_DSI_TOP_CLK_PIXCLK_EN,
67 MIPI_DSI_TOP_CLK_SYSCLK_EN | MIPI_DSI_TOP_CLK_PIXCLK_EN,
68 mipi_dsi->base + MIPI_DSI_TOP_CLK_CNTL);
70 /* Take memory out of power down */
71 writel_relaxed(0, mipi_dsi->base + MIPI_DSI_TOP_MEM_PD);
74 static int dw_mipi_dsi_phy_init(void *priv_data)
76 struct meson_dw_mipi_dsi *mipi_dsi = priv_data;
77 unsigned int dpi_data_format, venc_data_width;
80 /* Set the bit clock rate to hs_clk_rate */
81 ret = clk_set_rate(mipi_dsi->bit_clk,
82 mipi_dsi->phy_opts.mipi_dphy.hs_clk_rate);
84 dev_err(mipi_dsi->dev, "Failed to set DSI Bit clock rate %lu (ret %d)\n",
85 mipi_dsi->phy_opts.mipi_dphy.hs_clk_rate, ret);
89 /* Make sure the rate of the bit clock is not modified by someone else */
90 ret = clk_rate_exclusive_get(mipi_dsi->bit_clk);
92 dev_err(mipi_dsi->dev,
93 "Failed to set the exclusivity on the bit clock rate (ret %d)\n", ret);
97 ret = clk_set_rate(mipi_dsi->px_clk, mipi_dsi->mode->clock * 1000);
100 dev_err(mipi_dsi->dev, "Failed to set DSI Pixel clock rate %u (%d)\n",
101 mipi_dsi->mode->clock * 1000, ret);
105 switch (mipi_dsi->dsi_device->format) {
106 case MIPI_DSI_FMT_RGB888:
107 dpi_data_format = DPI_COLOR_24BIT;
108 venc_data_width = VENC_IN_COLOR_24B;
110 case MIPI_DSI_FMT_RGB666:
111 dpi_data_format = DPI_COLOR_18BIT_CFG_2;
112 venc_data_width = VENC_IN_COLOR_18B;
114 case MIPI_DSI_FMT_RGB666_PACKED:
115 case MIPI_DSI_FMT_RGB565:
119 /* Configure color format for DPI register */
120 writel_relaxed(FIELD_PREP(MIPI_DSI_TOP_DPI_COLOR_MODE, dpi_data_format) |
121 FIELD_PREP(MIPI_DSI_TOP_IN_COLOR_MODE, venc_data_width) |
122 FIELD_PREP(MIPI_DSI_TOP_COMP2_SEL, 2) |
123 FIELD_PREP(MIPI_DSI_TOP_COMP1_SEL, 1) |
124 FIELD_PREP(MIPI_DSI_TOP_COMP0_SEL, 0),
125 mipi_dsi->base + MIPI_DSI_TOP_CNTL);
127 return phy_configure(mipi_dsi->phy, &mipi_dsi->phy_opts);
130 static void dw_mipi_dsi_phy_power_on(void *priv_data)
132 struct meson_dw_mipi_dsi *mipi_dsi = priv_data;
134 if (phy_power_on(mipi_dsi->phy))
135 dev_warn(mipi_dsi->dev, "Failed to power on PHY\n");
138 static void dw_mipi_dsi_phy_power_off(void *priv_data)
140 struct meson_dw_mipi_dsi *mipi_dsi = priv_data;
142 if (phy_power_off(mipi_dsi->phy))
143 dev_warn(mipi_dsi->dev, "Failed to power off PHY\n");
145 /* Remove the exclusivity on the bit clock rate */
146 clk_rate_exclusive_put(mipi_dsi->bit_clk);
150 dw_mipi_dsi_get_lane_mbps(void *priv_data, const struct drm_display_mode *mode,
151 unsigned long mode_flags, u32 lanes, u32 format,
152 unsigned int *lane_mbps)
154 struct meson_dw_mipi_dsi *mipi_dsi = priv_data;
157 mipi_dsi->mode = mode;
159 bpp = mipi_dsi_pixel_format_to_bpp(mipi_dsi->dsi_device->format);
161 phy_mipi_dphy_get_default_config(mode->clock * 1000,
162 bpp, mipi_dsi->dsi_device->lanes,
163 &mipi_dsi->phy_opts.mipi_dphy);
165 *lane_mbps = DIV_ROUND_UP(mipi_dsi->phy_opts.mipi_dphy.hs_clk_rate, USEC_PER_SEC);
171 dw_mipi_dsi_phy_get_timing(void *priv_data, unsigned int lane_mbps,
172 struct dw_mipi_dsi_dphy_timing *timing)
174 struct meson_dw_mipi_dsi *mipi_dsi = priv_data;
176 switch (mipi_dsi->mode->hdisplay) {
181 timing->clk_lp2hs = 23;
182 timing->clk_hs2lp = 38;
183 timing->data_lp2hs = 15;
184 timing->data_hs2lp = 9;
188 timing->clk_lp2hs = 37;
189 timing->clk_hs2lp = 135;
190 timing->data_lp2hs = 50;
191 timing->data_hs2lp = 3;
198 dw_mipi_dsi_get_esc_clk_rate(void *priv_data, unsigned int *esc_clk_rate)
200 *esc_clk_rate = 4; /* Mhz */
205 static const struct dw_mipi_dsi_phy_ops meson_dw_mipi_dsi_phy_ops = {
206 .init = dw_mipi_dsi_phy_init,
207 .power_on = dw_mipi_dsi_phy_power_on,
208 .power_off = dw_mipi_dsi_phy_power_off,
209 .get_lane_mbps = dw_mipi_dsi_get_lane_mbps,
210 .get_timing = dw_mipi_dsi_phy_get_timing,
211 .get_esc_clk_rate = dw_mipi_dsi_get_esc_clk_rate,
214 static int meson_dw_mipi_dsi_host_attach(void *priv_data,
215 struct mipi_dsi_device *device)
217 struct meson_dw_mipi_dsi *mipi_dsi = priv_data;
220 mipi_dsi->dsi_device = device;
222 switch (device->format) {
223 case MIPI_DSI_FMT_RGB888:
225 case MIPI_DSI_FMT_RGB666:
227 case MIPI_DSI_FMT_RGB666_PACKED:
228 case MIPI_DSI_FMT_RGB565:
229 dev_err(mipi_dsi->dev, "invalid pixel format %d\n", device->format);
233 ret = phy_init(mipi_dsi->phy);
237 meson_dw_mipi_dsi_hw_init(mipi_dsi);
242 static int meson_dw_mipi_dsi_host_detach(void *priv_data,
243 struct mipi_dsi_device *device)
245 struct meson_dw_mipi_dsi *mipi_dsi = priv_data;
247 if (device == mipi_dsi->dsi_device)
248 mipi_dsi->dsi_device = NULL;
252 return phy_exit(mipi_dsi->phy);
255 static const struct dw_mipi_dsi_host_ops meson_dw_mipi_dsi_host_ops = {
256 .attach = meson_dw_mipi_dsi_host_attach,
257 .detach = meson_dw_mipi_dsi_host_detach,
260 static int meson_dw_mipi_dsi_probe(struct platform_device *pdev)
262 struct meson_dw_mipi_dsi *mipi_dsi;
263 struct device *dev = &pdev->dev;
265 mipi_dsi = devm_kzalloc(dev, sizeof(*mipi_dsi), GFP_KERNEL);
269 mipi_dsi->base = devm_platform_ioremap_resource(pdev, 0);
270 if (IS_ERR(mipi_dsi->base))
271 return PTR_ERR(mipi_dsi->base);
273 mipi_dsi->phy = devm_phy_get(dev, "dphy");
274 if (IS_ERR(mipi_dsi->phy))
275 return dev_err_probe(dev, PTR_ERR(mipi_dsi->phy),
276 "failed to get mipi dphy\n");
278 mipi_dsi->bit_clk = devm_clk_get_enabled(dev, "bit");
279 if (IS_ERR(mipi_dsi->bit_clk)) {
280 int ret = PTR_ERR(mipi_dsi->bit_clk);
282 /* TOFIX GP0 on some platforms fails to lock in early boot, defer probe */
286 return dev_err_probe(dev, ret, "Unable to get enabled bit_clk\n");
289 mipi_dsi->px_clk = devm_clk_get_enabled(dev, "px");
290 if (IS_ERR(mipi_dsi->px_clk))
291 return dev_err_probe(dev, PTR_ERR(mipi_dsi->px_clk),
292 "Unable to get enabled px_clk\n");
295 * We use a TOP reset signal because the APB reset signal
296 * is handled by the TOP control registers.
298 mipi_dsi->top_rst = devm_reset_control_get_exclusive(dev, "top");
299 if (IS_ERR(mipi_dsi->top_rst))
300 return dev_err_probe(dev, PTR_ERR(mipi_dsi->top_rst),
301 "Unable to get reset control\n");
303 reset_control_assert(mipi_dsi->top_rst);
304 usleep_range(10, 20);
305 reset_control_deassert(mipi_dsi->top_rst);
307 /* MIPI DSI Controller */
310 mipi_dsi->pdata.base = mipi_dsi->base;
311 mipi_dsi->pdata.max_data_lanes = 4;
312 mipi_dsi->pdata.phy_ops = &meson_dw_mipi_dsi_phy_ops;
313 mipi_dsi->pdata.host_ops = &meson_dw_mipi_dsi_host_ops;
314 mipi_dsi->pdata.priv_data = mipi_dsi;
315 platform_set_drvdata(pdev, mipi_dsi);
317 mipi_dsi->dmd = dw_mipi_dsi_probe(pdev, &mipi_dsi->pdata);
318 if (IS_ERR(mipi_dsi->dmd))
319 return dev_err_probe(dev, PTR_ERR(mipi_dsi->dmd),
320 "Failed to probe dw_mipi_dsi\n");
325 static int meson_dw_mipi_dsi_remove(struct platform_device *pdev)
327 struct meson_dw_mipi_dsi *mipi_dsi = platform_get_drvdata(pdev);
329 dw_mipi_dsi_remove(mipi_dsi->dmd);
334 static const struct of_device_id meson_dw_mipi_dsi_of_table[] = {
335 { .compatible = "amlogic,meson-g12a-dw-mipi-dsi", },
338 MODULE_DEVICE_TABLE(of, meson_dw_mipi_dsi_of_table);
340 static struct platform_driver meson_dw_mipi_dsi_platform_driver = {
341 .probe = meson_dw_mipi_dsi_probe,
342 .remove = meson_dw_mipi_dsi_remove,
345 .of_match_table = meson_dw_mipi_dsi_of_table,
348 module_platform_driver(meson_dw_mipi_dsi_platform_driver);
351 MODULE_DESCRIPTION(DRIVER_DESC);
352 MODULE_LICENSE("GPL");