goto exit;
        }
 
-       if (cfg_hnd->major == MSM_DSI_VER_MAJOR_6G &&
-           cfg_hnd->minor >= MSM_DSI_6G_VER_MINOR_V2_2_1) {
-               msm_host->byte_intf_clk = msm_clk_get(pdev, "byte_intf");
-               if (IS_ERR(msm_host->byte_intf_clk)) {
-                       ret = PTR_ERR(msm_host->byte_intf_clk);
-                       pr_err("%s: can't find byte_intf clock. ret=%d\n",
-                               __func__, ret);
-                       goto exit;
-               }
-       } else {
-               msm_host->byte_intf_clk = NULL;
-       }
-
        msm_host->byte_clk_src = clk_get_parent(msm_host->byte_clk);
        if (!msm_host->byte_clk_src) {
                ret = -ENODEV;
                goto exit;
        }
 
-       if (cfg_hnd->major == MSM_DSI_VER_MAJOR_V2) {
-               msm_host->src_clk = msm_clk_get(pdev, "src");
-               if (IS_ERR(msm_host->src_clk)) {
-                       ret = PTR_ERR(msm_host->src_clk);
-                       pr_err("%s: can't find src clock. ret=%d\n",
-                               __func__, ret);
-                       msm_host->src_clk = NULL;
-                       goto exit;
-               }
-
-               msm_host->esc_clk_src = clk_get_parent(msm_host->esc_clk);
-               if (!msm_host->esc_clk_src) {
-                       ret = -ENODEV;
-                       pr_err("%s: can't get esc clock parent. ret=%d\n",
-                               __func__, ret);
-                       goto exit;
-               }
-
-               msm_host->dsi_clk_src = clk_get_parent(msm_host->src_clk);
-               if (!msm_host->dsi_clk_src) {
-                       ret = -ENODEV;
-                       pr_err("%s: can't get src clock parent. ret=%d\n",
-                               __func__, ret);
-               }
-       }
+       if (cfg_hnd->ops->clk_init_ver)
+               ret = cfg_hnd->ops->clk_init_ver(msm_host);
 exit:
        return ret;
 }
        return ret;
 }
 
-static int dsi_link_clk_enable(struct msm_dsi_host *msm_host)
-{
-       const struct msm_dsi_cfg_handler *cfg_hnd = msm_host->cfg_hnd;
-
-       if (cfg_hnd->major == MSM_DSI_VER_MAJOR_6G)
-               return dsi_link_clk_enable_6g(msm_host);
-       else
-               return dsi_link_clk_enable_v2(msm_host);
-}
-
 void dsi_link_clk_disable_6g(struct msm_dsi_host *msm_host)
 {
        clk_disable_unprepare(msm_host->esc_clk);
        clk_disable_unprepare(msm_host->byte_clk);
 }
 
-static void dsi_link_clk_disable(struct msm_dsi_host *msm_host)
-{
-       const struct msm_dsi_cfg_handler *cfg_hnd = msm_host->cfg_hnd;
-
-       if (cfg_hnd->major == MSM_DSI_VER_MAJOR_6G) {
-               clk_disable_unprepare(msm_host->esc_clk);
-               clk_disable_unprepare(msm_host->pixel_clk);
-               if (msm_host->byte_intf_clk)
-                       clk_disable_unprepare(msm_host->byte_intf_clk);
-               clk_disable_unprepare(msm_host->byte_clk);
-       } else {
-               clk_disable_unprepare(msm_host->pixel_clk);
-               clk_disable_unprepare(msm_host->src_clk);
-               clk_disable_unprepare(msm_host->esc_clk);
-               clk_disable_unprepare(msm_host->byte_clk);
-       }
-}
-
 int dsi_calc_clk_rate_6g(struct msm_dsi_host *msm_host)
 {
        struct drm_display_mode *mode = msm_host->mode;
        return 0;
 }
 
