Commit 143e2f82 authored by Jan Kara's avatar Jan Kara Committed by Thadeu Lima de Souza Cascardo

block: Allow bdi re-registration

BugLink: http://bugs.launchpad.net/bugs/1659111

SCSI can call device_add_disk() several times for one request queue when
a device in unbound and bound, creating new gendisk each time. This will
lead to bdi being repeatedly registered and unregistered. This was not a
big problem until commit 165a5e22 "block: Move bdi_unregister() to
del_gendisk()" since bdi was only registered repeatedly (bdi_register()
handles repeated calls fine, only we ended up leaking reference to
gendisk due to overwriting bdi->owner) but unregistered only in
blk_cleanup_queue() which didn't get called repeatedly. After
165a5e22 we were doing correct bdi_register() - bdi_unregister()
cycles however bdi_unregister() is not prepared for it. So make sure
bdi_unregister() cleans up bdi in such a way that it is prepared for
a possible following bdi_register() call.

An easy way to provoke this behavior is to enable
CONFIG_DEBUG_TEST_DRIVER_REMOVE and use scsi_debug driver to create a
scsi disk which immediately hangs without this fix.

Fixes: 165a5e22Signed-off-by: default avatarJan Kara <jack@suse.cz>
Tested-by: default avatarOmar Sandoval <osandov@fb.com>
Signed-off-by: default avatarJens Axboe <axboe@fb.com>
(cherry picked from commit b6f8fec4)
Signed-off-by: default avatarThiago Jung Bauermann <bauerman@linux.vnet.ibm.com>
Signed-off-by: default avatarTim Gardner <tim.gardner@canonical.com>
Acked-by: default avatarSeth Forshee <seth.forshee@canonical.com>
Acked-by: default avatarBrad Figg <brad.figg@canonical.com>
Signed-off-by: default avatarThadeu Lima de Souza Cascardo <cascardo@canonical.com>
parent be2a2080
...@@ -709,6 +709,11 @@ static void cgwb_bdi_destroy(struct backing_dev_info *bdi) ...@@ -709,6 +709,11 @@ static void cgwb_bdi_destroy(struct backing_dev_info *bdi)
*/ */
atomic_dec(&bdi->usage_cnt); atomic_dec(&bdi->usage_cnt);
wait_event(cgwb_release_wait, !atomic_read(&bdi->usage_cnt)); wait_event(cgwb_release_wait, !atomic_read(&bdi->usage_cnt));
/*
* Grab back our reference so that we hold it when @bdi gets
* re-registered.
*/
atomic_inc(&bdi->usage_cnt);
} }
/** /**
...@@ -856,6 +861,8 @@ int bdi_register_owner(struct backing_dev_info *bdi, struct device *owner) ...@@ -856,6 +861,8 @@ int bdi_register_owner(struct backing_dev_info *bdi, struct device *owner)
MINOR(owner->devt)); MINOR(owner->devt));
if (rc) if (rc)
return rc; return rc;
/* Leaking owner reference... */
WARN_ON(bdi->owner);
bdi->owner = owner; bdi->owner = owner;
get_device(owner); get_device(owner);
return 0; return 0;
......
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