diff options
author | Signed-off-by: Jan Kara <jack@suse.cz> | 2010-09-16 14:36:36 -0400 |
---|---|---|
committer | Jens Axboe <jaxboe@fusionio.com> | 2010-09-16 14:36:36 -0400 |
commit | 01ea50638bc04ca5259f5711fcdedefcdde1cf43 (patch) | |
tree | c525dc13b05feab90d71b78a1d7adb0ddde081ea /block/genhd.c | |
parent | 2786c4e5e54802c34297e55050fef3e862a27b3f (diff) |
block: Fix race during disk initialization
When a new disk is being discovered, add_disk() first ties the bdev to gendisk
(via register_disk()->blkdev_get()) and only after that calls
bdi_register_bdev(). Because register_disk() also creates disk's kobject, it
can happen that userspace manages to open and modify the device's data (or
inode) before its BDI is properly initialized leading to a warning in
__mark_inode_dirty().
Fix the problem by registering BDI early enough.
This patch addresses https://bugzilla.kernel.org/show_bug.cgi?id=16312
Cc: stable@kernel.org
Reported-by: Larry Finger <Larry.Finger@lwfinger.net>
Signed-off-by: Jan Kara <jack@suse.cz>
Signed-off-by: Jens Axboe <jaxboe@fusionio.com>
Diffstat (limited to 'block/genhd.c')
-rw-r--r-- | block/genhd.c | 6 |
1 files changed, 4 insertions, 2 deletions
diff --git a/block/genhd.c b/block/genhd.c index 5c9c503de423..7923e720ddf5 100644 --- a/block/genhd.c +++ b/block/genhd.c | |||
@@ -541,13 +541,15 @@ void add_disk(struct gendisk *disk) | |||
541 | disk->major = MAJOR(devt); | 541 | disk->major = MAJOR(devt); |
542 | disk->first_minor = MINOR(devt); | 542 | disk->first_minor = MINOR(devt); |
543 | 543 | ||
544 | /* Register BDI before referencing it from bdev */ | ||
545 | bdi = &disk->queue->backing_dev_info; | ||
546 | bdi_register_dev(bdi, disk_devt(disk)); | ||
547 | |||
544 | blk_register_region(disk_devt(disk), disk->minors, NULL, | 548 | blk_register_region(disk_devt(disk), disk->minors, NULL, |
545 | exact_match, exact_lock, disk); | 549 | exact_match, exact_lock, disk); |
546 | register_disk(disk); | 550 | register_disk(disk); |
547 | blk_register_queue(disk); | 551 | blk_register_queue(disk); |
548 | 552 | ||
549 | bdi = &disk->queue->backing_dev_info; | ||
550 | bdi_register_dev(bdi, disk_devt(disk)); | ||
551 | retval = sysfs_create_link(&disk_to_dev(disk)->kobj, &bdi->dev->kobj, | 553 | retval = sysfs_create_link(&disk_to_dev(disk)->kobj, &bdi->dev->kobj, |
552 | "bdi"); | 554 | "bdi"); |
553 | WARN_ON(retval); | 555 | WARN_ON(retval); |