scsi: scsi_debug: Fix in_use bitmap corruption
authorDouglas Gilbert <dgilbert@interlog.com>
Thu, 2 Jul 2020 14:53:55 +0000 (10:53 -0400)
committerMartin K. Petersen <martin.petersen@oracle.com>
Fri, 3 Jul 2020 03:49:54 +0000 (23:49 -0400)
Heavy testing indicates the irqsave() spinlock around the __set_bit() is
insufficient to stop following clear_bit() calls being rarely applied
out-of-order. Also the nearby failed kzalloc() path leading to
SCSI_MLQUEUE_HOST_BUSY does not properly undo the in_use bitmap and
num_in_q, fix.

Link: https://lore.kernel.org/r/20200702145355.522283-1-dgilbert@interlog.com
Signed-off-by: Douglas Gilbert <dgilbert@interlog.com>
Signed-off-by: Martin K. Petersen <martin.petersen@oracle.com>
drivers/scsi/scsi_debug.c

index 843cccb..4692f5b 100644 (file)
@@ -5430,7 +5430,7 @@ static int schedule_resp(struct scsi_cmnd *cmnd, struct sdebug_dev_info *devip,
                else
                        return SCSI_MLQUEUE_HOST_BUSY;
        }
-       __set_bit(k, sqp->in_use_bm);
+       set_bit(k, sqp->in_use_bm);
        atomic_inc(&devip->num_in_q);
        sqcp = &sqp->qc_arr[k];
        sqcp->a_cmnd = cmnd;
@@ -5439,10 +5439,13 @@ static int schedule_resp(struct scsi_cmnd *cmnd, struct sdebug_dev_info *devip,
        spin_unlock_irqrestore(&sqp->qc_lock, iflags);
        if (unlikely(sdebug_every_nth && sdebug_any_injecting_opt))
                setup_inject(sqp, sqcp);
-       if (sd_dp == NULL) {
+       if (!sd_dp) {
                sd_dp = kzalloc(sizeof(*sd_dp), GFP_ATOMIC);
-               if (sd_dp == NULL)
+               if (!sd_dp) {
+                       atomic_dec(&devip->num_in_q);
+                       clear_bit(k, sqp->in_use_bm);
                        return SCSI_MLQUEUE_HOST_BUSY;
+               }
                new_sd_dp = true;
        } else {
                new_sd_dp = false;