On Mon, Mar 02, 2020 at 07:07:31PM +0100, Philipp Zabel wrote: > Hi, > > On Thu, 2020-02-27 at 19:14 +0100, Daniel Vetter wrote: > > On Thu, Feb 27, 2020 at 6:44 PM Lucas Stach <l.stach@xxxxxxxxxxxxxx> wrote: > > > Hi Daniel, > > > > > > On Do, 2020-02-27 at 18:29 +0100, Daniel Vetter wrote: > > > > On Thu, Feb 27, 2020 at 05:21:25PM +0100, Marco Felsch wrote: > > > > > Currently there is a race conditions if the panel can't be probed e.g. > > > > > it is not connected [1]. There where several attempts to fix this [2,3] > > > > > but non of them made it into mainline. > > > > > > > > > > The problem is the combination of the component framework and the drm > > > > > framework, as Philipp already explained [1]. To fix this we need to > > > > > drop the devres-kmalloc and move the plain kmalloc to let the drm > > > > > framework free the resources upon a drm_mode_config_cleanup(). So we need > > > > > to implement a .destroy() callback for each component. We also need to > > > > > reorder the master.unbind() callback to ensure that the driver states > > > > > are accessible during a component.unbind() call. This reordering also > > > > > aligns the master.unbind() with the error-cleanup path during > > > > > master.bind(). > > > > > > > > > > [1] https://www.spinics.net/lists/dri-devel/msg189388.html > > > > > [2] https://lkml.org/lkml/2018/10/16/1148 > > > > > [3] https://lkml.org/lkml/2019/4/2/612 > > > > > > > > > > Signed-off-by: Marco Felsch <m.felsch@xxxxxxxxxxxxxx> > > > > > > > > I think this collides quite badly with my managed drm device resources > > > > patch series I'm working on. Plus once we have that, you could use > > > > drmm_kzalloc and wouldn't need to sprinkle kfree() over everything. > > How does it collide, none of the patches touch imx-drm? > > > > > I think at least, I haven't rolled much further than just getting the > > > > baseline stuff figured out. So if it's not super-pressing to get this > > > > patch here landed I think it'd be better to base this on top of the drmm > > > > series. I'm working on sending out v3, I'll cc you on the imx parts so > > > > you'll get pinged. > > > > > > IMO this part of imx-drm has been broken for far too long already, so > > > we shouldn't delay this fixes series on a complete resource management > > > rework. > > Right, the use-after-free fixes should not have to be rebased onto > WIP drmm code. But could we move the fixes to the front, with just > minimal necessary changes? > That way they could be backported to stable without the cleanup and code > shuffling patches in-between. > We could then migrate the rework to drm managed resources without hurry. > > > Given it's patch 17/17 in a spring cleanup, and not patch 1/17 I'm not > > entirely sure it's really that high priority. > > This series fixes crashes on probe in case of defective device trees or > missing component drivers. I wouldn't get too hung up on the "spring > cleanup" name, but the actual fix being the last of a series of 17 > patches is a valid point. > > > Anyway would be great if you at least check out what the new drm > > managed resource stuff would mean for imx here, since you're blowing > > on devm_kzalloc exactly in the way that I'm trying to get sorted now > > (without tons of explicit kfree() everywhere). > > I concur. Marco, would the following workaround be enough to fix the > issue until we can switch to drm managed allocations? So what would be really useful for managed allocations with drmm if you folks could test-drive this with a component driver. I've started looking at some of them, but the load sequence is fairly tricky so I'm not sure whether we'll correctly unwind in all cases. But I do think since you're just putting a kfree() into the various drm_mode_object->destroy hooks it should work. At least as long as you keep the explicit drm_mode_config_cleanup call still (I'm working on that problem). Knowing that (with maybe some warts, but at least as a poc) drmm_kzalloc works correctly for component.c based drivers would be really useful. That's kinda why I jumped in here. -Daniel > > ----------8<---------- > From b1c98a9d7b29fc052491d7fe0f7a1af91e48d866 Mon Sep 17 00:00:00 2001 > From: Philipp Zabel <p.zabel@xxxxxxxxxxxxxx> > Date: Mon, 2 Mar 2020 17:12:44 +0100 > Subject: [PATCH] drm/imx: fix use after free > > Component driver structures allocated with devm_kmalloc() in bind() are > freed automatically after unbind(). Since the contained drm structures > are accessed afterwards in drm_mode_config_cleanup(), move the > allocation into probe() to extend the driver structure's lifetime to the > lifetime of the device. This should eventually be changed to use drm > resource managed allocations with lifetime of the drm device. > > Signed-off-by: Philipp Zabel <p.zabel@xxxxxxxxxxxxxx> > --- > drivers/gpu/drm/imx/dw_hdmi-imx.c | 15 ++++++++++----- > drivers/gpu/drm/imx/imx-ldb.c | 15 ++++++++++----- > drivers/gpu/drm/imx/imx-tve.c | 15 ++++++++++----- > drivers/gpu/drm/imx/ipuv3-crtc.c | 21 ++++++++++----------- > drivers/gpu/drm/imx/parallel-display.c | 15 ++++++++++----- > 5 files changed, 50 insertions(+), 31 deletions(-) > > diff --git a/drivers/gpu/drm/imx/dw_hdmi-imx.c b/drivers/gpu/drm/imx/dw_hdmi-imx.c > index f22cfbf9353e..2e12a4a3bfa1 100644 > --- a/drivers/gpu/drm/imx/dw_hdmi-imx.c > +++ b/drivers/gpu/drm/imx/dw_hdmi-imx.c > @@ -212,9 +212,8 @@ static int dw_hdmi_imx_bind(struct device *dev, struct device *master, > if (!pdev->dev.of_node) > return -ENODEV; > > - hdmi = devm_kzalloc(&pdev->dev, sizeof(*hdmi), GFP_KERNEL); > - if (!hdmi) > - return -ENOMEM; > + hdmi = dev_get_drvdata(dev); > + memset(hdmi, 0, sizeof(*hdmi)); > > match = of_match_node(dw_hdmi_imx_dt_ids, pdev->dev.of_node); > plat_data = match->data; > @@ -239,8 +238,6 @@ static int dw_hdmi_imx_bind(struct device *dev, struct device *master, > drm_encoder_init(drm, encoder, &dw_hdmi_imx_encoder_funcs, > DRM_MODE_ENCODER_TMDS, NULL); > > - platform_set_drvdata(pdev, hdmi); > - > hdmi->hdmi = dw_hdmi_bind(pdev, encoder, plat_data); > > /* > @@ -270,6 +267,14 @@ static const struct component_ops dw_hdmi_imx_ops = { > > static int dw_hdmi_imx_probe(struct platform_device *pdev) > { > + struct imx_hdmi *hdmi; > + > + hdmi = devm_kzalloc(&pdev->dev, sizeof(*hdmi), GFP_KERNEL); > + if (!hdmi) > + return -ENOMEM; > + > + platform_set_drvdata(pdev, hdmi); > + > return component_add(&pdev->dev, &dw_hdmi_imx_ops); > } > > diff --git a/drivers/gpu/drm/imx/imx-ldb.c b/drivers/gpu/drm/imx/imx-ldb.c > index 8cb2665b2c74..c42217fc9f47 100644 > --- a/drivers/gpu/drm/imx/imx-ldb.c > +++ b/drivers/gpu/drm/imx/imx-ldb.c > @@ -594,9 +594,8 @@ static int imx_ldb_bind(struct device *dev, struct device *master, void *data) > int ret; > int i; > > - imx_ldb = devm_kzalloc(dev, sizeof(*imx_ldb), GFP_KERNEL); > - if (!imx_ldb) > - return -ENOMEM; > + imx_ldb = dev_get_drvdata(dev); > + memset(imx_ldb, 0, sizeof(*imx_ldb)); > > imx_ldb->regmap = syscon_regmap_lookup_by_phandle(np, "gpr"); > if (IS_ERR(imx_ldb->regmap)) { > @@ -704,8 +703,6 @@ static int imx_ldb_bind(struct device *dev, struct device *master, void *data) > } > } > > - dev_set_drvdata(dev, imx_ldb); > - > return 0; > > free_child: > @@ -737,6 +734,14 @@ static const struct component_ops imx_ldb_ops = { > > static int imx_ldb_probe(struct platform_device *pdev) > { > + struct imx_ldb *imx_ldb; > + > + imx_ldb = devm_kzalloc(&pdev->dev, sizeof(*imx_ldb), GFP_KERNEL); > + if (!imx_ldb) > + return -ENOMEM; > + > + platform_set_drvdata(pdev, imx_ldb); > + > return component_add(&pdev->dev, &imx_ldb_ops); > } > > diff --git a/drivers/gpu/drm/imx/imx-tve.c b/drivers/gpu/drm/imx/imx-tve.c > index 5bbfaa2cd0f4..6593e75fc16e 100644 > --- a/drivers/gpu/drm/imx/imx-tve.c > +++ b/drivers/gpu/drm/imx/imx-tve.c > @@ -546,9 +546,8 @@ static int imx_tve_bind(struct device *dev, struct device *master, void *data) > int irq; > int ret; > > - tve = devm_kzalloc(dev, sizeof(*tve), GFP_KERNEL); > - if (!tve) > - return -ENOMEM; > + tve = dev_get_drvdata(dev); > + memset(tve, 0, sizeof(*tve)); > > tve->dev = dev; > spin_lock_init(&tve->lock); > @@ -659,8 +658,6 @@ static int imx_tve_bind(struct device *dev, struct device *master, void *data) > if (ret) > return ret; > > - dev_set_drvdata(dev, tve); > - > return 0; > } > > @@ -680,6 +677,14 @@ static const struct component_ops imx_tve_ops = { > > static int imx_tve_probe(struct platform_device *pdev) > { > + struct imx_tve *tve; > + > + tve = devm_kzalloc(dev, sizeof(*tve), GFP_KERNEL); > + if (!tve) > + return -ENOMEM; > + > + platform_set_drvdata(pdev, tve); > + > return component_add(&pdev->dev, &imx_tve_ops); > } > > diff --git a/drivers/gpu/drm/imx/ipuv3-crtc.c b/drivers/gpu/drm/imx/ipuv3-crtc.c > index 63c0284f8b3c..2256c9789fc2 100644 > --- a/drivers/gpu/drm/imx/ipuv3-crtc.c > +++ b/drivers/gpu/drm/imx/ipuv3-crtc.c > @@ -438,21 +438,13 @@ static int ipu_drm_bind(struct device *dev, struct device *master, void *data) > struct ipu_client_platformdata *pdata = dev->platform_data; > struct drm_device *drm = data; > struct ipu_crtc *ipu_crtc; > - int ret; > > - ipu_crtc = devm_kzalloc(dev, sizeof(*ipu_crtc), GFP_KERNEL); > - if (!ipu_crtc) > - return -ENOMEM; > + ipu_crtc = dev_get_drvdata(dev); > + memset(ipu_crtc, 0, sizeof(*ipu_crtc)); > > ipu_crtc->dev = dev; > > - ret = ipu_crtc_init(ipu_crtc, pdata, drm); > - if (ret) > - return ret; > - > - dev_set_drvdata(dev, ipu_crtc); > - > - return 0; > + return ipu_crtc_init(ipu_crtc, pdata, drm); > } > > static void ipu_drm_unbind(struct device *dev, struct device *master, > @@ -474,6 +466,7 @@ static const struct component_ops ipu_crtc_ops = { > static int ipu_drm_probe(struct platform_device *pdev) > { > struct device *dev = &pdev->dev; > + struct ipu_crtc *ipu_crtc; > int ret; > > if (!dev->platform_data) > @@ -483,6 +476,12 @@ static int ipu_drm_probe(struct platform_device *pdev) > if (ret) > return ret; > > + ipu_crtc = devm_kzalloc(dev, sizeof(*ipu_crtc), GFP_KERNEL); > + if (!ipu_crtc) > + return -ENOMEM; > + > + dev_set_drvdata(dev, ipu_crtc); > + > return component_add(dev, &ipu_crtc_ops); > } > > diff --git a/drivers/gpu/drm/imx/parallel-display.c b/drivers/gpu/drm/imx/parallel-display.c > index 3dca424059f7..e6e629bd4b2a 100644 > --- a/drivers/gpu/drm/imx/parallel-display.c > +++ b/drivers/gpu/drm/imx/parallel-display.c > @@ -205,9 +205,8 @@ static int imx_pd_bind(struct device *dev, struct device *master, void *data) > u32 bus_format = 0; > const char *fmt; > > - imxpd = devm_kzalloc(dev, sizeof(*imxpd), GFP_KERNEL); > - if (!imxpd) > - return -ENOMEM; > + imxpd = dev_get_drvdata(dev); > + memset(imxpd, 0, sizeof(*imxpd)); > > edidp = of_get_property(np, "edid", &imxpd->edid_len); > if (edidp) > @@ -237,8 +236,6 @@ static int imx_pd_bind(struct device *dev, struct device *master, void *data) > if (ret) > return ret; > > - dev_set_drvdata(dev, imxpd); > - > return 0; > } > > @@ -260,6 +257,14 @@ static const struct component_ops imx_pd_ops = { > > static int imx_pd_probe(struct platform_device *pdev) > { > + struct imx_parallel_display *imxpd; > + > + imxpd = devm_kzalloc(&pdev->dev, sizeof(*imxpd), GFP_KERNEL); > + if (!imxpd) > + return -ENOMEM; > + > + platform_set_drvdata(pdev, imxpd); > + > return component_add(&pdev->dev, &imx_pd_ops); > } > > > base-commit: 98d54f81e36ba3bf92172791eba5ca5bd813989b > -- > 2.20.1 > ---------->8---------- > > regards > Philipp -- Daniel Vetter Software Engineer, Intel Corporation http://blog.ffwll.ch _______________________________________________ dri-devel mailing list dri-devel@xxxxxxxxxxxxxxxxxxxxx https://lists.freedesktop.org/mailman/listinfo/dri-devel