diff options
| -rw-r--r-- | fs/adfs/inode.c | 3 | ||||
| -rw-r--r-- | fs/ecryptfs/inode.c | 4 | ||||
| -rw-r--r-- | fs/gfs2/aops.c | 8 | ||||
| -rw-r--r-- | fs/gfs2/ops_inode.c | 5 | ||||
| -rw-r--r-- | fs/jffs2/fs.c | 4 | ||||
| -rw-r--r-- | fs/ocfs2/file.c | 8 | ||||
| -rw-r--r-- | fs/smbfs/inode.c | 2 | ||||
| -rw-r--r-- | fs/ubifs/file.c | 13 | ||||
| -rw-r--r-- | fs/ubifs/ubifs.h | 2 | ||||
| -rw-r--r-- | fs/ufs/truncate.c | 10 |
10 files changed, 37 insertions, 22 deletions
diff --git a/fs/adfs/inode.c b/fs/adfs/inode.c index 0f5e30978135..6f850b06ab62 100644 --- a/fs/adfs/inode.c +++ b/fs/adfs/inode.c | |||
| @@ -322,8 +322,9 @@ adfs_notify_change(struct dentry *dentry, struct iattr *attr) | |||
| 322 | if (error) | 322 | if (error) |
| 323 | goto out; | 323 | goto out; |
| 324 | 324 | ||
| 325 | /* XXX: this is missing some actual on-disk truncation.. */ | ||
| 325 | if (ia_valid & ATTR_SIZE) | 326 | if (ia_valid & ATTR_SIZE) |
| 326 | error = vmtruncate(inode, attr->ia_size); | 327 | error = simple_setsize(inode, attr->ia_size); |
| 327 | 328 | ||
| 328 | if (error) | 329 | if (error) |
| 329 | goto out; | 330 | goto out; |
diff --git a/fs/ecryptfs/inode.c b/fs/ecryptfs/inode.c index 65dee2f336ae..31ef5252f0fe 100644 --- a/fs/ecryptfs/inode.c +++ b/fs/ecryptfs/inode.c | |||
| @@ -805,7 +805,7 @@ static int truncate_upper(struct dentry *dentry, struct iattr *ia, | |||
| 805 | - (ia->ia_size & ~PAGE_CACHE_MASK)); | 805 | - (ia->ia_size & ~PAGE_CACHE_MASK)); |
| 806 | 806 | ||
| 807 | if (!(crypt_stat->flags & ECRYPTFS_ENCRYPTED)) { | 807 | if (!(crypt_stat->flags & ECRYPTFS_ENCRYPTED)) { |
| 808 | rc = vmtruncate(inode, ia->ia_size); | 808 | rc = simple_setsize(inode, ia->ia_size); |
| 809 | if (rc) | 809 | if (rc) |
| 810 | goto out; | 810 | goto out; |
| 811 | lower_ia->ia_size = ia->ia_size; | 811 | lower_ia->ia_size = ia->ia_size; |
| @@ -830,7 +830,7 @@ static int truncate_upper(struct dentry *dentry, struct iattr *ia, | |||
| 830 | goto out; | 830 | goto out; |
| 831 | } | 831 | } |
| 832 | } | 832 | } |
| 833 | vmtruncate(inode, ia->ia_size); | 833 | simple_setsize(inode, ia->ia_size); |
| 834 | rc = ecryptfs_write_inode_size_to_metadata(inode); | 834 | rc = ecryptfs_write_inode_size_to_metadata(inode); |
| 835 | if (rc) { | 835 | if (rc) { |
| 836 | printk(KERN_ERR "Problem with " | 836 | printk(KERN_ERR "Problem with " |
diff --git a/fs/gfs2/aops.c b/fs/gfs2/aops.c index a739a0a48067..9f8b52500d63 100644 --- a/fs/gfs2/aops.c +++ b/fs/gfs2/aops.c | |||
| @@ -700,8 +700,14 @@ out: | |||
| 700 | return 0; | 700 | return 0; |
| 701 | 701 | ||
| 702 | page_cache_release(page); | 702 | page_cache_release(page); |
| 703 | |||
| 704 | /* | ||
| 705 | * XXX(hch): the call below should probably be replaced with | ||
| 706 | * a call to the gfs2-specific truncate blocks helper to actually | ||
| 707 | * release disk blocks.. | ||
| 708 | */ | ||
| 703 | if (pos + len > ip->i_inode.i_size) | 709 | if (pos + len > ip->i_inode.i_size) |
| 704 | vmtruncate(&ip->i_inode, ip->i_inode.i_size); | 710 | simple_setsize(&ip->i_inode, ip->i_inode.i_size); |
| 705 | out_endtrans: | 711 | out_endtrans: |
| 706 | gfs2_trans_end(sdp); | 712 | gfs2_trans_end(sdp); |
| 707 | out_trans_fail: | 713 | out_trans_fail: |
diff --git a/fs/gfs2/ops_inode.c b/fs/gfs2/ops_inode.c index 4e64352d49de..98cdd05f3316 100644 --- a/fs/gfs2/ops_inode.c +++ b/fs/gfs2/ops_inode.c | |||
| @@ -1071,6 +1071,9 @@ int gfs2_permission(struct inode *inode, int mask) | |||
| 1071 | return error; | 1071 | return error; |
| 1072 | } | 1072 | } |
| 1073 | 1073 | ||
| 1074 | /* | ||
| 1075 | * XXX: should be changed to have proper ordering by opencoding simple_setsize | ||
| 1076 | */ | ||
| 1074 | static int setattr_size(struct inode *inode, struct iattr *attr) | 1077 | static int setattr_size(struct inode *inode, struct iattr *attr) |
| 1075 | { | 1078 | { |
| 1076 | struct gfs2_inode *ip = GFS2_I(inode); | 1079 | struct gfs2_inode *ip = GFS2_I(inode); |
| @@ -1081,7 +1084,7 @@ static int setattr_size(struct inode *inode, struct iattr *attr) | |||
| 1081 | error = gfs2_trans_begin(sdp, 0, sdp->sd_jdesc->jd_blocks); | 1084 | error = gfs2_trans_begin(sdp, 0, sdp->sd_jdesc->jd_blocks); |
| 1082 | if (error) | 1085 | if (error) |
| 1083 | return error; | 1086 | return error; |
| 1084 | error = vmtruncate(inode, attr->ia_size); | 1087 | error = simple_setsize(inode, attr->ia_size); |
| 1085 | gfs2_trans_end(sdp); | 1088 | gfs2_trans_end(sdp); |
| 1086 | if (error) | 1089 | if (error) |
| 1087 | return error; | 1090 | return error; |
diff --git a/fs/jffs2/fs.c b/fs/jffs2/fs.c index 86e0821fc989..8bc2c80ab159 100644 --- a/fs/jffs2/fs.c +++ b/fs/jffs2/fs.c | |||
| @@ -169,13 +169,13 @@ int jffs2_do_setattr (struct inode *inode, struct iattr *iattr) | |||
| 169 | mutex_unlock(&f->sem); | 169 | mutex_unlock(&f->sem); |
| 170 | jffs2_complete_reservation(c); | 170 | jffs2_complete_reservation(c); |
| 171 | 171 | ||
| 172 | /* We have to do the vmtruncate() without f->sem held, since | 172 | /* We have to do the simple_setsize() without f->sem held, since |
| 173 | some pages may be locked and waiting for it in readpage(). | 173 | some pages may be locked and waiting for it in readpage(). |
| 174 | We are protected from a simultaneous write() extending i_size | 174 | We are protected from a simultaneous write() extending i_size |
| 175 | back past iattr->ia_size, because do_truncate() holds the | 175 | back past iattr->ia_size, because do_truncate() holds the |
| 176 | generic inode semaphore. */ | 176 | generic inode semaphore. */ |
| 177 | if (ivalid & ATTR_SIZE && inode->i_size > iattr->ia_size) { | 177 | if (ivalid & ATTR_SIZE && inode->i_size > iattr->ia_size) { |
| 178 | vmtruncate(inode, iattr->ia_size); | 178 | simple_setsize(inode, iattr->ia_size); |
| 179 | inode->i_blocks = (inode->i_size + 511) >> 9; | 179 | inode->i_blocks = (inode->i_size + 511) >> 9; |
| 180 | } | 180 | } |
| 181 | 181 | ||
diff --git a/fs/ocfs2/file.c b/fs/ocfs2/file.c index 1c6220a8e072..6a13ea64c447 100644 --- a/fs/ocfs2/file.c +++ b/fs/ocfs2/file.c | |||
| @@ -1052,7 +1052,7 @@ int ocfs2_setattr(struct dentry *dentry, struct iattr *attr) | |||
| 1052 | } | 1052 | } |
| 1053 | 1053 | ||
| 1054 | /* | 1054 | /* |
| 1055 | * This will intentionally not wind up calling vmtruncate(), | 1055 | * This will intentionally not wind up calling simple_setsize(), |
| 1056 | * since all the work for a size change has been done above. | 1056 | * since all the work for a size change has been done above. |
| 1057 | * Otherwise, we could get into problems with truncate as | 1057 | * Otherwise, we could get into problems with truncate as |
| 1058 | * ip_alloc_sem is used there to protect against i_size | 1058 | * ip_alloc_sem is used there to protect against i_size |
| @@ -2118,9 +2118,13 @@ relock: | |||
| 2118 | * direct write may have instantiated a few | 2118 | * direct write may have instantiated a few |
| 2119 | * blocks outside i_size. Trim these off again. | 2119 | * blocks outside i_size. Trim these off again. |
| 2120 | * Don't need i_size_read because we hold i_mutex. | 2120 | * Don't need i_size_read because we hold i_mutex. |
| 2121 | * | ||
| 2122 | * XXX(hch): this looks buggy because ocfs2 did not | ||
| 2123 | * actually implement ->truncate. Take a look at | ||
| 2124 | * the new truncate sequence and update this accordingly | ||
| 2121 | */ | 2125 | */ |
| 2122 | if (*ppos + count > inode->i_size) | 2126 | if (*ppos + count > inode->i_size) |
| 2123 | vmtruncate(inode, inode->i_size); | 2127 | simple_setsize(inode, inode->i_size); |
| 2124 | ret = written; | 2128 | ret = written; |
| 2125 | goto out_dio; | 2129 | goto out_dio; |
| 2126 | } | 2130 | } |
diff --git a/fs/smbfs/inode.c b/fs/smbfs/inode.c index dfa1d67f8fca..9551cb6f7fe4 100644 --- a/fs/smbfs/inode.c +++ b/fs/smbfs/inode.c | |||
| @@ -714,7 +714,7 @@ smb_notify_change(struct dentry *dentry, struct iattr *attr) | |||
| 714 | error = server->ops->truncate(inode, attr->ia_size); | 714 | error = server->ops->truncate(inode, attr->ia_size); |
| 715 | if (error) | 715 | if (error) |
| 716 | goto out; | 716 | goto out; |
| 717 | error = vmtruncate(inode, attr->ia_size); | 717 | error = simple_setsize(inode, attr->ia_size); |
| 718 | if (error) | 718 | if (error) |
| 719 | goto out; | 719 | goto out; |
| 720 | refresh = 1; | 720 | refresh = 1; |
diff --git a/fs/ubifs/file.c b/fs/ubifs/file.c index c726da68e6be..12f445cee9f7 100644 --- a/fs/ubifs/file.c +++ b/fs/ubifs/file.c | |||
| @@ -967,12 +967,15 @@ static int do_writepage(struct page *page, int len) | |||
| 967 | * the page locked, and it locks @ui_mutex. However, write-back does take inode | 967 | * the page locked, and it locks @ui_mutex. However, write-back does take inode |
| 968 | * @i_mutex, which means other VFS operations may be run on this inode at the | 968 | * @i_mutex, which means other VFS operations may be run on this inode at the |
| 969 | * same time. And the problematic one is truncation to smaller size, from where | 969 | * same time. And the problematic one is truncation to smaller size, from where |
| 970 | * we have to call 'vmtruncate()', which first changes @inode->i_size, then | 970 | * we have to call 'simple_setsize()', which first changes @inode->i_size, then |
| 971 | * drops the truncated pages. And while dropping the pages, it takes the page | 971 | * drops the truncated pages. And while dropping the pages, it takes the page |
| 972 | * lock. This means that 'do_truncation()' cannot call 'vmtruncate()' with | 972 | * lock. This means that 'do_truncation()' cannot call 'simple_setsize()' with |
| 973 | * @ui_mutex locked, because it would deadlock with 'ubifs_writepage()'. This | 973 | * @ui_mutex locked, because it would deadlock with 'ubifs_writepage()'. This |
| 974 | * means that @inode->i_size is changed while @ui_mutex is unlocked. | 974 | * means that @inode->i_size is changed while @ui_mutex is unlocked. |
| 975 | * | 975 | * |
| 976 | * XXX: with the new truncate the above is not true anymore, the simple_setsize | ||
| 977 | * calls can be replaced with the individual components. | ||
| 978 | * | ||
| 976 | * But in 'ubifs_writepage()' we have to guarantee that we do not write beyond | 979 | * But in 'ubifs_writepage()' we have to guarantee that we do not write beyond |
| 977 | * inode size. How do we do this if @inode->i_size may became smaller while we | 980 | * inode size. How do we do this if @inode->i_size may became smaller while we |
| 978 | * are in the middle of 'ubifs_writepage()'? The UBIFS solution is the | 981 | * are in the middle of 'ubifs_writepage()'? The UBIFS solution is the |
| @@ -1125,7 +1128,7 @@ static int do_truncation(struct ubifs_info *c, struct inode *inode, | |||
| 1125 | budgeted = 0; | 1128 | budgeted = 0; |
| 1126 | } | 1129 | } |
| 1127 | 1130 | ||
| 1128 | err = vmtruncate(inode, new_size); | 1131 | err = simple_setsize(inode, new_size); |
| 1129 | if (err) | 1132 | if (err) |
| 1130 | goto out_budg; | 1133 | goto out_budg; |
| 1131 | 1134 | ||
| @@ -1214,7 +1217,7 @@ static int do_setattr(struct ubifs_info *c, struct inode *inode, | |||
| 1214 | 1217 | ||
| 1215 | if (attr->ia_valid & ATTR_SIZE) { | 1218 | if (attr->ia_valid & ATTR_SIZE) { |
| 1216 | dbg_gen("size %lld -> %lld", inode->i_size, new_size); | 1219 | dbg_gen("size %lld -> %lld", inode->i_size, new_size); |
| 1217 | err = vmtruncate(inode, new_size); | 1220 | err = simple_setsize(inode, new_size); |
| 1218 | if (err) | 1221 | if (err) |
| 1219 | goto out; | 1222 | goto out; |
| 1220 | } | 1223 | } |
| @@ -1223,7 +1226,7 @@ static int do_setattr(struct ubifs_info *c, struct inode *inode, | |||
| 1223 | if (attr->ia_valid & ATTR_SIZE) { | 1226 | if (attr->ia_valid & ATTR_SIZE) { |
| 1224 | /* Truncation changes inode [mc]time */ | 1227 | /* Truncation changes inode [mc]time */ |
| 1225 | inode->i_mtime = inode->i_ctime = ubifs_current_time(inode); | 1228 | inode->i_mtime = inode->i_ctime = ubifs_current_time(inode); |
| 1226 | /* 'vmtruncate()' changed @i_size, update @ui_size */ | 1229 | /* 'simple_setsize()' changed @i_size, update @ui_size */ |
| 1227 | ui->ui_size = inode->i_size; | 1230 | ui->ui_size = inode->i_size; |
| 1228 | } | 1231 | } |
| 1229 | 1232 | ||
diff --git a/fs/ubifs/ubifs.h b/fs/ubifs/ubifs.h index b0904536cc1c..2eef553d50c8 100644 --- a/fs/ubifs/ubifs.h +++ b/fs/ubifs/ubifs.h | |||
| @@ -379,7 +379,7 @@ struct ubifs_gced_idx_leb { | |||
| 379 | * The @ui_size is a "shadow" variable for @inode->i_size and UBIFS uses | 379 | * The @ui_size is a "shadow" variable for @inode->i_size and UBIFS uses |
| 380 | * @ui_size instead of @inode->i_size. The reason for this is that UBIFS cannot | 380 | * @ui_size instead of @inode->i_size. The reason for this is that UBIFS cannot |
| 381 | * make sure @inode->i_size is always changed under @ui_mutex, because it | 381 | * make sure @inode->i_size is always changed under @ui_mutex, because it |
| 382 | * cannot call 'vmtruncate()' with @ui_mutex locked, because it would deadlock | 382 | * cannot call 'simple_setsize()' with @ui_mutex locked, because it would deadlock |
| 383 | * with 'ubifs_writepage()' (see file.c). All the other inode fields are | 383 | * with 'ubifs_writepage()' (see file.c). All the other inode fields are |
| 384 | * changed under @ui_mutex, so they do not need "shadow" fields. Note, one | 384 | * changed under @ui_mutex, so they do not need "shadow" fields. Note, one |
| 385 | * could consider to rework locking and base it on "shadow" fields. | 385 | * could consider to rework locking and base it on "shadow" fields. |
diff --git a/fs/ufs/truncate.c b/fs/ufs/truncate.c index f294c44577dc..3733057e7833 100644 --- a/fs/ufs/truncate.c +++ b/fs/ufs/truncate.c | |||
| @@ -501,12 +501,10 @@ out: | |||
| 501 | return err; | 501 | return err; |
| 502 | } | 502 | } |
| 503 | 503 | ||
| 504 | |||
| 505 | /* | 504 | /* |
| 506 | * We don't define our `inode->i_op->truncate', and call it here, | 505 | * TODO: |
| 507 | * because of: | 506 | * - truncate case should use proper ordering instead of using |
| 508 | * - there is no way to know old size | 507 | * simple_setsize |
| 509 | * - there is no way inform user about error, if it happens in `truncate' | ||
| 510 | */ | 508 | */ |
| 511 | int ufs_setattr(struct dentry *dentry, struct iattr *attr) | 509 | int ufs_setattr(struct dentry *dentry, struct iattr *attr) |
| 512 | { | 510 | { |
| @@ -530,7 +528,7 @@ int ufs_setattr(struct dentry *dentry, struct iattr *attr) | |||
| 530 | if (ia_valid & ATTR_SIZE && attr->ia_size != inode->i_size) { | 528 | if (ia_valid & ATTR_SIZE && attr->ia_size != inode->i_size) { |
| 531 | loff_t old_i_size = inode->i_size; | 529 | loff_t old_i_size = inode->i_size; |
| 532 | 530 | ||
| 533 | error = vmtruncate(inode, attr->ia_size); | 531 | error = simple_setsize(inode, attr->ia_size); |
| 534 | if (error) | 532 | if (error) |
| 535 | return error; | 533 | return error; |
| 536 | error = ufs_truncate(inode, old_i_size); | 534 | error = ufs_truncate(inode, old_i_size); |