-static int dsi_calc_clk_rate(struct msm_dsi_host *msm_host)
-{
-       struct drm_display_mode *mode = msm_host->mode;
-       const struct msm_dsi_cfg_handler *cfg_hnd = msm_host->cfg_hnd;
-       u8 lanes = msm_host->lanes;
-       u32 bpp = dsi_get_bpp(msm_host->format);
-       u32 pclk_rate;
-
-       if (!mode) {
-               pr_err("%s: mode not set\n", __func__);
-               return -EINVAL;
-       }
-
-       pclk_rate = mode->clock * 1000;
-       if (lanes > 0) {
-               msm_host->byte_clk_rate = (pclk_rate * bpp) / (8 * lanes);
-       } else {
-               pr_err("%s: forcing mdss_dsi lanes to 1\n", __func__);
-               msm_host->byte_clk_rate = (pclk_rate * bpp) / 8;
-       }
-
-       DBG("pclk=%d, bclk=%d", pclk_rate, msm_host->byte_clk_rate);
-
-       msm_host->esc_clk_rate = clk_get_rate(msm_host->esc_clk);
-
-       if (cfg_hnd->major == MSM_DSI_VER_MAJOR_V2) {
-               unsigned int esc_mhz, esc_div;
-               unsigned long byte_mhz;
-
-               msm_host->src_clk_rate = (pclk_rate * bpp) / 8;
-
-               /*
-                * esc clock is byte clock followed by a 4 bit divider,
-                * we need to find an escape clock frequency within the
-                * mipi DSI spec range within the maximum divider limit
-                * We iterate here between an escape clock frequencey
-                * between 20 Mhz to 5 Mhz and pick up the first one
-                * that can be supported by our divider
-                */
-
-               byte_mhz = msm_host->byte_clk_rate / 1000000;
-
-               for (esc_mhz = 20; esc_mhz >= 5; esc_mhz--) {
-                       esc_div = DIV_ROUND_UP(byte_mhz, esc_mhz);
-
-                       /*
-                        * TODO: Ideally, we shouldn't know what sort of divider
-                        * is available in mmss_cc, we're just assuming that
-                        * it'll always be a 4 bit divider. Need to come up with
-                        * a better way here.
-                        */
-                       if (esc_div >= 1 && esc_div <= 16)
-                               break;
-               }
-
-               if (esc_mhz < 5)
-                       return -EINVAL;
-
-               msm_host->esc_clk_rate = msm_host->byte_clk_rate / esc_div;
-
-               DBG("esc=%d, src=%d", msm_host->esc_clk_rate,
-                       msm_host->src_clk_rate);
-       }
-
-       return 0;
-}
-
 static void dsi_intr_ctrl(struct msm_dsi_host *msm_host, u32 mask, int enable)
 {
        u32 intr;
        return 0;
 }
 
-/* dsi_cmd */
-static int dsi_tx_buf_alloc(struct msm_dsi_host *msm_host, int size)
-{
-       struct drm_device *dev = msm_host->dev;
-       struct msm_drm_private *priv = dev->dev_private;
-       const struct msm_dsi_cfg_handler *cfg_hnd = msm_host->cfg_hnd;
-       int ret;
-       uint64_t iova;
-
-       if (cfg_hnd->major == MSM_DSI_VER_MAJOR_6G) {
-               msm_host->tx_gem_obj = msm_gem_new(dev, size, MSM_BO_UNCACHED);
-               if (IS_ERR(msm_host->tx_gem_obj)) {
-                       ret = PTR_ERR(msm_host->tx_gem_obj);
-                       pr_err("%s: failed to allocate gem, %d\n",
-                               __func__, ret);
-                       msm_host->tx_gem_obj = NULL;
-                       return ret;
-               }
-
-               ret = msm_gem_get_iova(msm_host->tx_gem_obj,
-                               priv->kms->aspace, &iova);
-               if (ret) {
-                       pr_err("%s: failed to get iova, %d\n", __func__, ret);
-                       return ret;
-               }
-
-               if (iova & 0x07) {
-                       pr_err("%s: buf NOT 8 bytes aligned\n", __func__);
-                       return -EINVAL;
-               }
-
-               msm_host->tx_size = msm_host->tx_gem_obj->size;
-       } else {
-               msm_host->tx_buf = dma_alloc_coherent(dev->dev, size,
-                                       &msm_host->tx_buf_paddr, GFP_KERNEL);
-               if (!msm_host->tx_buf) {
-                       ret = -ENOMEM;
-                       pr_err("%s: failed to allocate tx buf, %d\n",
-                               __func__, ret);
-                       return ret;
-               }
-
-               msm_host->tx_size = size;
-       }
-
-       return 0;
-}
-
 static void dsi_tx_buf_free(struct msm_dsi_host *msm_host)
 {
        struct drm_device *dev = msm_host->dev;
                return -EINVAL;
        }
 
