From: Brian Foster <bfoster@redhat.com>
To: Christoph Hellwig <hch@lst.de>
Cc: linux-xfs@vger.kernel.org
Subject: Re: [PATCH 05/17] xfs: use xfs_bmap_del_extent_delay for the data fork as well
Date: Thu, 7 Sep 2017 11:46:58 -0400 [thread overview]
Message-ID: <20170907154657.GE61662@bfoster.bfoster> (raw)
In-Reply-To: <20170903072956.3175-6-hch@lst.de>
On Sun, Sep 03, 2017 at 09:29:44AM +0200, Christoph Hellwig wrote:
> 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: Christoph Hellwig <hch@lst.de>
> ---
Reviewed-by: Brian Foster <bfoster@redhat.com>
> fs/xfs/libxfs/xfs_bmap.c | 331 ++++++++++++++++-------------------------------
> 1 file changed, 114 insertions(+), 217 deletions(-)
>
> diff --git a/fs/xfs/libxfs/xfs_bmap.c b/fs/xfs/libxfs/xfs_bmap.c
> index 6c65954fc42c..e6feb71298ba 100644
> --- a/fs/xfs/libxfs/xfs_bmap.c
> +++ b/fs/xfs/libxfs/xfs_bmap.c
> @@ -5061,10 +5061,10 @@ xfs_bmap_del_extent_cow(
>
> /*
> * 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 */
> -xfs_bmap_del_extent(
> +xfs_bmap_del_extent_real(
> xfs_inode_t *ip, /* incore inode pointer */
> xfs_trans_t *tp, /* current transaction pointer */
> xfs_extnum_t *idx, /* extent number to update/delete */
> @@ -5075,11 +5075,8 @@ xfs_bmap_del_extent(
> int whichfork, /* data or attr fork */
> 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_fileoff_t del_endoff; /* first offset past del */
> - int delay; /* current block is delayed allocated */
> int do_fx; /* free extent at end of routine */
> xfs_bmbt_rec_host_t *ep; /* current extent entry pointer */
> int error; /* error return value */
> @@ -5114,63 +5111,40 @@ xfs_bmap_del_extent(
> del_endoff = del->br_startoff + del->br_blockcount;
> got_endoff = got.br_startoff + got.br_blockcount;
> ASSERT(got_endoff >= del_endoff);
> - delay = isnullstartblock(got.br_startblock);
> - ASSERT(isnullstartblock(del->br_startblock) == delay);
> - flags = 0;
> + ASSERT(!isnullstartblock(got.br_startblock));
> + flags = XFS_ILOG_CORE;
> qfield = 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)) {
> - xfs_fsblock_t bno;
> - xfs_filblks_t len;
> -
> - ASSERT(do_mod(del->br_blockcount,
> - mp->m_sb.sb_rextsize) == 0);
> - ASSERT(do_mod(del->br_startblock,
> - mp->m_sb.sb_rextsize) == 0);
> - bno = del->br_startblock;
> - len = del->br_blockcount;
> - do_div(bno, mp->m_sb.sb_rextsize);
> - do_div(len, mp->m_sb.sb_rextsize);
> - error = xfs_rtfree_extent(tp, bno, (xfs_extlen_t)len);
> - if (error)
> - goto done;
> - do_fx = 0;
> - nblks = len * mp->m_sb.sb_rextsize;
> - qfield = XFS_TRANS_DQ_RTBCOUNT;
> - }
> - /*
> - * Ordinary allocation.
> - */
> - else {
> - do_fx = 1;
> - nblks = del->br_blockcount;
> - qfield = XFS_TRANS_DQ_BCOUNT;
> - }
> - /*
> - * Set up del_endblock and cur for later.
> - */
> - del_endblock = del->br_startblock + del->br_blockcount;
> - if (cur) {
> - if ((error = xfs_bmbt_lookup_eq(cur, got.br_startoff,
> - got.br_startblock, got.br_blockcount,
> - &i)))
> - goto 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;
> +
> + if (whichfork == XFS_DATA_FORK && XFS_IS_REALTIME_INODE(ip)) {
> + xfs_fsblock_t bno;
> + xfs_filblks_t len;
> +
> + ASSERT(do_mod(del->br_blockcount, mp->m_sb.sb_rextsize) == 0);
> + ASSERT(do_mod(del->br_startblock, mp->m_sb.sb_rextsize) == 0);
> + bno = del->br_startblock;
> + len = del->br_blockcount;
> + do_div(bno, mp->m_sb.sb_rextsize);
> + do_div(len, mp->m_sb.sb_rextsize);
> + error = xfs_rtfree_extent(tp, bno, (xfs_extlen_t)len);
> + if (error)
> + goto done;
> do_fx = 0;
> + nblks = len * mp->m_sb.sb_rextsize;
> + qfield = XFS_TRANS_DQ_RTBCOUNT;
> + } else {
> + do_fx = 1;
> + nblks = del->br_blockcount;
> + qfield = XFS_TRANS_DQ_BCOUNT;
> + }
> +
> + del_endblock = del->br_startblock + del->br_blockcount;
> + if (cur) {
> + error = xfs_bmbt_lookup_eq(cur, got.br_startoff,
> + got.br_startblock, got.br_blockcount, &i);
> + if (error)
> + goto done;
> + XFS_WANT_CORRUPTED_GOTO(mp, i == 1, done);
> }
>
> /*
> @@ -5187,8 +5161,6 @@ xfs_bmap_del_extent(
> xfs_iext_remove(ip, *idx, 1,
> whichfork == XFS_ATTR_FORK ? BMAP_ATTRFORK : 0);
> --*idx;
> - if (delay)
> - break;
>
> XFS_IFORK_NEXT_SET(ip, whichfork,
> XFS_IFORK_NEXTENTS(ip, whichfork) - 1);
> @@ -5210,14 +5182,6 @@ xfs_bmap_del_extent(
> xfs_bmbt_set_startoff(ep, del_endoff);
> temp = got.br_blockcount - del->br_blockcount;
> 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);
> trace_xfs_bmap_post_update(ip, *idx, state, _THIS_IP_);
> if (!cur) {
> @@ -5237,14 +5201,6 @@ xfs_bmap_del_extent(
> temp = got.br_blockcount - del->br_blockcount;
> trace_xfs_bmap_pre_update(ip, *idx, state, _THIS_IP_);
> 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_);
> if (!cur) {
> flags |= xfs_ilog_fext(whichfork);
> @@ -5268,89 +5224,60 @@ xfs_bmap_del_extent(
> temp2 = got_endoff - del_endoff;
> new.br_blockcount = temp2;
> new.br_state = got.br_state;
> - if (!delay) {
> - new.br_startblock = del_endblock;
> - flags |= XFS_ILOG_CORE;
> - if (cur) {
> - if ((error = xfs_bmbt_update(cur,
> - got.br_startoff,
> - got.br_startblock, temp,
> - got.br_state)))
> - goto done;
> - if ((error = xfs_btree_increment(cur, 0, &i)))
> - goto done;
> - cur->bc_rec.b = new;
> - error = xfs_btree_insert(cur, &i);
> - if (error && error != -ENOSPC)
> - goto done;
> + new.br_startblock = del_endblock;
> + flags |= XFS_ILOG_CORE;
> + if (cur) {
> + error = xfs_bmbt_update(cur, got.br_startoff,
> + got.br_startblock, temp,
> + got.br_state);
> + if (error)
> + goto done;
> + error = xfs_btree_increment(cur, 0, &i);
> + if (error)
> + goto done;
> + cur->bc_rec.b = new;
> + error = xfs_btree_insert(cur, &i);
> + if (error && error != -ENOSPC)
> + goto done;
> + /*
> + * If get no-space back from btree insert, it tried a
> + * split, and we have a zero block reservation. Fix up
> + * our state and return the error.
> + */
> + if (error == -ENOSPC) {
> /*
> - * If get no-space back from btree insert,
> - * it tried a split, and we have a zero
> - * block reservation.
> - * Fix up our state and return the error.
> + * Reset the cursor, don't trust it after any
> + * insert operation.
> */
> - if (error == -ENOSPC) {
> - /*
> - * Reset the cursor, don't trust
> - * it after any insert operation.
> - */
> - if ((error = xfs_bmbt_lookup_eq(cur,
> - got.br_startoff,
> - got.br_startblock,
> - temp, &i)))
> - goto done;
> - XFS_WANT_CORRUPTED_GOTO(mp,
> - i == 1, done);
> - /*
> - * Update the btree record back
> - * to the original value.
> - */
> - if ((error = xfs_bmbt_update(cur,
> - got.br_startoff,
> - got.br_startblock,
> - got.br_blockcount,
> - got.br_state)))
> - goto done;
> - /*
> - * Reset the extent record back
> - * to the original value.
> - */
> - xfs_bmbt_set_blockcount(ep,
> - got.br_blockcount);
> - flags = 0;
> - error = -ENOSPC;
> + error = xfs_bmbt_lookup_eq(cur, got.br_startoff,
> + got.br_startblock, temp, &i);
> + if (error)
> goto done;
> - }
> XFS_WANT_CORRUPTED_GOTO(mp, i == 1, done);
> - } else
> - flags |= xfs_ilog_fext(whichfork);
> - XFS_IFORK_NEXT_SET(ip, whichfork,
> - 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);
> - }
> + /*
> + * Update the btree record back
> + * to the original value.
> + */
> + error = xfs_bmbt_update(cur, got.br_startoff,
> + got.br_startblock,
> + got.br_blockcount,
> + got.br_state);
> + if (error)
> + goto done;
> + /*
> + * Reset the extent record back
> + * to the original value.
> + */
> + xfs_bmbt_set_blockcount(ep, got.br_blockcount);
> + flags = 0;
> + error = -ENOSPC;
> + goto done;
> + }
> + XFS_WANT_CORRUPTED_GOTO(mp, i == 1, done);
> + } else
> + flags |= xfs_ilog_fext(whichfork);
> + XFS_IFORK_NEXT_SET(ip, whichfork,
> + XFS_IFORK_NEXTENTS(ip, whichfork) + 1);
> trace_xfs_bmap_post_update(ip, *idx, state, _THIS_IP_);
> xfs_iext_insert(ip, *idx + 1, 1, &new, state);
> ++*idx;
> @@ -5358,11 +5285,9 @@ xfs_bmap_del_extent(
> }
>
> /* remove reverse mapping */
> - if (!delay) {
> - error = xfs_rmap_unmap_extent(mp, dfops, ip, whichfork, del);
> - if (error)
> - goto done;
> - }
> + error = xfs_rmap_unmap_extent(mp, dfops, ip, whichfork, del);
> + if (error)
> + goto done;
>
> /*
> * If we need to, add to list of extents to delete.
> @@ -5388,13 +5313,6 @@ xfs_bmap_del_extent(
> if (qfield && !(bflags & XFS_BMAPI_REMAP))
> 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:
> *logflagsp = flags;
> return error;
> @@ -5679,62 +5597,41 @@ __xfs_bunmapi(
> }
> }
>
> - /*
> - * If it's the case where the directory code is running
> - * with no block reservation, and the deleted block is in
> - * the middle of its extent, and the resulting insert
> - * of an extent would cause transformation to btree format,
> - * then reject it. The calling code will then swap
> - * blocks around instead.
> - * We have to do this now, rather than waiting for the
> - * conversion to btree format, since the transaction
> - * will be dirty.
> - */
> - if (!wasdel && tp->t_blk_res == 0 &&
> - XFS_IFORK_FORMAT(ip, whichfork) == XFS_DINODE_FMT_EXTENTS &&
> - XFS_IFORK_NEXTENTS(ip, whichfork) >= /* Note the >= */
> - XFS_IFORK_MAXEXT(ip, whichfork) &&
> - del.br_startoff > got.br_startoff &&
> - del.br_startoff + del.br_blockcount <
> - got.br_startoff + got.br_blockcount) {
> - error = -ENOSPC;
> - goto error0;
> - }
> -
> - /*
> - * Unreserve quota and update realtime free space, if
> - * appropriate. If delayed allocation, update the inode delalloc
> - * counter now and wait to update the sb counters as
> - * 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);
> + error = xfs_bmap_del_extent_delay(ip, whichfork, &lastx,
> + &got, &del);
> + } else {
> + /*
> + * If it's the case where the directory code is running
> + * with no block reservation, and the deleted block is
> + * in the middle of its extent, and the resulting insert
> + * of an extent would cause transformation to btree
> + * format, then reject it. The calling code will then
> + * swap blocks around instead. We have to do this now,
> + * rather than waiting for the conversion to btree
> + * format, since the transaction will be dirty.
> + */
> + if (tp->t_blk_res == 0 &&
> + XFS_IFORK_FORMAT(ip, whichfork) ==
> + XFS_DINODE_FMT_EXTENTS &&
> + XFS_IFORK_NEXTENTS(ip, whichfork) >=
> + XFS_IFORK_MAXEXT(ip, whichfork) &&
> + del.br_startoff > got.br_startoff &&
> + del.br_startoff + del.br_blockcount <
> + got.br_startoff + got.br_blockcount) {
> + error = -ENOSPC;
> + goto error0;
> }
> - ip->i_delayed_blks -= del.br_blockcount;
> +
> + error = xfs_bmap_del_extent_real(ip, tp, &lastx, dfops,
> + cur, &del, &tmp_logflags, whichfork,
> + flags);
> + logflags |= tmp_logflags;
> }
>
> - error = xfs_bmap_del_extent(ip, tp, &lastx, dfops, cur, &del,
> - &tmp_logflags, whichfork, flags);
> - logflags |= tmp_logflags;
> if (error)
> goto error0;
>
> - if (!isrt && wasdel)
> - xfs_mod_fdblocks(mp, (int64_t)del.br_blockcount, false);
> -
> max_len -= del.br_blockcount;
> end = del.br_startoff - 1;
> nodelete:
> --
> 2.11.0
>
> --
> To unsubscribe from this list: send the line "unsubscribe linux-xfs" in
> the body of a message to majordomo@vger.kernel.org
> More majordomo info at http://vger.kernel.org/majordomo-info.html
next prev parent reply other threads:[~2017-09-07 15:46 UTC|newest]
Thread overview: 37+ messages / expand[flat|nested] mbox.gz Atom feed top
2017-09-03 7:29 refactor extent manipulation Christoph Hellwig
2017-09-03 7:29 ` [PATCH 01/17] xfs: fix incorrect extent state in xfs_bmap_add_extent_unwritten_real Christoph Hellwig
2017-09-07 15:46 ` Brian Foster
2017-09-03 7:29 ` [PATCH 02/17] xfs: use xfs_iext_get_extent instead of open coding it Christoph Hellwig
2017-09-07 15:46 ` Brian Foster
2017-09-03 7:29 ` [PATCH 03/17] xfs: don't set XFS_BTCUR_BPRV_WASDEL in xfs_bunmapi Christoph Hellwig
2017-09-07 15:46 ` Brian Foster
2017-09-03 7:29 ` [PATCH 04/17] xfs: rename bno to end in __xfs_bunmapi Christoph Hellwig
2017-09-07 15:46 ` Brian Foster
2017-09-03 7:29 ` [PATCH 05/17] xfs: use xfs_bmap_del_extent_delay for the data fork as well Christoph Hellwig
2017-09-07 15:46 ` Brian Foster [this message]
2017-09-03 7:29 ` [PATCH 06/17] xfs: move some more code into xfs_bmap_del_extent_real Christoph Hellwig
2017-09-07 15:47 ` Brian Foster
2017-09-03 7:29 ` [PATCH 07/17] xfs: use the proper state defines in xfs_bmap_del_extent_real Christoph Hellwig
2017-09-07 15:47 ` Brian Foster
2017-09-08 7:33 ` Christoph Hellwig
2017-09-03 7:29 ` [PATCH 08/17] xfs: refactor xfs_del_extent_real Christoph Hellwig
2017-09-07 20:21 ` Brian Foster
2017-09-03 7:29 ` [PATCH 09/17] xfs: refactor xfs_bmap_add_extent_hole_delay Christoph Hellwig
2017-09-07 20:21 ` Brian Foster
2017-09-03 7:29 ` [PATCH 10/17] xfs: refactor xfs_bmap_add_extent_hole_real Christoph Hellwig
2017-09-07 20:21 ` Brian Foster
2017-09-03 7:29 ` [PATCH 11/17] xfs: refactor xfs_bmap_add_extent_delay_real Christoph Hellwig
2017-09-08 17:18 ` Brian Foster
2017-09-14 13:23 ` Christoph Hellwig
2017-09-03 7:29 ` [PATCH 12/17] xfs: refactor xfs_bmap_add_extent_unwritten_real Christoph Hellwig
2017-09-08 17:19 ` Brian Foster
2017-09-03 7:29 ` [PATCH 13/17] xfs: pass a struct xfs_bmbt_irec to xfs_bmbt_update Christoph Hellwig
2017-09-08 17:19 ` Brian Foster
2017-09-03 7:29 ` [PATCH 14/17] xfs: pass a struct xfs_bmbt_irec to xfs_bmbt_lookup_eq Christoph Hellwig
2017-09-08 17:19 ` Brian Foster
2017-09-03 7:29 ` [PATCH 15/17] xfs: replace xfs_bmbt_lookup_ge with xfs_bmbt_lookup_first Christoph Hellwig
2017-09-08 17:19 ` Brian Foster
2017-09-03 7:29 ` [PATCH 16/17] xfs: remove all xfs_bmbt_set_* helpers except for xfs_bmbt_set_all Christoph Hellwig
2017-09-08 17:19 ` Brian Foster
2017-09-03 7:29 ` [PATCH 17/17] xfs: remove xfs_bmbt_get_state Christoph Hellwig
2017-09-08 17:19 ` Brian Foster
Reply instructions:
You may reply publicly to this message via plain-text email
using any one of the following methods:
* Save the following mbox file, import it into your mail client,
and reply-to-all from there: mbox
Avoid top-posting and favor interleaved quoting:
https://en.wikipedia.org/wiki/Posting_style#Interleaved_style
* Reply using the --to, --cc, and --in-reply-to
switches of git-send-email(1):
git send-email \
--in-reply-to=20170907154657.GE61662@bfoster.bfoster \
--to=bfoster@redhat.com \
--cc=hch@lst.de \
--cc=linux-xfs@vger.kernel.org \
/path/to/YOUR_REPLY
https://kernel.org/pub/software/scm/git/docs/git-send-email.html
* If your mail client supports setting the In-Reply-To header
via mailto: links, try the mailto: link
Be sure your reply has a Subject: header at the top and a blank line
before the message body.
This is an external index of several public inboxes,
see mirroring instructions on how to clone and mirror
all data and code used by this external index.