mirror of
https://git.kernel.org/pub/scm/linux/kernel/git/torvalds/linux.git
synced 2025-01-07 13:53:24 +00:00
xfs: refactor f_op->release handling
Currently f_op->release is split in not very obvious ways. Fix that by folding xfs_release into xfs_file_release. Signed-off-by: Christoph Hellwig <hch@lst.de> Reviewed-by: Darrick J. Wong <djwong@kernel.org> Signed-off-by: Chandan Babu R <chandanbabu@kernel.org>
This commit is contained in:
parent
6e13dbebd5
commit
5d3ca62611
@ -1177,10 +1177,75 @@ xfs_dir_open(
|
||||
|
||||
STATIC int
|
||||
xfs_file_release(
|
||||
struct inode *inode,
|
||||
struct file *filp)
|
||||
struct inode *inode,
|
||||
struct file *file)
|
||||
{
|
||||
return xfs_release(XFS_I(inode));
|
||||
struct xfs_inode *ip = XFS_I(inode);
|
||||
struct xfs_mount *mp = ip->i_mount;
|
||||
int error;
|
||||
|
||||
/* If this is a read-only mount, don't generate I/O */
|
||||
if (xfs_is_readonly(mp))
|
||||
return 0;
|
||||
|
||||
/*
|
||||
* If we previously truncated this file and removed old data in the
|
||||
* process, we want to initiate "early" writeout on the last close.
|
||||
* This is an attempt to combat the notorious NULL files problem which
|
||||
* is particularly noticeable from a truncate down, buffered (re-)write
|
||||
* (delalloc), followed by a crash. What we are effectively doing here
|
||||
* is significantly reducing the time window where we'd otherwise be
|
||||
* exposed to that problem.
|
||||
*/
|
||||
if (!xfs_is_shutdown(mp) &&
|
||||
xfs_iflags_test_and_clear(ip, XFS_ITRUNCATED)) {
|
||||
xfs_iflags_clear(ip, XFS_IDIRTY_RELEASE);
|
||||
if (ip->i_delayed_blks > 0) {
|
||||
error = filemap_flush(inode->i_mapping);
|
||||
if (error)
|
||||
return error;
|
||||
}
|
||||
}
|
||||
|
||||
/*
|
||||
* XFS aggressively preallocates post-EOF space to generate contiguous
|
||||
* allocations for writers that append to the end of the file and we
|
||||
* try to free these when an open file context is released.
|
||||
*
|
||||
* There is no point in freeing blocks here for open but unlinked files
|
||||
* as they will be taken care of by the inactivation path soon.
|
||||
*
|
||||
* If we can't get the iolock just skip truncating the blocks past EOF
|
||||
* because we could deadlock with the mmap_lock otherwise. We'll get
|
||||
* another chance to drop them once the last reference to the inode is
|
||||
* dropped, so we'll never leak blocks permanently.
|
||||
*/
|
||||
if (inode->i_nlink && xfs_ilock_nowait(ip, XFS_IOLOCK_EXCL)) {
|
||||
if (xfs_can_free_eofblocks(ip) &&
|
||||
!xfs_iflags_test(ip, XFS_IDIRTY_RELEASE)) {
|
||||
/*
|
||||
* Check if the inode is being opened, written and
|
||||
* closed frequently and we have delayed allocation
|
||||
* blocks outstanding (e.g. streaming writes from the
|
||||
* NFS server), truncating the blocks past EOF will
|
||||
* cause fragmentation to occur.
|
||||
*
|
||||
* In this case don't do the truncation, but we have to
|
||||
* be careful how we detect this case. Blocks beyond EOF
|
||||
* show up as i_delayed_blks even when the inode is
|
||||
* clean, so we need to truncate them away first before
|
||||
* checking for a dirty release. Hence on the first
|
||||
* dirty close we will still remove the speculative
|
||||
* allocation, but after that we will leave it in place.
|
||||
*/
|
||||
error = xfs_free_eofblocks(ip);
|
||||
if (!error && ip->i_delayed_blks)
|
||||
xfs_iflags_set(ip, XFS_IDIRTY_RELEASE);
|
||||
}
|
||||
xfs_iunlock(ip, XFS_IOLOCK_EXCL);
|
||||
}
|
||||
|
||||
return error;
|
||||
}
|
||||
|
||||
STATIC int
|
||||
|
@ -1079,85 +1079,6 @@ xfs_itruncate_extents_flags(
|
||||
return error;
|
||||
}
|
||||
|
||||
int
|
||||
xfs_release(
|
||||
xfs_inode_t *ip)
|
||||
{
|
||||
xfs_mount_t *mp = ip->i_mount;
|
||||
int error = 0;
|
||||
|
||||
/* If this is a read-only mount, don't do this (would generate I/O) */
|
||||
if (xfs_is_readonly(mp))
|
||||
return 0;
|
||||
|
||||
if (!xfs_is_shutdown(mp)) {
|
||||
int truncated;
|
||||
|
||||
/*
|
||||
* If we previously truncated this file and removed old data
|
||||
* in the process, we want to initiate "early" writeout on
|
||||
* the last close. This is an attempt to combat the notorious
|
||||
* NULL files problem which is particularly noticeable from a
|
||||
* truncate down, buffered (re-)write (delalloc), followed by
|
||||
* a crash. What we are effectively doing here is
|
||||
* significantly reducing the time window where we'd otherwise
|
||||
* be exposed to that problem.
|
||||
*/
|
||||
truncated = xfs_iflags_test_and_clear(ip, XFS_ITRUNCATED);
|
||||
if (truncated) {
|
||||
xfs_iflags_clear(ip, XFS_IDIRTY_RELEASE);
|
||||
if (ip->i_delayed_blks > 0) {
|
||||
error = filemap_flush(VFS_I(ip)->i_mapping);
|
||||
if (error)
|
||||
return error;
|
||||
}
|
||||
}
|
||||
}
|
||||
|
||||
if (VFS_I(ip)->i_nlink == 0)
|
||||
return 0;
|
||||
|
||||
/*
|
||||
* If we can't get the iolock just skip truncating the blocks past EOF
|
||||
* because we could deadlock with the mmap_lock otherwise. We'll get
|
||||
* another chance to drop them once the last reference to the inode is
|
||||
* dropped, so we'll never leak blocks permanently.
|
||||
*/
|
||||
if (!xfs_ilock_nowait(ip, XFS_IOLOCK_EXCL))
|
||||
return 0;
|
||||
|
||||
if (xfs_can_free_eofblocks(ip)) {
|
||||
/*
|
||||
* Check if the inode is being opened, written and closed
|
||||
* frequently and we have delayed allocation blocks outstanding
|
||||
* (e.g. streaming writes from the NFS server), truncating the
|
||||
* blocks past EOF will cause fragmentation to occur.
|
||||
*
|
||||
* In this case don't do the truncation, but we have to be
|
||||
* careful how we detect this case. Blocks beyond EOF show up as
|
||||
* i_delayed_blks even when the inode is clean, so we need to
|
||||
* truncate them away first before checking for a dirty release.
|
||||
* Hence on the first dirty close we will still remove the
|
||||
* speculative allocation, but after that we will leave it in
|
||||
* place.
|
||||
*/
|
||||
if (xfs_iflags_test(ip, XFS_IDIRTY_RELEASE))
|
||||
goto out_unlock;
|
||||
|
||||
error = xfs_free_eofblocks(ip);
|
||||
if (error)
|
||||
goto out_unlock;
|
||||
|
||||
/* delalloc blocks after truncation means it really is dirty */
|
||||
if (ip->i_delayed_blks)
|
||||
xfs_iflags_set(ip, XFS_IDIRTY_RELEASE);
|
||||
}
|
||||
|
||||
out_unlock:
|
||||
xfs_iunlock(ip, XFS_IOLOCK_EXCL);
|
||||
return error;
|
||||
}
|
||||
|
||||
/*
|
||||
* Mark all the buffers attached to this directory stale. In theory we should
|
||||
* never be freeing a directory with any blocks at all, but this covers the
|
||||
|
@ -513,7 +513,6 @@ enum layout_break_reason {
|
||||
#define XFS_INHERIT_GID(pip) \
|
||||
(xfs_has_grpid((pip)->i_mount) || (VFS_I(pip)->i_mode & S_ISGID))
|
||||
|
||||
int xfs_release(struct xfs_inode *ip);
|
||||
int xfs_inactive(struct xfs_inode *ip);
|
||||
int xfs_lookup(struct xfs_inode *dp, const struct xfs_name *name,
|
||||
struct xfs_inode **ipp, struct xfs_name *ci_name);
|
||||
|
Loading…
Reference in New Issue
Block a user