]> git.ipfire.org Git - thirdparty/kernel/linux.git/commitdiff
drm/nouveau: fix reversed error cleanup order in ucopy functions
authorJunrui Luo <moonafterrain@outlook.com>
Wed, 10 Jun 2026 10:01:28 +0000 (18:01 +0800)
committerDanilo Krummrich <dakr@kernel.org>
Sun, 21 Jun 2026 13:15:23 +0000 (15:15 +0200)
nouveau_uvmm_vm_bind_ucopy() and nouveau_exec_ucopy() place their error
cleanup labels in allocation order rather than reverse allocation order.
On a u_memcpya() failure for in_sync.s, the goto to err_free_ops (or
err_free_pushs) frees the first allocation and then falls through to
err_free_ins, which calls u_free() on args->in_sync.s.

Since args->in_sync.s still holds the ERR_PTR returned by the failed
u_memcpya(), and ERR_PTR values are not caught by ZERO_OR_NULL_PTR(),
kvfree() proceeds to dereference it, which can result in a kernel oops.
A failure for out_sync.s instead jumps to err_free_ins and skips freeing
the first allocation, leading to a memory leak.

Fix by swapping the cleanup label order so resources are freed in the
correct reverse allocation sequence.

Fixes: b88baab82871 ("drm/nouveau: implement new VM_BIND uAPI")
Reported-by: Yuhao Jiang <danisjiang@gmail.com>
Cc: stable@vger.kernel.org
Signed-off-by: Junrui Luo <moonafterrain@outlook.com>
Link: https://patch.msgid.link/SYBPR01MB7881484D91A6F80271415F71AF1A2@SYBPR01MB7881.ausprd01.prod.outlook.com
Signed-off-by: Danilo Krummrich <dakr@kernel.org>
drivers/gpu/drm/nouveau/nouveau_exec.c
drivers/gpu/drm/nouveau/nouveau_uvmm.c

index c01a01aee32be80997833ebe756b9df291516c8a..a08ab1cfea9be872f4f26573313a26346f83b80b 100644 (file)
@@ -331,10 +331,10 @@ nouveau_exec_ucopy(struct nouveau_exec_job_args *args,
 
        return 0;
 
-err_free_pushs:
-       u_free(args->push.s);
 err_free_ins:
        u_free(args->in_sync.s);
+err_free_pushs:
+       u_free(args->push.s);
        return ret;
 }
 
index 36445915aa58c179d64f5de8022f5029b4cc958d..f5e4756b4de4ae31f99f79c8f757c6da04b5d288 100644 (file)
@@ -1779,10 +1779,10 @@ nouveau_uvmm_vm_bind_ucopy(struct nouveau_uvmm_bind_job_args *args,
 
        return 0;
 
-err_free_ops:
-       u_free(args->op.s);
 err_free_ins:
        u_free(args->in_sync.s);
+err_free_ops:
+       u_free(args->op.s);
        return ret;
 }