Skip to content

Commit e2daec4

Browse files
Ye Binaxboe
authored andcommitted
nbd: Fix hungtask when nbd_config_put
I got follow issue: [ 247.381177] INFO: task kworker/u10:0:47 blocked for more than 120 seconds. [ 247.382644] Not tainted 4.19.90-dirty #140 [ 247.383502] "echo 0 > /proc/sys/kernel/hung_task_timeout_secs" disables this message. [ 247.385027] Call Trace: [ 247.388384] schedule+0xb8/0x3c0 [ 247.388966] schedule_timeout+0x2b4/0x380 [ 247.392815] wait_for_completion+0x367/0x510 [ 247.397713] flush_workqueue+0x32b/0x1340 [ 247.402700] drain_workqueue+0xda/0x3c0 [ 247.403442] destroy_workqueue+0x7b/0x690 [ 247.405014] nbd_config_put.cold+0x2f9/0x5b6 [ 247.405823] recv_work+0x1fd/0x2b0 [ 247.406485] process_one_work+0x70b/0x1610 [ 247.407262] worker_thread+0x5a9/0x1060 [ 247.408699] kthread+0x35e/0x430 [ 247.410918] ret_from_fork+0x1f/0x30 We can reproduce issue as follows: 1. Inject memory fault in nbd_start_device -1244,10 +1248,18 @@ static int nbd_start_device(struct nbd_device *nbd) nbd_dev_dbg_init(nbd); for (i = 0; i < num_connections; i++) { struct recv_thread_args *args; - - args = kzalloc(sizeof(*args), GFP_KERNEL); + + if (i == 1) { + args = NULL; + printk("%s: inject malloc error\n", __func__); + } + else + args = kzalloc(sizeof(*args), GFP_KERNEL); 2. Inject delay in recv_work -757,6 +760,8 @@ static void recv_work(struct work_struct *work) blk_mq_complete_request(blk_mq_rq_from_pdu(cmd)); } + printk("%s: comm=%s pid=%d\n", __func__, current->comm, current->pid); + mdelay(5 * 1000); nbd_config_put(nbd); atomic_dec(&config->recv_threads); wake_up(&config->recv_wq); 3. Create nbd server nbd-server 8000 /tmp/disk 4. Create nbd client nbd-client localhost 8000 /dev/nbd1 Then will trigger above issue. Reason is when add delay in recv_work, lead to release the last reference of 'nbd->config_refs'. nbd_config_put will call flush_workqueue to make all work finish. Obviously, it will lead to deadloop. To solve this issue, according to Josef's suggestion move 'recv_work' init from start device to nbd_dev_add, then destroy 'recv_work'when nbd device teardown. Signed-off-by: Ye Bin <[email protected]> Reviewed-by: Josef Bacik <[email protected]> Link: https://lore.kernel.org/r/[email protected] Signed-off-by: Jens Axboe <[email protected]>
1 parent 69beb62 commit e2daec4

File tree

1 file changed

+16
-20
lines changed

1 file changed

+16
-20
lines changed

drivers/block/nbd.c

Lines changed: 16 additions & 20 deletions
Original file line numberDiff line numberDiff line change
@@ -260,7 +260,7 @@ static void nbd_dev_remove(struct nbd_device *nbd)
260260
mutex_lock(&nbd_index_mutex);
261261
idr_remove(&nbd_index_idr, nbd->index);
262262
mutex_unlock(&nbd_index_mutex);
263-
263+
destroy_workqueue(nbd->recv_workq);
264264
kfree(nbd);
265265
}
266266

@@ -1319,10 +1319,6 @@ static void nbd_config_put(struct nbd_device *nbd)
13191319
kfree(nbd->config);
13201320
nbd->config = NULL;
13211321

1322-
if (nbd->recv_workq)
1323-
destroy_workqueue(nbd->recv_workq);
1324-
nbd->recv_workq = NULL;
1325-
13261322
nbd->tag_set.timeout = 0;
13271323
nbd->disk->queue->limits.discard_granularity = 0;
13281324
nbd->disk->queue->limits.discard_alignment = 0;
@@ -1351,14 +1347,6 @@ static int nbd_start_device(struct nbd_device *nbd)
13511347
return -EINVAL;
13521348
}
13531349

1354-
nbd->recv_workq = alloc_workqueue("knbd%d-recv",
1355-
WQ_MEM_RECLAIM | WQ_HIGHPRI |
1356-
WQ_UNBOUND, 0, nbd->index);
1357-
if (!nbd->recv_workq) {
1358-
dev_err(disk_to_dev(nbd->disk), "Could not allocate knbd recv work queue.\n");
1359-
return -ENOMEM;
1360-
}
1361-
13621350
blk_mq_update_nr_hw_queues(&nbd->tag_set, config->num_connections);
13631351
nbd->pid = task_pid_nr(current);
13641352

@@ -1784,6 +1772,15 @@ static struct nbd_device *nbd_dev_add(int index, unsigned int refs)
17841772
}
17851773
nbd->disk = disk;
17861774

1775+
nbd->recv_workq = alloc_workqueue("nbd%d-recv",
1776+
WQ_MEM_RECLAIM | WQ_HIGHPRI |
1777+
WQ_UNBOUND, 0, nbd->index);
1778+
if (!nbd->recv_workq) {
1779+
dev_err(disk_to_dev(nbd->disk), "Could not allocate knbd recv work queue.\n");
1780+
err = -ENOMEM;
1781+
goto out_err_disk;
1782+
}
1783+
17871784
/*
17881785
* Tell the block layer that we are not a rotational device
17891786
*/
@@ -1814,7 +1811,7 @@ static struct nbd_device *nbd_dev_add(int index, unsigned int refs)
18141811
disk->first_minor = index << part_shift;
18151812
if (disk->first_minor < index || disk->first_minor > MINORMASK) {
18161813
err = -EINVAL;
1817-
goto out_err_disk;
1814+
goto out_free_work;
18181815
}
18191816

18201817
disk->minors = 1 << part_shift;
@@ -1823,7 +1820,7 @@ static struct nbd_device *nbd_dev_add(int index, unsigned int refs)
18231820
sprintf(disk->disk_name, "nbd%d", index);
18241821
err = add_disk(disk);
18251822
if (err)
1826-
goto out_err_disk;
1823+
goto out_free_work;
18271824

18281825
/*
18291826
* Now publish the device.
@@ -1832,6 +1829,8 @@ static struct nbd_device *nbd_dev_add(int index, unsigned int refs)
18321829
nbd_total_devices++;
18331830
return nbd;
18341831

1832+
out_free_work:
1833+
destroy_workqueue(nbd->recv_workq);
18351834
out_err_disk:
18361835
blk_cleanup_disk(disk);
18371836
out_free_idr:
@@ -2087,13 +2086,10 @@ static void nbd_disconnect_and_put(struct nbd_device *nbd)
20872086
nbd_disconnect(nbd);
20882087
sock_shutdown(nbd);
20892088
/*
2090-
* Make sure recv thread has finished, so it does not drop the last
2091-
* config ref and try to destroy the workqueue from inside the work
2092-
* queue. And this also ensure that we can safely call nbd_clear_que()
2089+
* Make sure recv thread has finished, we can safely call nbd_clear_que()
20932090
* to cancel the inflight I/Os.
20942091
*/
2095-
if (nbd->recv_workq)
2096-
flush_workqueue(nbd->recv_workq);
2092+
flush_workqueue(nbd->recv_workq);
20972093
nbd_clear_que(nbd);
20982094
nbd->task_setup = NULL;
20992095
mutex_unlock(&nbd->config_lock);

0 commit comments

Comments
 (0)