Commit e3c3b6e6 authored by Dave Airlie's avatar Dave Airlie

Merge tag 'exynos-drm-fixes-for-v5.6-rc5-v2' of...

Merge tag 'exynos-drm-fixes-for-v5.6-rc5-v2' of git://git.kernel.org/pub/scm/linux/kernel/git/daeinki/drm-exynos into drm-fixes

Fix IOMMU initialization failure when Exynos DRM driver is rebound,
and also fix memory leak to iommu mapping object, which was
detected by kmemleak detector.
Signed-off-by: default avatarDave Airlie <airlied@redhat.com>

From: Inki Dae <inki.dae@samsung.com>
Link: https://patchwork.freedesktop.org/patch/msgid/1583887109-4148-1-git-send-email-inki.dae@samsung.com
parents 2c523b34 07dc3678
...@@ -55,6 +55,7 @@ static const char * const decon_clks_name[] = { ...@@ -55,6 +55,7 @@ static const char * const decon_clks_name[] = {
struct decon_context { struct decon_context {
struct device *dev; struct device *dev;
struct drm_device *drm_dev; struct drm_device *drm_dev;
void *dma_priv;
struct exynos_drm_crtc *crtc; struct exynos_drm_crtc *crtc;
struct exynos_drm_plane planes[WINDOWS_NR]; struct exynos_drm_plane planes[WINDOWS_NR];
struct exynos_drm_plane_config configs[WINDOWS_NR]; struct exynos_drm_plane_config configs[WINDOWS_NR];
...@@ -644,7 +645,7 @@ static int decon_bind(struct device *dev, struct device *master, void *data) ...@@ -644,7 +645,7 @@ static int decon_bind(struct device *dev, struct device *master, void *data)
decon_clear_channels(ctx->crtc); decon_clear_channels(ctx->crtc);
return exynos_drm_register_dma(drm_dev, dev); return exynos_drm_register_dma(drm_dev, dev, &ctx->dma_priv);
} }
static void decon_unbind(struct device *dev, struct device *master, void *data) static void decon_unbind(struct device *dev, struct device *master, void *data)
...@@ -654,7 +655,7 @@ static void decon_unbind(struct device *dev, struct device *master, void *data) ...@@ -654,7 +655,7 @@ static void decon_unbind(struct device *dev, struct device *master, void *data)
decon_atomic_disable(ctx->crtc); decon_atomic_disable(ctx->crtc);
/* detach this sub driver from iommu mapping if supported. */ /* detach this sub driver from iommu mapping if supported. */
exynos_drm_unregister_dma(ctx->drm_dev, ctx->dev); exynos_drm_unregister_dma(ctx->drm_dev, ctx->dev, &ctx->dma_priv);
} }
static const struct component_ops decon_component_ops = { static const struct component_ops decon_component_ops = {
......
...@@ -40,6 +40,7 @@ ...@@ -40,6 +40,7 @@
struct decon_context { struct decon_context {
struct device *dev; struct device *dev;
struct drm_device *drm_dev; struct drm_device *drm_dev;
void *dma_priv;
struct exynos_drm_crtc *crtc; struct exynos_drm_crtc *crtc;
struct exynos_drm_plane planes[WINDOWS_NR]; struct exynos_drm_plane planes[WINDOWS_NR];
struct exynos_drm_plane_config configs[WINDOWS_NR]; struct exynos_drm_plane_config configs[WINDOWS_NR];
...@@ -127,13 +128,13 @@ static int decon_ctx_initialize(struct decon_context *ctx, ...@@ -127,13 +128,13 @@ static int decon_ctx_initialize(struct decon_context *ctx,
decon_clear_channels(ctx->crtc); decon_clear_channels(ctx->crtc);
return exynos_drm_register_dma(drm_dev, ctx->dev); return exynos_drm_register_dma(drm_dev, ctx->dev, &ctx->dma_priv);
} }
static void decon_ctx_remove(struct decon_context *ctx) static void decon_ctx_remove(struct decon_context *ctx)
{ {
/* detach this sub driver from iommu mapping if supported. */ /* detach this sub driver from iommu mapping if supported. */
exynos_drm_unregister_dma(ctx->drm_dev, ctx->dev); exynos_drm_unregister_dma(ctx->drm_dev, ctx->dev, &ctx->dma_priv);
} }
static u32 decon_calc_clkdiv(struct decon_context *ctx, static u32 decon_calc_clkdiv(struct decon_context *ctx,
......
...@@ -58,7 +58,7 @@ static inline void clear_dma_max_seg_size(struct device *dev) ...@@ -58,7 +58,7 @@ static inline void clear_dma_max_seg_size(struct device *dev)
* mapping. * mapping.
*/ */
static int drm_iommu_attach_device(struct drm_device *drm_dev, static int drm_iommu_attach_device(struct drm_device *drm_dev,
struct device *subdrv_dev) struct device *subdrv_dev, void **dma_priv)
{ {
struct exynos_drm_private *priv = drm_dev->dev_private; struct exynos_drm_private *priv = drm_dev->dev_private;
int ret; int ret;
...@@ -74,7 +74,14 @@ static int drm_iommu_attach_device(struct drm_device *drm_dev, ...@@ -74,7 +74,14 @@ static int drm_iommu_attach_device(struct drm_device *drm_dev,
return ret; return ret;
if (IS_ENABLED(CONFIG_ARM_DMA_USE_IOMMU)) { if (IS_ENABLED(CONFIG_ARM_DMA_USE_IOMMU)) {
if (to_dma_iommu_mapping(subdrv_dev)) /*
* Keep the original DMA mapping of the sub-device and
* restore it on Exynos DRM detach, otherwise the DMA
* framework considers it as IOMMU-less during the next
* probe (in case of deferred probe or modular build)
*/
*dma_priv = to_dma_iommu_mapping(subdrv_dev);
if (*dma_priv)
arm_iommu_detach_device(subdrv_dev); arm_iommu_detach_device(subdrv_dev);
ret = arm_iommu_attach_device(subdrv_dev, priv->mapping); ret = arm_iommu_attach_device(subdrv_dev, priv->mapping);
...@@ -98,19 +105,21 @@ static int drm_iommu_attach_device(struct drm_device *drm_dev, ...@@ -98,19 +105,21 @@ static int drm_iommu_attach_device(struct drm_device *drm_dev,
* mapping * mapping
*/ */
static void drm_iommu_detach_device(struct drm_device *drm_dev, static void drm_iommu_detach_device(struct drm_device *drm_dev,
struct device *subdrv_dev) struct device *subdrv_dev, void **dma_priv)
{ {
struct exynos_drm_private *priv = drm_dev->dev_private; struct exynos_drm_private *priv = drm_dev->dev_private;
if (IS_ENABLED(CONFIG_ARM_DMA_USE_IOMMU)) if (IS_ENABLED(CONFIG_ARM_DMA_USE_IOMMU)) {
arm_iommu_detach_device(subdrv_dev); arm_iommu_detach_device(subdrv_dev);
else if (IS_ENABLED(CONFIG_IOMMU_DMA)) arm_iommu_attach_device(subdrv_dev, *dma_priv);
} else if (IS_ENABLED(CONFIG_IOMMU_DMA))
iommu_detach_device(priv->mapping, subdrv_dev); iommu_detach_device(priv->mapping, subdrv_dev);
clear_dma_max_seg_size(subdrv_dev); clear_dma_max_seg_size(subdrv_dev);
} }
int exynos_drm_register_dma(struct drm_device *drm, struct device *dev) int exynos_drm_register_dma(struct drm_device *drm, struct device *dev,
void **dma_priv)
{ {
struct exynos_drm_private *priv = drm->dev_private; struct exynos_drm_private *priv = drm->dev_private;
...@@ -137,13 +146,14 @@ int exynos_drm_register_dma(struct drm_device *drm, struct device *dev) ...@@ -137,13 +146,14 @@ int exynos_drm_register_dma(struct drm_device *drm, struct device *dev)
priv->mapping = mapping; priv->mapping = mapping;
} }
return drm_iommu_attach_device(drm, dev); return drm_iommu_attach_device(drm, dev, dma_priv);
} }
void exynos_drm_unregister_dma(struct drm_device *drm, struct device *dev) void exynos_drm_unregister_dma(struct drm_device *drm, struct device *dev,
void **dma_priv)
{ {
if (IS_ENABLED(CONFIG_EXYNOS_IOMMU)) if (IS_ENABLED(CONFIG_EXYNOS_IOMMU))
drm_iommu_detach_device(drm, dev); drm_iommu_detach_device(drm, dev, dma_priv);
} }
void exynos_drm_cleanup_dma(struct drm_device *drm) void exynos_drm_cleanup_dma(struct drm_device *drm)
......
...@@ -223,8 +223,10 @@ static inline bool is_drm_iommu_supported(struct drm_device *drm_dev) ...@@ -223,8 +223,10 @@ static inline bool is_drm_iommu_supported(struct drm_device *drm_dev)
return priv->mapping ? true : false; return priv->mapping ? true : false;
} }
int exynos_drm_register_dma(struct drm_device *drm, struct device *dev); int exynos_drm_register_dma(struct drm_device *drm, struct device *dev,
void exynos_drm_unregister_dma(struct drm_device *drm, struct device *dev); void **dma_priv);
void exynos_drm_unregister_dma(struct drm_device *drm, struct device *dev,
void **dma_priv);
void exynos_drm_cleanup_dma(struct drm_device *drm); void exynos_drm_cleanup_dma(struct drm_device *drm);
#ifdef CONFIG_DRM_EXYNOS_DPI #ifdef CONFIG_DRM_EXYNOS_DPI
......
...@@ -97,6 +97,7 @@ struct fimc_scaler { ...@@ -97,6 +97,7 @@ struct fimc_scaler {
struct fimc_context { struct fimc_context {
struct exynos_drm_ipp ipp; struct exynos_drm_ipp ipp;
struct drm_device *drm_dev; struct drm_device *drm_dev;
void *dma_priv;
struct device *dev; struct device *dev;
struct exynos_drm_ipp_task *task; struct exynos_drm_ipp_task *task;
struct exynos_drm_ipp_formats *formats; struct exynos_drm_ipp_formats *formats;
...@@ -1133,7 +1134,7 @@ static int fimc_bind(struct device *dev, struct device *master, void *data) ...@@ -1133,7 +1134,7 @@ static int fimc_bind(struct device *dev, struct device *master, void *data)
ctx->drm_dev = drm_dev; ctx->drm_dev = drm_dev;
ipp->drm_dev = drm_dev; ipp->drm_dev = drm_dev;
exynos_drm_register_dma(drm_dev, dev); exynos_drm_register_dma(drm_dev, dev, &ctx->dma_priv);
exynos_drm_ipp_register(dev, ipp, &ipp_funcs, exynos_drm_ipp_register(dev, ipp, &ipp_funcs,
DRM_EXYNOS_IPP_CAP_CROP | DRM_EXYNOS_IPP_CAP_ROTATE | DRM_EXYNOS_IPP_CAP_CROP | DRM_EXYNOS_IPP_CAP_ROTATE |
...@@ -1153,7 +1154,7 @@ static void fimc_unbind(struct device *dev, struct device *master, ...@@ -1153,7 +1154,7 @@ static void fimc_unbind(struct device *dev, struct device *master,
struct exynos_drm_ipp *ipp = &ctx->ipp; struct exynos_drm_ipp *ipp = &ctx->ipp;
exynos_drm_ipp_unregister(dev, ipp); exynos_drm_ipp_unregister(dev, ipp);
exynos_drm_unregister_dma(drm_dev, dev); exynos_drm_unregister_dma(drm_dev, dev, &ctx->dma_priv);
} }
static const struct component_ops fimc_component_ops = { static const struct component_ops fimc_component_ops = {
......
...@@ -167,6 +167,7 @@ static struct fimd_driver_data exynos5420_fimd_driver_data = { ...@@ -167,6 +167,7 @@ static struct fimd_driver_data exynos5420_fimd_driver_data = {
struct fimd_context { struct fimd_context {
struct device *dev; struct device *dev;
struct drm_device *drm_dev; struct drm_device *drm_dev;
void *dma_priv;
struct exynos_drm_crtc *crtc; struct exynos_drm_crtc *crtc;
struct exynos_drm_plane planes[WINDOWS_NR]; struct exynos_drm_plane planes[WINDOWS_NR];
struct exynos_drm_plane_config configs[WINDOWS_NR]; struct exynos_drm_plane_config configs[WINDOWS_NR];
...@@ -1090,7 +1091,7 @@ static int fimd_bind(struct device *dev, struct device *master, void *data) ...@@ -1090,7 +1091,7 @@ static int fimd_bind(struct device *dev, struct device *master, void *data)
if (is_drm_iommu_supported(drm_dev)) if (is_drm_iommu_supported(drm_dev))
fimd_clear_channels(ctx->crtc); fimd_clear_channels(ctx->crtc);
return exynos_drm_register_dma(drm_dev, dev); return exynos_drm_register_dma(drm_dev, dev, &ctx->dma_priv);
} }
static void fimd_unbind(struct device *dev, struct device *master, static void fimd_unbind(struct device *dev, struct device *master,
...@@ -1100,7 +1101,7 @@ static void fimd_unbind(struct device *dev, struct device *master, ...@@ -1100,7 +1101,7 @@ static void fimd_unbind(struct device *dev, struct device *master,
fimd_atomic_disable(ctx->crtc); fimd_atomic_disable(ctx->crtc);
exynos_drm_unregister_dma(ctx->drm_dev, ctx->dev); exynos_drm_unregister_dma(ctx->drm_dev, ctx->dev, &ctx->dma_priv);
if (ctx->encoder) if (ctx->encoder)
exynos_dpi_remove(ctx->encoder); exynos_dpi_remove(ctx->encoder);
......
...@@ -232,6 +232,7 @@ struct g2d_runqueue_node { ...@@ -232,6 +232,7 @@ struct g2d_runqueue_node {
struct g2d_data { struct g2d_data {
struct device *dev; struct device *dev;
void *dma_priv;
struct clk *gate_clk; struct clk *gate_clk;
void __iomem *regs; void __iomem *regs;
int irq; int irq;
...@@ -1409,7 +1410,7 @@ static int g2d_bind(struct device *dev, struct device *master, void *data) ...@@ -1409,7 +1410,7 @@ static int g2d_bind(struct device *dev, struct device *master, void *data)
return ret; return ret;
} }
ret = exynos_drm_register_dma(drm_dev, dev); ret = exynos_drm_register_dma(drm_dev, dev, &g2d->dma_priv);
if (ret < 0) { if (ret < 0) {
dev_err(dev, "failed to enable iommu.\n"); dev_err(dev, "failed to enable iommu.\n");
g2d_fini_cmdlist(g2d); g2d_fini_cmdlist(g2d);
...@@ -1434,7 +1435,7 @@ static void g2d_unbind(struct device *dev, struct device *master, void *data) ...@@ -1434,7 +1435,7 @@ static void g2d_unbind(struct device *dev, struct device *master, void *data)
priv->g2d_dev = NULL; priv->g2d_dev = NULL;
cancel_work_sync(&g2d->runqueue_work); cancel_work_sync(&g2d->runqueue_work);
exynos_drm_unregister_dma(g2d->drm_dev, dev); exynos_drm_unregister_dma(g2d->drm_dev, dev, &g2d->dma_priv);
} }
static const struct component_ops g2d_component_ops = { static const struct component_ops g2d_component_ops = {
......
...@@ -97,6 +97,7 @@ struct gsc_scaler { ...@@ -97,6 +97,7 @@ struct gsc_scaler {
struct gsc_context { struct gsc_context {
struct exynos_drm_ipp ipp; struct exynos_drm_ipp ipp;
struct drm_device *drm_dev; struct drm_device *drm_dev;
void *dma_priv;
struct device *dev; struct device *dev;
struct exynos_drm_ipp_task *task; struct exynos_drm_ipp_task *task;
struct exynos_drm_ipp_formats *formats; struct exynos_drm_ipp_formats *formats;
...@@ -1169,7 +1170,7 @@ static int gsc_bind(struct device *dev, struct device *master, void *data) ...@@ -1169,7 +1170,7 @@ static int gsc_bind(struct device *dev, struct device *master, void *data)
ctx->drm_dev = drm_dev; ctx->drm_dev = drm_dev;
ctx->drm_dev = drm_dev; ctx->drm_dev = drm_dev;
exynos_drm_register_dma(drm_dev, dev); exynos_drm_register_dma(drm_dev, dev, &ctx->dma_priv);
exynos_drm_ipp_register(dev, ipp, &ipp_funcs, exynos_drm_ipp_register(dev, ipp, &ipp_funcs,
DRM_EXYNOS_IPP_CAP_CROP | DRM_EXYNOS_IPP_CAP_ROTATE | DRM_EXYNOS_IPP_CAP_CROP | DRM_EXYNOS_IPP_CAP_ROTATE |
...@@ -1189,7 +1190,7 @@ static void gsc_unbind(struct device *dev, struct device *master, ...@@ -1189,7 +1190,7 @@ static void gsc_unbind(struct device *dev, struct device *master,
struct exynos_drm_ipp *ipp = &ctx->ipp; struct exynos_drm_ipp *ipp = &ctx->ipp;
exynos_drm_ipp_unregister(dev, ipp); exynos_drm_ipp_unregister(dev, ipp);
exynos_drm_unregister_dma(drm_dev, dev); exynos_drm_unregister_dma(drm_dev, dev, &ctx->dma_priv);
} }
static const struct component_ops gsc_component_ops = { static const struct component_ops gsc_component_ops = {
......
...@@ -56,6 +56,7 @@ struct rot_variant { ...@@ -56,6 +56,7 @@ struct rot_variant {
struct rot_context { struct rot_context {
struct exynos_drm_ipp ipp; struct exynos_drm_ipp ipp;
struct drm_device *drm_dev; struct drm_device *drm_dev;
void *dma_priv;
struct device *dev; struct device *dev;
void __iomem *regs; void __iomem *regs;
struct clk *clock; struct clk *clock;
...@@ -243,7 +244,7 @@ static int rotator_bind(struct device *dev, struct device *master, void *data) ...@@ -243,7 +244,7 @@ static int rotator_bind(struct device *dev, struct device *master, void *data)
rot->drm_dev = drm_dev; rot->drm_dev = drm_dev;
ipp->drm_dev = drm_dev; ipp->drm_dev = drm_dev;
exynos_drm_register_dma(drm_dev, dev); exynos_drm_register_dma(drm_dev, dev, &rot->dma_priv);
exynos_drm_ipp_register(dev, ipp, &ipp_funcs, exynos_drm_ipp_register(dev, ipp, &ipp_funcs,
DRM_EXYNOS_IPP_CAP_CROP | DRM_EXYNOS_IPP_CAP_ROTATE, DRM_EXYNOS_IPP_CAP_CROP | DRM_EXYNOS_IPP_CAP_ROTATE,
...@@ -261,7 +262,7 @@ static void rotator_unbind(struct device *dev, struct device *master, ...@@ -261,7 +262,7 @@ static void rotator_unbind(struct device *dev, struct device *master,
struct exynos_drm_ipp *ipp = &rot->ipp; struct exynos_drm_ipp *ipp = &rot->ipp;
exynos_drm_ipp_unregister(dev, ipp); exynos_drm_ipp_unregister(dev, ipp);
exynos_drm_unregister_dma(rot->drm_dev, rot->dev); exynos_drm_unregister_dma(rot->drm_dev, rot->dev, &rot->dma_priv);
} }
static const struct component_ops rotator_component_ops = { static const struct component_ops rotator_component_ops = {
......
...@@ -39,6 +39,7 @@ struct scaler_data { ...@@ -39,6 +39,7 @@ struct scaler_data {
struct scaler_context { struct scaler_context {
struct exynos_drm_ipp ipp; struct exynos_drm_ipp ipp;
struct drm_device *drm_dev; struct drm_device *drm_dev;
void *dma_priv;
struct device *dev; struct device *dev;
void __iomem *regs; void __iomem *regs;
struct clk *clock[SCALER_MAX_CLK]; struct clk *clock[SCALER_MAX_CLK];
...@@ -450,7 +451,7 @@ static int scaler_bind(struct device *dev, struct device *master, void *data) ...@@ -450,7 +451,7 @@ static int scaler_bind(struct device *dev, struct device *master, void *data)
scaler->drm_dev = drm_dev; scaler->drm_dev = drm_dev;
ipp->drm_dev = drm_dev; ipp->drm_dev = drm_dev;
exynos_drm_register_dma(drm_dev, dev); exynos_drm_register_dma(drm_dev, dev, &scaler->dma_priv);
exynos_drm_ipp_register(dev, ipp, &ipp_funcs, exynos_drm_ipp_register(dev, ipp, &ipp_funcs,
DRM_EXYNOS_IPP_CAP_CROP | DRM_EXYNOS_IPP_CAP_ROTATE | DRM_EXYNOS_IPP_CAP_CROP | DRM_EXYNOS_IPP_CAP_ROTATE |
...@@ -470,7 +471,8 @@ static void scaler_unbind(struct device *dev, struct device *master, ...@@ -470,7 +471,8 @@ static void scaler_unbind(struct device *dev, struct device *master,
struct exynos_drm_ipp *ipp = &scaler->ipp; struct exynos_drm_ipp *ipp = &scaler->ipp;
exynos_drm_ipp_unregister(dev, ipp); exynos_drm_ipp_unregister(dev, ipp);
exynos_drm_unregister_dma(scaler->drm_dev, scaler->dev); exynos_drm_unregister_dma(scaler->drm_dev, scaler->dev,
&scaler->dma_priv);
} }
static const struct component_ops scaler_component_ops = { static const struct component_ops scaler_component_ops = {
......
...@@ -94,6 +94,7 @@ struct mixer_context { ...@@ -94,6 +94,7 @@ struct mixer_context {
struct platform_device *pdev; struct platform_device *pdev;
struct device *dev; struct device *dev;
struct drm_device *drm_dev; struct drm_device *drm_dev;
void *dma_priv;
struct exynos_drm_crtc *crtc; struct exynos_drm_crtc *crtc;
struct exynos_drm_plane planes[MIXER_WIN_NR]; struct exynos_drm_plane planes[MIXER_WIN_NR];
unsigned long flags; unsigned long flags;
...@@ -894,12 +895,14 @@ static int mixer_initialize(struct mixer_context *mixer_ctx, ...@@ -894,12 +895,14 @@ static int mixer_initialize(struct mixer_context *mixer_ctx,
} }
} }
return exynos_drm_register_dma(drm_dev, mixer_ctx->dev); return exynos_drm_register_dma(drm_dev, mixer_ctx->dev,
&mixer_ctx->dma_priv);
} }
static void mixer_ctx_remove(struct mixer_context *mixer_ctx) static void mixer_ctx_remove(struct mixer_context *mixer_ctx)
{ {
exynos_drm_unregister_dma(mixer_ctx->drm_dev, mixer_ctx->dev); exynos_drm_unregister_dma(mixer_ctx->drm_dev, mixer_ctx->dev,
&mixer_ctx->dma_priv);
} }
static int mixer_enable_vblank(struct exynos_drm_crtc *crtc) static int mixer_enable_vblank(struct exynos_drm_crtc *crtc)
......
Markdown is supported
0%
or
You are about to add 0 people to the discussion. Proceed with caution.
Finish editing this message first!
Please register or to comment