Commit ea7cdd7b authored by Darrick J. Wong's avatar Darrick J. Wong

xfs: separate function to check if inode shares extents

Separate the "clear reflink flag" function into one function that checks
if the flag is needed, and a second function that checks and clears the
flag.  The inode scrub code will want to check the necessity of the flag
without clearing it.
Signed-off-by: default avatarDarrick J. Wong <darrick.wong@oracle.com>
Reviewed-by: default avatarBrian Foster <bfoster@redhat.com>
parent 92ff7285
...@@ -1406,57 +1406,73 @@ xfs_reflink_dirty_extents( ...@@ -1406,57 +1406,73 @@ xfs_reflink_dirty_extents(
return error; return error;
} }
/* Clear the inode reflink flag if there are no shared extents. */ /* Does this inode need the reflink flag? */
int int
xfs_reflink_clear_inode_flag( xfs_reflink_inode_has_shared_extents(
struct xfs_trans *tp,
struct xfs_inode *ip, struct xfs_inode *ip,
struct xfs_trans **tpp) bool *has_shared)
{ {
struct xfs_bmbt_irec got;
struct xfs_mount *mp = ip->i_mount; struct xfs_mount *mp = ip->i_mount;
xfs_fileoff_t fbno; struct xfs_ifork *ifp;
xfs_filblks_t end;
xfs_agnumber_t agno; xfs_agnumber_t agno;
xfs_agblock_t agbno; xfs_agblock_t agbno;
xfs_extlen_t aglen; xfs_extlen_t aglen;
xfs_agblock_t rbno; xfs_agblock_t rbno;
xfs_extlen_t rlen; xfs_extlen_t rlen;
struct xfs_bmbt_irec map; xfs_extnum_t idx;
int nmaps; bool found;
int error = 0; int error;
ASSERT(xfs_is_reflink_inode(ip));
fbno = 0; ifp = XFS_IFORK_PTR(ip, XFS_DATA_FORK);
end = XFS_B_TO_FSB(mp, i_size_read(VFS_I(ip))); if (!(ifp->if_flags & XFS_IFEXTENTS)) {
while (end - fbno > 0) { error = xfs_iread_extents(tp, ip, XFS_DATA_FORK);
nmaps = 1;
/*
* Look for extents in the file. Skip holes, delalloc, or
* unwritten extents; they can't be reflinked.
*/
error = xfs_bmapi_read(ip, fbno, end - fbno, &map, &nmaps, 0);
if (error) if (error)
return error; return error;
if (nmaps == 0) }
break;
if (!xfs_bmap_is_real_extent(&map))
goto next;
agno = XFS_FSB_TO_AGNO(mp, map.br_startblock); *has_shared = false;
agbno = XFS_FSB_TO_AGBNO(mp, map.br_startblock); found = xfs_iext_lookup_extent(ip, ifp, 0, &idx, &got);
aglen = map.br_blockcount; while (found) {
if (isnullstartblock(got.br_startblock) ||
got.br_state != XFS_EXT_NORM)
goto next;
agno = XFS_FSB_TO_AGNO(mp, got.br_startblock);
agbno = XFS_FSB_TO_AGBNO(mp, got.br_startblock);
aglen = got.br_blockcount;
error = xfs_reflink_find_shared(mp, *tpp, agno, agbno, aglen, error = xfs_reflink_find_shared(mp, tp, agno, agbno, aglen,
&rbno, &rlen, false); &rbno, &rlen, false);
if (error) if (error)
return error; return error;
/* Is there still a shared block here? */ /* Is there still a shared block here? */
if (rbno != NULLAGBLOCK) if (rbno != NULLAGBLOCK) {
*has_shared = true;
return 0; return 0;
}
next: next:
fbno = map.br_startoff + map.br_blockcount; found = xfs_iext_get_extent(ifp, ++idx, &got);
} }
return 0;
}
/* Clear the inode reflink flag if there are no shared extents. */
int
xfs_reflink_clear_inode_flag(
struct xfs_inode *ip,
struct xfs_trans **tpp)
{
bool needs_flag;
int error = 0;
ASSERT(xfs_is_reflink_inode(ip));
error = xfs_reflink_inode_has_shared_extents(*tpp, ip, &needs_flag);
if (error || needs_flag)
return error;
/* /*
* We didn't find any shared blocks so turn off the reflink flag. * We didn't find any shared blocks so turn off the reflink flag.
* First, get rid of any leftover CoW mappings. * First, get rid of any leftover CoW mappings.
......
...@@ -47,6 +47,8 @@ extern int xfs_reflink_end_cow(struct xfs_inode *ip, xfs_off_t offset, ...@@ -47,6 +47,8 @@ extern int xfs_reflink_end_cow(struct xfs_inode *ip, xfs_off_t offset,
extern int xfs_reflink_recover_cow(struct xfs_mount *mp); extern int xfs_reflink_recover_cow(struct xfs_mount *mp);
extern int xfs_reflink_remap_range(struct file *file_in, loff_t pos_in, extern int xfs_reflink_remap_range(struct file *file_in, loff_t pos_in,
struct file *file_out, loff_t pos_out, u64 len, bool is_dedupe); struct file *file_out, loff_t pos_out, u64 len, bool is_dedupe);
extern int xfs_reflink_inode_has_shared_extents(struct xfs_trans *tp,
struct xfs_inode *ip, bool *has_shared);
extern int xfs_reflink_clear_inode_flag(struct xfs_inode *ip, extern int xfs_reflink_clear_inode_flag(struct xfs_inode *ip,
struct xfs_trans **tpp); struct xfs_trans **tpp);
extern int xfs_reflink_unshare(struct xfs_inode *ip, xfs_off_t offset, extern int xfs_reflink_unshare(struct xfs_inode *ip, xfs_off_t offset,
......
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