[Freedreno] [DPU PATCH 04/11] drm/msm/dpu: create new platform driver for dpu device
Sean Paul
seanpaul at chromium.org
Thu May 10 15:00:48 UTC 2018
On Thu, May 10, 2018 at 01:59:38PM +0530, Rajesh Yadav wrote:
> Current MSM display controller HW matches a tree like
> hierarchy where MDSS top level wrapper is parent device
> and mdp5/dpu, dsi, dp are child devices.
>
> Each child device like mdp5, dsi etc. have a separate driver,
> but currently dpu handling is tied to a single driver which
> was managing both mdss and dpu resources.
>
> Inorder to have the cleaner one to one device and driver
> association, this change adds a new platform_driver for dpu
> child device node which implements the kms functionality.
>
> The dpu driver implements runtime_pm support for managing clocks
> and bus bandwidth etc.
>
> Signed-off-by: Rajesh Yadav <ryadav at codeaurora.org>
> ---
> drivers/gpu/drm/msm/disp/dpu1/dpu_kms.c | 251 ++++++++++++++++++++++++++------
> drivers/gpu/drm/msm/disp/dpu1/dpu_kms.h | 4 +
> drivers/gpu/drm/msm/msm_drv.c | 2 +
> drivers/gpu/drm/msm/msm_drv.h | 3 +
> 4 files changed, 214 insertions(+), 46 deletions(-)
>
> diff --git a/drivers/gpu/drm/msm/disp/dpu1/dpu_kms.c b/drivers/gpu/drm/msm/disp/dpu1/dpu_kms.c
> index e4ab753..2cd51fc 100644
> --- a/drivers/gpu/drm/msm/disp/dpu1/dpu_kms.c
> +++ b/drivers/gpu/drm/msm/disp/dpu1/dpu_kms.c
> @@ -1030,14 +1030,13 @@ static long dpu_kms_round_pixclk(struct msm_kms *kms, unsigned long rate,
> return rate;
> }
>
> -static void _dpu_kms_hw_destroy(struct dpu_kms *dpu_kms,
> - struct platform_device *pdev)
> +static void _dpu_kms_hw_destroy(struct dpu_kms *dpu_kms)
> {
> struct drm_device *dev;
> struct msm_drm_private *priv;
> int i;
>
> - if (!dpu_kms || !pdev)
> + if (!dpu_kms)
This isn't possible, please remove.
> return;
>
> dev = dpu_kms->dev;
> @@ -1091,15 +1090,15 @@ static void _dpu_kms_hw_destroy(struct dpu_kms *dpu_kms,
> dpu_kms->core_client = NULL;
>
> if (dpu_kms->vbif[VBIF_NRT])
> - msm_iounmap(pdev, dpu_kms->vbif[VBIF_NRT]);
> + msm_iounmap(dpu_kms->pdev, dpu_kms->vbif[VBIF_NRT]);
> dpu_kms->vbif[VBIF_NRT] = NULL;
>
> if (dpu_kms->vbif[VBIF_RT])
> - msm_iounmap(pdev, dpu_kms->vbif[VBIF_RT]);
> + msm_iounmap(dpu_kms->pdev, dpu_kms->vbif[VBIF_RT]);
> dpu_kms->vbif[VBIF_RT] = NULL;
>
> if (dpu_kms->mmio)
> - msm_iounmap(pdev, dpu_kms->mmio);
> + msm_iounmap(dpu_kms->pdev, dpu_kms->mmio);
> dpu_kms->mmio = NULL;
>
> dpu_reg_dma_deinit();
<snip />
> +static void dpu_destroy(struct platform_device *pdev)
> +{
> + struct dpu_kms *dpu_kms = platform_get_drvdata(pdev);
> + struct dss_module_power *mp = &dpu_kms->mp;
> +
> + msm_dss_put_clk(mp->clk_config, mp->num_clk);
> + devm_kfree(&pdev->dev, mp->clk_config);
> + mp->num_clk = 0;
> +
> + if (dpu_kms->rpm_enabled)
> + pm_runtime_disable(&pdev->dev);
> +
> + devm_kfree(&pdev->dev, dpu_kms);
devm_kfrees are not necessary
> +}
>
> - dpu_kms = kzalloc(sizeof(*dpu_kms), GFP_KERNEL);
> +static int dpu_init(struct platform_device *pdev, struct drm_device *dev)
> +{
> + struct msm_drm_private *priv = dev->dev_private;
> + struct dpu_kms *dpu_kms;
> + struct dss_module_power *mp;
> + int ret = 0;
> +
> + dpu_kms = devm_kzalloc(&pdev->dev, sizeof(*dpu_kms), GFP_KERNEL);
> if (!dpu_kms) {
> DPU_ERROR("failed to allocate dpu kms\n");
> - return ERR_PTR(-ENOMEM);
> + return -ENOMEM;
> + }
> +
> + mp = &dpu_kms->mp;
> + ret = msm_dss_parse_clock(pdev, mp);
> + if (ret) {
> + DPU_ERROR("failed to parse clocks, ret=%d\n", ret);
> + goto clk_parse_error;
> + }
> +
> + ret = msm_dss_get_clk(&pdev->dev, mp->clk_config, mp->num_clk);
> + if (ret) {
> + pr_err("failed to get clocks, ret=%d\n", ret);
> + goto clk_get_error;
> + }
> +
> + ret = msm_dss_clk_set_rate(mp->clk_config, mp->num_clk);
> + if (ret) {
> + pr_err("failed to set clock rate, ret=%d\n", ret);
> + goto clk_rate_error;
> }
>
> + platform_set_drvdata(pdev, dpu_kms);
> +
> msm_kms_init(&dpu_kms->base, &kms_funcs);
> dpu_kms->dev = dev;
> - dpu_kms->base.irq = irq;
> + dpu_kms->pdev = pdev;
>
> - return &dpu_kms->base;
> + pm_runtime_enable(&pdev->dev);
> + dpu_kms->rpm_enabled = true;
> +
> + priv->kms = &dpu_kms->base;
> +
> + return ret;
> +
> +clk_rate_error:
> + msm_dss_put_clk(mp->clk_config, mp->num_clk);
> +clk_get_error:
> + devm_kfree(&pdev->dev, mp->clk_config);
> + mp->num_clk = 0;
> +clk_parse_error:
> + devm_kfree(&pdev->dev, dpu_kms);
> +
> + return ret;
> +}
> +
> +static int dpu_bind(struct device *dev, struct device *master, void *data)
> +{
> + struct drm_device *ddev = dev_get_drvdata(master);
> + struct platform_device *pdev = to_platform_device(dev);
> +
> + return dpu_init(pdev, ddev);
No need to have dpu_init separate, just move its contents inline.
> }
>
> +static void dpu_unbind(struct device *dev, struct device *master, void *data)
> +{
> + struct platform_device *pdev = to_platform_device(dev);
> +
> + dpu_destroy(pdev);
Same here.
> +}
> +
> +static const struct component_ops dpu_ops = {
> + .bind = dpu_bind,
> + .unbind = dpu_unbind,
> +};
> +
> +static int dpu_dev_probe(struct platform_device *pdev)
> +{
> + return component_add(&pdev->dev, &dpu_ops);
> +}
> +
> +static int dpu_dev_remove(struct platform_device *pdev)
> +{
> + component_del(&pdev->dev, &dpu_ops);
> + return 0;
> +}
> +
> +static int dpu_runtime_suspend(struct device *dev)
> +{
> + int rc = -1;
> + struct platform_device *pdev = to_platform_device(dev);
> + struct dpu_kms *dpu_kms = platform_get_drvdata(pdev);
> + struct drm_device *ddev;
> + struct msm_drm_private *priv;
> + struct dss_module_power *mp = &dpu_kms->mp;
> +
> + ddev = dpu_kms->dev;
> + if (!ddev) {
> + DPU_ERROR("invalid drm_device\n");
> + goto exit;
> + }
> + priv = ddev->dev_private;
> +
> + rc = dpu_power_resource_enable(&priv->phandle,
> + dpu_kms->core_client, false);
> + if (rc)
> + DPU_ERROR("resource disable failed: %d\n", rc);
> +
> + rc = msm_dss_enable_clk(mp->clk_config, mp->num_clk, false);
> + if (rc)
> + DPU_ERROR("clock disable failed rc:%d\n", rc);
> +
> +exit:
> + return rc;
> +}
> +
> +static int dpu_runtime_resume(struct device *dev)
> +{
> + int rc = -1;
> + struct platform_device *pdev = to_platform_device(dev);
> + struct dpu_kms *dpu_kms = platform_get_drvdata(pdev);
> + struct drm_device *ddev;
> + struct msm_drm_private *priv;
> + struct dss_module_power *mp = &dpu_kms->mp;
> +
> + ddev = dpu_kms->dev;
> + if (!ddev) {
> + DPU_ERROR("invalid drm_device\n");
> + goto exit;
> + }
> + priv = ddev->dev_private;
> +
> + rc = msm_dss_enable_clk(mp->clk_config, mp->num_clk, true);
> + if (rc) {
> + DPU_ERROR("clock enable failed rc:%d\n", rc);
> + goto exit;
> + }
> +
> + rc = dpu_power_resource_enable(&priv->phandle, dpu_kms->core_client,
> + true);
> + if (rc)
> + DPU_ERROR("resource enable failed: %d\n", rc);
> +
> +exit:
> + return rc;
> +}
> +
> +static const struct dev_pm_ops dpu_pm_ops = {
> + SET_RUNTIME_PM_OPS(dpu_runtime_suspend, dpu_runtime_resume, NULL)
> +};
> +
> +static const struct of_device_id dpu_dt_match[] = {
> + { .compatible = "qcom,dpu", },
Update the dt bindings?
> + {}
> +};
> +MODULE_DEVICE_TABLE(of, dpu_dt_match);
> +
> +static struct platform_driver dpu_driver = {
> + .probe = dpu_dev_probe,
> + .remove = dpu_dev_remove,
> + .driver = {
> + .name = "msm_dpu",
> + .of_match_table = dpu_dt_match,
> + .pm = &dpu_pm_ops,
> + },
> +};
> +
> +void __init msm_dpu_register(void)
> +{
> + platform_driver_register(&dpu_driver);
> +}
> +
> +void __exit msm_dpu_unregister(void)
> +{
> + platform_driver_unregister(&dpu_driver);
> +}
> diff --git a/drivers/gpu/drm/msm/disp/dpu1/dpu_kms.h b/drivers/gpu/drm/msm/disp/dpu1/dpu_kms.h
> index a1c0910..3c69921 100644
> --- a/drivers/gpu/drm/msm/disp/dpu1/dpu_kms.h
> +++ b/drivers/gpu/drm/msm/disp/dpu1/dpu_kms.h
> @@ -200,6 +200,10 @@ struct dpu_kms {
> struct dpu_hw_mdp *hw_mdp;
>
> bool has_danger_ctrl;
> +
> + struct platform_device *pdev;
> + bool rpm_enabled;
> + struct dss_module_power mp;
> };
>
> struct vsync_info {
> diff --git a/drivers/gpu/drm/msm/msm_drv.c b/drivers/gpu/drm/msm/msm_drv.c
> index a0e73ea..5470529 100644
> --- a/drivers/gpu/drm/msm/msm_drv.c
> +++ b/drivers/gpu/drm/msm/msm_drv.c
> @@ -1731,6 +1731,7 @@ static int __init msm_drm_register(void)
>
> DBG("init");
> msm_mdp_register();
> + msm_dpu_register();
> msm_dsi_register();
> msm_edp_register();
> msm_hdmi_register();
> @@ -1747,6 +1748,7 @@ static void __exit msm_drm_unregister(void)
> msm_edp_unregister();
> msm_dsi_unregister();
> msm_mdp_unregister();
> + msm_dpu_unregister();
> }
>
> module_init(msm_drm_register);
> diff --git a/drivers/gpu/drm/msm/msm_drv.h b/drivers/gpu/drm/msm/msm_drv.h
> index e8e5e73..22a3096 100644
> --- a/drivers/gpu/drm/msm/msm_drv.h
> +++ b/drivers/gpu/drm/msm/msm_drv.h
> @@ -682,6 +682,9 @@ static inline int msm_dsi_modeset_init(struct msm_dsi *msm_dsi,
> void __init msm_mdp_register(void);
> void __exit msm_mdp_unregister(void);
>
> +void __init msm_dpu_register(void);
> +void __exit msm_dpu_unregister(void);
> +
> #ifdef CONFIG_DEBUG_FS
> void msm_gem_describe(struct drm_gem_object *obj, struct seq_file *m);
> void msm_gem_describe_objects(struct list_head *list, struct seq_file *m);
> --
> The Qualcomm Innovation Center, Inc. is a member of the Code Aurora Forum,
> a Linux Foundation Collaborative Project
>
--
Sean Paul, Software Engineer, Google / Chromium OS
More information about the Freedreno
mailing list