diff options
Diffstat (limited to 'fs/btrfs/file.c')
| -rw-r--r-- | fs/btrfs/file.c | 81 |
1 files changed, 55 insertions, 26 deletions
diff --git a/fs/btrfs/file.c b/fs/btrfs/file.c index 571ad3c13b47..06550affbd27 100644 --- a/fs/btrfs/file.c +++ b/fs/btrfs/file.c | |||
| @@ -123,7 +123,10 @@ static noinline int dirty_and_release_pages(struct btrfs_trans_handle *trans, | |||
| 123 | root->sectorsize - 1) & ~((u64)root->sectorsize - 1); | 123 | root->sectorsize - 1) & ~((u64)root->sectorsize - 1); |
| 124 | 124 | ||
| 125 | end_of_last_block = start_pos + num_bytes - 1; | 125 | end_of_last_block = start_pos + num_bytes - 1; |
| 126 | btrfs_set_extent_delalloc(inode, start_pos, end_of_last_block); | 126 | err = btrfs_set_extent_delalloc(inode, start_pos, end_of_last_block); |
| 127 | if (err) | ||
| 128 | return err; | ||
| 129 | |||
| 127 | for (i = 0; i < num_pages; i++) { | 130 | for (i = 0; i < num_pages; i++) { |
| 128 | struct page *p = pages[i]; | 131 | struct page *p = pages[i]; |
| 129 | SetPageUptodate(p); | 132 | SetPageUptodate(p); |
| @@ -875,7 +878,8 @@ again: | |||
| 875 | btrfs_put_ordered_extent(ordered); | 878 | btrfs_put_ordered_extent(ordered); |
| 876 | 879 | ||
| 877 | clear_extent_bits(&BTRFS_I(inode)->io_tree, start_pos, | 880 | clear_extent_bits(&BTRFS_I(inode)->io_tree, start_pos, |
| 878 | last_pos - 1, EXTENT_DIRTY | EXTENT_DELALLOC, | 881 | last_pos - 1, EXTENT_DIRTY | EXTENT_DELALLOC | |
| 882 | EXTENT_DO_ACCOUNTING, | ||
| 879 | GFP_NOFS); | 883 | GFP_NOFS); |
| 880 | unlock_extent(&BTRFS_I(inode)->io_tree, | 884 | unlock_extent(&BTRFS_I(inode)->io_tree, |
| 881 | start_pos, last_pos - 1, GFP_NOFS); | 885 | start_pos, last_pos - 1, GFP_NOFS); |
| @@ -917,21 +921,35 @@ static ssize_t btrfs_file_write(struct file *file, const char __user *buf, | |||
| 917 | start_pos = pos; | 921 | start_pos = pos; |
| 918 | 922 | ||
| 919 | vfs_check_frozen(inode->i_sb, SB_FREEZE_WRITE); | 923 | vfs_check_frozen(inode->i_sb, SB_FREEZE_WRITE); |
| 924 | |||
| 925 | /* do the reserve before the mutex lock in case we have to do some | ||
| 926 | * flushing. We wouldn't deadlock, but this is more polite. | ||
| 927 | */ | ||
| 928 | err = btrfs_reserve_metadata_for_delalloc(root, inode, 1); | ||
| 929 | if (err) | ||
| 930 | goto out_nolock; | ||
| 931 | |||
| 932 | mutex_lock(&inode->i_mutex); | ||
| 933 | |||
| 920 | current->backing_dev_info = inode->i_mapping->backing_dev_info; | 934 | current->backing_dev_info = inode->i_mapping->backing_dev_info; |
| 921 | err = generic_write_checks(file, &pos, &count, S_ISBLK(inode->i_mode)); | 935 | err = generic_write_checks(file, &pos, &count, S_ISBLK(inode->i_mode)); |
| 922 | if (err) | 936 | if (err) |
| 923 | goto out_nolock; | 937 | goto out; |
| 938 | |||
| 924 | if (count == 0) | 939 | if (count == 0) |
| 925 | goto out_nolock; | 940 | goto out; |
| 926 | 941 | ||
| 927 | err = file_remove_suid(file); | 942 | err = file_remove_suid(file); |
| 928 | if (err) | 943 | if (err) |
| 929 | goto out_nolock; | 944 | goto out; |
| 945 | |||
| 930 | file_update_time(file); | 946 | file_update_time(file); |
| 931 | 947 | ||
| 932 | pages = kmalloc(nrptrs * sizeof(struct page *), GFP_KERNEL); | 948 | pages = kmalloc(nrptrs * sizeof(struct page *), GFP_KERNEL); |
| 933 | 949 | ||
| 934 | mutex_lock(&inode->i_mutex); | 950 | /* generic_write_checks can change our pos */ |
| 951 | start_pos = pos; | ||
| 952 | |||
| 935 | BTRFS_I(inode)->sequence++; | 953 | BTRFS_I(inode)->sequence++; |
| 936 | first_index = pos >> PAGE_CACHE_SHIFT; | 954 | first_index = pos >> PAGE_CACHE_SHIFT; |
| 937 | last_index = (pos + count) >> PAGE_CACHE_SHIFT; | 955 | last_index = (pos + count) >> PAGE_CACHE_SHIFT; |
| @@ -1005,9 +1023,8 @@ static ssize_t btrfs_file_write(struct file *file, const char __user *buf, | |||
| 1005 | } | 1023 | } |
| 1006 | 1024 | ||
| 1007 | if (will_write) { | 1025 | if (will_write) { |
| 1008 | btrfs_fdatawrite_range(inode->i_mapping, pos, | 1026 | filemap_fdatawrite_range(inode->i_mapping, pos, |
| 1009 | pos + write_bytes - 1, | 1027 | pos + write_bytes - 1); |
| 1010 | WB_SYNC_ALL); | ||
| 1011 | } else { | 1028 | } else { |
| 1012 | balance_dirty_pages_ratelimited_nr(inode->i_mapping, | 1029 | balance_dirty_pages_ratelimited_nr(inode->i_mapping, |
| 1013 | num_pages); | 1030 | num_pages); |
| @@ -1028,6 +1045,7 @@ out: | |||
| 1028 | mutex_unlock(&inode->i_mutex); | 1045 | mutex_unlock(&inode->i_mutex); |
| 1029 | if (ret) | 1046 | if (ret) |
| 1030 | err = ret; | 1047 | err = ret; |
| 1048 | btrfs_unreserve_metadata_for_delalloc(root, inode, 1); | ||
| 1031 | 1049 | ||
| 1032 | out_nolock: | 1050 | out_nolock: |
| 1033 | kfree(pages); | 1051 | kfree(pages); |
| @@ -1068,8 +1086,10 @@ out_nolock: | |||
| 1068 | btrfs_end_transaction(trans, root); | 1086 | btrfs_end_transaction(trans, root); |
| 1069 | else | 1087 | else |
| 1070 | btrfs_commit_transaction(trans, root); | 1088 | btrfs_commit_transaction(trans, root); |
| 1071 | } else { | 1089 | } else if (ret != BTRFS_NO_LOG_SYNC) { |
| 1072 | btrfs_commit_transaction(trans, root); | 1090 | btrfs_commit_transaction(trans, root); |
| 1091 | } else { | ||
| 1092 | btrfs_end_transaction(trans, root); | ||
| 1073 | } | 1093 | } |
| 1074 | } | 1094 | } |
| 1075 | if (file->f_flags & O_DIRECT) { | 1095 | if (file->f_flags & O_DIRECT) { |
| @@ -1119,6 +1139,13 @@ int btrfs_sync_file(struct file *file, struct dentry *dentry, int datasync) | |||
| 1119 | int ret = 0; | 1139 | int ret = 0; |
| 1120 | struct btrfs_trans_handle *trans; | 1140 | struct btrfs_trans_handle *trans; |
| 1121 | 1141 | ||
| 1142 | |||
| 1143 | /* we wait first, since the writeback may change the inode */ | ||
| 1144 | root->log_batch++; | ||
| 1145 | /* the VFS called filemap_fdatawrite for us */ | ||
| 1146 | btrfs_wait_ordered_range(inode, 0, (u64)-1); | ||
| 1147 | root->log_batch++; | ||
| 1148 | |||
| 1122 | /* | 1149 | /* |
| 1123 | * check the transaction that last modified this inode | 1150 | * check the transaction that last modified this inode |
| 1124 | * and see if its already been committed | 1151 | * and see if its already been committed |
| @@ -1126,6 +1153,11 @@ int btrfs_sync_file(struct file *file, struct dentry *dentry, int datasync) | |||
| 1126 | if (!BTRFS_I(inode)->last_trans) | 1153 | if (!BTRFS_I(inode)->last_trans) |
| 1127 | goto out; | 1154 | goto out; |
| 1128 | 1155 | ||
| 1156 | /* | ||
| 1157 | * if the last transaction that changed this file was before | ||
| 1158 | * the current transaction, we can bail out now without any | ||
| 1159 | * syncing | ||
| 1160 | */ | ||
| 1129 | mutex_lock(&root->fs_info->trans_mutex); | 1161 | mutex_lock(&root->fs_info->trans_mutex); |
| 1130 | if (BTRFS_I(inode)->last_trans <= | 1162 | if (BTRFS_I(inode)->last_trans <= |
| 1131 | root->fs_info->last_trans_committed) { | 1163 | root->fs_info->last_trans_committed) { |
| @@ -1135,13 +1167,6 @@ int btrfs_sync_file(struct file *file, struct dentry *dentry, int datasync) | |||
| 1135 | } | 1167 | } |
| 1136 | mutex_unlock(&root->fs_info->trans_mutex); | 1168 | mutex_unlock(&root->fs_info->trans_mutex); |
| 1137 | 1169 | ||
| 1138 | root->log_batch++; | ||
| 1139 | filemap_fdatawrite(inode->i_mapping); | ||
| 1140 | btrfs_wait_ordered_range(inode, 0, (u64)-1); | ||
| 1141 | root->log_batch++; | ||
| 1142 | |||
| 1143 | if (datasync && !(inode->i_state & I_DIRTY_PAGES)) | ||
| 1144 | goto out; | ||
| 1145 | /* | 1170 | /* |
| 1146 | * ok we haven't committed the transaction yet, lets do a commit | 1171 | * ok we haven't committed the transaction yet, lets do a commit |
| 1147 | */ | 1172 | */ |
| @@ -1170,21 +1195,25 @@ int btrfs_sync_file(struct file *file, struct dentry *dentry, int datasync) | |||
| 1170 | */ | 1195 | */ |
| 1171 | mutex_unlock(&dentry->d_inode->i_mutex); | 1196 | mutex_unlock(&dentry->d_inode->i_mutex); |
| 1172 | 1197 | ||
| 1173 | if (ret > 0) { | 1198 | if (ret != BTRFS_NO_LOG_SYNC) { |
| 1174 | ret = btrfs_commit_transaction(trans, root); | 1199 | if (ret > 0) { |
| 1175 | } else { | ||
| 1176 | ret = btrfs_sync_log(trans, root); | ||
| 1177 | if (ret == 0) | ||
| 1178 | ret = btrfs_end_transaction(trans, root); | ||
| 1179 | else | ||
| 1180 | ret = btrfs_commit_transaction(trans, root); | 1200 | ret = btrfs_commit_transaction(trans, root); |
| 1201 | } else { | ||
| 1202 | ret = btrfs_sync_log(trans, root); | ||
| 1203 | if (ret == 0) | ||
| 1204 | ret = btrfs_end_transaction(trans, root); | ||
| 1205 | else | ||
| 1206 | ret = btrfs_commit_transaction(trans, root); | ||
| 1207 | } | ||
| 1208 | } else { | ||
| 1209 | ret = btrfs_end_transaction(trans, root); | ||
| 1181 | } | 1210 | } |
| 1182 | mutex_lock(&dentry->d_inode->i_mutex); | 1211 | mutex_lock(&dentry->d_inode->i_mutex); |
| 1183 | out: | 1212 | out: |
| 1184 | return ret > 0 ? EIO : ret; | 1213 | return ret > 0 ? EIO : ret; |
| 1185 | } | 1214 | } |
| 1186 | 1215 | ||
| 1187 | static struct vm_operations_struct btrfs_file_vm_ops = { | 1216 | static const struct vm_operations_struct btrfs_file_vm_ops = { |
| 1188 | .fault = filemap_fault, | 1217 | .fault = filemap_fault, |
| 1189 | .page_mkwrite = btrfs_page_mkwrite, | 1218 | .page_mkwrite = btrfs_page_mkwrite, |
| 1190 | }; | 1219 | }; |
| @@ -1196,7 +1225,7 @@ static int btrfs_file_mmap(struct file *filp, struct vm_area_struct *vma) | |||
| 1196 | return 0; | 1225 | return 0; |
| 1197 | } | 1226 | } |
| 1198 | 1227 | ||
| 1199 | struct file_operations btrfs_file_operations = { | 1228 | const struct file_operations btrfs_file_operations = { |
| 1200 | .llseek = generic_file_llseek, | 1229 | .llseek = generic_file_llseek, |
| 1201 | .read = do_sync_read, | 1230 | .read = do_sync_read, |
| 1202 | .aio_read = generic_file_aio_read, | 1231 | .aio_read = generic_file_aio_read, |
