[PATCH 1/3] drm/tegra: Fix error handling cleanup

Sean Paul seanpaul at chromium.org
Thu Nov 6 10:56:16 PST 2014


On Thu, Nov 06, 2014 at 04:57:14PM +0100, Thierry Reding wrote:
> From: Thierry Reding <treding at nvidia.com>
> 
> The DRM driver's ->load() implementation didn't do a good job (no job at
> all really) cleaning up on failure. Fix that by undoing any prior setup
> when an error occurs. This requires a bit of rework to make it possible
> to clean up fbdev midway.
> 
> This was tested by injecting errors at various points during the
> initialization sequence and verifying that error cleanup didn't crash
> and no memory leaked (using kmemleak).
> 
> Reported-by: Stéphane Marchesin <marcheu at google.com>
> Reported-by: Sean Paul <seanpaul at google.com>
> Signed-off-by: Thierry Reding <treding at nvidia.com>

Looks much better :-)

Reviewed-by: Sean Paul <seanpaul at chromium.org>

> ---
>  drivers/gpu/drm/tegra/drm.c | 20 ++++++++++++++++----
>  drivers/gpu/drm/tegra/drm.h |  1 +
>  drivers/gpu/drm/tegra/fb.c  | 20 +++++++++++++++++---
>  3 files changed, 34 insertions(+), 7 deletions(-)
> 
> diff --git a/drivers/gpu/drm/tegra/drm.c b/drivers/gpu/drm/tegra/drm.c
> index 59736bb810cd..e1632fb03a89 100644
> --- a/drivers/gpu/drm/tegra/drm.c
> +++ b/drivers/gpu/drm/tegra/drm.c
> @@ -42,13 +42,13 @@ static int tegra_drm_load(struct drm_device *drm, unsigned long flags)
>  
>  	err = tegra_drm_fb_prepare(drm);
>  	if (err < 0)
> -		return err;
> +		goto config;
>  
>  	drm_kms_helper_poll_init(drm);
>  
>  	err = host1x_device_init(device);
>  	if (err < 0)
> -		return err;
> +		goto fbdev;
>  
>  	/*
>  	 * We don't use the drm_irq_install() helpers provided by the DRM
> @@ -59,13 +59,25 @@ static int tegra_drm_load(struct drm_device *drm, unsigned long flags)
>  
>  	err = drm_vblank_init(drm, drm->mode_config.num_crtc);
>  	if (err < 0)
> -		return err;
> +		goto device;
>  
>  	err = tegra_drm_fb_init(drm);
>  	if (err < 0)
> -		return err;
> +		goto vblank;
>  
>  	return 0;
> +
> +vblank:
> +	drm_vblank_cleanup(drm);
> +device:
> +	host1x_device_exit(device);
> +fbdev:
> +	drm_kms_helper_poll_fini(drm);
> +	tegra_drm_fb_free(drm);
> +config:
> +	drm_mode_config_cleanup(drm);
> +	kfree(tegra);
> +	return err;
>  }
>  
>  static int tegra_drm_unload(struct drm_device *drm)
> diff --git a/drivers/gpu/drm/tegra/drm.h b/drivers/gpu/drm/tegra/drm.h
> index b994c017971d..ef2faaef5936 100644
> --- a/drivers/gpu/drm/tegra/drm.h
> +++ b/drivers/gpu/drm/tegra/drm.h
> @@ -288,6 +288,7 @@ bool tegra_fb_is_bottom_up(struct drm_framebuffer *framebuffer);
>  int tegra_fb_get_tiling(struct drm_framebuffer *framebuffer,
>  			struct tegra_bo_tiling *tiling);
>  int tegra_drm_fb_prepare(struct drm_device *drm);
> +void tegra_drm_fb_free(struct drm_device *drm);
>  int tegra_drm_fb_init(struct drm_device *drm);
>  void tegra_drm_fb_exit(struct drm_device *drm);
>  #ifdef CONFIG_DRM_TEGRA_FBDEV
> diff --git a/drivers/gpu/drm/tegra/fb.c b/drivers/gpu/drm/tegra/fb.c
> index 3513d12d5aa1..0474241ac2a4 100644
> --- a/drivers/gpu/drm/tegra/fb.c
> +++ b/drivers/gpu/drm/tegra/fb.c
> @@ -289,6 +289,11 @@ static struct tegra_fbdev *tegra_fbdev_create(struct drm_device *drm)
>  	return fbdev;
>  }
>  
> +static void tegra_fbdev_free(struct tegra_fbdev *fbdev)
> +{
> +	kfree(fbdev);
> +}
> +
>  static int tegra_fbdev_init(struct tegra_fbdev *fbdev,
>  			    unsigned int preferred_bpp,
>  			    unsigned int num_crtc,
> @@ -322,7 +327,7 @@ fini:
>  	return err;
>  }
>  
> -static void tegra_fbdev_free(struct tegra_fbdev *fbdev)
> +static void tegra_fbdev_exit(struct tegra_fbdev *fbdev)
>  {
>  	struct fb_info *info = fbdev->base.fbdev;
>  
> @@ -345,7 +350,7 @@ static void tegra_fbdev_free(struct tegra_fbdev *fbdev)
>  	}
>  
>  	drm_fb_helper_fini(&fbdev->base);
> -	kfree(fbdev);
> +	tegra_fbdev_free(fbdev);
>  }
>  
>  void tegra_fbdev_restore_mode(struct tegra_fbdev *fbdev)
> @@ -393,6 +398,15 @@ int tegra_drm_fb_prepare(struct drm_device *drm)
>  	return 0;
>  }
>  
> +void tegra_drm_fb_free(struct drm_device *drm)
> +{
> +#ifdef CONFIG_DRM_TEGRA_FBDEV
> +	struct tegra_drm *tegra = drm->dev_private;
> +
> +	tegra_fbdev_free(tegra->fbdev);
> +#endif
> +}
> +
>  int tegra_drm_fb_init(struct drm_device *drm)
>  {
>  #ifdef CONFIG_DRM_TEGRA_FBDEV
> @@ -413,6 +427,6 @@ void tegra_drm_fb_exit(struct drm_device *drm)
>  #ifdef CONFIG_DRM_TEGRA_FBDEV
>  	struct tegra_drm *tegra = drm->dev_private;
>  
> -	tegra_fbdev_free(tegra->fbdev);
> +	tegra_fbdev_exit(tegra->fbdev);
>  #endif
>  }
> -- 
> 2.1.3
> 
> _______________________________________________
> dri-devel mailing list
> dri-devel at lists.freedesktop.org
> http://lists.freedesktop.org/mailman/listinfo/dri-devel


More information about the dri-devel mailing list