提交 e1d7553f 编写于 作者: C Christoph Hellwig 提交者: Darrick J. Wong

xfs: use xfs_bmap_del_extent_delay for the data fork as well

And remove the delalloc code from xfs_bmap_del_extent, which gets renamed
to xfs_bmap_del_extent_real to fit the naming scheme used by the other
xfs_bmap_{add,del}_extent_* routines.
Signed-off-by: NChristoph Hellwig <hch@lst.de>
Reviewed-by: NBrian Foster <bfoster@redhat.com>
Reviewed-by: NDarrick J. Wong <darrick.wong@oracle.com>
Signed-off-by: NDarrick J. Wong <darrick.wong@oracle.com>
上级 8280f6ed
...@@ -5059,10 +5059,10 @@ xfs_bmap_del_extent_cow( ...@@ -5059,10 +5059,10 @@ xfs_bmap_del_extent_cow(
/* /*
* Called by xfs_bmapi to update file extent records and the btree * Called by xfs_bmapi to update file extent records and the btree
* after removing space (or undoing a delayed allocation). * after removing space.
*/ */
STATIC int /* error */ STATIC int /* error */
xfs_bmap_del_extent( xfs_bmap_del_extent_real(
xfs_inode_t *ip, /* incore inode pointer */ xfs_inode_t *ip, /* incore inode pointer */
xfs_trans_t *tp, /* current transaction pointer */ xfs_trans_t *tp, /* current transaction pointer */
xfs_extnum_t *idx, /* extent number to update/delete */ xfs_extnum_t *idx, /* extent number to update/delete */
...@@ -5073,11 +5073,8 @@ xfs_bmap_del_extent( ...@@ -5073,11 +5073,8 @@ xfs_bmap_del_extent(
int whichfork, /* data or attr fork */ int whichfork, /* data or attr fork */
int bflags) /* bmapi flags */ int bflags) /* bmapi flags */
{ {
xfs_filblks_t da_new; /* new delay-alloc indirect blocks */
xfs_filblks_t da_old; /* old delay-alloc indirect blocks */
xfs_fsblock_t del_endblock=0; /* first block past del */ xfs_fsblock_t del_endblock=0; /* first block past del */
xfs_fileoff_t del_endoff; /* first offset past del */ xfs_fileoff_t del_endoff; /* first offset past del */
int delay; /* current block is delayed allocated */
int do_fx; /* free extent at end of routine */ int do_fx; /* free extent at end of routine */
xfs_bmbt_rec_host_t *ep; /* current extent entry pointer */ xfs_bmbt_rec_host_t *ep; /* current extent entry pointer */
int error; /* error return value */ int error; /* error return value */
...@@ -5112,27 +5109,17 @@ xfs_bmap_del_extent( ...@@ -5112,27 +5109,17 @@ xfs_bmap_del_extent(
del_endoff = del->br_startoff + del->br_blockcount; del_endoff = del->br_startoff + del->br_blockcount;
got_endoff = got.br_startoff + got.br_blockcount; got_endoff = got.br_startoff + got.br_blockcount;
ASSERT(got_endoff >= del_endoff); ASSERT(got_endoff >= del_endoff);
delay = isnullstartblock(got.br_startblock); ASSERT(!isnullstartblock(got.br_startblock));
ASSERT(isnullstartblock(del->br_startblock) == delay); flags = XFS_ILOG_CORE;
flags = 0;
qfield = 0; qfield = 0;
error = 0; error = 0;
/*
* If deleting a real allocation, must free up the disk space.
*/
if (!delay) {
flags = XFS_ILOG_CORE;
/*
* Realtime allocation. Free it and record di_nblocks update.
*/
if (whichfork == XFS_DATA_FORK && XFS_IS_REALTIME_INODE(ip)) { if (whichfork == XFS_DATA_FORK && XFS_IS_REALTIME_INODE(ip)) {
xfs_fsblock_t bno; xfs_fsblock_t bno;
xfs_filblks_t len; xfs_filblks_t len;
ASSERT(do_mod(del->br_blockcount, ASSERT(do_mod(del->br_blockcount, mp->m_sb.sb_rextsize) == 0);
mp->m_sb.sb_rextsize) == 0); ASSERT(do_mod(del->br_startblock, mp->m_sb.sb_rextsize) == 0);
ASSERT(do_mod(del->br_startblock,
mp->m_sb.sb_rextsize) == 0);
bno = del->br_startblock; bno = del->br_startblock;
len = del->br_blockcount; len = del->br_blockcount;
do_div(bno, mp->m_sb.sb_rextsize); do_div(bno, mp->m_sb.sb_rextsize);
...@@ -5143,33 +5130,20 @@ xfs_bmap_del_extent( ...@@ -5143,33 +5130,20 @@ xfs_bmap_del_extent(
do_fx = 0; do_fx = 0;
nblks = len * mp->m_sb.sb_rextsize; nblks = len * mp->m_sb.sb_rextsize;
qfield = XFS_TRANS_DQ_RTBCOUNT; qfield = XFS_TRANS_DQ_RTBCOUNT;
} } else {
/*
* Ordinary allocation.
*/
else {
do_fx = 1; do_fx = 1;
nblks = del->br_blockcount; nblks = del->br_blockcount;
qfield = XFS_TRANS_DQ_BCOUNT; qfield = XFS_TRANS_DQ_BCOUNT;
} }
/*
* Set up del_endblock and cur for later.
*/
del_endblock = del->br_startblock + del->br_blockcount; del_endblock = del->br_startblock + del->br_blockcount;
if (cur) { if (cur) {
if ((error = xfs_bmbt_lookup_eq(cur, got.br_startoff, error = xfs_bmbt_lookup_eq(cur, got.br_startoff,
got.br_startblock, got.br_blockcount, got.br_startblock, got.br_blockcount, &i);
&i))) if (error)
goto done; goto done;
XFS_WANT_CORRUPTED_GOTO(mp, i == 1, done); XFS_WANT_CORRUPTED_GOTO(mp, i == 1, done);
} }
da_old = da_new = 0;
} else {
da_old = startblockval(got.br_startblock);
da_new = 0;
nblks = 0;
do_fx = 0;
}
/* /*
* Set flag value to use in switch statement. * Set flag value to use in switch statement.
...@@ -5185,8 +5159,6 @@ xfs_bmap_del_extent( ...@@ -5185,8 +5159,6 @@ xfs_bmap_del_extent(
xfs_iext_remove(ip, *idx, 1, xfs_iext_remove(ip, *idx, 1,
whichfork == XFS_ATTR_FORK ? BMAP_ATTRFORK : 0); whichfork == XFS_ATTR_FORK ? BMAP_ATTRFORK : 0);
--*idx; --*idx;
if (delay)
break;
XFS_IFORK_NEXT_SET(ip, whichfork, XFS_IFORK_NEXT_SET(ip, whichfork,
XFS_IFORK_NEXTENTS(ip, whichfork) - 1); XFS_IFORK_NEXTENTS(ip, whichfork) - 1);
...@@ -5208,14 +5180,6 @@ xfs_bmap_del_extent( ...@@ -5208,14 +5180,6 @@ xfs_bmap_del_extent(
xfs_bmbt_set_startoff(ep, del_endoff); xfs_bmbt_set_startoff(ep, del_endoff);
temp = got.br_blockcount - del->br_blockcount; temp = got.br_blockcount - del->br_blockcount;
xfs_bmbt_set_blockcount(ep, temp); xfs_bmbt_set_blockcount(ep, temp);
if (delay) {
temp = XFS_FILBLKS_MIN(xfs_bmap_worst_indlen(ip, temp),
da_old);
xfs_bmbt_set_startblock(ep, nullstartblock((int)temp));
trace_xfs_bmap_post_update(ip, *idx, state, _THIS_IP_);
da_new = temp;
break;
}
xfs_bmbt_set_startblock(ep, del_endblock); xfs_bmbt_set_startblock(ep, del_endblock);
trace_xfs_bmap_post_update(ip, *idx, state, _THIS_IP_); trace_xfs_bmap_post_update(ip, *idx, state, _THIS_IP_);
if (!cur) { if (!cur) {
...@@ -5235,14 +5199,6 @@ xfs_bmap_del_extent( ...@@ -5235,14 +5199,6 @@ xfs_bmap_del_extent(
temp = got.br_blockcount - del->br_blockcount; temp = got.br_blockcount - del->br_blockcount;
trace_xfs_bmap_pre_update(ip, *idx, state, _THIS_IP_); trace_xfs_bmap_pre_update(ip, *idx, state, _THIS_IP_);
xfs_bmbt_set_blockcount(ep, temp); xfs_bmbt_set_blockcount(ep, temp);
if (delay) {
temp = XFS_FILBLKS_MIN(xfs_bmap_worst_indlen(ip, temp),
da_old);
xfs_bmbt_set_startblock(ep, nullstartblock((int)temp));
trace_xfs_bmap_post_update(ip, *idx, state, _THIS_IP_);
da_new = temp;
break;
}
trace_xfs_bmap_post_update(ip, *idx, state, _THIS_IP_); trace_xfs_bmap_post_update(ip, *idx, state, _THIS_IP_);
if (!cur) { if (!cur) {
flags |= xfs_ilog_fext(whichfork); flags |= xfs_ilog_fext(whichfork);
...@@ -5266,55 +5222,51 @@ xfs_bmap_del_extent( ...@@ -5266,55 +5222,51 @@ xfs_bmap_del_extent(
temp2 = got_endoff - del_endoff; temp2 = got_endoff - del_endoff;
new.br_blockcount = temp2; new.br_blockcount = temp2;
new.br_state = got.br_state; new.br_state = got.br_state;
if (!delay) {
new.br_startblock = del_endblock; new.br_startblock = del_endblock;
flags |= XFS_ILOG_CORE; flags |= XFS_ILOG_CORE;
if (cur) { if (cur) {
if ((error = xfs_bmbt_update(cur, error = xfs_bmbt_update(cur, got.br_startoff,
got.br_startoff,
got.br_startblock, temp, got.br_startblock, temp,
got.br_state))) got.br_state);
if (error)
goto done; goto done;
if ((error = xfs_btree_increment(cur, 0, &i))) error = xfs_btree_increment(cur, 0, &i);
if (error)
goto done; goto done;
cur->bc_rec.b = new; cur->bc_rec.b = new;
error = xfs_btree_insert(cur, &i); error = xfs_btree_insert(cur, &i);
if (error && error != -ENOSPC) if (error && error != -ENOSPC)
goto done; goto done;
/* /*
* If get no-space back from btree insert, * If get no-space back from btree insert, it tried a
* it tried a split, and we have a zero * split, and we have a zero block reservation. Fix up
* block reservation. * our state and return the error.
* Fix up our state and return the error.
*/ */
if (error == -ENOSPC) { if (error == -ENOSPC) {
/* /*
* Reset the cursor, don't trust * Reset the cursor, don't trust it after any
* it after any insert operation. * insert operation.
*/ */
if ((error = xfs_bmbt_lookup_eq(cur, error = xfs_bmbt_lookup_eq(cur, got.br_startoff,
got.br_startoff, got.br_startblock, temp, &i);
got.br_startblock, if (error)
temp, &i)))
goto done; goto done;
XFS_WANT_CORRUPTED_GOTO(mp, XFS_WANT_CORRUPTED_GOTO(mp, i == 1, done);
i == 1, done);
/* /*
* Update the btree record back * Update the btree record back
* to the original value. * to the original value.
*/ */
if ((error = xfs_bmbt_update(cur, error = xfs_bmbt_update(cur, got.br_startoff,
got.br_startoff,
got.br_startblock, got.br_startblock,
got.br_blockcount, got.br_blockcount,
got.br_state))) got.br_state);
if (error)
goto done; goto done;
/* /*
* Reset the extent record back * Reset the extent record back
* to the original value. * to the original value.
*/ */
xfs_bmbt_set_blockcount(ep, xfs_bmbt_set_blockcount(ep, got.br_blockcount);
got.br_blockcount);
flags = 0; flags = 0;
error = -ENOSPC; error = -ENOSPC;
goto done; goto done;
...@@ -5324,31 +5276,6 @@ xfs_bmap_del_extent( ...@@ -5324,31 +5276,6 @@ xfs_bmap_del_extent(
flags |= xfs_ilog_fext(whichfork); flags |= xfs_ilog_fext(whichfork);
XFS_IFORK_NEXT_SET(ip, whichfork, XFS_IFORK_NEXT_SET(ip, whichfork,
XFS_IFORK_NEXTENTS(ip, whichfork) + 1); XFS_IFORK_NEXTENTS(ip, whichfork) + 1);
} else {
xfs_filblks_t stolen;
ASSERT(whichfork == XFS_DATA_FORK);
/*
* Distribute the original indlen reservation across the
* two new extents. Steal blocks from the deleted extent
* if necessary. Stealing blocks simply fudges the
* fdblocks accounting in xfs_bunmapi().
*/
temp = xfs_bmap_worst_indlen(ip, got.br_blockcount);
temp2 = xfs_bmap_worst_indlen(ip, new.br_blockcount);
stolen = xfs_bmap_split_indlen(da_old, &temp, &temp2,
del->br_blockcount);
da_new = temp + temp2 - stolen;
del->br_blockcount -= stolen;
/*
* Set the reservation for each extent. Warn if either
* is zero as this can lead to delalloc problems.
*/
WARN_ON_ONCE(!temp || !temp2);
xfs_bmbt_set_startblock(ep, nullstartblock((int)temp));
new.br_startblock = nullstartblock((int)temp2);
}
trace_xfs_bmap_post_update(ip, *idx, state, _THIS_IP_); trace_xfs_bmap_post_update(ip, *idx, state, _THIS_IP_);
xfs_iext_insert(ip, *idx + 1, 1, &new, state); xfs_iext_insert(ip, *idx + 1, 1, &new, state);
++*idx; ++*idx;
...@@ -5356,11 +5283,9 @@ xfs_bmap_del_extent( ...@@ -5356,11 +5283,9 @@ xfs_bmap_del_extent(
} }
/* remove reverse mapping */ /* remove reverse mapping */
if (!delay) {
error = xfs_rmap_unmap_extent(mp, dfops, ip, whichfork, del); error = xfs_rmap_unmap_extent(mp, dfops, ip, whichfork, del);
if (error) if (error)
goto done; goto done;
}
/* /*
* If we need to, add to list of extents to delete. * If we need to, add to list of extents to delete.
...@@ -5386,13 +5311,6 @@ xfs_bmap_del_extent( ...@@ -5386,13 +5311,6 @@ xfs_bmap_del_extent(
if (qfield && !(bflags & XFS_BMAPI_REMAP)) if (qfield && !(bflags & XFS_BMAPI_REMAP))
xfs_trans_mod_dquot_byino(tp, ip, qfield, (long)-nblks); xfs_trans_mod_dquot_byino(tp, ip, qfield, (long)-nblks);
/*
* Account for change in delayed indirect blocks.
* Nothing to do for disk quota accounting here.
*/
ASSERT(da_old >= da_new);
if (da_old > da_new)
xfs_mod_fdblocks(mp, (int64_t)(da_old - da_new), false);
done: done:
*logflagsp = flags; *logflagsp = flags;
return error; return error;
...@@ -5677,20 +5595,24 @@ __xfs_bunmapi( ...@@ -5677,20 +5595,24 @@ __xfs_bunmapi(
} }
} }
if (wasdel) {
error = xfs_bmap_del_extent_delay(ip, whichfork, &lastx,
&got, &del);
} else {
/* /*
* If it's the case where the directory code is running * If it's the case where the directory code is running
* with no block reservation, and the deleted block is in * with no block reservation, and the deleted block is
* the middle of its extent, and the resulting insert * in the middle of its extent, and the resulting insert
* of an extent would cause transformation to btree format, * of an extent would cause transformation to btree
* then reject it. The calling code will then swap * format, then reject it. The calling code will then
* blocks around instead. * swap blocks around instead. We have to do this now,
* We have to do this now, rather than waiting for the * rather than waiting for the conversion to btree
* conversion to btree format, since the transaction * format, since the transaction will be dirty.
* will be dirty. */
*/ if (tp->t_blk_res == 0 &&
if (!wasdel && tp->t_blk_res == 0 && XFS_IFORK_FORMAT(ip, whichfork) ==
XFS_IFORK_FORMAT(ip, whichfork) == XFS_DINODE_FMT_EXTENTS && XFS_DINODE_FMT_EXTENTS &&
XFS_IFORK_NEXTENTS(ip, whichfork) >= /* Note the >= */ XFS_IFORK_NEXTENTS(ip, whichfork) >=
XFS_IFORK_MAXEXT(ip, whichfork) && XFS_IFORK_MAXEXT(ip, whichfork) &&
del.br_startoff > got.br_startoff && del.br_startoff > got.br_startoff &&
del.br_startoff + del.br_blockcount < del.br_startoff + del.br_blockcount <
...@@ -5699,40 +5621,15 @@ __xfs_bunmapi( ...@@ -5699,40 +5621,15 @@ __xfs_bunmapi(
goto error0; goto error0;
} }
/* error = xfs_bmap_del_extent_real(ip, tp, &lastx, dfops,
* Unreserve quota and update realtime free space, if cur, &del, &tmp_logflags, whichfork,
* appropriate. If delayed allocation, update the inode delalloc flags);
* counter now and wait to update the sb counters as logflags |= tmp_logflags;
* xfs_bmap_del_extent() might need to borrow some blocks.
*/
if (wasdel) {
ASSERT(startblockval(del.br_startblock) > 0);
if (isrt) {
xfs_filblks_t rtexts;
rtexts = XFS_FSB_TO_B(mp, del.br_blockcount);
do_div(rtexts, mp->m_sb.sb_rextsize);
xfs_mod_frextents(mp, (int64_t)rtexts);
(void)xfs_trans_reserve_quota_nblks(NULL,
ip, -((long)del.br_blockcount), 0,
XFS_QMOPT_RES_RTBLKS);
} else {
(void)xfs_trans_reserve_quota_nblks(NULL,
ip, -((long)del.br_blockcount), 0,
XFS_QMOPT_RES_REGBLKS);
}
ip->i_delayed_blks -= del.br_blockcount;
} }
error = xfs_bmap_del_extent(ip, tp, &lastx, dfops, cur, &del,
&tmp_logflags, whichfork, flags);
logflags |= tmp_logflags;
if (error) if (error)
goto error0; goto error0;
if (!isrt && wasdel)
xfs_mod_fdblocks(mp, (int64_t)del.br_blockcount, false);
max_len -= del.br_blockcount; max_len -= del.br_blockcount;
end = del.br_startoff - 1; end = del.br_startoff - 1;
nodelete: nodelete:
......
Markdown is supported
0% .
You are about to add 0 people to the discussion. Proceed with caution.
先完成此消息的编辑!
想要评论请 注册