diff options
| author | Eric Sandeen <sandeen@redhat.com> | 2009-12-14 14:01:05 -0500 | 
|---|---|---|
| committer | Jan Kara <jack@suse.cz> | 2009-12-23 07:44:12 -0500 | 
| commit | 96d2a495c25d525873529b736cdb63ad502b101c (patch) | |
| tree | 7ceb418c1e26e682cc0c43d47efe4eb2cc639de9 | |
| parent | b8a052d01669977f224255b0f9f2737018171ddb (diff) | |
ext3: Replace lock/unlock_super() with an explicit lock for resizing
Use a separate lock to protect s_groups_count and the other block
group descriptors which get changed via an on-line resize operation,
so we can stop overloading the use of lock_super().
Port of ext4 commit 32ed5058ce90024efcd811254b4b1de0468099df by
Theodore Ts'o <tytso@mit.edu>.
CC: Theodore Ts'o <tytso@mit.edu>
Signed-off-by: Eric Sandeen <sandeen@redhat.com>
Signed-off-by: Jan Kara <jack@suse.cz>
| -rw-r--r-- | fs/ext3/resize.c | 35 | ||||
| -rw-r--r-- | fs/ext3/super.c | 1 | ||||
| -rw-r--r-- | include/linux/ext3_fs_sb.h | 1 | 
3 files changed, 20 insertions, 17 deletions
| diff --git a/fs/ext3/resize.c b/fs/ext3/resize.c index 5f83b6179178..54351ac7cef9 100644 --- a/fs/ext3/resize.c +++ b/fs/ext3/resize.c | |||
| @@ -209,7 +209,7 @@ static int setup_new_group_blocks(struct super_block *sb, | |||
| 209 | if (IS_ERR(handle)) | 209 | if (IS_ERR(handle)) | 
| 210 | return PTR_ERR(handle); | 210 | return PTR_ERR(handle); | 
| 211 | 211 | ||
| 212 | lock_super(sb); | 212 | mutex_lock(&sbi->s_resize_lock); | 
| 213 | if (input->group != sbi->s_groups_count) { | 213 | if (input->group != sbi->s_groups_count) { | 
| 214 | err = -EBUSY; | 214 | err = -EBUSY; | 
| 215 | goto exit_journal; | 215 | goto exit_journal; | 
| @@ -324,7 +324,7 @@ exit_bh: | |||
| 324 | brelse(bh); | 324 | brelse(bh); | 
| 325 | 325 | ||
| 326 | exit_journal: | 326 | exit_journal: | 
| 327 | unlock_super(sb); | 327 | mutex_unlock(&sbi->s_resize_lock); | 
| 328 | if ((err2 = ext3_journal_stop(handle)) && !err) | 328 | if ((err2 = ext3_journal_stop(handle)) && !err) | 
| 329 | err = err2; | 329 | err = err2; | 
| 330 | 330 | ||
| @@ -662,11 +662,12 @@ exit_free: | |||
| 662 | * important part is that the new block and inode counts are in the backup | 662 | * important part is that the new block and inode counts are in the backup | 
| 663 | * superblocks, and the location of the new group metadata in the GDT backups. | 663 | * superblocks, and the location of the new group metadata in the GDT backups. | 
| 664 | * | 664 | * | 
| 665 | * We do not need lock_super() for this, because these blocks are not | 665 | * We do not need take the s_resize_lock for this, because these | 
| 666 | * otherwise touched by the filesystem code when it is mounted. We don't | 666 | * blocks are not otherwise touched by the filesystem code when it is | 
| 667 | * need to worry about last changing from sbi->s_groups_count, because the | 667 | * mounted. We don't need to worry about last changing from | 
| 668 | * worst that can happen is that we do not copy the full number of backups | 668 | * sbi->s_groups_count, because the worst that can happen is that we | 
| 669 | * at this time. The resize which changed s_groups_count will backup again. | 669 | * do not copy the full number of backups at this time. The resize | 
| 670 | * which changed s_groups_count will backup again. | ||
| 670 | */ | 671 | */ | 
| 671 | static void update_backups(struct super_block *sb, | 672 | static void update_backups(struct super_block *sb, | 
| 672 | int blk_off, char *data, int size) | 673 | int blk_off, char *data, int size) | 
| @@ -825,7 +826,7 @@ int ext3_group_add(struct super_block *sb, struct ext3_new_group_data *input) | |||
| 825 | goto exit_put; | 826 | goto exit_put; | 
| 826 | } | 827 | } | 
| 827 | 828 | ||
| 828 | lock_super(sb); | 829 | mutex_lock(&sbi->s_resize_lock); | 
| 829 | if (input->group != sbi->s_groups_count) { | 830 | if (input->group != sbi->s_groups_count) { | 
| 830 | ext3_warning(sb, __func__, | 831 | ext3_warning(sb, __func__, | 
| 831 | "multiple resizers run on filesystem!"); | 832 | "multiple resizers run on filesystem!"); | 
| @@ -856,7 +857,7 @@ int ext3_group_add(struct super_block *sb, struct ext3_new_group_data *input) | |||
| 856 | /* | 857 | /* | 
| 857 | * OK, now we've set up the new group. Time to make it active. | 858 | * OK, now we've set up the new group. Time to make it active. | 
| 858 | * | 859 | * | 
| 859 | * Current kernels don't lock all allocations via lock_super(), | 860 | * We do not lock all allocations via s_resize_lock | 
| 860 | * so we have to be safe wrt. concurrent accesses the group | 861 | * so we have to be safe wrt. concurrent accesses the group | 
| 861 | * data. So we need to be careful to set all of the relevant | 862 | * data. So we need to be careful to set all of the relevant | 
| 862 | * group descriptor data etc. *before* we enable the group. | 863 | * group descriptor data etc. *before* we enable the group. | 
| @@ -900,12 +901,12 @@ int ext3_group_add(struct super_block *sb, struct ext3_new_group_data *input) | |||
| 900 | * | 901 | * | 
| 901 | * The precise rules we use are: | 902 | * The precise rules we use are: | 
| 902 | * | 903 | * | 
| 903 | * * Writers of s_groups_count *must* hold lock_super | 904 | * * Writers of s_groups_count *must* hold s_resize_lock | 
| 904 | * AND | 905 | * AND | 
| 905 | * * Writers must perform a smp_wmb() after updating all dependent | 906 | * * Writers must perform a smp_wmb() after updating all dependent | 
| 906 | * data and before modifying the groups count | 907 | * data and before modifying the groups count | 
| 907 | * | 908 | * | 
| 908 | * * Readers must hold lock_super() over the access | 909 | * * Readers must hold s_resize_lock over the access | 
| 909 | * OR | 910 | * OR | 
| 910 | * * Readers must perform an smp_rmb() after reading the groups count | 911 | * * Readers must perform an smp_rmb() after reading the groups count | 
| 911 | * and before reading any dependent data. | 912 | * and before reading any dependent data. | 
| @@ -936,7 +937,7 @@ int ext3_group_add(struct super_block *sb, struct ext3_new_group_data *input) | |||
| 936 | ext3_journal_dirty_metadata(handle, sbi->s_sbh); | 937 | ext3_journal_dirty_metadata(handle, sbi->s_sbh); | 
| 937 | 938 | ||
| 938 | exit_journal: | 939 | exit_journal: | 
| 939 | unlock_super(sb); | 940 | mutex_unlock(&sbi->s_resize_lock); | 
| 940 | if ((err2 = ext3_journal_stop(handle)) && !err) | 941 | if ((err2 = ext3_journal_stop(handle)) && !err) | 
| 941 | err = err2; | 942 | err = err2; | 
| 942 | if (!err) { | 943 | if (!err) { | 
| @@ -973,7 +974,7 @@ int ext3_group_extend(struct super_block *sb, struct ext3_super_block *es, | |||
| 973 | 974 | ||
| 974 | /* We don't need to worry about locking wrt other resizers just | 975 | /* We don't need to worry about locking wrt other resizers just | 
| 975 | * yet: we're going to revalidate es->s_blocks_count after | 976 | * yet: we're going to revalidate es->s_blocks_count after | 
| 976 | * taking lock_super() below. */ | 977 | * taking the s_resize_lock below. */ | 
| 977 | o_blocks_count = le32_to_cpu(es->s_blocks_count); | 978 | o_blocks_count = le32_to_cpu(es->s_blocks_count); | 
| 978 | o_groups_count = EXT3_SB(sb)->s_groups_count; | 979 | o_groups_count = EXT3_SB(sb)->s_groups_count; | 
| 979 | 980 | ||
| @@ -1045,11 +1046,11 @@ int ext3_group_extend(struct super_block *sb, struct ext3_super_block *es, | |||
| 1045 | goto exit_put; | 1046 | goto exit_put; | 
| 1046 | } | 1047 | } | 
| 1047 | 1048 | ||
| 1048 | lock_super(sb); | 1049 | mutex_lock(&EXT3_SB(sb)->s_resize_lock); | 
| 1049 | if (o_blocks_count != le32_to_cpu(es->s_blocks_count)) { | 1050 | if (o_blocks_count != le32_to_cpu(es->s_blocks_count)) { | 
| 1050 | ext3_warning(sb, __func__, | 1051 | ext3_warning(sb, __func__, | 
| 1051 | "multiple resizers run on filesystem!"); | 1052 | "multiple resizers run on filesystem!"); | 
| 1052 | unlock_super(sb); | 1053 | mutex_unlock(&EXT3_SB(sb)->s_resize_lock); | 
| 1053 | ext3_journal_stop(handle); | 1054 | ext3_journal_stop(handle); | 
| 1054 | err = -EBUSY; | 1055 | err = -EBUSY; | 
| 1055 | goto exit_put; | 1056 | goto exit_put; | 
| @@ -1059,13 +1060,13 @@ int ext3_group_extend(struct super_block *sb, struct ext3_super_block *es, | |||
| 1059 | EXT3_SB(sb)->s_sbh))) { | 1060 | EXT3_SB(sb)->s_sbh))) { | 
| 1060 | ext3_warning(sb, __func__, | 1061 | ext3_warning(sb, __func__, | 
| 1061 | "error %d on journal write access", err); | 1062 | "error %d on journal write access", err); | 
| 1062 | unlock_super(sb); | 1063 | mutex_unlock(&EXT3_SB(sb)->s_resize_lock); | 
| 1063 | ext3_journal_stop(handle); | 1064 | ext3_journal_stop(handle); | 
| 1064 | goto exit_put; | 1065 | goto exit_put; | 
| 1065 | } | 1066 | } | 
| 1066 | es->s_blocks_count = cpu_to_le32(o_blocks_count + add); | 1067 | es->s_blocks_count = cpu_to_le32(o_blocks_count + add); | 
| 1067 | ext3_journal_dirty_metadata(handle, EXT3_SB(sb)->s_sbh); | 1068 | ext3_journal_dirty_metadata(handle, EXT3_SB(sb)->s_sbh); | 
| 1068 | unlock_super(sb); | 1069 | mutex_unlock(&EXT3_SB(sb)->s_resize_lock); | 
| 1069 | ext3_debug("freeing blocks %lu through "E3FSBLK"\n", o_blocks_count, | 1070 | ext3_debug("freeing blocks %lu through "E3FSBLK"\n", o_blocks_count, | 
| 1070 | o_blocks_count + add); | 1071 | o_blocks_count + add); | 
| 1071 | ext3_free_blocks_sb(handle, sb, o_blocks_count, add, &freed_blocks); | 1072 | ext3_free_blocks_sb(handle, sb, o_blocks_count, add, &freed_blocks); | 
| diff --git a/fs/ext3/super.c b/fs/ext3/super.c index 97dd3828384c..afa2b569da10 100644 --- a/fs/ext3/super.c +++ b/fs/ext3/super.c | |||
| @@ -1929,6 +1929,7 @@ static int ext3_fill_super (struct super_block *sb, void *data, int silent) | |||
| 1929 | #endif | 1929 | #endif | 
| 1930 | INIT_LIST_HEAD(&sbi->s_orphan); /* unlinked but open files */ | 1930 | INIT_LIST_HEAD(&sbi->s_orphan); /* unlinked but open files */ | 
| 1931 | mutex_init(&sbi->s_orphan_lock); | 1931 | mutex_init(&sbi->s_orphan_lock); | 
| 1932 | mutex_init(&sbi->s_resize_lock); | ||
| 1932 | 1933 | ||
| 1933 | sb->s_root = NULL; | 1934 | sb->s_root = NULL; | 
| 1934 | 1935 | ||
| diff --git a/include/linux/ext3_fs_sb.h b/include/linux/ext3_fs_sb.h index dd61d83026a0..258088ab3c6b 100644 --- a/include/linux/ext3_fs_sb.h +++ b/include/linux/ext3_fs_sb.h | |||
| @@ -73,6 +73,7 @@ struct ext3_sb_info { | |||
| 73 | struct journal_s * s_journal; | 73 | struct journal_s * s_journal; | 
| 74 | struct list_head s_orphan; | 74 | struct list_head s_orphan; | 
| 75 | struct mutex s_orphan_lock; | 75 | struct mutex s_orphan_lock; | 
| 76 | struct mutex s_resize_lock; | ||
| 76 | unsigned long s_commit_interval; | 77 | unsigned long s_commit_interval; | 
| 77 | struct block_device *journal_bdev; | 78 | struct block_device *journal_bdev; | 
| 78 | #ifdef CONFIG_JBD_DEBUG | 79 | #ifdef CONFIG_JBD_DEBUG | 
