[PATCH] md: do not set mddev->bitmap until bitmap is fully initialised

When hot-adding a bitmap, bitmap_daemon_work could get called while the bitmap
is being created, so don't set mddev->bitmap until the bitmap is ready.

This requires freeing the bitmap inside bitmap_create if creation failed
part-way through.

Signed-off-by: Neil Brown <neilb@cse.unsw.edu.au>
Signed-off-by: Andrew Morton <akpm@osdl.org>
Signed-off-by: Linus Torvalds <torvalds@osdl.org>
This commit is contained in:
NeilBrown 2005-09-09 16:23:50 -07:00 коммит произвёл Linus Torvalds
Родитель 585f0dd5a9
Коммит 3178b0dbdf
1 изменённых файлов: 24 добавлений и 9 удалений

Просмотреть файл

@ -1503,17 +1503,14 @@ void bitmap_flush(mddev_t *mddev)
/* /*
* free memory that was allocated * free memory that was allocated
*/ */
void bitmap_destroy(mddev_t *mddev) static void bitmap_free(struct bitmap *bitmap)
{ {
unsigned long k, pages; unsigned long k, pages;
struct bitmap_page *bp; struct bitmap_page *bp;
struct bitmap *bitmap = mddev->bitmap;
if (!bitmap) /* there was no bitmap */ if (!bitmap) /* there was no bitmap */
return; return;
mddev->bitmap = NULL; /* disconnect from the md device */
/* release the bitmap file and kill the daemon */ /* release the bitmap file and kill the daemon */
bitmap_file_put(bitmap); bitmap_file_put(bitmap);
@ -1531,6 +1528,17 @@ void bitmap_destroy(mddev_t *mddev)
kfree(bp); kfree(bp);
kfree(bitmap); kfree(bitmap);
} }
void bitmap_destroy(mddev_t *mddev)
{
struct bitmap *bitmap = mddev->bitmap;
if (!bitmap) /* there was no bitmap */
return;
mddev->bitmap = NULL; /* disconnect from the md device */
bitmap_free(bitmap);
}
/* /*
* initialize the bitmap structure * initialize the bitmap structure
@ -1561,15 +1569,15 @@ int bitmap_create(mddev_t *mddev)
spin_lock_init(&bitmap->lock); spin_lock_init(&bitmap->lock);
bitmap->mddev = mddev; bitmap->mddev = mddev;
mddev->bitmap = bitmap;
spin_lock_init(&bitmap->write_lock); spin_lock_init(&bitmap->write_lock);
INIT_LIST_HEAD(&bitmap->complete_pages); INIT_LIST_HEAD(&bitmap->complete_pages);
init_waitqueue_head(&bitmap->write_wait); init_waitqueue_head(&bitmap->write_wait);
bitmap->write_pool = mempool_create(WRITE_POOL_SIZE, write_pool_alloc, bitmap->write_pool = mempool_create(WRITE_POOL_SIZE, write_pool_alloc,
write_pool_free, NULL); write_pool_free, NULL);
err = -ENOMEM;
if (!bitmap->write_pool) if (!bitmap->write_pool)
return -ENOMEM; goto error;
bitmap->file = file; bitmap->file = file;
bitmap->offset = mddev->bitmap_offset; bitmap->offset = mddev->bitmap_offset;
@ -1577,7 +1585,7 @@ int bitmap_create(mddev_t *mddev)
/* read superblock from bitmap file (this sets bitmap->chunksize) */ /* read superblock from bitmap file (this sets bitmap->chunksize) */
err = bitmap_read_sb(bitmap); err = bitmap_read_sb(bitmap);
if (err) if (err)
return err; goto error;
bitmap->chunkshift = find_first_bit(&bitmap->chunksize, bitmap->chunkshift = find_first_bit(&bitmap->chunksize,
sizeof(bitmap->chunksize)); sizeof(bitmap->chunksize));
@ -1601,8 +1609,9 @@ int bitmap_create(mddev_t *mddev)
#else #else
bitmap->bp = kmalloc(pages * sizeof(*bitmap->bp), GFP_KERNEL); bitmap->bp = kmalloc(pages * sizeof(*bitmap->bp), GFP_KERNEL);
#endif #endif
err = -ENOMEM;
if (!bitmap->bp) if (!bitmap->bp)
return -ENOMEM; goto error;
memset(bitmap->bp, 0, pages * sizeof(*bitmap->bp)); memset(bitmap->bp, 0, pages * sizeof(*bitmap->bp));
bitmap->flags |= BITMAP_ACTIVE; bitmap->flags |= BITMAP_ACTIVE;
@ -1617,16 +1626,22 @@ int bitmap_create(mddev_t *mddev)
err = bitmap_init_from_disk(bitmap, start); err = bitmap_init_from_disk(bitmap, start);
if (err) if (err)
return err; goto error;
printk(KERN_INFO "created bitmap (%lu pages) for device %s\n", printk(KERN_INFO "created bitmap (%lu pages) for device %s\n",
pages, bmname(bitmap)); pages, bmname(bitmap));
mddev->bitmap = bitmap;
/* kick off the bitmap daemons */ /* kick off the bitmap daemons */
err = bitmap_start_daemons(bitmap); err = bitmap_start_daemons(bitmap);
if (err) if (err)
return err; return err;
return bitmap_update_sb(bitmap); return bitmap_update_sb(bitmap);
error:
bitmap_free(bitmap);
return err;
} }
/* the bitmap API -- for raid personalities */ /* the bitmap API -- for raid personalities */