Commit 4049dc96 authored by Daniel Wagner's avatar Daniel Wagner Committed by Keith Busch

nvmet-fc: defer cleanup using RCU properly

When the target executes a disconnect and the host triggers a reconnect
immediately, the reconnect command still finds an existing association.

The reconnect crashes later on because nvmet_fc_delete_target_assoc
blindly removes resources while the reconnect code wants to use it.

To address this, nvmet_fc_find_target_assoc should not be able to
lookup an association which is being removed. The association list
is already under RCU lifetime management, so let's properly use it
and remove the association from the list and wait for a grace period
before cleaning up all. This means we also can drop the RCU management
on the queues, because this is now handled via the association itself.

A second step split the execution context so that the initial disconnect
command can complete without running the reconnect code in the same
context. As usual, this is done by deferring the ->done to a workqueue.
Reviewed-by: default avatarChristoph Hellwig <hch@lst.de>
Reviewed-by: default avatarHannes Reinecke <hare@suse.de>
Signed-off-by: default avatarDaniel Wagner <dwagner@suse.de>
Signed-off-by: default avatarKeith Busch <kbusch@kernel.org>
parent c691e6d7
...@@ -166,7 +166,7 @@ struct nvmet_fc_tgt_assoc { ...@@ -166,7 +166,7 @@ struct nvmet_fc_tgt_assoc {
struct nvmet_fc_hostport *hostport; struct nvmet_fc_hostport *hostport;
struct nvmet_fc_ls_iod *rcv_disconn; struct nvmet_fc_ls_iod *rcv_disconn;
struct list_head a_list; struct list_head a_list;
struct nvmet_fc_tgt_queue __rcu *queues[NVMET_NR_QUEUES + 1]; struct nvmet_fc_tgt_queue *queues[NVMET_NR_QUEUES + 1];
struct kref ref; struct kref ref;
struct work_struct del_work; struct work_struct del_work;
struct rcu_head rcu; struct rcu_head rcu;
...@@ -803,14 +803,11 @@ nvmet_fc_alloc_target_queue(struct nvmet_fc_tgt_assoc *assoc, ...@@ -803,14 +803,11 @@ nvmet_fc_alloc_target_queue(struct nvmet_fc_tgt_assoc *assoc,
if (!queue) if (!queue)
return NULL; return NULL;
if (!nvmet_fc_tgt_a_get(assoc))
goto out_free_queue;
queue->work_q = alloc_workqueue("ntfc%d.%d.%d", 0, 0, queue->work_q = alloc_workqueue("ntfc%d.%d.%d", 0, 0,
assoc->tgtport->fc_target_port.port_num, assoc->tgtport->fc_target_port.port_num,
assoc->a_id, qid); assoc->a_id, qid);
if (!queue->work_q) if (!queue->work_q)
goto out_a_put; goto out_free_queue;
queue->qid = qid; queue->qid = qid;
queue->sqsize = sqsize; queue->sqsize = sqsize;
...@@ -832,15 +829,13 @@ nvmet_fc_alloc_target_queue(struct nvmet_fc_tgt_assoc *assoc, ...@@ -832,15 +829,13 @@ nvmet_fc_alloc_target_queue(struct nvmet_fc_tgt_assoc *assoc,
goto out_fail_iodlist; goto out_fail_iodlist;
WARN_ON(assoc->queues[qid]); WARN_ON(assoc->queues[qid]);
rcu_assign_pointer(assoc->queues[qid], queue); assoc->queues[qid] = queue;
return queue; return queue;
out_fail_iodlist: out_fail_iodlist:
nvmet_fc_destroy_fcp_iodlist(assoc->tgtport, queue); nvmet_fc_destroy_fcp_iodlist(assoc->tgtport, queue);
destroy_workqueue(queue->work_q); destroy_workqueue(queue->work_q);
out_a_put:
nvmet_fc_tgt_a_put(assoc);
out_free_queue: out_free_queue:
kfree(queue); kfree(queue);
return NULL; return NULL;
...@@ -853,12 +848,8 @@ nvmet_fc_tgt_queue_free(struct kref *ref) ...@@ -853,12 +848,8 @@ nvmet_fc_tgt_queue_free(struct kref *ref)
struct nvmet_fc_tgt_queue *queue = struct nvmet_fc_tgt_queue *queue =
container_of(ref, struct nvmet_fc_tgt_queue, ref); container_of(ref, struct nvmet_fc_tgt_queue, ref);
rcu_assign_pointer(queue->assoc->queues[queue->qid], NULL);
nvmet_fc_destroy_fcp_iodlist(queue->assoc->tgtport, queue); nvmet_fc_destroy_fcp_iodlist(queue->assoc->tgtport, queue);
nvmet_fc_tgt_a_put(queue->assoc);
destroy_workqueue(queue->work_q); destroy_workqueue(queue->work_q);
kfree_rcu(queue, rcu); kfree_rcu(queue, rcu);
...@@ -970,7 +961,7 @@ nvmet_fc_find_target_queue(struct nvmet_fc_tgtport *tgtport, ...@@ -970,7 +961,7 @@ nvmet_fc_find_target_queue(struct nvmet_fc_tgtport *tgtport,
rcu_read_lock(); rcu_read_lock();
list_for_each_entry_rcu(assoc, &tgtport->assoc_list, a_list) { list_for_each_entry_rcu(assoc, &tgtport->assoc_list, a_list) {
if (association_id == assoc->association_id) { if (association_id == assoc->association_id) {
queue = rcu_dereference(assoc->queues[qid]); queue = assoc->queues[qid];
if (queue && if (queue &&
(!atomic_read(&queue->connected) || (!atomic_read(&queue->connected) ||
!nvmet_fc_tgt_q_get(queue))) !nvmet_fc_tgt_q_get(queue)))
...@@ -1173,13 +1164,18 @@ nvmet_fc_target_assoc_free(struct kref *ref) ...@@ -1173,13 +1164,18 @@ nvmet_fc_target_assoc_free(struct kref *ref)
struct nvmet_fc_tgtport *tgtport = assoc->tgtport; struct nvmet_fc_tgtport *tgtport = assoc->tgtport;
struct nvmet_fc_ls_iod *oldls; struct nvmet_fc_ls_iod *oldls;
unsigned long flags; unsigned long flags;
int i;
for (i = NVMET_NR_QUEUES; i >= 0; i--) {
if (assoc->queues[i])
nvmet_fc_delete_target_queue(assoc->queues[i]);
}
/* Send Disconnect now that all i/o has completed */ /* Send Disconnect now that all i/o has completed */
nvmet_fc_xmt_disconnect_assoc(assoc); nvmet_fc_xmt_disconnect_assoc(assoc);
nvmet_fc_free_hostport(assoc->hostport); nvmet_fc_free_hostport(assoc->hostport);
spin_lock_irqsave(&tgtport->lock, flags); spin_lock_irqsave(&tgtport->lock, flags);
list_del_rcu(&assoc->a_list);
oldls = assoc->rcv_disconn; oldls = assoc->rcv_disconn;
spin_unlock_irqrestore(&tgtport->lock, flags); spin_unlock_irqrestore(&tgtport->lock, flags);
/* if pending Rcv Disconnect Association LS, send rsp now */ /* if pending Rcv Disconnect Association LS, send rsp now */
...@@ -1209,7 +1205,7 @@ static void ...@@ -1209,7 +1205,7 @@ static void
nvmet_fc_delete_target_assoc(struct nvmet_fc_tgt_assoc *assoc) nvmet_fc_delete_target_assoc(struct nvmet_fc_tgt_assoc *assoc)
{ {
struct nvmet_fc_tgtport *tgtport = assoc->tgtport; struct nvmet_fc_tgtport *tgtport = assoc->tgtport;
struct nvmet_fc_tgt_queue *queue; unsigned long flags;
int i, terminating; int i, terminating;
terminating = atomic_xchg(&assoc->terminating, 1); terminating = atomic_xchg(&assoc->terminating, 1);
...@@ -1218,29 +1214,21 @@ nvmet_fc_delete_target_assoc(struct nvmet_fc_tgt_assoc *assoc) ...@@ -1218,29 +1214,21 @@ nvmet_fc_delete_target_assoc(struct nvmet_fc_tgt_assoc *assoc)
if (terminating) if (terminating)
return; return;
spin_lock_irqsave(&tgtport->lock, flags);
list_del_rcu(&assoc->a_list);
spin_unlock_irqrestore(&tgtport->lock, flags);
for (i = NVMET_NR_QUEUES; i >= 0; i--) { synchronize_rcu();
rcu_read_lock();
queue = rcu_dereference(assoc->queues[i]);
if (!queue) {
rcu_read_unlock();
continue;
}
if (!nvmet_fc_tgt_q_get(queue)) { /* ensure all in-flight I/Os have been processed */
rcu_read_unlock(); for (i = NVMET_NR_QUEUES; i >= 0; i--) {
continue; if (assoc->queues[i])
} flush_workqueue(assoc->queues[i]->work_q);
rcu_read_unlock();
nvmet_fc_delete_target_queue(queue);
nvmet_fc_tgt_q_put(queue);
} }
dev_info(tgtport->dev, dev_info(tgtport->dev,
"{%d:%d} Association deleted\n", "{%d:%d} Association deleted\n",
tgtport->fc_target_port.port_num, assoc->a_id); tgtport->fc_target_port.port_num, assoc->a_id);
nvmet_fc_tgt_a_put(assoc);
} }
static struct nvmet_fc_tgt_assoc * static struct nvmet_fc_tgt_assoc *
...@@ -1493,9 +1481,8 @@ __nvmet_fc_free_assocs(struct nvmet_fc_tgtport *tgtport) ...@@ -1493,9 +1481,8 @@ __nvmet_fc_free_assocs(struct nvmet_fc_tgtport *tgtport)
list_for_each_entry_rcu(assoc, &tgtport->assoc_list, a_list) { list_for_each_entry_rcu(assoc, &tgtport->assoc_list, a_list) {
if (!nvmet_fc_tgt_a_get(assoc)) if (!nvmet_fc_tgt_a_get(assoc))
continue; continue;
if (!queue_work(nvmet_wq, &assoc->del_work)) queue_work(nvmet_wq, &assoc->del_work);
/* already deleting - release local reference */ nvmet_fc_tgt_a_put(assoc);
nvmet_fc_tgt_a_put(assoc);
} }
rcu_read_unlock(); rcu_read_unlock();
} }
...@@ -1548,9 +1535,8 @@ nvmet_fc_invalidate_host(struct nvmet_fc_target_port *target_port, ...@@ -1548,9 +1535,8 @@ nvmet_fc_invalidate_host(struct nvmet_fc_target_port *target_port,
continue; continue;
assoc->hostport->invalid = 1; assoc->hostport->invalid = 1;
noassoc = false; noassoc = false;
if (!queue_work(nvmet_wq, &assoc->del_work)) queue_work(nvmet_wq, &assoc->del_work);
/* already deleting - release local reference */ nvmet_fc_tgt_a_put(assoc);
nvmet_fc_tgt_a_put(assoc);
} }
spin_unlock_irqrestore(&tgtport->lock, flags); spin_unlock_irqrestore(&tgtport->lock, flags);
...@@ -1582,7 +1568,7 @@ nvmet_fc_delete_ctrl(struct nvmet_ctrl *ctrl) ...@@ -1582,7 +1568,7 @@ nvmet_fc_delete_ctrl(struct nvmet_ctrl *ctrl)
rcu_read_lock(); rcu_read_lock();
list_for_each_entry_rcu(assoc, &tgtport->assoc_list, a_list) { list_for_each_entry_rcu(assoc, &tgtport->assoc_list, a_list) {
queue = rcu_dereference(assoc->queues[0]); queue = assoc->queues[0];
if (queue && queue->nvme_sq.ctrl == ctrl) { if (queue && queue->nvme_sq.ctrl == ctrl) {
if (nvmet_fc_tgt_a_get(assoc)) if (nvmet_fc_tgt_a_get(assoc))
found_ctrl = true; found_ctrl = true;
...@@ -1594,9 +1580,8 @@ nvmet_fc_delete_ctrl(struct nvmet_ctrl *ctrl) ...@@ -1594,9 +1580,8 @@ nvmet_fc_delete_ctrl(struct nvmet_ctrl *ctrl)
nvmet_fc_tgtport_put(tgtport); nvmet_fc_tgtport_put(tgtport);
if (found_ctrl) { if (found_ctrl) {
if (!queue_work(nvmet_wq, &assoc->del_work)) queue_work(nvmet_wq, &assoc->del_work);
/* already deleting - release local reference */ nvmet_fc_tgt_a_put(assoc);
nvmet_fc_tgt_a_put(assoc);
return; return;
} }
...@@ -1626,6 +1611,8 @@ nvmet_fc_unregister_targetport(struct nvmet_fc_target_port *target_port) ...@@ -1626,6 +1611,8 @@ nvmet_fc_unregister_targetport(struct nvmet_fc_target_port *target_port)
/* terminate any outstanding associations */ /* terminate any outstanding associations */
__nvmet_fc_free_assocs(tgtport); __nvmet_fc_free_assocs(tgtport);
flush_workqueue(nvmet_wq);
/* /*
* should terminate LS's as well. However, LS's will be generated * should terminate LS's as well. However, LS's will be generated
* at the tail end of association termination, so they likely don't * at the tail end of association termination, so they likely don't
...@@ -1871,9 +1858,6 @@ nvmet_fc_ls_disconnect(struct nvmet_fc_tgtport *tgtport, ...@@ -1871,9 +1858,6 @@ nvmet_fc_ls_disconnect(struct nvmet_fc_tgtport *tgtport,
sizeof(struct fcnvme_ls_disconnect_assoc_acc)), sizeof(struct fcnvme_ls_disconnect_assoc_acc)),
FCNVME_LS_DISCONNECT_ASSOC); FCNVME_LS_DISCONNECT_ASSOC);
/* release get taken in nvmet_fc_find_target_assoc */
nvmet_fc_tgt_a_put(assoc);
/* /*
* The rules for LS response says the response cannot * The rules for LS response says the response cannot
* go back until ABTS's have been sent for all outstanding * go back until ABTS's have been sent for all outstanding
...@@ -1888,8 +1872,6 @@ nvmet_fc_ls_disconnect(struct nvmet_fc_tgtport *tgtport, ...@@ -1888,8 +1872,6 @@ nvmet_fc_ls_disconnect(struct nvmet_fc_tgtport *tgtport,
assoc->rcv_disconn = iod; assoc->rcv_disconn = iod;
spin_unlock_irqrestore(&tgtport->lock, flags); spin_unlock_irqrestore(&tgtport->lock, flags);
nvmet_fc_delete_target_assoc(assoc);
if (oldls) { if (oldls) {
dev_info(tgtport->dev, dev_info(tgtport->dev,
"{%d:%d} Multiple Disconnect Association LS's " "{%d:%d} Multiple Disconnect Association LS's "
...@@ -1905,6 +1887,9 @@ nvmet_fc_ls_disconnect(struct nvmet_fc_tgtport *tgtport, ...@@ -1905,6 +1887,9 @@ nvmet_fc_ls_disconnect(struct nvmet_fc_tgtport *tgtport,
nvmet_fc_xmt_ls_rsp(tgtport, oldls); nvmet_fc_xmt_ls_rsp(tgtport, oldls);
} }
queue_work(nvmet_wq, &assoc->del_work);
nvmet_fc_tgt_a_put(assoc);
return false; return false;
} }
...@@ -2903,6 +2888,9 @@ nvmet_fc_remove_port(struct nvmet_port *port) ...@@ -2903,6 +2888,9 @@ nvmet_fc_remove_port(struct nvmet_port *port)
nvmet_fc_portentry_unbind(pe); nvmet_fc_portentry_unbind(pe);
/* terminate any outstanding associations */
__nvmet_fc_free_assocs(pe->tgtport);
kfree(pe); kfree(pe);
} }
...@@ -2934,6 +2922,9 @@ static int __init nvmet_fc_init_module(void) ...@@ -2934,6 +2922,9 @@ static int __init nvmet_fc_init_module(void)
static void __exit nvmet_fc_exit_module(void) static void __exit nvmet_fc_exit_module(void)
{ {
/* ensure any shutdown operation, e.g. delete ctrls have finished */
flush_workqueue(nvmet_wq);
/* sanity check - all lports should be removed */ /* sanity check - all lports should be removed */
if (!list_empty(&nvmet_fc_target_list)) if (!list_empty(&nvmet_fc_target_list))
pr_warn("%s: targetport list not empty\n", __func__); pr_warn("%s: targetport list not empty\n", __func__);
......
Markdown is supported
0%
or
You are about to add 0 people to the discussion. Proceed with caution.
Finish editing this message first!
Please register or to comment