-       if (cfg_hnd->major == MSM_DSI_VER_MAJOR_6G) {
-               data = msm_gem_get_vaddr(msm_host->tx_gem_obj);
-               if (IS_ERR(data)) {
-                       ret = PTR_ERR(data);
-                       pr_err("%s: get vaddr failed, %d\n", __func__, ret);
-                       return ret;
-               }
-       } else {
-               data = msm_host->tx_buf;
+       data = cfg_hnd->ops->tx_buf_get(msm_host);
+       if (IS_ERR(data)) {
+               ret = PTR_ERR(data);
+               pr_err("%s: get vaddr failed, %d\n", __func__, ret);
+               return ret;
        }
 
        /* MSM specific command format in memory */
        if (packet.size < len)
                memset(data + packet.size, 0xff, len - packet.size);
 
-       if (cfg_hnd->major == MSM_DSI_VER_MAJOR_6G)
-               msm_gem_put_vaddr(msm_host->tx_gem_obj);
+       if (cfg_hnd->ops->tx_buf_put)
+               cfg_hnd->ops->tx_buf_put(msm_host);
 
        return len;
 }
 static int dsi_cmd_dma_tx(struct msm_dsi_host *msm_host, int len)
 {
        const struct msm_dsi_cfg_handler *cfg_hnd = msm_host->cfg_hnd;
-       struct drm_device *dev = msm_host->dev;
-       struct msm_drm_private *priv = dev->dev_private;
        int ret;
        uint64_t dma_base;
        bool triggered;
 
-       if (cfg_hnd->major == MSM_DSI_VER_MAJOR_6G) {
-               ret = msm_gem_get_iova(msm_host->tx_gem_obj,
-                               priv->kms->aspace, &dma_base);
-               if (ret) {
-                       pr_err("%s: failed to get iova: %d\n", __func__, ret);
-                       return ret;
-               }
-       } else {
-               dma_base = msm_host->tx_buf_paddr;
+       ret = cfg_hnd->ops->dma_base_get(msm_host, &dma_base);
+       if (ret) {
+               pr_err("%s: failed to get iova: %d\n", __func__, ret);
+               return ret;
        }
 
        reinit_completion(&msm_host->dma_comp);
                                        struct drm_device *dev)
 {
        struct msm_dsi_host *msm_host = to_msm_dsi_host(host);
+       const struct msm_dsi_cfg_handler *cfg_hnd = msm_host->cfg_hnd;
        struct platform_device *pdev = msm_host->pdev;
        int ret;
 
        }
 
        msm_host->dev = dev;
-       ret = dsi_tx_buf_alloc(msm_host, SZ_4K);
+       ret = cfg_hnd->ops->tx_buf_alloc(msm_host, SZ_4K);
        if (ret) {
                pr_err("%s: alloc tx gem obj failed, %d\n", __func__, ret);
                return ret;
                                const struct mipi_dsi_msg *msg)
 {
        struct msm_dsi_host *msm_host = to_msm_dsi_host(host);
+       const struct msm_dsi_cfg_handler *cfg_hnd = msm_host->cfg_hnd;
 
        /* TODO: make sure dsi_cmd_mdp is idle.
         * Since DSI6G v1.2.0, we can set DSI_TRIG_CTRL.BLOCK_DMA_WITHIN_FRAME
         * mdp clock need to be enabled to receive dsi interrupt
         */
        pm_runtime_get_sync(&msm_host->pdev->dev);
-       dsi_link_clk_enable(msm_host);
+       cfg_hnd->ops->link_clk_enable(msm_host);
 
        /* TODO: vote for bus bandwidth */
 
                                const struct mipi_dsi_msg *msg)
 {
        struct msm_dsi_host *msm_host = to_msm_dsi_host(host);
+       const struct msm_dsi_cfg_handler *cfg_hnd = msm_host->cfg_hnd;
 
        dsi_intr_ctrl(msm_host, DSI_IRQ_MASK_CMD_DMA_DONE, 0);
        dsi_write(msm_host, REG_DSI_CTRL, msm_host->dma_cmd_ctrl_restore);
 
        /* TODO: unvote for bus bandwidth */
 
-       dsi_link_clk_disable(msm_host);
+       cfg_hnd->ops->link_clk_disable(msm_host);
        pm_runtime_put_autosuspend(&msm_host->pdev->dev);
 }
 
        struct msm_dsi_pll *src_pll)
 {
        struct msm_dsi_host *msm_host = to_msm_dsi_host(host);
-       const struct msm_dsi_cfg_handler *cfg_hnd = msm_host->cfg_hnd;
        struct clk *byte_clk_provider, *pixel_clk_provider;
        int ret;
 
                goto exit;
        }
 
