Commit 55edcd1f authored by Darrick J. Wong's avatar Darrick J. Wong

xfs: split xfs_bmap_add_attrfork into two pieces

Split this function into two pieces -- one to make the actual changes to
the inode core to add the attr fork, and another one to deal with
getting the transaction and locking the inodes.

The next couple of patches will need this to be split into two.  One
patch implements committing new parent pointer recordsets to damaged
files.  If one file has an attr fork and the other does not, we have to
create the missing attr fork before the atomic swap transaction, and can
use the behavior encoded in the current xfs_bmap_add_attrfork.

The second patch adapts /lost+found adoptions to handle parent pointers
correctly.  The adoption process will add a parent pointer to a child
that is being moved to /lost+found, but this requires that the attr fork
already exists.  We don't know if we're actually going to commit the
adoption until we've already reserved a transaction and taken the
ILOCKs, which means that we must have a way to bypass the start of the
current xfs_bmap_add_attrfork.

Therefore, create xfs_attr_add_fork as the helper that creates a
transaction and takes locks; and make xfs_bmap_add_attrfork the function
that updates the inode core and allocates the incore attr fork.
Signed-off-by: default avatarDarrick J. Wong <djwong@kernel.org>
Reviewed-by: default avatarChristoph Hellwig <hch@lst.de>
parent 13db7007
...@@ -948,6 +948,43 @@ xfs_attr_lookup( ...@@ -948,6 +948,43 @@ xfs_attr_lookup(
return error; return error;
} }
STATIC int
xfs_attr_add_fork(
struct xfs_inode *ip, /* incore inode pointer */
int size, /* space new attribute needs */
int rsvd) /* xact may use reserved blks */
{
struct xfs_mount *mp = ip->i_mount;
struct xfs_trans *tp; /* transaction pointer */
unsigned int blks; /* space reservation */
int error; /* error return value */
ASSERT(!XFS_NOT_DQATTACHED(mp, ip));
blks = XFS_ADDAFORK_SPACE_RES(mp);
error = xfs_trans_alloc_inode(ip, &M_RES(mp)->tr_addafork, blks, 0,
rsvd, &tp);
if (error)
return error;
if (xfs_inode_has_attr_fork(ip))
goto trans_cancel;
error = xfs_bmap_add_attrfork(tp, ip, size, rsvd);
if (error)
goto trans_cancel;
error = xfs_trans_commit(tp);
xfs_iunlock(ip, XFS_ILOCK_EXCL);
return error;
trans_cancel:
xfs_trans_cancel(tp);
xfs_iunlock(ip, XFS_ILOCK_EXCL);
return error;
}
/* /*
* Make a change to the xattr structure. * Make a change to the xattr structure.
* *
...@@ -989,7 +1026,7 @@ xfs_attr_set( ...@@ -989,7 +1026,7 @@ xfs_attr_set(
xfs_attr_sf_entsize_byname(args->namelen, xfs_attr_sf_entsize_byname(args->namelen,
args->valuelen); args->valuelen);
error = xfs_bmap_add_attrfork(dp, sf_size, rsvd); error = xfs_attr_add_fork(dp, sf_size, rsvd);
if (error) if (error)
return error; return error;
} }
......
...@@ -1025,38 +1025,29 @@ xfs_bmap_set_attrforkoff( ...@@ -1025,38 +1025,29 @@ xfs_bmap_set_attrforkoff(
} }
/* /*
* Convert inode from non-attributed to attributed. * Convert inode from non-attributed to attributed. Caller must hold the
* Must not be in a transaction, ip must not be locked. * ILOCK_EXCL and the file cannot have an attr fork.
*/ */
int /* error code */ int /* error code */
xfs_bmap_add_attrfork( xfs_bmap_add_attrfork(
xfs_inode_t *ip, /* incore inode pointer */ struct xfs_trans *tp,
struct xfs_inode *ip, /* incore inode pointer */
int size, /* space new attribute needs */ int size, /* space new attribute needs */
int rsvd) /* xact may use reserved blks */ int rsvd) /* xact may use reserved blks */
{ {
xfs_mount_t *mp; /* mount structure */ struct xfs_mount *mp = tp->t_mountp;
xfs_trans_t *tp; /* transaction pointer */
int blks; /* space reservation */
int version = 1; /* superblock attr version */ int version = 1; /* superblock attr version */
int logflags; /* logging flags */ int logflags; /* logging flags */
int error; /* error return value */ int error; /* error return value */
mp = ip->i_mount; xfs_assert_ilocked(ip, XFS_ILOCK_EXCL);
ASSERT(!XFS_NOT_DQATTACHED(mp, ip)); ASSERT(!XFS_NOT_DQATTACHED(mp, ip));
ASSERT(!xfs_inode_has_attr_fork(ip));
blks = XFS_ADDAFORK_SPACE_RES(mp);
error = xfs_trans_alloc_inode(ip, &M_RES(mp)->tr_addafork, blks, 0,
rsvd, &tp);
if (error)
return error;
if (xfs_inode_has_attr_fork(ip))
goto trans_cancel;
xfs_trans_log_inode(tp, ip, XFS_ILOG_CORE); xfs_trans_log_inode(tp, ip, XFS_ILOG_CORE);
error = xfs_bmap_set_attrforkoff(ip, size, &version); error = xfs_bmap_set_attrforkoff(ip, size, &version);
if (error) if (error)
goto trans_cancel; return error;
xfs_ifork_init_attr(ip, XFS_DINODE_FMT_EXTENTS, 0); xfs_ifork_init_attr(ip, XFS_DINODE_FMT_EXTENTS, 0);
logflags = 0; logflags = 0;
...@@ -1077,7 +1068,7 @@ xfs_bmap_add_attrfork( ...@@ -1077,7 +1068,7 @@ xfs_bmap_add_attrfork(
if (logflags) if (logflags)
xfs_trans_log_inode(tp, ip, logflags); xfs_trans_log_inode(tp, ip, logflags);
if (error) if (error)
goto trans_cancel; return error;
if (!xfs_has_attr(mp) || if (!xfs_has_attr(mp) ||
(!xfs_has_attr2(mp) && version == 2)) { (!xfs_has_attr2(mp) && version == 2)) {
bool log_sb = false; bool log_sb = false;
...@@ -1096,14 +1087,7 @@ xfs_bmap_add_attrfork( ...@@ -1096,14 +1087,7 @@ xfs_bmap_add_attrfork(
xfs_log_sb(tp); xfs_log_sb(tp);
} }
error = xfs_trans_commit(tp); return 0;
xfs_iunlock(ip, XFS_ILOCK_EXCL);
return error;
trans_cancel:
xfs_trans_cancel(tp);
xfs_iunlock(ip, XFS_ILOCK_EXCL);
return error;
} }
/* /*
......
...@@ -176,7 +176,8 @@ int xfs_bmap_longest_free_extent(struct xfs_perag *pag, ...@@ -176,7 +176,8 @@ int xfs_bmap_longest_free_extent(struct xfs_perag *pag,
void xfs_trim_extent(struct xfs_bmbt_irec *irec, xfs_fileoff_t bno, void xfs_trim_extent(struct xfs_bmbt_irec *irec, xfs_fileoff_t bno,
xfs_filblks_t len); xfs_filblks_t len);
unsigned int xfs_bmap_compute_attr_offset(struct xfs_mount *mp); unsigned int xfs_bmap_compute_attr_offset(struct xfs_mount *mp);
int xfs_bmap_add_attrfork(struct xfs_inode *ip, int size, int rsvd); int xfs_bmap_add_attrfork(struct xfs_trans *tp, struct xfs_inode *ip,
int size, int rsvd);
void xfs_bmap_local_to_extents_empty(struct xfs_trans *tp, void xfs_bmap_local_to_extents_empty(struct xfs_trans *tp,
struct xfs_inode *ip, int whichfork); struct xfs_inode *ip, int whichfork);
int xfs_bmap_local_to_extents(struct xfs_trans *tp, struct xfs_inode *ip, int xfs_bmap_local_to_extents(struct xfs_trans *tp, struct xfs_inode *ip,
......
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