Skip to content

Commit afb54c1

Browse files
committed
Merge tag 'scsi-fixes' of git://git.kernel.org/pub/scm/linux/kernel/git/jejb/scsi
Pull SCSI fixes from James Bottomley: "Driver (and enclosure) only fixes. Most are obvious. The big change is in the tcm_loop driver to add command draining to error handling (the lack of which was causing hangs with the potential for double use crashes)" * tag 'scsi-fixes' of git://git.kernel.org/pub/scm/linux/kernel/git/jejb/scsi: scsi: target: file: Use kzalloc_flex for aio_cmd scsi: scsi_transport_sas: Fix the maximum channel scanning issue scsi: target: tcm_loop: Drain commands in target_reset handler scsi: ibmvfc: Fix OOB access in ibmvfc_discover_targets_done() scsi: ses: Handle positive SCSI error from ses_recv_diag()
2 parents 26df51a + 01f784f commit afb54c1

5 files changed

Lines changed: 51 additions & 10 deletions

File tree

drivers/scsi/ibmvscsi/ibmvfc.c

Lines changed: 2 additions & 1 deletion
Original file line numberDiff line numberDiff line change
@@ -4966,7 +4966,8 @@ static void ibmvfc_discover_targets_done(struct ibmvfc_event *evt)
49664966
switch (mad_status) {
49674967
case IBMVFC_MAD_SUCCESS:
49684968
ibmvfc_dbg(vhost, "Discover Targets succeeded\n");
4969-
vhost->num_targets = be32_to_cpu(rsp->num_written);
4969+
vhost->num_targets = min_t(u32, be32_to_cpu(rsp->num_written),
4970+
max_targets);
49704971
ibmvfc_set_host_action(vhost, IBMVFC_HOST_ACTION_ALLOC_TGTS);
49714972
break;
49724973
case IBMVFC_MAD_FAILED:

drivers/scsi/scsi_transport_sas.c

Lines changed: 1 addition & 1 deletion
Original file line numberDiff line numberDiff line change
@@ -1734,7 +1734,7 @@ static int sas_user_scan(struct Scsi_Host *shost, uint channel,
17341734
break;
17351735

17361736
default:
1737-
if (channel < shost->max_channel) {
1737+
if (channel <= shost->max_channel) {
17381738
res = scsi_scan_host_selected(shost, channel, id, lun,
17391739
SCSI_SCAN_MANUAL);
17401740
} else {

drivers/scsi/ses.c

Lines changed: 1 addition & 1 deletion
Original file line numberDiff line numberDiff line change
@@ -215,7 +215,7 @@ static unsigned char *ses_get_page2_descriptor(struct enclosure_device *edev,
215215
unsigned char *type_ptr = ses_dev->page1_types;
216216
unsigned char *desc_ptr = ses_dev->page2 + 8;
217217

218-
if (ses_recv_diag(sdev, 2, ses_dev->page2, ses_dev->page2_len) < 0)
218+
if (ses_recv_diag(sdev, 2, ses_dev->page2, ses_dev->page2_len))
219219
return NULL;
220220

221221
for (i = 0; i < ses_dev->page1_num_types; i++, type_ptr += 4) {

drivers/target/loopback/tcm_loop.c

Lines changed: 46 additions & 6 deletions
Original file line numberDiff line numberDiff line change
@@ -26,6 +26,7 @@
2626
#include <linux/slab.h>
2727
#include <linux/types.h>
2828
#include <linux/configfs.h>
29+
#include <linux/blk-mq.h>
2930
#include <scsi/scsi.h>
3031
#include <scsi/scsi_tcq.h>
3132
#include <scsi/scsi_host.h>
@@ -269,15 +270,27 @@ static int tcm_loop_device_reset(struct scsi_cmnd *sc)
269270
return (ret == TMR_FUNCTION_COMPLETE) ? SUCCESS : FAILED;
270271
}
271272

273+
static bool tcm_loop_flush_work_iter(struct request *rq, void *data)
274+
{
275+
struct scsi_cmnd *sc = blk_mq_rq_to_pdu(rq);
276+
struct tcm_loop_cmd *tl_cmd = scsi_cmd_priv(sc);
277+
struct se_cmd *se_cmd = &tl_cmd->tl_se_cmd;
278+
279+
flush_work(&se_cmd->work);
280+
return true;
281+
}
282+
272283
static int tcm_loop_target_reset(struct scsi_cmnd *sc)
273284
{
274285
struct tcm_loop_hba *tl_hba;
275286
struct tcm_loop_tpg *tl_tpg;
287+
struct Scsi_Host *sh = sc->device->host;
288+
int ret;
276289

277290
/*
278291
* Locate the tcm_loop_hba_t pointer
279292
*/
280-
tl_hba = *(struct tcm_loop_hba **)shost_priv(sc->device->host);
293+
tl_hba = *(struct tcm_loop_hba **)shost_priv(sh);
281294
if (!tl_hba) {
282295
pr_err("Unable to perform device reset without active I_T Nexus\n");
283296
return FAILED;
@@ -286,11 +299,38 @@ static int tcm_loop_target_reset(struct scsi_cmnd *sc)
286299
* Locate the tl_tpg pointer from TargetID in sc->device->id
287300
*/
288301
tl_tpg = &tl_hba->tl_hba_tpgs[sc->device->id];
289-
if (tl_tpg) {
290-
tl_tpg->tl_transport_status = TCM_TRANSPORT_ONLINE;
291-
return SUCCESS;
292-
}
293-
return FAILED;
302+
if (!tl_tpg)
303+
return FAILED;
304+
305+
/*
306+
* Issue a LUN_RESET to drain all commands that the target core
307+
* knows about. This handles commands not yet marked CMD_T_COMPLETE.
308+
*/
309+
ret = tcm_loop_issue_tmr(tl_tpg, sc->device->lun, 0, TMR_LUN_RESET);
310+
if (ret != TMR_FUNCTION_COMPLETE)
311+
return FAILED;
312+
313+
/*
314+
* Flush any deferred target core completion work that may still be
315+
* queued. Commands that already had CMD_T_COMPLETE set before the TMR
316+
* are skipped by the TMR drain, but their async completion work
317+
* (transport_lun_remove_cmd → percpu_ref_put, release_cmd → scsi_done)
318+
* may still be pending in target_completion_wq.
319+
*
320+
* The SCSI EH will reuse in-flight scsi_cmnd structures for recovery
321+
* commands (e.g. TUR) immediately after this handler returns SUCCESS —
322+
* if deferred work is still pending, the memset in queuecommand would
323+
* zero the se_cmd while the work accesses it, leaking the LUN
324+
* percpu_ref and hanging configfs unlink forever.
325+
*
326+
* Use blk_mq_tagset_busy_iter() to find all started requests and
327+
* flush_work() on each — the same pattern used by mpi3mr, scsi_debug,
328+
* and other SCSI drivers to drain outstanding commands during reset.
329+
*/
330+
blk_mq_tagset_busy_iter(&sh->tag_set, tcm_loop_flush_work_iter, NULL);
331+
332+
tl_tpg->tl_transport_status = TCM_TRANSPORT_ONLINE;
333+
return SUCCESS;
294334
}
295335

296336
static const struct scsi_host_template tcm_loop_driver_template = {

drivers/target/target_core_file.c

Lines changed: 1 addition & 1 deletion
Original file line numberDiff line numberDiff line change
@@ -276,7 +276,7 @@ fd_execute_rw_aio(struct se_cmd *cmd, struct scatterlist *sgl, u32 sgl_nents,
276276
ssize_t len = 0;
277277
int ret = 0, i;
278278

279-
aio_cmd = kmalloc_flex(*aio_cmd, bvecs, sgl_nents);
279+
aio_cmd = kzalloc_flex(*aio_cmd, bvecs, sgl_nents);
280280
if (!aio_cmd)
281281
return TCM_LOGICAL_UNIT_COMMUNICATION_FAILURE;
282282

0 commit comments

Comments
 (0)