-       if (cfg_hnd->major == MSM_DSI_VER_MAJOR_V2) {
+       if (msm_host->dsi_clk_src) {
                ret = clk_set_parent(msm_host->dsi_clk_src, pixel_clk_provider);
                if (ret) {
                        pr_err("%s: can't set parent to dsi_clk_src. ret=%d\n",
                                __func__, ret);
                        goto exit;
                }
+       }
 
+       if (msm_host->esc_clk_src) {
                ret = clk_set_parent(msm_host->esc_clk_src, byte_clk_provider);
                if (ret) {
                        pr_err("%s: can't set parent to esc_clk_src. ret=%d\n",
        struct msm_dsi_phy_clk_request *clk_req)
 {
        struct msm_dsi_host *msm_host = to_msm_dsi_host(host);
+       const struct msm_dsi_cfg_handler *cfg_hnd = msm_host->cfg_hnd;
        int ret;
 
-       ret = dsi_calc_clk_rate(msm_host);
+       ret = cfg_hnd->ops->calc_clk_rate(msm_host);
        if (ret) {
                pr_err("%s: unable to calc clk rate, %d\n", __func__, ret);
                return;
                        struct msm_dsi_phy_shared_timings *phy_shared_timings)
 {
        struct msm_dsi_host *msm_host = to_msm_dsi_host(host);
+       const struct msm_dsi_cfg_handler *cfg_hnd = msm_host->cfg_hnd;
        int ret = 0;
 
        mutex_lock(&msm_host->dev_mutex);
        }
 
        pm_runtime_get_sync(&msm_host->pdev->dev);
-       ret = dsi_link_clk_enable(msm_host);
+       ret = cfg_hnd->ops->link_clk_enable(msm_host);
        if (ret) {
                pr_err("%s: failed to enable link clocks. ret=%d\n",
                       __func__, ret);
        return 0;
 
 fail_disable_clk:
-       dsi_link_clk_disable(msm_host);
+       cfg_hnd->ops->link_clk_disable(msm_host);
        pm_runtime_put_autosuspend(&msm_host->pdev->dev);
 fail_disable_reg:
        dsi_host_regulator_disable(msm_host);
 int msm_dsi_host_power_off(struct mipi_dsi_host *host)
 {
        struct msm_dsi_host *msm_host = to_msm_dsi_host(host);
+       const struct msm_dsi_cfg_handler *cfg_hnd = msm_host->cfg_hnd;
 
        mutex_lock(&msm_host->dev_mutex);
        if (!msm_host->power_on) {
 
        pinctrl_pm_select_sleep_state(&msm_host->pdev->dev);
 
-       dsi_link_clk_disable(msm_host);
+       cfg_hnd->ops->link_clk_disable(msm_host);
        pm_runtime_put_autosuspend(&msm_host->pdev->dev);
 
        dsi_host_regulator_disable(msm_host);