kill spurious reference to vmtruncate
Lots of filesystems calls vmtruncate despite not implementing the old ->truncate method. Switch them to use simple_setsize and add some comments about the truncate code where it seems fitting. Signed-off-by: Christoph Hellwig <hch@lst.de> Signed-off-by: Nick Piggin <npiggin@suse.de> Signed-off-by: Al Viro <viro@zeniv.linux.org.uk>
This commit is contained in:
parent
7bb46a6734
commit
15c6fd9786
|
@ -322,8 +322,9 @@ adfs_notify_change(struct dentry *dentry, struct iattr *attr)
|
|||
if (error)
|
||||
goto out;
|
||||
|
||||
/* XXX: this is missing some actual on-disk truncation.. */
|
||||
if (ia_valid & ATTR_SIZE)
|
||||
error = vmtruncate(inode, attr->ia_size);
|
||||
error = simple_setsize(inode, attr->ia_size);
|
||||
|
||||
if (error)
|
||||
goto out;
|
||||
|
|
|
@ -805,7 +805,7 @@ static int truncate_upper(struct dentry *dentry, struct iattr *ia,
|
|||
- (ia->ia_size & ~PAGE_CACHE_MASK));
|
||||
|
||||
if (!(crypt_stat->flags & ECRYPTFS_ENCRYPTED)) {
|
||||
rc = vmtruncate(inode, ia->ia_size);
|
||||
rc = simple_setsize(inode, ia->ia_size);
|
||||
if (rc)
|
||||
goto out;
|
||||
lower_ia->ia_size = ia->ia_size;
|
||||
|
@ -830,7 +830,7 @@ static int truncate_upper(struct dentry *dentry, struct iattr *ia,
|
|||
goto out;
|
||||
}
|
||||
}
|
||||
vmtruncate(inode, ia->ia_size);
|
||||
simple_setsize(inode, ia->ia_size);
|
||||
rc = ecryptfs_write_inode_size_to_metadata(inode);
|
||||
if (rc) {
|
||||
printk(KERN_ERR "Problem with "
|
||||
|
|
|
@ -700,8 +700,14 @@ out:
|
|||
return 0;
|
||||
|
||||
page_cache_release(page);
|
||||
|
||||
/*
|
||||
* XXX(hch): the call below should probably be replaced with
|
||||
* a call to the gfs2-specific truncate blocks helper to actually
|
||||
* release disk blocks..
|
||||
*/
|
||||
if (pos + len > ip->i_inode.i_size)
|
||||
vmtruncate(&ip->i_inode, ip->i_inode.i_size);
|
||||
simple_setsize(&ip->i_inode, ip->i_inode.i_size);
|
||||
out_endtrans:
|
||||
gfs2_trans_end(sdp);
|
||||
out_trans_fail:
|
||||
|
|
|
@ -1071,6 +1071,9 @@ int gfs2_permission(struct inode *inode, int mask)
|
|||
return error;
|
||||
}
|
||||
|
||||
/*
|
||||
* XXX: should be changed to have proper ordering by opencoding simple_setsize
|
||||
*/
|
||||
static int setattr_size(struct inode *inode, struct iattr *attr)
|
||||
{
|
||||
struct gfs2_inode *ip = GFS2_I(inode);
|
||||
|
@ -1081,7 +1084,7 @@ static int setattr_size(struct inode *inode, struct iattr *attr)
|
|||
error = gfs2_trans_begin(sdp, 0, sdp->sd_jdesc->jd_blocks);
|
||||
if (error)
|
||||
return error;
|
||||
error = vmtruncate(inode, attr->ia_size);
|
||||
error = simple_setsize(inode, attr->ia_size);
|
||||
gfs2_trans_end(sdp);
|
||||
if (error)
|
||||
return error;
|
||||
|
|
|
@ -169,13 +169,13 @@ int jffs2_do_setattr (struct inode *inode, struct iattr *iattr)
|
|||
mutex_unlock(&f->sem);
|
||||
jffs2_complete_reservation(c);
|
||||
|
||||
/* We have to do the vmtruncate() without f->sem held, since
|
||||
/* We have to do the simple_setsize() without f->sem held, since
|
||||
some pages may be locked and waiting for it in readpage().
|
||||
We are protected from a simultaneous write() extending i_size
|
||||
back past iattr->ia_size, because do_truncate() holds the
|
||||
generic inode semaphore. */
|
||||
if (ivalid & ATTR_SIZE && inode->i_size > iattr->ia_size) {
|
||||
vmtruncate(inode, iattr->ia_size);
|
||||
simple_setsize(inode, iattr->ia_size);
|
||||
inode->i_blocks = (inode->i_size + 511) >> 9;
|
||||
}
|
||||
|
||||
|
|
|
@ -1052,7 +1052,7 @@ int ocfs2_setattr(struct dentry *dentry, struct iattr *attr)
|
|||
}
|
||||
|
||||
/*
|
||||
* This will intentionally not wind up calling vmtruncate(),
|
||||
* This will intentionally not wind up calling simple_setsize(),
|
||||
* since all the work for a size change has been done above.
|
||||
* Otherwise, we could get into problems with truncate as
|
||||
* ip_alloc_sem is used there to protect against i_size
|
||||
|
@ -2118,9 +2118,13 @@ relock:
|
|||
* direct write may have instantiated a few
|
||||
* blocks outside i_size. Trim these off again.
|
||||
* Don't need i_size_read because we hold i_mutex.
|
||||
*
|
||||
* XXX(hch): this looks buggy because ocfs2 did not
|
||||
* actually implement ->truncate. Take a look at
|
||||
* the new truncate sequence and update this accordingly
|
||||
*/
|
||||
if (*ppos + count > inode->i_size)
|
||||
vmtruncate(inode, inode->i_size);
|
||||
simple_setsize(inode, inode->i_size);
|
||||
ret = written;
|
||||
goto out_dio;
|
||||
}
|
||||
|
|
|
@ -714,7 +714,7 @@ smb_notify_change(struct dentry *dentry, struct iattr *attr)
|
|||
error = server->ops->truncate(inode, attr->ia_size);
|
||||
if (error)
|
||||
goto out;
|
||||
error = vmtruncate(inode, attr->ia_size);
|
||||
error = simple_setsize(inode, attr->ia_size);
|
||||
if (error)
|
||||
goto out;
|
||||
refresh = 1;
|
||||
|
|
|
@ -967,12 +967,15 @@ static int do_writepage(struct page *page, int len)
|
|||
* the page locked, and it locks @ui_mutex. However, write-back does take inode
|
||||
* @i_mutex, which means other VFS operations may be run on this inode at the
|
||||
* same time. And the problematic one is truncation to smaller size, from where
|
||||
* we have to call 'vmtruncate()', which first changes @inode->i_size, then
|
||||
* we have to call 'simple_setsize()', which first changes @inode->i_size, then
|
||||
* drops the truncated pages. And while dropping the pages, it takes the page
|
||||
* lock. This means that 'do_truncation()' cannot call 'vmtruncate()' with
|
||||
* lock. This means that 'do_truncation()' cannot call 'simple_setsize()' with
|
||||
* @ui_mutex locked, because it would deadlock with 'ubifs_writepage()'. This
|
||||
* means that @inode->i_size is changed while @ui_mutex is unlocked.
|
||||
*
|
||||
* XXX: with the new truncate the above is not true anymore, the simple_setsize
|
||||
* calls can be replaced with the individual components.
|
||||
*
|
||||
* But in 'ubifs_writepage()' we have to guarantee that we do not write beyond
|
||||
* inode size. How do we do this if @inode->i_size may became smaller while we
|
||||
* 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,
|
|||
budgeted = 0;
|
||||
}
|
||||
|
||||
err = vmtruncate(inode, new_size);
|
||||
err = simple_setsize(inode, new_size);
|
||||
if (err)
|
||||
goto out_budg;
|
||||
|
||||
|
@ -1214,7 +1217,7 @@ static int do_setattr(struct ubifs_info *c, struct inode *inode,
|
|||
|
||||
if (attr->ia_valid & ATTR_SIZE) {
|
||||
dbg_gen("size %lld -> %lld", inode->i_size, new_size);
|
||||
err = vmtruncate(inode, new_size);
|
||||
err = simple_setsize(inode, new_size);
|
||||
if (err)
|
||||
goto out;
|
||||
}
|
||||
|
@ -1223,7 +1226,7 @@ static int do_setattr(struct ubifs_info *c, struct inode *inode,
|
|||
if (attr->ia_valid & ATTR_SIZE) {
|
||||
/* Truncation changes inode [mc]time */
|
||||
inode->i_mtime = inode->i_ctime = ubifs_current_time(inode);
|
||||
/* 'vmtruncate()' changed @i_size, update @ui_size */
|
||||
/* 'simple_setsize()' changed @i_size, update @ui_size */
|
||||
ui->ui_size = inode->i_size;
|
||||
}
|
||||
|
||||
|
|
|
@ -379,7 +379,7 @@ struct ubifs_gced_idx_leb {
|
|||
* The @ui_size is a "shadow" variable for @inode->i_size and UBIFS uses
|
||||
* @ui_size instead of @inode->i_size. The reason for this is that UBIFS cannot
|
||||
* make sure @inode->i_size is always changed under @ui_mutex, because it
|
||||
* cannot call 'vmtruncate()' with @ui_mutex locked, because it would deadlock
|
||||
* cannot call 'simple_setsize()' with @ui_mutex locked, because it would deadlock
|
||||
* with 'ubifs_writepage()' (see file.c). All the other inode fields are
|
||||
* changed under @ui_mutex, so they do not need "shadow" fields. Note, one
|
||||
* could consider to rework locking and base it on "shadow" fields.
|
||||
|
|
|
@ -501,12 +501,10 @@ out:
|
|||
return err;
|
||||
}
|
||||
|
||||
|
||||
/*
|
||||
* We don't define our `inode->i_op->truncate', and call it here,
|
||||
* because of:
|
||||
* - there is no way to know old size
|
||||
* - there is no way inform user about error, if it happens in `truncate'
|
||||
* TODO:
|
||||
* - truncate case should use proper ordering instead of using
|
||||
* simple_setsize
|
||||
*/
|
||||
int ufs_setattr(struct dentry *dentry, struct iattr *attr)
|
||||
{
|
||||
|
@ -530,7 +528,7 @@ int ufs_setattr(struct dentry *dentry, struct iattr *attr)
|
|||
if (ia_valid & ATTR_SIZE && attr->ia_size != inode->i_size) {
|
||||
loff_t old_i_size = inode->i_size;
|
||||
|
||||
error = vmtruncate(inode, attr->ia_size);
|
||||
error = simple_setsize(inode, attr->ia_size);
|
||||
if (error)
|
||||
return error;
|
||||
error = ufs_truncate(inode, old_i_size);
|
||||
|
|
Loading…
Reference in New Issue