On 22.01.2018 16:08, Philippe Cornu wrote:
> From: Philippe CORNU <philippe.co...@st.com>
>
> Add support for the Synopsys DesignWare MIPI DSI version 1.31
> Two registers need to be updated/added for supporting 1.31:
> * PHY_TMR_CFG 0x9c (updated)
>   1.30 [31:24] phy_hs2lp_time
>        [23:16] phy_lp2hs_time
>        [14: 0] max_rd_time
>
>   1.31 [25:16] phy_hs2lp_time
>        [ 9: 0] phy_lp2hs_time
>
> * PHY_TMR_RD_CFG 0xf4 (new)
>   1.31 [14: 0] max_rd_time
>
> Signed-off-by: Philippe Cornu <philippe.co...@st.com>
> ---
>  drivers/gpu/drm/bridge/synopsys/dw-mipi-dsi.c | 52 
> +++++++++++++++++++++++----
>  1 file changed, 46 insertions(+), 6 deletions(-)
>
> diff --git a/drivers/gpu/drm/bridge/synopsys/dw-mipi-dsi.c 
> b/drivers/gpu/drm/bridge/synopsys/dw-mipi-dsi.c
> index 735f38429c06..20a2ca14a7ad 100644
> --- a/drivers/gpu/drm/bridge/synopsys/dw-mipi-dsi.c
> +++ b/drivers/gpu/drm/bridge/synopsys/dw-mipi-dsi.c
> @@ -25,7 +25,13 @@
>  #include <drm/bridge/dw_mipi_dsi.h>
>  #include <video/mipi_display.h>
>  
> +#define HWVER_130                    0x31333000      /* IP version 1.30 */
> +#define HWVER_131                    0x31333100      /* IP version 1.31 */
> +#define HWVER_OLDEST                 HWVER_130
> +#define HWVER_NEWEST                 HWVER_131
> +
>  #define DSI_VERSION                  0x00
> +#define VERSION                              GENMASK(31, 8)
>  
>  #define DSI_PWR_UP                   0x04
>  #define RESET                                0
> @@ -161,11 +167,12 @@
>  #define PHY_CLKHS2LP_TIME(lbcc)              (((lbcc) & 0x3ff) << 16)
>  #define PHY_CLKLP2HS_TIME(lbcc)              ((lbcc) & 0x3ff)
>  
> -/* TODO Next register is slightly different between 1.30 & 1.31 IP version */
>  #define DSI_PHY_TMR_CFG                      0x9c
> -#define PHY_HS2LP_TIME(lbcc)         (((lbcc) & 0xff) << 24)
> -#define PHY_LP2HS_TIME(lbcc)         (((lbcc) & 0xff) << 16)
> -#define MAX_RD_TIME(lbcc)            ((lbcc) & 0x7fff)
> +#define PHY_HS2LP_TIME_V130(lbcc)    (((lbcc) & 0xff) << 24)
> +#define PHY_LP2HS_TIME_V130(lbcc)    (((lbcc) & 0xff) << 16)
> +#define MAX_RD_TIME_V130(lbcc)               ((lbcc) & 0x7fff)
> +#define PHY_HS2LP_TIME_V131(lbcc)    (((lbcc) & 0x3ff) << 16)
> +#define PHY_LP2HS_TIME_V131(lbcc)    ((lbcc) & 0x3ff)
>  
>  #define DSI_PHY_RSTZ                 0xa0
>  #define PHY_DISFORCEPLL                      0
> @@ -204,7 +211,9 @@
>  #define DSI_INT_ST1                  0xc0
>  #define DSI_INT_MSK0                 0xc4
>  #define DSI_INT_MSK1                 0xc8
> +
>  #define DSI_PHY_TMR_RD_CFG           0xf4
> +#define MAX_RD_TIME_V131(lbcc)               ((lbcc) & 0x7fff)
>  
>  #define PHY_STATUS_TIMEOUT_US                10000
>  #define CMD_PKT_STATUS_TIMEOUT_US    20000
> @@ -215,6 +224,7 @@ struct dw_mipi_dsi {
>       struct drm_bridge *panel_bridge;
>       struct device *dev;
>       void __iomem *base;
> +     u32 hw_version;
>  
>       struct clk *pclk;
>       struct clk *px_clk;
> @@ -616,8 +626,14 @@ static void dw_mipi_dsi_dphy_timing_config(struct 
> dw_mipi_dsi *dsi)
>        * note: DSI_PHY_TMR_CFG.MAX_RD_TIME should be in line with
>        * DSI_CMD_MODE_CFG.MAX_RD_PKT_SIZE_LP (see CMD_MODE_ALL_LP)
>        */
> -     dsi_write(dsi, DSI_PHY_TMR_CFG, PHY_HS2LP_TIME(0x40)
> -               | PHY_LP2HS_TIME(0x40) | MAX_RD_TIME(10000));
> +     if (dsi->hw_version == HWVER_131) {
> +             dsi_write(dsi, DSI_PHY_TMR_CFG, PHY_HS2LP_TIME_V131(0x40) |
> +                       PHY_LP2HS_TIME_V131(0x40));
> +             dsi_write(dsi, DSI_PHY_TMR_RD_CFG, MAX_RD_TIME_V131(10000));
> +     } else {
> +             dsi_write(dsi, DSI_PHY_TMR_CFG, PHY_HS2LP_TIME_V130(0x40) |
> +                       PHY_LP2HS_TIME_V130(0x40) | MAX_RD_TIME_V130(10000));
> +     }
>  
>       dsi_write(dsi, DSI_PHY_TMR_LPCLK_CFG, PHY_CLKHS2LP_TIME(0x40)
>                 | PHY_CLKLP2HS_TIME(0x40));
> @@ -791,6 +807,28 @@ static const struct drm_bridge_funcs 
> dw_mipi_dsi_bridge_funcs = {
>       .attach       = dw_mipi_dsi_bridge_attach,
>  };
>  
> +static void dsi_get_version(struct dw_mipi_dsi *dsi)
> +{
> +     u32 hw_version;
> +
> +     clk_prepare_enable(dsi->pclk);
> +     hw_version = dsi_read(dsi, DSI_VERSION) & VERSION;
> +     clk_disable_unprepare(dsi->pclk);
> +
> +     if (hw_version > HWVER_NEWEST) {
> +             DRM_DEBUG("hw version: use 0x%08x for this recent 0x%08x\n",
> +                       HWVER_NEWEST, hw_version);
> +             hw_version = HWVER_NEWEST;
> +
> +     } else if (hw_version < HWVER_OLDEST) {
> +             DRM_DEBUG("hw version: use 0x%08x for this old 0x%08x\n",
> +                       HWVER_OLDEST, hw_version);
> +             hw_version = HWVER_OLDEST;
> +     }
> +
> +     dsi->hw_version = hw_version;
> +}
> +
>  static struct dw_mipi_dsi *
>  __dw_mipi_dsi_probe(struct platform_device *pdev,
>                   const struct dw_mipi_dsi_plat_data *plat_data)
> @@ -870,6 +908,8 @@ __dw_mipi_dsi_probe(struct platform_device *pdev,
>               clk_disable_unprepare(dsi->pclk);
>       }
>  
> +     dsi_get_version(dsi);
> +
>       pm_runtime_enable(dev);

Two questions:
1. You have generally two variants:
- older than 131,
- not older than 131.
Wouldn't be better to just assume that since 131 registers slightly differ?
So you just stores:
    dsi->hw_version = dsi_read(dsi, DSI_VERSION) & VERSION;
And if neccessary performs comparison:
    if (dsi->hw_version >= HWVER_131)
        ...
    else
        ...

This way you can remove these HWVER_(NEWEST|OLDEST).

2. You are using pm_runtime, but not in dsi_get_version. I guess it is
SoC dependant  but I suppose performing registry read should be safer
after runtime_get.

Regards
Andrzej

>  
>       dsi->dsi_host.ops = &dw_mipi_dsi_host_ops;


_______________________________________________
dri-devel mailing list
dri-devel@lists.freedesktop.org
https://lists.freedesktop.org/mailman/listinfo/dri-devel

Reply via email to