]> git.karo-electronics.de Git - karo-tx-linux.git/commitdiff
drm/vc4: Fix resource leak in 'vc4_get_hang_state_ioctl()' in error handling path
authorChristophe JAILLET <christophe.jaillet@wanadoo.fr>
Fri, 12 May 2017 12:38:03 +0000 (14:38 +0200)
committerEric Anholt <eric@anholt.net>
Tue, 13 Jun 2017 00:11:39 +0000 (17:11 -0700)
If one 'drm_gem_handle_create()' fails, we leak somes handles and some
memory.

In order to fix it:
  - move the 'free(bo_state)' at the end of the function so that it is also
    called in the eror handling path. This has the side effect to also try
    to free it if the first 'kcalloc' fails. This is harmless.
  - add a new label, err_delete_handle, in order to delete already
    allocated handles in error handling path
  - remove the now useless 'err' label

The way the code is now written will also delete the handles if the
'copy_to_user()' call fails.

Signed-off-by: Christophe JAILLET <christophe.jaillet@wanadoo.fr>
Reviewed-by: Eric Anholt <eric@anholt.net>
Link: http://patchwork.freedesktop.org/patch/msgid/20170512123803.1886-1-christophe.jaillet@wanadoo.fr
drivers/gpu/drm/vc4/vc4_gem.c

index 9dc7646d49edd29888729ccbf42d2a9d248048bd..d5b821ad06af63a28d442edc4cc97658a26f67b2 100644 (file)
@@ -111,8 +111,8 @@ vc4_get_hang_state_ioctl(struct drm_device *dev, void *data,
                                            &handle);
 
                if (ret) {
-                       state->bo_count = i - 1;
-                       goto err;
+                       state->bo_count = i;
+                       goto err_delete_handle;
                }
                bo_state[i].handle = handle;
                bo_state[i].paddr = vc4_bo->base.paddr;
@@ -124,13 +124,16 @@ vc4_get_hang_state_ioctl(struct drm_device *dev, void *data,
                         state->bo_count * sizeof(*bo_state)))
                ret = -EFAULT;
 
-       kfree(bo_state);
+err_delete_handle:
+       if (ret) {
+               for (i = 0; i < state->bo_count; i++)
+                       drm_gem_handle_delete(file_priv, bo_state[i].handle);
+       }
 
 err_free:
-
        vc4_free_hang_state(dev, kernel_state);
+       kfree(bo_state);
 
-err:
        return ret;
 }