Commit 9eef772f authored by Darrick J. Wong's avatar Darrick J. Wong

xfs: add an explicit owner field to xfs_da_args

Add an explicit owner field to xfs_da_args, which will make it easier
for online fsck to set the owner field of the temporary directory and
xattr structures that it builds to repair damaged metadata.

Note: I hopefully found all the xfs_da_args definitions by looking for
automatic stack variable declarations and xfs_da_args.dp assignments:

git grep -E '(args.*dp =|struct xfs_da_args[[:space:]]*[a-z0-9][a-z0-9]*)'

Note that callers of xfs_attr_{get,set,change} can set the owner to zero
(or leave it unset) to have the default set to args->dp.
Signed-off-by: default avatarDarrick J. Wong <djwong@kernel.org>
Reviewed-by: default avatarChristoph Hellwig <hch@lst.de>
parent abf039e2
...@@ -264,6 +264,8 @@ xfs_attr_get( ...@@ -264,6 +264,8 @@ xfs_attr_get(
if (xfs_is_shutdown(args->dp->i_mount)) if (xfs_is_shutdown(args->dp->i_mount))
return -EIO; return -EIO;
if (!args->owner)
args->owner = args->dp->i_ino;
args->geo = args->dp->i_mount->m_attr_geo; args->geo = args->dp->i_mount->m_attr_geo;
args->whichfork = XFS_ATTR_FORK; args->whichfork = XFS_ATTR_FORK;
args->hashval = xfs_da_hashname(args->name, args->namelen); args->hashval = xfs_da_hashname(args->name, args->namelen);
...@@ -937,6 +939,8 @@ xfs_attr_set( ...@@ -937,6 +939,8 @@ xfs_attr_set(
if (error) if (error)
return error; return error;
if (!args->owner)
args->owner = args->dp->i_ino;
args->geo = mp->m_attr_geo; args->geo = mp->m_attr_geo;
args->whichfork = XFS_ATTR_FORK; args->whichfork = XFS_ATTR_FORK;
args->hashval = xfs_da_hashname(args->name, args->namelen); args->hashval = xfs_da_hashname(args->name, args->namelen);
......
...@@ -904,6 +904,7 @@ xfs_attr_shortform_to_leaf( ...@@ -904,6 +904,7 @@ xfs_attr_shortform_to_leaf(
nargs.whichfork = XFS_ATTR_FORK; nargs.whichfork = XFS_ATTR_FORK;
nargs.trans = args->trans; nargs.trans = args->trans;
nargs.op_flags = XFS_DA_OP_OKNOENT; nargs.op_flags = XFS_DA_OP_OKNOENT;
nargs.owner = args->owner;
sfe = xfs_attr_sf_firstentry(sf); sfe = xfs_attr_sf_firstentry(sf);
for (i = 0; i < sf->count; i++) { for (i = 0; i < sf->count; i++) {
...@@ -1106,6 +1107,7 @@ xfs_attr3_leaf_to_shortform( ...@@ -1106,6 +1107,7 @@ xfs_attr3_leaf_to_shortform(
nargs.whichfork = XFS_ATTR_FORK; nargs.whichfork = XFS_ATTR_FORK;
nargs.trans = args->trans; nargs.trans = args->trans;
nargs.op_flags = XFS_DA_OP_OKNOENT; nargs.op_flags = XFS_DA_OP_OKNOENT;
nargs.owner = args->owner;
for (i = 0; i < ichdr.count; entry++, i++) { for (i = 0; i < ichdr.count; entry++, i++) {
if (entry->flags & XFS_ATTR_INCOMPLETE) if (entry->flags & XFS_ATTR_INCOMPLETE)
......
...@@ -976,6 +976,7 @@ xfs_bmap_add_attrfork_local( ...@@ -976,6 +976,7 @@ xfs_bmap_add_attrfork_local(
dargs.total = dargs.geo->fsbcount; dargs.total = dargs.geo->fsbcount;
dargs.whichfork = XFS_DATA_FORK; dargs.whichfork = XFS_DATA_FORK;
dargs.trans = tp; dargs.trans = tp;
dargs.owner = ip->i_ino;
return xfs_dir2_sf_to_block(&dargs); return xfs_dir2_sf_to_block(&dargs);
} }
......
...@@ -79,6 +79,7 @@ typedef struct xfs_da_args { ...@@ -79,6 +79,7 @@ typedef struct xfs_da_args {
int rmtvaluelen2; /* remote attr value length in bytes */ int rmtvaluelen2; /* remote attr value length in bytes */
uint32_t op_flags; /* operation flags */ uint32_t op_flags; /* operation flags */
enum xfs_dacmp cmpresult; /* name compare result for lookups */ enum xfs_dacmp cmpresult; /* name compare result for lookups */
xfs_ino_t owner; /* inode that owns the dir/attr data */
} xfs_da_args_t; } xfs_da_args_t;
/* /*
......
...@@ -250,6 +250,7 @@ xfs_dir_init( ...@@ -250,6 +250,7 @@ xfs_dir_init(
args->geo = dp->i_mount->m_dir_geo; args->geo = dp->i_mount->m_dir_geo;
args->dp = dp; args->dp = dp;
args->trans = tp; args->trans = tp;
args->owner = dp->i_ino;
error = xfs_dir2_sf_create(args, pdp->i_ino); error = xfs_dir2_sf_create(args, pdp->i_ino);
kfree(args); kfree(args);
return error; return error;
...@@ -295,6 +296,7 @@ xfs_dir_createname( ...@@ -295,6 +296,7 @@ xfs_dir_createname(
args->whichfork = XFS_DATA_FORK; args->whichfork = XFS_DATA_FORK;
args->trans = tp; args->trans = tp;
args->op_flags = XFS_DA_OP_ADDNAME | XFS_DA_OP_OKNOENT; args->op_flags = XFS_DA_OP_ADDNAME | XFS_DA_OP_OKNOENT;
args->owner = dp->i_ino;
if (!inum) if (!inum)
args->op_flags |= XFS_DA_OP_JUSTCHECK; args->op_flags |= XFS_DA_OP_JUSTCHECK;
...@@ -383,6 +385,7 @@ xfs_dir_lookup( ...@@ -383,6 +385,7 @@ xfs_dir_lookup(
args->whichfork = XFS_DATA_FORK; args->whichfork = XFS_DATA_FORK;
args->trans = tp; args->trans = tp;
args->op_flags = XFS_DA_OP_OKNOENT; args->op_flags = XFS_DA_OP_OKNOENT;
args->owner = dp->i_ino;
if (ci_name) if (ci_name)
args->op_flags |= XFS_DA_OP_CILOOKUP; args->op_flags |= XFS_DA_OP_CILOOKUP;
...@@ -456,6 +459,7 @@ xfs_dir_removename( ...@@ -456,6 +459,7 @@ xfs_dir_removename(
args->total = total; args->total = total;
args->whichfork = XFS_DATA_FORK; args->whichfork = XFS_DATA_FORK;
args->trans = tp; args->trans = tp;
args->owner = dp->i_ino;
if (dp->i_df.if_format == XFS_DINODE_FMT_LOCAL) { if (dp->i_df.if_format == XFS_DINODE_FMT_LOCAL) {
rval = xfs_dir2_sf_removename(args); rval = xfs_dir2_sf_removename(args);
...@@ -517,6 +521,7 @@ xfs_dir_replace( ...@@ -517,6 +521,7 @@ xfs_dir_replace(
args->total = total; args->total = total;
args->whichfork = XFS_DATA_FORK; args->whichfork = XFS_DATA_FORK;
args->trans = tp; args->trans = tp;
args->owner = dp->i_ino;
if (dp->i_df.if_format == XFS_DINODE_FMT_LOCAL) { if (dp->i_df.if_format == XFS_DINODE_FMT_LOCAL) {
rval = xfs_dir2_sf_replace(args); rval = xfs_dir2_sf_replace(args);
......
...@@ -429,6 +429,7 @@ xfs_exchmaps_attr_to_sf( ...@@ -429,6 +429,7 @@ xfs_exchmaps_attr_to_sf(
.geo = tp->t_mountp->m_attr_geo, .geo = tp->t_mountp->m_attr_geo,
.whichfork = XFS_ATTR_FORK, .whichfork = XFS_ATTR_FORK,
.trans = tp, .trans = tp,
.owner = xmi->xmi_ip2->i_ino,
}; };
struct xfs_buf *bp; struct xfs_buf *bp;
int forkoff; int forkoff;
...@@ -459,6 +460,7 @@ xfs_exchmaps_dir_to_sf( ...@@ -459,6 +460,7 @@ xfs_exchmaps_dir_to_sf(
.geo = tp->t_mountp->m_dir_geo, .geo = tp->t_mountp->m_dir_geo,
.whichfork = XFS_DATA_FORK, .whichfork = XFS_DATA_FORK,
.trans = tp, .trans = tp,
.owner = xmi->xmi_ip2->i_ino,
}; };
struct xfs_dir2_sf_hdr sfh; struct xfs_dir2_sf_hdr sfh;
struct xfs_buf *bp; struct xfs_buf *bp;
......
...@@ -169,6 +169,7 @@ xchk_xattr_listent( ...@@ -169,6 +169,7 @@ xchk_xattr_listent(
.hashval = xfs_da_hashname(name, namelen), .hashval = xfs_da_hashname(name, namelen),
.trans = context->tp, .trans = context->tp,
.valuelen = valuelen, .valuelen = valuelen,
.owner = context->dp->i_ino,
}; };
struct xchk_xattr_buf *ab; struct xchk_xattr_buf *ab;
struct xchk_xattr *sx; struct xchk_xattr *sx;
......
...@@ -494,6 +494,7 @@ xchk_da_btree( ...@@ -494,6 +494,7 @@ xchk_da_btree(
ds->dargs.whichfork = whichfork; ds->dargs.whichfork = whichfork;
ds->dargs.trans = sc->tp; ds->dargs.trans = sc->tp;
ds->dargs.op_flags = XFS_DA_OP_OKNOENT; ds->dargs.op_flags = XFS_DA_OP_OKNOENT;
ds->dargs.owner = sc->ip->i_ino;
ds->state = xfs_da_state_alloc(&ds->dargs); ds->state = xfs_da_state_alloc(&ds->dargs);
ds->sc = sc; ds->sc = sc;
ds->private = private; ds->private = private;
......
...@@ -621,10 +621,11 @@ xchk_directory_blocks( ...@@ -621,10 +621,11 @@ xchk_directory_blocks(
{ {
struct xfs_bmbt_irec got; struct xfs_bmbt_irec got;
struct xfs_da_args args = { struct xfs_da_args args = {
.dp = sc ->ip, .dp = sc->ip,
.whichfork = XFS_DATA_FORK, .whichfork = XFS_DATA_FORK,
.geo = sc->mp->m_dir_geo, .geo = sc->mp->m_dir_geo,
.trans = sc->tp, .trans = sc->tp,
.owner = sc->ip->i_ino,
}; };
struct xfs_ifork *ifp = xfs_ifork_ptr(sc->ip, XFS_DATA_FORK); struct xfs_ifork *ifp = xfs_ifork_ptr(sc->ip, XFS_DATA_FORK);
struct xfs_mount *mp = sc->mp; struct xfs_mount *mp = sc->mp;
......
...@@ -273,6 +273,7 @@ xchk_dir_walk( ...@@ -273,6 +273,7 @@ xchk_dir_walk(
.dp = dp, .dp = dp,
.geo = dp->i_mount->m_dir_geo, .geo = dp->i_mount->m_dir_geo,
.trans = sc->tp, .trans = sc->tp,
.owner = dp->i_ino,
}; };
bool isblock; bool isblock;
int error; int error;
...@@ -324,6 +325,7 @@ xchk_dir_lookup( ...@@ -324,6 +325,7 @@ xchk_dir_lookup(
.hashval = xfs_dir2_hashname(dp->i_mount, name), .hashval = xfs_dir2_hashname(dp->i_mount, name),
.whichfork = XFS_DATA_FORK, .whichfork = XFS_DATA_FORK,
.op_flags = XFS_DA_OP_OKNOENT, .op_flags = XFS_DA_OP_OKNOENT,
.owner = dp->i_ino,
}; };
bool isblock, isleaf; bool isblock, isleaf;
int error; int error;
......
...@@ -540,6 +540,7 @@ xfs_attri_recover_work( ...@@ -540,6 +540,7 @@ xfs_attri_recover_work(
args->attr_filter = attrp->alfi_attr_filter & XFS_ATTRI_FILTER_MASK; args->attr_filter = attrp->alfi_attr_filter & XFS_ATTRI_FILTER_MASK;
args->op_flags = XFS_DA_OP_RECOVERY | XFS_DA_OP_OKNOENT | args->op_flags = XFS_DA_OP_RECOVERY | XFS_DA_OP_OKNOENT |
XFS_DA_OP_LOGGED; XFS_DA_OP_LOGGED;
args->owner = args->dp->i_ino;
ASSERT(xfs_sb_version_haslogxattrs(&mp->m_sb)); ASSERT(xfs_sb_version_haslogxattrs(&mp->m_sb));
......
...@@ -532,6 +532,7 @@ xfs_readdir( ...@@ -532,6 +532,7 @@ xfs_readdir(
args.dp = dp; args.dp = dp;
args.geo = dp->i_mount->m_dir_geo; args.geo = dp->i_mount->m_dir_geo;
args.trans = tp; args.trans = tp;
args.owner = dp->i_ino;
if (dp->i_df.if_format == XFS_DINODE_FMT_LOCAL) if (dp->i_df.if_format == XFS_DINODE_FMT_LOCAL)
return xfs_dir2_sf_getdents(&args, ctx); return xfs_dir2_sf_getdents(&args, ctx);
......
...@@ -1931,6 +1931,7 @@ DECLARE_EVENT_CLASS(xfs_da_class, ...@@ -1931,6 +1931,7 @@ DECLARE_EVENT_CLASS(xfs_da_class,
__field(xfs_dahash_t, hashval) __field(xfs_dahash_t, hashval)
__field(xfs_ino_t, inumber) __field(xfs_ino_t, inumber)
__field(uint32_t, op_flags) __field(uint32_t, op_flags)
__field(xfs_ino_t, owner)
), ),
TP_fast_assign( TP_fast_assign(
__entry->dev = VFS_I(args->dp)->i_sb->s_dev; __entry->dev = VFS_I(args->dp)->i_sb->s_dev;
...@@ -1941,9 +1942,10 @@ DECLARE_EVENT_CLASS(xfs_da_class, ...@@ -1941,9 +1942,10 @@ DECLARE_EVENT_CLASS(xfs_da_class,
__entry->hashval = args->hashval; __entry->hashval = args->hashval;
__entry->inumber = args->inumber; __entry->inumber = args->inumber;
__entry->op_flags = args->op_flags; __entry->op_flags = args->op_flags;
__entry->owner = args->owner;
), ),
TP_printk("dev %d:%d ino 0x%llx name %.*s namelen %d hashval 0x%x " TP_printk("dev %d:%d ino 0x%llx name %.*s namelen %d hashval 0x%x "
"inumber 0x%llx op_flags %s", "inumber 0x%llx op_flags %s owner 0x%llx",
MAJOR(__entry->dev), MINOR(__entry->dev), MAJOR(__entry->dev), MINOR(__entry->dev),
__entry->ino, __entry->ino,
__entry->namelen, __entry->namelen,
...@@ -1951,7 +1953,8 @@ DECLARE_EVENT_CLASS(xfs_da_class, ...@@ -1951,7 +1953,8 @@ DECLARE_EVENT_CLASS(xfs_da_class,
__entry->namelen, __entry->namelen,
__entry->hashval, __entry->hashval,
__entry->inumber, __entry->inumber,
__print_flags(__entry->op_flags, "|", XFS_DA_OP_FLAGS)) __print_flags(__entry->op_flags, "|", XFS_DA_OP_FLAGS),
__entry->owner)
) )
#define DEFINE_DIR2_EVENT(name) \ #define DEFINE_DIR2_EVENT(name) \
......
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