提交 8f04c47a 编写于 作者: C Christoph Hellwig

xfs: split xfs_itruncate_finish

Split the guts of xfs_itruncate_finish that loop over the existing extents
and calls xfs_bunmapi on them into a new helper, xfs_itruncate_externs.
Make xfs_attr_inactive call it directly instead of xfs_itruncate_finish,
which allows to simplify the latter a lot, by only letting it deal with
the data fork.  As a result xfs_itruncate_finish is renamed to
xfs_itruncate_data to make its use case more obvious.

Also remove the sync parameter from xfs_itruncate_data, which has been
unessecary since the introduction of the busy extent list in 2002, and
completely dead code since 2003 when the XFS_BMAPI_ASYNC parameter was
made a no-op.

I can't actually see why the xfs_attr_inactive needs to set the transaction
sync, but let's keep this patch simple and without changes in behaviour.

Also avoid passing a useless argument to xfs_isize_check, and make it
private to xfs_inode.c.
Signed-off-by: NChristoph Hellwig <hch@lst.de>
Reviewed-by: NAlex Elder <aelder@sgi.com>
Reviewed-by: NDave Chinner <dchinner@redhat.com>
上级 857b9778
...@@ -879,15 +879,7 @@ xfs_setattr_size( ...@@ -879,15 +879,7 @@ xfs_setattr_size(
ip->i_size = iattr->ia_size; ip->i_size = iattr->ia_size;
} else if (iattr->ia_size <= ip->i_size || } else if (iattr->ia_size <= ip->i_size ||
(iattr->ia_size == 0 && ip->i_d.di_nextents)) { (iattr->ia_size == 0 && ip->i_d.di_nextents)) {
/* error = xfs_itruncate_data(&tp, ip, iattr->ia_size);
* Signal a sync transaction unless we are truncating an
* already unlinked file on a wsync filesystem.
*/
error = xfs_itruncate_finish(&tp, ip, iattr->ia_size,
XFS_DATA_FORK,
((ip->i_d.di_nlink != 0 ||
!(mp->m_flags & XFS_MOUNT_WSYNC))
? 1 : 0));
if (error) if (error)
goto out_trans_abort; goto out_trans_abort;
......
...@@ -1055,8 +1055,8 @@ DECLARE_EVENT_CLASS(xfs_itrunc_class, ...@@ -1055,8 +1055,8 @@ DECLARE_EVENT_CLASS(xfs_itrunc_class,
DEFINE_EVENT(xfs_itrunc_class, name, \ DEFINE_EVENT(xfs_itrunc_class, name, \
TP_PROTO(struct xfs_inode *ip, xfs_fsize_t new_size), \ TP_PROTO(struct xfs_inode *ip, xfs_fsize_t new_size), \
TP_ARGS(ip, new_size)) TP_ARGS(ip, new_size))
DEFINE_ITRUNC_EVENT(xfs_itruncate_finish_start); DEFINE_ITRUNC_EVENT(xfs_itruncate_data_start);
DEFINE_ITRUNC_EVENT(xfs_itruncate_finish_end); DEFINE_ITRUNC_EVENT(xfs_itruncate_data_end);
TRACE_EVENT(xfs_pagecache_inval, TRACE_EVENT(xfs_pagecache_inval,
TP_PROTO(struct xfs_inode *ip, xfs_off_t start, xfs_off_t finish), TP_PROTO(struct xfs_inode *ip, xfs_off_t start, xfs_off_t finish),
......
...@@ -263,7 +263,7 @@ xfs_qm_scall_trunc_qfile( ...@@ -263,7 +263,7 @@ xfs_qm_scall_trunc_qfile(
xfs_ilock(ip, XFS_ILOCK_EXCL); xfs_ilock(ip, XFS_ILOCK_EXCL);
xfs_trans_ijoin(tp, ip); xfs_trans_ijoin(tp, ip);
error = xfs_itruncate_finish(&tp, ip, 0, XFS_DATA_FORK, 1); error = xfs_itruncate_data(&tp, ip, 0);
if (error) { if (error) {
xfs_trans_cancel(tp, XFS_TRANS_RELEASE_LOG_RES | xfs_trans_cancel(tp, XFS_TRANS_RELEASE_LOG_RES |
XFS_TRANS_ABORT); XFS_TRANS_ABORT);
......
...@@ -822,17 +822,21 @@ xfs_attr_inactive(xfs_inode_t *dp) ...@@ -822,17 +822,21 @@ xfs_attr_inactive(xfs_inode_t *dp)
error = xfs_attr_root_inactive(&trans, dp); error = xfs_attr_root_inactive(&trans, dp);
if (error) if (error)
goto out; goto out;
/* /*
* signal synchronous inactive transactions unless this * Signal synchronous inactive transactions unless this is a
* is a synchronous mount filesystem in which case we * synchronous mount filesystem in which case we know that we're here
* know that we're here because we've been called out of * because we've been called out of xfs_inactive which means that the
* xfs_inactive which means that the last reference is gone * last reference is gone and the unlink transaction has already hit
* and the unlink transaction has already hit the disk so * the disk so async inactive transactions are safe.
* async inactive transactions are safe.
*/ */
if ((error = xfs_itruncate_finish(&trans, dp, 0LL, XFS_ATTR_FORK, if (!(mp->m_flags & XFS_MOUNT_WSYNC)) {
(!(mp->m_flags & XFS_MOUNT_WSYNC) if (dp->i_d.di_anextents > 0)
? 1 : 0)))) xfs_trans_set_sync(trans);
}
error = xfs_itruncate_extents(&trans, dp, XFS_ATTR_FORK, 0);
if (error)
goto out; goto out;
/* /*
......
...@@ -52,7 +52,7 @@ kmem_zone_t *xfs_ifork_zone; ...@@ -52,7 +52,7 @@ kmem_zone_t *xfs_ifork_zone;
kmem_zone_t *xfs_inode_zone; kmem_zone_t *xfs_inode_zone;
/* /*
* Used in xfs_itruncate(). This is the maximum number of extents * Used in xfs_itruncate_extents(). This is the maximum number of extents
* freed from a file in a single transaction. * freed from a file in a single transaction.
*/ */
#define XFS_ITRUNC_MAX_EXTENTS 2 #define XFS_ITRUNC_MAX_EXTENTS 2
...@@ -1179,12 +1179,12 @@ xfs_ialloc( ...@@ -1179,12 +1179,12 @@ xfs_ialloc(
* at least do it for regular files. * at least do it for regular files.
*/ */
#ifdef DEBUG #ifdef DEBUG
void STATIC void
xfs_isize_check( xfs_isize_check(
xfs_mount_t *mp, struct xfs_inode *ip,
xfs_inode_t *ip,
xfs_fsize_t isize) xfs_fsize_t isize)
{ {
struct xfs_mount *mp = ip->i_mount;
xfs_fileoff_t map_first; xfs_fileoff_t map_first;
int nimaps; int nimaps;
xfs_bmbt_irec_t imaps[2]; xfs_bmbt_irec_t imaps[2];
...@@ -1214,11 +1214,14 @@ xfs_isize_check( ...@@ -1214,11 +1214,14 @@ xfs_isize_check(
ASSERT(nimaps == 1); ASSERT(nimaps == 1);
ASSERT(imaps[0].br_startblock == HOLESTARTBLOCK); ASSERT(imaps[0].br_startblock == HOLESTARTBLOCK);
} }
#else /* DEBUG */
#define xfs_isize_check(ip, isize)
#endif /* DEBUG */ #endif /* DEBUG */
/* /*
* Free up the underlying blocks past new_size. The new size must be * Free up the underlying blocks past new_size. The new size must be smaller
* smaller than the current size. * than the current size. This routine can be used both for the attribute and
* data fork, and does not modify the inode size, which is left to the caller.
* *
* The transaction passed to this routine must have made a permanent log * The transaction passed to this routine must have made a permanent log
* reservation of at least XFS_ITRUNCATE_LOG_RES. This routine may commit the * reservation of at least XFS_ITRUNCATE_LOG_RES. This routine may commit the
...@@ -1230,31 +1233,6 @@ xfs_isize_check( ...@@ -1230,31 +1233,6 @@ xfs_isize_check(
* will be "held" within the returned transaction. This routine does NOT * will be "held" within the returned transaction. This routine does NOT
* require any disk space to be reserved for it within the transaction. * require any disk space to be reserved for it within the transaction.
* *
* The fork parameter must be either XFS_ATTR_FORK or XFS_DATA_FORK, and it
* indicates the fork which is to be truncated. For the attribute fork we only
* support truncation to size 0.
*
* We use the sync parameter to indicate whether or not the first transaction
* we perform might have to be synchronous. For the attr fork, it needs to be
* so if the unlink of the inode is not yet known to be permanent in the log.
* This keeps us from freeing and reusing the blocks of the attribute fork
* before the unlink of the inode becomes permanent.
*
* For the data fork, we normally have to run synchronously if we're being
* called out of the inactive path or we're being called out of the create path
* where we're truncating an existing file. Either way, the truncate needs to
* be sync so blocks don't reappear in the file with altered data in case of a
* crash. wsync filesystems can run the first case async because anything that
* shrinks the inode has to run sync so by the time we're called here from
* inactive, the inode size is permanently set to 0.
*
* Calls from the truncate path always need to be sync unless we're in a wsync
* filesystem and the file has already been unlinked.
*
* The caller is responsible for correctly setting the sync parameter. It gets
* too hard for us to guess here which path we're being called out of just
* based on inode state.
*
* If we get an error, we must return with the inode locked and linked into the * If we get an error, we must return with the inode locked and linked into the
* current transaction. This keeps things simple for the higher level code, * current transaction. This keeps things simple for the higher level code,
* because it always knows that the inode is locked and held in the transaction * because it always knows that the inode is locked and held in the transaction
...@@ -1262,124 +1240,31 @@ xfs_isize_check( ...@@ -1262,124 +1240,31 @@ xfs_isize_check(
* dirty on error so that transactions can be easily aborted if possible. * dirty on error so that transactions can be easily aborted if possible.
*/ */
int int
xfs_itruncate_finish( xfs_itruncate_extents(
xfs_trans_t **tp, struct xfs_trans **tpp,
xfs_inode_t *ip, struct xfs_inode *ip,
xfs_fsize_t new_size, int whichfork,
int fork, xfs_fsize_t new_size)
int sync)
{ {
struct xfs_mount *mp = ip->i_mount;
struct xfs_trans *tp = *tpp;
struct xfs_trans *ntp;
xfs_bmap_free_t free_list;
xfs_fsblock_t first_block; xfs_fsblock_t first_block;
xfs_fileoff_t first_unmap_block; xfs_fileoff_t first_unmap_block;
xfs_fileoff_t last_block; xfs_fileoff_t last_block;
xfs_filblks_t unmap_len=0; xfs_filblks_t unmap_len;
xfs_mount_t *mp;
xfs_trans_t *ntp;
int done;
int committed; int committed;
xfs_bmap_free_t free_list; int error = 0;
int error; int done = 0;
ASSERT(xfs_isilocked(ip, XFS_ILOCK_EXCL|XFS_IOLOCK_EXCL)); ASSERT(xfs_isilocked(ip, XFS_ILOCK_EXCL|XFS_IOLOCK_EXCL));
ASSERT((new_size == 0) || (new_size <= ip->i_size)); ASSERT(new_size <= ip->i_size);
ASSERT(*tp != NULL); ASSERT(tp->t_flags & XFS_TRANS_PERM_LOG_RES);
ASSERT((*tp)->t_flags & XFS_TRANS_PERM_LOG_RES); ASSERT(ip->i_transp == tp);
ASSERT(ip->i_transp == *tp);
ASSERT(ip->i_itemp != NULL); ASSERT(ip->i_itemp != NULL);
ASSERT(ip->i_itemp->ili_lock_flags == 0); ASSERT(ip->i_itemp->ili_lock_flags == 0);
ASSERT(!XFS_NOT_DQATTACHED(mp, ip));
ntp = *tp;
mp = (ntp)->t_mountp;
ASSERT(! XFS_NOT_DQATTACHED(mp, ip));
/*
* We only support truncating the entire attribute fork.
*/
if (fork == XFS_ATTR_FORK) {
new_size = 0LL;
}
first_unmap_block = XFS_B_TO_FSB(mp, (xfs_ufsize_t)new_size);
trace_xfs_itruncate_finish_start(ip, new_size);
/*
* The first thing we do is set the size to new_size permanently
* on disk. This way we don't have to worry about anyone ever
* being able to look at the data being freed even in the face
* of a crash. What we're getting around here is the case where
* we free a block, it is allocated to another file, it is written
* to, and then we crash. If the new data gets written to the
* file but the log buffers containing the free and reallocation
* don't, then we'd end up with garbage in the blocks being freed.
* As long as we make the new_size permanent before actually
* freeing any blocks it doesn't matter if they get written to.
*
* The callers must signal into us whether or not the size
* setting here must be synchronous. There are a few cases
* where it doesn't have to be synchronous. Those cases
* occur if the file is unlinked and we know the unlink is
* permanent or if the blocks being truncated are guaranteed
* to be beyond the inode eof (regardless of the link count)
* and the eof value is permanent. Both of these cases occur
* only on wsync-mounted filesystems. In those cases, we're
* guaranteed that no user will ever see the data in the blocks
* that are being truncated so the truncate can run async.
* In the free beyond eof case, the file may wind up with
* more blocks allocated to it than it needs if we crash
* and that won't get fixed until the next time the file
* is re-opened and closed but that's ok as that shouldn't
* be too many blocks.
*
* However, we can't just make all wsync xactions run async
* because there's one call out of the create path that needs
* to run sync where it's truncating an existing file to size
* 0 whose size is > 0.
*
* It's probably possible to come up with a test in this
* routine that would correctly distinguish all the above
* cases from the values of the function parameters and the
* inode state but for sanity's sake, I've decided to let the
* layers above just tell us. It's simpler to correctly figure
* out in the layer above exactly under what conditions we
* can run async and I think it's easier for others read and
* follow the logic in case something has to be changed.
* cscope is your friend -- rcc.
*
* The attribute fork is much simpler.
*
* For the attribute fork we allow the caller to tell us whether
* the unlink of the inode that led to this call is yet permanent
* in the on disk log. If it is not and we will be freeing extents
* in this inode then we make the first transaction synchronous
* to make sure that the unlink is permanent by the time we free
* the blocks.
*/
if (fork == XFS_DATA_FORK) {
if (ip->i_d.di_nextents > 0) {
/*
* If we are not changing the file size then do
* not update the on-disk file size - we may be
* called from xfs_inactive_free_eofblocks(). If we
* update the on-disk file size and then the system
* crashes before the contents of the file are
* flushed to disk then the files may be full of
* holes (ie NULL files bug).
*/
if (ip->i_size != new_size) {
ip->i_d.di_size = new_size;
ip->i_size = new_size;
xfs_trans_log_inode(ntp, ip, XFS_ILOG_CORE);
}
}
} else if (sync) {
ASSERT(!(mp->m_flags & XFS_MOUNT_WSYNC));
if (ip->i_d.di_anextents > 0)
xfs_trans_set_sync(ntp);
}
ASSERT(fork == XFS_DATA_FORK ||
(fork == XFS_ATTR_FORK &&
((sync && !(mp->m_flags & XFS_MOUNT_WSYNC)) ||
(sync == 0 && (mp->m_flags & XFS_MOUNT_WSYNC)))));
/* /*
* Since it is possible for space to become allocated beyond * Since it is possible for space to become allocated beyond
...@@ -1390,128 +1275,142 @@ xfs_itruncate_finish( ...@@ -1390,128 +1275,142 @@ xfs_itruncate_finish(
* beyond the maximum file size (ie it is the same as last_block), * beyond the maximum file size (ie it is the same as last_block),
* then there is nothing to do. * then there is nothing to do.
*/ */
first_unmap_block = XFS_B_TO_FSB(mp, (xfs_ufsize_t)new_size);
last_block = XFS_B_TO_FSB(mp, (xfs_ufsize_t)XFS_MAXIOFFSET(mp)); last_block = XFS_B_TO_FSB(mp, (xfs_ufsize_t)XFS_MAXIOFFSET(mp));
ASSERT(first_unmap_block <= last_block); if (first_unmap_block == last_block)
done = 0; return 0;
if (last_block == first_unmap_block) {
done = 1; ASSERT(first_unmap_block < last_block);
} else {
unmap_len = last_block - first_unmap_block + 1; unmap_len = last_block - first_unmap_block + 1;
}
while (!done) { while (!done) {
/*
* Free up up to XFS_ITRUNC_MAX_EXTENTS. xfs_bunmapi()
* will tell us whether it freed the entire range or
* not. If this is a synchronous mount (wsync),
* then we can tell bunmapi to keep all the
* transactions asynchronous since the unlink
* transaction that made this inode inactive has
* already hit the disk. There's no danger of
* the freed blocks being reused, there being a
* crash, and the reused blocks suddenly reappearing
* in this file with garbage in them once recovery
* runs.
*/
xfs_bmap_init(&free_list, &first_block); xfs_bmap_init(&free_list, &first_block);
error = xfs_bunmapi(ntp, ip, error = xfs_bunmapi(tp, ip,
first_unmap_block, unmap_len, first_unmap_block, unmap_len,
xfs_bmapi_aflag(fork), xfs_bmapi_aflag(whichfork),
XFS_ITRUNC_MAX_EXTENTS, XFS_ITRUNC_MAX_EXTENTS,
&first_block, &free_list, &first_block, &free_list,
&done); &done);
if (error) { if (error)
/* goto out_bmap_cancel;
* If the bunmapi call encounters an error,
* return to the caller where the transaction
* can be properly aborted. We just need to
* make sure we're not holding any resources
* that we were not when we came in.
*/
xfs_bmap_cancel(&free_list);
return error;
}
/* /*
* Duplicate the transaction that has the permanent * Duplicate the transaction that has the permanent
* reservation and commit the old transaction. * reservation and commit the old transaction.
*/ */
error = xfs_bmap_finish(tp, &free_list, &committed); error = xfs_bmap_finish(&tp, &free_list, &committed);
ntp = *tp;
if (committed) if (committed)
xfs_trans_ijoin(ntp, ip); xfs_trans_ijoin(tp, ip);
if (error)
if (error) { goto out_bmap_cancel;
/*
* If the bmap finish call encounters an error, return
* to the caller where the transaction can be properly
* aborted. We just need to make sure we're not
* holding any resources that we were not when we came
* in.
*
* Aborting from this point might lose some blocks in
* the file system, but oh well.
*/
xfs_bmap_cancel(&free_list);
return error;
}
if (committed) { if (committed) {
/* /*
* Mark the inode dirty so it will be logged and * Mark the inode dirty so it will be logged and
* moved forward in the log as part of every commit. * moved forward in the log as part of every commit.
*/ */
xfs_trans_log_inode(ntp, ip, XFS_ILOG_CORE); xfs_trans_log_inode(tp, ip, XFS_ILOG_CORE);
} }
ntp = xfs_trans_dup(ntp); ntp = xfs_trans_dup(tp);
error = xfs_trans_commit(*tp, 0); error = xfs_trans_commit(tp, 0);
*tp = ntp; tp = ntp;
xfs_trans_ijoin(ntp, ip); xfs_trans_ijoin(tp, ip);
if (error) if (error)
return error; goto out;
/* /*
* transaction commit worked ok so we can drop the extra ticket * Transaction commit worked ok so we can drop the extra ticket
* reference that we gained in xfs_trans_dup() * reference that we gained in xfs_trans_dup()
*/ */
xfs_log_ticket_put(ntp->t_ticket); xfs_log_ticket_put(tp->t_ticket);
error = xfs_trans_reserve(ntp, 0, error = xfs_trans_reserve(tp, 0,
XFS_ITRUNCATE_LOG_RES(mp), 0, XFS_ITRUNCATE_LOG_RES(mp), 0,
XFS_TRANS_PERM_LOG_RES, XFS_TRANS_PERM_LOG_RES,
XFS_ITRUNCATE_LOG_COUNT); XFS_ITRUNCATE_LOG_COUNT);
if (error) if (error)
return error; goto out;
} }
out:
*tpp = tp;
return error;
out_bmap_cancel:
/* /*
* Only update the size in the case of the data fork, but * If the bunmapi call encounters an error, return to the caller where
* always re-log the inode so that our permanent transaction * the transaction can be properly aborted. We just need to make sure
* can keep on rolling it forward in the log. * we're not holding any resources that we were not when we came in.
*/ */
if (fork == XFS_DATA_FORK) { xfs_bmap_cancel(&free_list);
xfs_isize_check(mp, ip, new_size); goto out;
}
int
xfs_itruncate_data(
struct xfs_trans **tpp,
struct xfs_inode *ip,
xfs_fsize_t new_size)
{
int error;
trace_xfs_itruncate_data_start(ip, new_size);
/*
* The first thing we do is set the size to new_size permanently on
* disk. This way we don't have to worry about anyone ever being able
* to look at the data being freed even in the face of a crash.
* What we're getting around here is the case where we free a block, it
* is allocated to another file, it is written to, and then we crash.
* If the new data gets written to the file but the log buffers
* containing the free and reallocation don't, then we'd end up with
* garbage in the blocks being freed. As long as we make the new_size
* permanent before actually freeing any blocks it doesn't matter if
* they get written to.
*/
if (ip->i_d.di_nextents > 0) {
/* /*
* If we are not changing the file size then do * If we are not changing the file size then do not update
* not update the on-disk file size - we may be * the on-disk file size - we may be called from
* called from xfs_inactive_free_eofblocks(). If we * xfs_inactive_free_eofblocks(). If we update the on-disk
* update the on-disk file size and then the system * file size and then the system crashes before the contents
* crashes before the contents of the file are * of the file are flushed to disk then the files may be
* flushed to disk then the files may be full of * full of holes (ie NULL files bug).
* holes (ie NULL files bug).
*/ */
if (ip->i_size != new_size) { if (ip->i_size != new_size) {
ip->i_d.di_size = new_size; ip->i_d.di_size = new_size;
ip->i_size = new_size; ip->i_size = new_size;
xfs_trans_log_inode(*tpp, ip, XFS_ILOG_CORE);
} }
} }
xfs_trans_log_inode(ntp, ip, XFS_ILOG_CORE);
ASSERT((new_size != 0) || error = xfs_itruncate_extents(tpp, ip, XFS_DATA_FORK, new_size);
(fork == XFS_ATTR_FORK) || if (error)
(ip->i_delayed_blks == 0)); return error;
ASSERT((new_size != 0) ||
(fork == XFS_ATTR_FORK) || /*
(ip->i_d.di_nextents == 0)); * If we are not changing the file size then do not update the on-disk
trace_xfs_itruncate_finish_end(ip, new_size); * file size - we may be called from xfs_inactive_free_eofblocks().
* If we update the on-disk file size and then the system crashes
* before the contents of the file are flushed to disk then the files
* may be full of holes (ie NULL files bug).
*/
xfs_isize_check(ip, new_size);
if (ip->i_size != new_size) {
ip->i_d.di_size = new_size;
ip->i_size = new_size;
}
ASSERT(new_size != 0 || ip->i_delayed_blks == 0);
ASSERT(new_size != 0 || ip->i_d.di_nextents == 0);
/*
* Always re-log the inode so that our permanent transaction can keep
* on rolling it forward in the log.
*/
xfs_trans_log_inode(*tpp, ip, XFS_ILOG_CORE);
trace_xfs_itruncate_data_end(ip, new_size);
return 0; return 0;
} }
......
...@@ -491,8 +491,10 @@ uint xfs_ip2xflags(struct xfs_inode *); ...@@ -491,8 +491,10 @@ uint xfs_ip2xflags(struct xfs_inode *);
uint xfs_dic2xflags(struct xfs_dinode *); uint xfs_dic2xflags(struct xfs_dinode *);
int xfs_ifree(struct xfs_trans *, xfs_inode_t *, int xfs_ifree(struct xfs_trans *, xfs_inode_t *,
struct xfs_bmap_free *); struct xfs_bmap_free *);
int xfs_itruncate_finish(struct xfs_trans **, xfs_inode_t *, int xfs_itruncate_extents(struct xfs_trans **, struct xfs_inode *,
xfs_fsize_t, int, int); int, xfs_fsize_t);
int xfs_itruncate_data(struct xfs_trans **, struct xfs_inode *,
xfs_fsize_t);
int xfs_iunlink(struct xfs_trans *, xfs_inode_t *); int xfs_iunlink(struct xfs_trans *, xfs_inode_t *);
void xfs_iext_realloc(xfs_inode_t *, int, int); void xfs_iext_realloc(xfs_inode_t *, int, int);
...@@ -568,13 +570,6 @@ void xfs_iext_irec_update_extoffs(xfs_ifork_t *, int, int); ...@@ -568,13 +570,6 @@ void xfs_iext_irec_update_extoffs(xfs_ifork_t *, int, int);
#define xfs_ipincount(ip) ((unsigned int) atomic_read(&ip->i_pincount)) #define xfs_ipincount(ip) ((unsigned int) atomic_read(&ip->i_pincount))
#ifdef DEBUG
void xfs_isize_check(struct xfs_mount *, struct xfs_inode *,
xfs_fsize_t);
#else /* DEBUG */
#define xfs_isize_check(mp, ip, isize)
#endif /* DEBUG */
#if defined(DEBUG) #if defined(DEBUG)
void xfs_inobp_check(struct xfs_mount *, struct xfs_buf *); void xfs_inobp_check(struct xfs_mount *, struct xfs_buf *);
#else #else
......
...@@ -220,15 +220,12 @@ xfs_free_eofblocks( ...@@ -220,15 +220,12 @@ xfs_free_eofblocks(
xfs_ilock(ip, XFS_ILOCK_EXCL); xfs_ilock(ip, XFS_ILOCK_EXCL);
xfs_trans_ijoin(tp, ip); xfs_trans_ijoin(tp, ip);
error = xfs_itruncate_finish(&tp, ip, error = xfs_itruncate_data(&tp, ip, ip->i_size);
ip->i_size, if (error) {
XFS_DATA_FORK,
0);
/* /*
* If we get an error at this point we * If we get an error at this point we simply don't
* simply don't bother truncating the file. * bother truncating the file.
*/ */
if (error) {
xfs_trans_cancel(tp, xfs_trans_cancel(tp,
(XFS_TRANS_RELEASE_LOG_RES | (XFS_TRANS_RELEASE_LOG_RES |
XFS_TRANS_ABORT)); XFS_TRANS_ABORT));
...@@ -665,16 +662,7 @@ xfs_inactive( ...@@ -665,16 +662,7 @@ xfs_inactive(
xfs_ilock(ip, XFS_ILOCK_EXCL); xfs_ilock(ip, XFS_ILOCK_EXCL);
xfs_trans_ijoin(tp, ip); xfs_trans_ijoin(tp, ip);
/* error = xfs_itruncate_data(&tp, ip, 0);
* normally, we have to run xfs_itruncate_finish sync.
* But if filesystem is wsync and we're in the inactive
* path, then we know that nlink == 0, and that the
* xaction that made nlink == 0 is permanently committed
* since xfs_remove runs as a synchronous transaction.
*/
error = xfs_itruncate_finish(&tp, ip, 0, XFS_DATA_FORK,
(!(mp->m_flags & XFS_MOUNT_WSYNC) ? 1 : 0));
if (error) { if (error) {
xfs_trans_cancel(tp, xfs_trans_cancel(tp,
XFS_TRANS_RELEASE_LOG_RES | XFS_TRANS_ABORT); XFS_TRANS_RELEASE_LOG_RES | XFS_TRANS_ABORT);
......
Markdown is supported
0% .
You are about to add 0 people to the discussion. Proceed with caution.
先完成此消息的编辑!
想要评论请 注册