bdi: Fix use-after-free in wb_congested_put()
bdi_writeback_congested structures get created for each blkcg and bdi regardless whether bdi is registered or not. When they are created in unregistered bdi and the request queue (and thus bdi) is then destroyed while blkg still holds reference to bdi_writeback_congested structure, this structure will be referencing freed bdi and last wb_congested_put() will try to remove the structure from already freed bdi. With commit165a5e22fa
"block: Move bdi_unregister() to del_gendisk()", SCSI started to destroy bdis without calling bdi_unregister() first (previously it was calling bdi_unregister() even for unregistered bdis) and thus the code detaching bdi_writeback_congested in cgwb_bdi_destroy() was not triggered and we started hitting this use-after-free bug. It is enough to boot a KVM instance with virtio-scsi device to trigger this behavior. Fix the problem by detaching bdi_writeback_congested structures in bdi_exit() instead of bdi_unregister(). This is also more logical as they can get attached to bdi regardless whether it ever got registered or not. Fixes:165a5e22fa
Signed-off-by: Jan Kara <jack@suse.cz> Tested-by: Omar Sandoval <osandov@fb.com> Signed-off-by: Jens Axboe <axboe@fb.com>
This commit is contained in:
parent
b6f8fec444
commit
df23de5561
|
@ -683,30 +683,18 @@ static int cgwb_bdi_init(struct backing_dev_info *bdi)
|
|||
static void cgwb_bdi_destroy(struct backing_dev_info *bdi)
|
||||
{
|
||||
struct radix_tree_iter iter;
|
||||
struct rb_node *rbn;
|
||||
void **slot;
|
||||
|
||||
WARN_ON(test_bit(WB_registered, &bdi->wb.state));
|
||||
|
||||
spin_lock_irq(&cgwb_lock);
|
||||
|
||||
radix_tree_for_each_slot(slot, &bdi->cgwb_tree, &iter, 0)
|
||||
cgwb_kill(*slot);
|
||||
|
||||
while ((rbn = rb_first(&bdi->cgwb_congested_tree))) {
|
||||
struct bdi_writeback_congested *congested =
|
||||
rb_entry(rbn, struct bdi_writeback_congested, rb_node);
|
||||
|
||||
rb_erase(rbn, &bdi->cgwb_congested_tree);
|
||||
congested->bdi = NULL; /* mark @congested unlinked */
|
||||
}
|
||||
|
||||
spin_unlock_irq(&cgwb_lock);
|
||||
|
||||
/*
|
||||
* All cgwb's and their congested states must be shutdown and
|
||||
* released before returning. Drain the usage counter to wait for
|
||||
* all cgwb's and cgwb_congested's ever created on @bdi.
|
||||
* All cgwb's must be shutdown and released before returning. Drain
|
||||
* the usage counter to wait for all cgwb's ever created on @bdi.
|
||||
*/
|
||||
atomic_dec(&bdi->usage_cnt);
|
||||
wait_event(cgwb_release_wait, !atomic_read(&bdi->usage_cnt));
|
||||
|
@ -754,6 +742,21 @@ void wb_blkcg_offline(struct blkcg *blkcg)
|
|||
spin_unlock_irq(&cgwb_lock);
|
||||
}
|
||||
|
||||
static void cgwb_bdi_exit(struct backing_dev_info *bdi)
|
||||
{
|
||||
struct rb_node *rbn;
|
||||
|
||||
spin_lock_irq(&cgwb_lock);
|
||||
while ((rbn = rb_first(&bdi->cgwb_congested_tree))) {
|
||||
struct bdi_writeback_congested *congested =
|
||||
rb_entry(rbn, struct bdi_writeback_congested, rb_node);
|
||||
|
||||
rb_erase(rbn, &bdi->cgwb_congested_tree);
|
||||
congested->bdi = NULL; /* mark @congested unlinked */
|
||||
}
|
||||
spin_unlock_irq(&cgwb_lock);
|
||||
}
|
||||
|
||||
#else /* CONFIG_CGROUP_WRITEBACK */
|
||||
|
||||
static int cgwb_bdi_init(struct backing_dev_info *bdi)
|
||||
|
@ -774,7 +777,9 @@ static int cgwb_bdi_init(struct backing_dev_info *bdi)
|
|||
return 0;
|
||||
}
|
||||
|
||||
static void cgwb_bdi_destroy(struct backing_dev_info *bdi)
|
||||
static void cgwb_bdi_destroy(struct backing_dev_info *bdi) { }
|
||||
|
||||
static void cgwb_bdi_exit(struct backing_dev_info *bdi)
|
||||
{
|
||||
wb_congested_put(bdi->wb_congested);
|
||||
}
|
||||
|
@ -905,6 +910,7 @@ static void bdi_exit(struct backing_dev_info *bdi)
|
|||
{
|
||||
WARN_ON_ONCE(bdi->dev);
|
||||
wb_exit(&bdi->wb);
|
||||
cgwb_bdi_exit(bdi);
|
||||
}
|
||||
|
||||
static void release_bdi(struct kref *ref)
|
||||
|
|
Loading…
Reference in New Issue