]> git-server-git.apps.pok.os.sepia.ceph.com Git - ceph-client.git/commitdiff
scsi: pm8001: Fix use-after-free in pm8001_queue_command()
authorSalomon Dushimirimana <salomondush@google.com>
Fri, 13 Feb 2026 19:28:06 +0000 (19:28 +0000)
committerMartin K. Petersen <martin.petersen@oracle.com>
Wed, 18 Feb 2026 02:35:43 +0000 (21:35 -0500)
Commit e29c47fe8946 ("scsi: pm8001: Simplify pm8001_task_exec()") refactors
pm8001_queue_command(), however it introduces a potential cause of a double
free scenario when it changes the function to return -ENODEV in case of phy
down/device gone state.

In this path, pm8001_queue_command() updates task status and calls
task_done to indicate to upper layer that the task has been handled.
However, this also frees the underlying SAS task. A -ENODEV is then
returned to the caller. When libsas sas_ata_qc_issue() receives this error
value, it assumes the task wasn't handled/queued by LLDD and proceeds to
clean up and free the task again, resulting in a double free.

Since pm8001_queue_command() handles the SAS task in this case, it should
return 0 to the caller indicating that the task has been handled.

Fixes: e29c47fe8946 ("scsi: pm8001: Simplify pm8001_task_exec()")
Signed-off-by: Salomon Dushimirimana <salomondush@google.com>
Reviewed-by: Damien Le Moal <dlemoal@kernel.org>
Link: https://patch.msgid.link/20260213192806.439432-1-salomondush@google.com
Signed-off-by: Martin K. Petersen <martin.petersen@oracle.com>
drivers/scsi/pm8001/pm8001_sas.c

index 6a8d35aea93a554db6f5683a8d10beaea00655a9..645524f3fe2d06160fa2286a83c30a9d4a0f71b4 100644 (file)
@@ -525,8 +525,9 @@ int pm8001_queue_command(struct sas_task *task, gfp_t gfp_flags)
                } else {
                        task->task_done(task);
                }
-               rc = -ENODEV;
-               goto err_out;
+               spin_unlock_irqrestore(&pm8001_ha->lock, flags);
+               pm8001_dbg(pm8001_ha, IO, "pm8001_task_exec device gone\n");
+               return 0;
        }
 
        ccb = pm8001_ccb_alloc(pm8001_ha, pm8001_dev, task);