Skip to content

Commit 1333eee

Browse files
josefbacikmartinkpetersen
authored andcommitted
scsi: target: tcm_loop: Drain commands in target_reset handler
tcm_loop_target_reset() violates the SCSI EH contract: it returns SUCCESS without draining any in-flight commands. The SCSI EH documentation (scsi_eh.rst) requires that when a reset handler returns SUCCESS the driver has made lower layers "forget about timed out scmds" and is ready for new commands. Every other SCSI LLD (virtio_scsi, mpt3sas, ipr, scsi_debug, mpi3mr) enforces this by draining or completing outstanding commands before returning SUCCESS. Because tcm_loop_target_reset() doesn't drain, the SCSI EH reuses in-flight scsi_cmnd structures for recovery commands (e.g. TUR) while the target core still has async completion work queued for the old se_cmd. The memset in queuecommand zeroes se_lun and lun_ref_active, causing transport_lun_remove_cmd() to skip its percpu_ref_put(). The leaked LUN reference prevents transport_clear_lun_ref() from completing, hanging configfs LUN unlink forever in D-state: INFO: task rm:264 blocked for more than 122 seconds. rm D 0 264 258 0x00004000 Call Trace: __schedule+0x3d0/0x8e0 schedule+0x36/0xf0 transport_clear_lun_ref+0x78/0x90 [target_core_mod] core_tpg_remove_lun+0x28/0xb0 [target_core_mod] target_fabric_port_unlink+0x50/0x60 [target_core_mod] configfs_unlink+0x156/0x1f0 [configfs] vfs_unlink+0x109/0x290 do_unlinkat+0x1d5/0x2d0 Fix this by making tcm_loop_target_reset() actually drain commands: 1. Issue TMR_LUN_RESET via tcm_loop_issue_tmr() to drain all commands that the target core knows about (those not yet CMD_T_COMPLETE). 2. Use blk_mq_tagset_busy_iter() to iterate all started requests and flush_work() on each se_cmd — this drains any deferred completion work for commands that already had CMD_T_COMPLETE set before the TMR (which the TMR skips via __target_check_io_state()). This is the same pattern used by mpi3mr, scsi_debug, and libsas to drain outstanding commands during reset. Fixes: e0eb5d3 ("scsi: target: tcm_loop: Use block cmd allocator for se_cmds") Cc: [email protected] Assisted-by: Claude:claude-opus-4-6 Signed-off-by: Josef Bacik <[email protected]> Link: https://patch.msgid.link/27011aa34c8f6b1b94d2e3cf5655b6d037f53428.1773706803.git.josef@toxicpanda.com Signed-off-by: Martin K. Petersen <[email protected]>
1 parent 61d099a commit 1333eee

1 file changed

Lines changed: 46 additions & 6 deletions

File tree

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 = {

0 commit comments

Comments
 (0)