diff options
| author | Chris Mason <chris.mason@oracle.com> | 2009-10-01 12:29:10 -0400 |
|---|---|---|
| committer | Chris Mason <chris.mason@oracle.com> | 2009-10-01 12:29:10 -0400 |
| commit | ab93dbecfba72bbc04b7036343d180aaff1b61a3 (patch) | |
| tree | 6bb523dcf9be0885b6fa9cb6cc9ac694ee089028 | |
| parent | 35d62a942db5ae03104929fe7397835b572c4bc4 (diff) | |
Btrfs: take i_mutex before generic_write_checks
btrfs_file_write was incorrectly calling generic_write_checks without
taking i_mutex. This lead to problems with racing around i_size when
doing O_APPEND writes.
The fix here is to move i_mutex higher.
Signed-off-by: Chris Mason <chris.mason@oracle.com>
| -rw-r--r-- | fs/btrfs/file.c | 25 |
1 files changed, 17 insertions, 8 deletions
diff --git a/fs/btrfs/file.c b/fs/btrfs/file.c index 1be96ba6f6bb..f155179877a6 100644 --- a/fs/btrfs/file.c +++ b/fs/btrfs/file.c | |||
| @@ -920,26 +920,35 @@ static ssize_t btrfs_file_write(struct file *file, const char __user *buf, | |||
| 920 | start_pos = pos; | 920 | start_pos = pos; |
| 921 | 921 | ||
| 922 | vfs_check_frozen(inode->i_sb, SB_FREEZE_WRITE); | 922 | vfs_check_frozen(inode->i_sb, SB_FREEZE_WRITE); |
| 923 | |||
| 924 | /* do the reserve before the mutex lock in case we have to do some | ||
| 925 | * flushing. We wouldn't deadlock, but this is more polite. | ||
| 926 | */ | ||
| 927 | err = btrfs_reserve_metadata_for_delalloc(root, inode, 1); | ||
| 928 | if (err) | ||
| 929 | goto out_nolock; | ||
| 930 | |||
| 931 | mutex_lock(&inode->i_mutex); | ||
| 932 | |||
| 923 | current->backing_dev_info = inode->i_mapping->backing_dev_info; | 933 | current->backing_dev_info = inode->i_mapping->backing_dev_info; |
| 924 | err = generic_write_checks(file, &pos, &count, S_ISBLK(inode->i_mode)); | 934 | err = generic_write_checks(file, &pos, &count, S_ISBLK(inode->i_mode)); |
| 925 | if (err) | 935 | if (err) |
| 926 | goto out_nolock; | 936 | goto out; |
| 937 | |||
| 927 | if (count == 0) | 938 | if (count == 0) |
| 928 | goto out_nolock; | 939 | goto out; |
| 929 | 940 | ||
| 930 | err = file_remove_suid(file); | 941 | err = file_remove_suid(file); |
| 931 | if (err) | 942 | if (err) |
| 932 | goto out_nolock; | 943 | goto out; |
| 933 | |||
| 934 | err = btrfs_reserve_metadata_for_delalloc(root, inode, 1); | ||
| 935 | if (err) | ||
| 936 | goto out_nolock; | ||
| 937 | 944 | ||
| 938 | file_update_time(file); | 945 | file_update_time(file); |
| 939 | 946 | ||
| 940 | pages = kmalloc(nrptrs * sizeof(struct page *), GFP_KERNEL); | 947 | pages = kmalloc(nrptrs * sizeof(struct page *), GFP_KERNEL); |
| 941 | 948 | ||
| 942 | mutex_lock(&inode->i_mutex); | 949 | /* generic_write_checks can change our pos */ |
| 950 | start_pos = pos; | ||
| 951 | |||
| 943 | BTRFS_I(inode)->sequence++; | 952 | BTRFS_I(inode)->sequence++; |
| 944 | first_index = pos >> PAGE_CACHE_SHIFT; | 953 | first_index = pos >> PAGE_CACHE_SHIFT; |
| 945 | last_index = (pos + count) >> PAGE_CACHE_SHIFT; | 954 | last_index = (pos + count) >> PAGE_CACHE_SHIFT; |
