Commit 25576c54 authored by Christoph Hellwig's avatar Christoph Hellwig Committed by Chandan Babu R

xfs: simplify iext overflow checking and upgrade

Currently the calls to xfs_iext_count_may_overflow and
xfs_iext_count_upgrade are always paired.  Merge them into a single
function to simplify the callers and the actual check and upgrade
logic itself.
Signed-off-by: default avatarChristoph Hellwig <hch@lst.de>
Reviewed-by: default avatar"Darrick J. Wong" <djwong@kernel.org>
Reviewed-by: default avatarDave Chinner <dchinner@redhat.com>
Signed-off-by: default avatarChandan Babu R <chandanbabu@kernel.org>
parent 86de8484
...@@ -1050,10 +1050,7 @@ xfs_attr_set( ...@@ -1050,10 +1050,7 @@ xfs_attr_set(
return error; return error;
if (op != XFS_ATTRUPDATE_REMOVE || xfs_inode_hasattr(dp)) { if (op != XFS_ATTRUPDATE_REMOVE || xfs_inode_hasattr(dp)) {
error = xfs_iext_count_may_overflow(dp, XFS_ATTR_FORK, error = xfs_iext_count_extend(args->trans, dp, XFS_ATTR_FORK,
XFS_IEXT_ATTR_MANIP_CNT(rmt_blks));
if (error == -EFBIG)
error = xfs_iext_count_upgrade(args->trans, dp,
XFS_IEXT_ATTR_MANIP_CNT(rmt_blks)); XFS_IEXT_ATTR_MANIP_CNT(rmt_blks));
if (error) if (error)
goto out_trans_cancel; goto out_trans_cancel;
......
...@@ -4651,10 +4651,7 @@ xfs_bmapi_convert_one_delalloc( ...@@ -4651,10 +4651,7 @@ xfs_bmapi_convert_one_delalloc(
xfs_ilock(ip, XFS_ILOCK_EXCL); xfs_ilock(ip, XFS_ILOCK_EXCL);
xfs_trans_ijoin(tp, ip, 0); xfs_trans_ijoin(tp, ip, 0);
error = xfs_iext_count_may_overflow(ip, whichfork, error = xfs_iext_count_extend(tp, ip, whichfork,
XFS_IEXT_ADD_NOSPLIT_CNT);
if (error == -EFBIG)
error = xfs_iext_count_upgrade(tp, ip,
XFS_IEXT_ADD_NOSPLIT_CNT); XFS_IEXT_ADD_NOSPLIT_CNT);
if (error) if (error)
goto out_trans_cancel; goto out_trans_cancel;
......
...@@ -765,53 +765,46 @@ xfs_ifork_verify_local_attr( ...@@ -765,53 +765,46 @@ xfs_ifork_verify_local_attr(
return 0; return 0;
} }
/*
* Check if the inode fork supports adding nr_to_add more extents.
*
* If it doesn't but we can upgrade it to large extent counters, do the upgrade.
* If we can't upgrade or are already using big counters but still can't fit the
* additional extents, return -EFBIG.
*/
int int
xfs_iext_count_may_overflow( xfs_iext_count_extend(
struct xfs_trans *tp,
struct xfs_inode *ip, struct xfs_inode *ip,
int whichfork, int whichfork,
int nr_to_add) uint nr_to_add)
{ {
struct xfs_mount *mp = ip->i_mount;
bool has_large =
xfs_inode_has_large_extent_counts(ip);
struct xfs_ifork *ifp = xfs_ifork_ptr(ip, whichfork); struct xfs_ifork *ifp = xfs_ifork_ptr(ip, whichfork);
uint64_t max_exts;
uint64_t nr_exts; uint64_t nr_exts;
ASSERT(nr_to_add <= XFS_MAX_EXTCNT_UPGRADE_NR);
if (whichfork == XFS_COW_FORK) if (whichfork == XFS_COW_FORK)
return 0; return 0;
max_exts = xfs_iext_max_nextents(xfs_inode_has_large_extent_counts(ip), /* no point in upgrading if if_nextents overflows */
whichfork);
if (XFS_TEST_ERROR(false, ip->i_mount, XFS_ERRTAG_REDUCE_MAX_IEXTENTS))
max_exts = 10;
nr_exts = ifp->if_nextents + nr_to_add; nr_exts = ifp->if_nextents + nr_to_add;
if (nr_exts < ifp->if_nextents || nr_exts > max_exts) if (nr_exts < ifp->if_nextents)
return -EFBIG; return -EFBIG;
return 0; if (XFS_TEST_ERROR(false, mp, XFS_ERRTAG_REDUCE_MAX_IEXTENTS) &&
} nr_exts > 10)
/*
* Upgrade this inode's extent counter fields to be able to handle a potential
* increase in the extent count by nr_to_add. Normally this is the same
* quantity that caused xfs_iext_count_may_overflow() to return -EFBIG.
*/
int
xfs_iext_count_upgrade(
struct xfs_trans *tp,
struct xfs_inode *ip,
uint nr_to_add)
{
ASSERT(nr_to_add <= XFS_MAX_EXTCNT_UPGRADE_NR);
if (!xfs_has_large_extent_counts(ip->i_mount) ||
xfs_inode_has_large_extent_counts(ip) ||
XFS_TEST_ERROR(false, ip->i_mount, XFS_ERRTAG_REDUCE_MAX_IEXTENTS))
return -EFBIG; return -EFBIG;
if (nr_exts > xfs_iext_max_nextents(has_large, whichfork)) {
if (has_large || !xfs_has_large_extent_counts(mp))
return -EFBIG;
ip->i_diflags2 |= XFS_DIFLAG2_NREXT64; ip->i_diflags2 |= XFS_DIFLAG2_NREXT64;
xfs_trans_log_inode(tp, ip, XFS_ILOG_CORE); xfs_trans_log_inode(tp, ip, XFS_ILOG_CORE);
}
return 0; return 0;
} }
......
...@@ -256,10 +256,8 @@ extern void xfs_ifork_init_cow(struct xfs_inode *ip); ...@@ -256,10 +256,8 @@ extern void xfs_ifork_init_cow(struct xfs_inode *ip);
int xfs_ifork_verify_local_data(struct xfs_inode *ip); int xfs_ifork_verify_local_data(struct xfs_inode *ip);
int xfs_ifork_verify_local_attr(struct xfs_inode *ip); int xfs_ifork_verify_local_attr(struct xfs_inode *ip);
int xfs_iext_count_may_overflow(struct xfs_inode *ip, int whichfork, int xfs_iext_count_extend(struct xfs_trans *tp, struct xfs_inode *ip,
int nr_to_add); int whichfork, uint nr_to_add);
int xfs_iext_count_upgrade(struct xfs_trans *tp, struct xfs_inode *ip,
uint nr_to_add);
bool xfs_ifork_is_realtime(struct xfs_inode *ip, int whichfork); bool xfs_ifork_is_realtime(struct xfs_inode *ip, int whichfork);
/* returns true if the fork has extents but they are not read in yet. */ /* returns true if the fork has extents but they are not read in yet. */
......
...@@ -524,9 +524,7 @@ xfs_bmap_recover_work( ...@@ -524,9 +524,7 @@ xfs_bmap_recover_work(
else else
iext_delta = XFS_IEXT_PUNCH_HOLE_CNT; iext_delta = XFS_IEXT_PUNCH_HOLE_CNT;
error = xfs_iext_count_may_overflow(ip, work->bi_whichfork, iext_delta); error = xfs_iext_count_extend(tp, ip, work->bi_whichfork, iext_delta);
if (error == -EFBIG)
error = xfs_iext_count_upgrade(tp, ip, iext_delta);
if (error) if (error)
goto err_cancel; goto err_cancel;
......
...@@ -710,10 +710,7 @@ xfs_alloc_file_space( ...@@ -710,10 +710,7 @@ xfs_alloc_file_space(
if (error) if (error)
break; break;
error = xfs_iext_count_may_overflow(ip, XFS_DATA_FORK, error = xfs_iext_count_extend(tp, ip, XFS_DATA_FORK,
XFS_IEXT_ADD_NOSPLIT_CNT);
if (error == -EFBIG)
error = xfs_iext_count_upgrade(tp, ip,
XFS_IEXT_ADD_NOSPLIT_CNT); XFS_IEXT_ADD_NOSPLIT_CNT);
if (error) if (error)
goto error; goto error;
...@@ -771,10 +768,8 @@ xfs_unmap_extent( ...@@ -771,10 +768,8 @@ xfs_unmap_extent(
if (error) if (error)
return error; return error;
error = xfs_iext_count_may_overflow(ip, XFS_DATA_FORK, error = xfs_iext_count_extend(tp, ip, XFS_DATA_FORK,
XFS_IEXT_PUNCH_HOLE_CNT); XFS_IEXT_PUNCH_HOLE_CNT);
if (error == -EFBIG)
error = xfs_iext_count_upgrade(tp, ip, XFS_IEXT_PUNCH_HOLE_CNT);
if (error) if (error)
goto out_trans_cancel; goto out_trans_cancel;
...@@ -1050,10 +1045,8 @@ xfs_insert_file_space( ...@@ -1050,10 +1045,8 @@ xfs_insert_file_space(
xfs_ilock(ip, XFS_ILOCK_EXCL); xfs_ilock(ip, XFS_ILOCK_EXCL);
xfs_trans_ijoin(tp, ip, 0); xfs_trans_ijoin(tp, ip, 0);
error = xfs_iext_count_may_overflow(ip, XFS_DATA_FORK, error = xfs_iext_count_extend(tp, ip, XFS_DATA_FORK,
XFS_IEXT_PUNCH_HOLE_CNT); XFS_IEXT_PUNCH_HOLE_CNT);
if (error == -EFBIG)
error = xfs_iext_count_upgrade(tp, ip, XFS_IEXT_PUNCH_HOLE_CNT);
if (error) if (error)
goto out_trans_cancel; goto out_trans_cancel;
...@@ -1279,23 +1272,17 @@ xfs_swap_extent_rmap( ...@@ -1279,23 +1272,17 @@ xfs_swap_extent_rmap(
trace_xfs_swap_extent_rmap_remap_piece(tip, &uirec); trace_xfs_swap_extent_rmap_remap_piece(tip, &uirec);
if (xfs_bmap_is_real_extent(&uirec)) { if (xfs_bmap_is_real_extent(&uirec)) {
error = xfs_iext_count_may_overflow(ip, error = xfs_iext_count_extend(tp, ip,
XFS_DATA_FORK, XFS_DATA_FORK,
XFS_IEXT_SWAP_RMAP_CNT); XFS_IEXT_SWAP_RMAP_CNT);
if (error == -EFBIG)
error = xfs_iext_count_upgrade(tp, ip,
XFS_IEXT_SWAP_RMAP_CNT);
if (error) if (error)
goto out; goto out;
} }
if (xfs_bmap_is_real_extent(&irec)) { if (xfs_bmap_is_real_extent(&irec)) {
error = xfs_iext_count_may_overflow(tip, error = xfs_iext_count_extend(tp, tip,
XFS_DATA_FORK, XFS_DATA_FORK,
XFS_IEXT_SWAP_RMAP_CNT); XFS_IEXT_SWAP_RMAP_CNT);
if (error == -EFBIG)
error = xfs_iext_count_upgrade(tp, ip,
XFS_IEXT_SWAP_RMAP_CNT);
if (error) if (error)
goto out; goto out;
} }
......
...@@ -341,10 +341,7 @@ xfs_dquot_disk_alloc( ...@@ -341,10 +341,7 @@ xfs_dquot_disk_alloc(
goto err_cancel; goto err_cancel;
} }
error = xfs_iext_count_may_overflow(quotip, XFS_DATA_FORK, error = xfs_iext_count_extend(tp, quotip, XFS_DATA_FORK,
XFS_IEXT_ADD_NOSPLIT_CNT);
if (error == -EFBIG)
error = xfs_iext_count_upgrade(tp, quotip,
XFS_IEXT_ADD_NOSPLIT_CNT); XFS_IEXT_ADD_NOSPLIT_CNT);
if (error) if (error)
goto err_cancel; goto err_cancel;
......
...@@ -299,9 +299,7 @@ xfs_iomap_write_direct( ...@@ -299,9 +299,7 @@ xfs_iomap_write_direct(
if (error) if (error)
return error; return error;
error = xfs_iext_count_may_overflow(ip, XFS_DATA_FORK, nr_exts); error = xfs_iext_count_extend(tp, ip, XFS_DATA_FORK, nr_exts);
if (error == -EFBIG)
error = xfs_iext_count_upgrade(tp, ip, nr_exts);
if (error) if (error)
goto out_trans_cancel; goto out_trans_cancel;
...@@ -617,10 +615,7 @@ xfs_iomap_write_unwritten( ...@@ -617,10 +615,7 @@ xfs_iomap_write_unwritten(
if (error) if (error)
return error; return error;
error = xfs_iext_count_may_overflow(ip, XFS_DATA_FORK, error = xfs_iext_count_extend(tp, ip, XFS_DATA_FORK,
XFS_IEXT_WRITE_UNWRITTEN_CNT);
if (error == -EFBIG)
error = xfs_iext_count_upgrade(tp, ip,
XFS_IEXT_WRITE_UNWRITTEN_CNT); XFS_IEXT_WRITE_UNWRITTEN_CNT);
if (error) if (error)
goto error_on_bmapi_transaction; goto error_on_bmapi_transaction;
......
...@@ -754,10 +754,7 @@ xfs_reflink_end_cow_extent( ...@@ -754,10 +754,7 @@ xfs_reflink_end_cow_extent(
del = got; del = got;
xfs_trim_extent(&del, *offset_fsb, end_fsb - *offset_fsb); xfs_trim_extent(&del, *offset_fsb, end_fsb - *offset_fsb);
error = xfs_iext_count_may_overflow(ip, XFS_DATA_FORK, error = xfs_iext_count_extend(tp, ip, XFS_DATA_FORK,
XFS_IEXT_REFLINK_END_COW_CNT);
if (error == -EFBIG)
error = xfs_iext_count_upgrade(tp, ip,
XFS_IEXT_REFLINK_END_COW_CNT); XFS_IEXT_REFLINK_END_COW_CNT);
if (error) if (error)
goto out_cancel; goto out_cancel;
...@@ -1258,9 +1255,7 @@ xfs_reflink_remap_extent( ...@@ -1258,9 +1255,7 @@ xfs_reflink_remap_extent(
if (dmap_written) if (dmap_written)
++iext_delta; ++iext_delta;
error = xfs_iext_count_may_overflow(ip, XFS_DATA_FORK, iext_delta); error = xfs_iext_count_extend(tp, ip, XFS_DATA_FORK, iext_delta);
if (error == -EFBIG)
error = xfs_iext_count_upgrade(tp, ip, iext_delta);
if (error) if (error)
goto out_cancel; goto out_cancel;
......
...@@ -695,10 +695,7 @@ xfs_growfs_rt_alloc( ...@@ -695,10 +695,7 @@ xfs_growfs_rt_alloc(
xfs_ilock(ip, XFS_ILOCK_EXCL); xfs_ilock(ip, XFS_ILOCK_EXCL);
xfs_trans_ijoin(tp, ip, XFS_ILOCK_EXCL); xfs_trans_ijoin(tp, ip, XFS_ILOCK_EXCL);
error = xfs_iext_count_may_overflow(ip, XFS_DATA_FORK, error = xfs_iext_count_extend(tp, ip, XFS_DATA_FORK,
XFS_IEXT_ADD_NOSPLIT_CNT);
if (error == -EFBIG)
error = xfs_iext_count_upgrade(tp, ip,
XFS_IEXT_ADD_NOSPLIT_CNT); XFS_IEXT_ADD_NOSPLIT_CNT);
if (error) if (error)
goto out_trans_cancel; goto out_trans_cancel;
......
Markdown is supported
0%
or
You are about to add 0 people to the discussion. Proceed with caution.
Finish editing this message first!
Please register or to comment