mirror of
git://git.yoctoproject.org/linux-yocto.git
synced 2025-10-22 23:13:01 +02:00
nvmet-fc: move lsop put work to nvmet_fc_ls_req_op
[ Upstream commit db5a5406fb7e5337a074385c7a3e53c77f2c1bd3 ]
It’s possible for more than one async command to be in flight from
__nvmet_fc_send_ls_req. For each command, a tgtport reference is taken.
In the current code, only one put work item is queued at a time, which
results in a leaked reference.
To fix this, move the work item to the nvmet_fc_ls_req_op struct, which
already tracks all resources related to the command.
Fixes: 710c69dbac
("nvmet-fc: avoid deadlock on delete association path")
Reviewed-by: Hannes Reinecke <hare@suse.de>
Signed-off-by: Daniel Wagner <wagi@kernel.org>
Signed-off-by: Keith Busch <kbusch@kernel.org>
Signed-off-by: Sasha Levin <sashal@kernel.org>
This commit is contained in:
parent
0c13b417fe
commit
a28112cc55
|
@ -54,6 +54,8 @@ struct nvmet_fc_ls_req_op { /* for an LS RQST XMT */
|
||||||
int ls_error;
|
int ls_error;
|
||||||
struct list_head lsreq_list; /* tgtport->ls_req_list */
|
struct list_head lsreq_list; /* tgtport->ls_req_list */
|
||||||
bool req_queued;
|
bool req_queued;
|
||||||
|
|
||||||
|
struct work_struct put_work;
|
||||||
};
|
};
|
||||||
|
|
||||||
|
|
||||||
|
@ -111,8 +113,6 @@ struct nvmet_fc_tgtport {
|
||||||
struct nvmet_fc_port_entry *pe;
|
struct nvmet_fc_port_entry *pe;
|
||||||
struct kref ref;
|
struct kref ref;
|
||||||
u32 max_sg_cnt;
|
u32 max_sg_cnt;
|
||||||
|
|
||||||
struct work_struct put_work;
|
|
||||||
};
|
};
|
||||||
|
|
||||||
struct nvmet_fc_port_entry {
|
struct nvmet_fc_port_entry {
|
||||||
|
@ -236,12 +236,13 @@ static int nvmet_fc_tgt_a_get(struct nvmet_fc_tgt_assoc *assoc);
|
||||||
static void nvmet_fc_tgt_q_put(struct nvmet_fc_tgt_queue *queue);
|
static void nvmet_fc_tgt_q_put(struct nvmet_fc_tgt_queue *queue);
|
||||||
static int nvmet_fc_tgt_q_get(struct nvmet_fc_tgt_queue *queue);
|
static int nvmet_fc_tgt_q_get(struct nvmet_fc_tgt_queue *queue);
|
||||||
static void nvmet_fc_tgtport_put(struct nvmet_fc_tgtport *tgtport);
|
static void nvmet_fc_tgtport_put(struct nvmet_fc_tgtport *tgtport);
|
||||||
static void nvmet_fc_put_tgtport_work(struct work_struct *work)
|
static void nvmet_fc_put_lsop_work(struct work_struct *work)
|
||||||
{
|
{
|
||||||
struct nvmet_fc_tgtport *tgtport =
|
struct nvmet_fc_ls_req_op *lsop =
|
||||||
container_of(work, struct nvmet_fc_tgtport, put_work);
|
container_of(work, struct nvmet_fc_ls_req_op, put_work);
|
||||||
|
|
||||||
nvmet_fc_tgtport_put(tgtport);
|
nvmet_fc_tgtport_put(lsop->tgtport);
|
||||||
|
kfree(lsop);
|
||||||
}
|
}
|
||||||
static int nvmet_fc_tgtport_get(struct nvmet_fc_tgtport *tgtport);
|
static int nvmet_fc_tgtport_get(struct nvmet_fc_tgtport *tgtport);
|
||||||
static void nvmet_fc_handle_fcp_rqst(struct nvmet_fc_tgtport *tgtport,
|
static void nvmet_fc_handle_fcp_rqst(struct nvmet_fc_tgtport *tgtport,
|
||||||
|
@ -368,7 +369,7 @@ __nvmet_fc_finish_ls_req(struct nvmet_fc_ls_req_op *lsop)
|
||||||
DMA_BIDIRECTIONAL);
|
DMA_BIDIRECTIONAL);
|
||||||
|
|
||||||
out_putwork:
|
out_putwork:
|
||||||
queue_work(nvmet_wq, &tgtport->put_work);
|
queue_work(nvmet_wq, &lsop->put_work);
|
||||||
}
|
}
|
||||||
|
|
||||||
static int
|
static int
|
||||||
|
@ -389,6 +390,7 @@ __nvmet_fc_send_ls_req(struct nvmet_fc_tgtport *tgtport,
|
||||||
lsreq->done = done;
|
lsreq->done = done;
|
||||||
lsop->req_queued = false;
|
lsop->req_queued = false;
|
||||||
INIT_LIST_HEAD(&lsop->lsreq_list);
|
INIT_LIST_HEAD(&lsop->lsreq_list);
|
||||||
|
INIT_WORK(&lsop->put_work, nvmet_fc_put_lsop_work);
|
||||||
|
|
||||||
lsreq->rqstdma = fc_dma_map_single(tgtport->dev, lsreq->rqstaddr,
|
lsreq->rqstdma = fc_dma_map_single(tgtport->dev, lsreq->rqstaddr,
|
||||||
lsreq->rqstlen + lsreq->rsplen,
|
lsreq->rqstlen + lsreq->rsplen,
|
||||||
|
@ -448,8 +450,6 @@ nvmet_fc_disconnect_assoc_done(struct nvmefc_ls_req *lsreq, int status)
|
||||||
__nvmet_fc_finish_ls_req(lsop);
|
__nvmet_fc_finish_ls_req(lsop);
|
||||||
|
|
||||||
/* fc-nvme target doesn't care about success or failure of cmd */
|
/* fc-nvme target doesn't care about success or failure of cmd */
|
||||||
|
|
||||||
kfree(lsop);
|
|
||||||
}
|
}
|
||||||
|
|
||||||
/*
|
/*
|
||||||
|
@ -1407,7 +1407,6 @@ nvmet_fc_register_targetport(struct nvmet_fc_port_info *pinfo,
|
||||||
kref_init(&newrec->ref);
|
kref_init(&newrec->ref);
|
||||||
ida_init(&newrec->assoc_cnt);
|
ida_init(&newrec->assoc_cnt);
|
||||||
newrec->max_sg_cnt = template->max_sgl_segments;
|
newrec->max_sg_cnt = template->max_sgl_segments;
|
||||||
INIT_WORK(&newrec->put_work, nvmet_fc_put_tgtport_work);
|
|
||||||
|
|
||||||
ret = nvmet_fc_alloc_ls_iodlist(newrec);
|
ret = nvmet_fc_alloc_ls_iodlist(newrec);
|
||||||
if (ret) {
|
if (ret) {
|
||||||
|
|
Loading…
Reference in New Issue
Block a user