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

xfs: validate explicit directory data buffer owners

Port the existing directory data header checking function to accept an
owner number instead of an xfs_inode, then update the callsites to use
xfs_da_args.owner when possible.
Signed-off-by: default avatarDarrick J. Wong <djwong@kernel.org>
Reviewed-by: default avatarChristoph Hellwig <hch@lst.de>
parent 402eef10
...@@ -102,6 +102,7 @@ extern struct xfs_dir2_data_free *xfs_dir2_data_freefind( ...@@ -102,6 +102,7 @@ extern struct xfs_dir2_data_free *xfs_dir2_data_freefind(
extern int xfs_dir_ino_validate(struct xfs_mount *mp, xfs_ino_t ino); extern int xfs_dir_ino_validate(struct xfs_mount *mp, xfs_ino_t ino);
xfs_failaddr_t xfs_dir3_leaf_header_check(struct xfs_buf *bp, xfs_ino_t owner); xfs_failaddr_t xfs_dir3_leaf_header_check(struct xfs_buf *bp, xfs_ino_t owner);
xfs_failaddr_t xfs_dir3_data_header_check(struct xfs_buf *bp, xfs_ino_t owner);
extern const struct xfs_buf_ops xfs_dir3_block_buf_ops; extern const struct xfs_buf_ops xfs_dir3_block_buf_ops;
extern const struct xfs_buf_ops xfs_dir3_leafn_buf_ops; extern const struct xfs_buf_ops xfs_dir3_leafn_buf_ops;
......
...@@ -982,7 +982,8 @@ xfs_dir2_leaf_to_block( ...@@ -982,7 +982,8 @@ xfs_dir2_leaf_to_block(
* Read the data block if we don't already have it, give up if it fails. * Read the data block if we don't already have it, give up if it fails.
*/ */
if (!dbp) { if (!dbp) {
error = xfs_dir3_data_read(tp, dp, args->geo->datablk, 0, &dbp); error = xfs_dir3_data_read(tp, dp, args->owner,
args->geo->datablk, 0, &dbp);
if (error) if (error)
return error; return error;
} }
......
...@@ -395,17 +395,20 @@ static const struct xfs_buf_ops xfs_dir3_data_reada_buf_ops = { ...@@ -395,17 +395,20 @@ static const struct xfs_buf_ops xfs_dir3_data_reada_buf_ops = {
.verify_write = xfs_dir3_data_write_verify, .verify_write = xfs_dir3_data_write_verify,
}; };
static xfs_failaddr_t xfs_failaddr_t
xfs_dir3_data_header_check( xfs_dir3_data_header_check(
struct xfs_inode *dp, struct xfs_buf *bp,
struct xfs_buf *bp) xfs_ino_t owner)
{ {
struct xfs_mount *mp = dp->i_mount; struct xfs_mount *mp = bp->b_mount;
if (xfs_has_crc(mp)) { if (xfs_has_crc(mp)) {
struct xfs_dir3_data_hdr *hdr3 = bp->b_addr; struct xfs_dir3_data_hdr *hdr3 = bp->b_addr;
if (be64_to_cpu(hdr3->hdr.owner) != dp->i_ino) if (hdr3->hdr.magic != cpu_to_be32(XFS_DIR3_DATA_MAGIC))
return __this_address;
if (be64_to_cpu(hdr3->hdr.owner) != owner)
return __this_address; return __this_address;
} }
...@@ -416,6 +419,7 @@ int ...@@ -416,6 +419,7 @@ int
xfs_dir3_data_read( xfs_dir3_data_read(
struct xfs_trans *tp, struct xfs_trans *tp,
struct xfs_inode *dp, struct xfs_inode *dp,
xfs_ino_t owner,
xfs_dablk_t bno, xfs_dablk_t bno,
unsigned int flags, unsigned int flags,
struct xfs_buf **bpp) struct xfs_buf **bpp)
...@@ -429,7 +433,7 @@ xfs_dir3_data_read( ...@@ -429,7 +433,7 @@ xfs_dir3_data_read(
return err; return err;
/* Check things that we can't do in the verifier. */ /* Check things that we can't do in the verifier. */
fa = xfs_dir3_data_header_check(dp, *bpp); fa = xfs_dir3_data_header_check(*bpp, owner);
if (fa) { if (fa) {
__xfs_buf_mark_corrupt(*bpp, fa); __xfs_buf_mark_corrupt(*bpp, fa);
xfs_trans_brelse(tp, *bpp); xfs_trans_brelse(tp, *bpp);
......
...@@ -885,9 +885,9 @@ xfs_dir2_leaf_addname( ...@@ -885,9 +885,9 @@ xfs_dir2_leaf_addname(
* Already had space in some data block. * Already had space in some data block.
* Just read that one in. * Just read that one in.
*/ */
error = xfs_dir3_data_read(tp, dp, error = xfs_dir3_data_read(tp, dp, args->owner,
xfs_dir2_db_to_da(args->geo, use_block), xfs_dir2_db_to_da(args->geo, use_block), 0,
0, &dbp); &dbp);
if (error) { if (error) {
xfs_trans_brelse(tp, lbp); xfs_trans_brelse(tp, lbp);
return error; return error;
...@@ -1328,9 +1328,9 @@ xfs_dir2_leaf_lookup_int( ...@@ -1328,9 +1328,9 @@ xfs_dir2_leaf_lookup_int(
if (newdb != curdb) { if (newdb != curdb) {
if (dbp) if (dbp)
xfs_trans_brelse(tp, dbp); xfs_trans_brelse(tp, dbp);
error = xfs_dir3_data_read(tp, dp, error = xfs_dir3_data_read(tp, dp, args->owner,
xfs_dir2_db_to_da(args->geo, newdb), xfs_dir2_db_to_da(args->geo, newdb), 0,
0, &dbp); &dbp);
if (error) { if (error) {
xfs_trans_brelse(tp, lbp); xfs_trans_brelse(tp, lbp);
return error; return error;
...@@ -1370,9 +1370,9 @@ xfs_dir2_leaf_lookup_int( ...@@ -1370,9 +1370,9 @@ xfs_dir2_leaf_lookup_int(
ASSERT(cidb != -1); ASSERT(cidb != -1);
if (cidb != curdb) { if (cidb != curdb) {
xfs_trans_brelse(tp, dbp); xfs_trans_brelse(tp, dbp);
error = xfs_dir3_data_read(tp, dp, error = xfs_dir3_data_read(tp, dp, args->owner,
xfs_dir2_db_to_da(args->geo, cidb), xfs_dir2_db_to_da(args->geo, cidb), 0,
0, &dbp); &dbp);
if (error) { if (error) {
xfs_trans_brelse(tp, lbp); xfs_trans_brelse(tp, lbp);
return error; return error;
...@@ -1666,7 +1666,8 @@ xfs_dir2_leaf_trim_data( ...@@ -1666,7 +1666,8 @@ xfs_dir2_leaf_trim_data(
/* /*
* Read the offending data block. We need its buffer. * Read the offending data block. We need its buffer.
*/ */
error = xfs_dir3_data_read(tp, dp, xfs_dir2_db_to_da(geo, db), 0, &dbp); error = xfs_dir3_data_read(tp, dp, args->owner,
xfs_dir2_db_to_da(geo, db), 0, &dbp);
if (error) if (error)
return error; return error;
......
...@@ -863,7 +863,7 @@ xfs_dir2_leafn_lookup_for_entry( ...@@ -863,7 +863,7 @@ xfs_dir2_leafn_lookup_for_entry(
ASSERT(state->extravalid); ASSERT(state->extravalid);
curbp = state->extrablk.bp; curbp = state->extrablk.bp;
} else { } else {
error = xfs_dir3_data_read(tp, dp, error = xfs_dir3_data_read(tp, dp, args->owner,
xfs_dir2_db_to_da(args->geo, xfs_dir2_db_to_da(args->geo,
newdb), newdb),
0, &curbp); 0, &curbp);
...@@ -1949,9 +1949,8 @@ xfs_dir2_node_addname_int( ...@@ -1949,9 +1949,8 @@ xfs_dir2_node_addname_int(
&freehdr, &findex); &freehdr, &findex);
} else { } else {
/* Read the data block in. */ /* Read the data block in. */
error = xfs_dir3_data_read(tp, dp, error = xfs_dir3_data_read(tp, dp, args->owner,
xfs_dir2_db_to_da(args->geo, dbno), xfs_dir2_db_to_da(args->geo, dbno), 0, &dbp);
0, &dbp);
} }
if (error) if (error)
return error; return error;
......
...@@ -78,7 +78,8 @@ extern void xfs_dir3_data_check(struct xfs_inode *dp, struct xfs_buf *bp); ...@@ -78,7 +78,8 @@ extern void xfs_dir3_data_check(struct xfs_inode *dp, struct xfs_buf *bp);
extern xfs_failaddr_t __xfs_dir3_data_check(struct xfs_inode *dp, extern xfs_failaddr_t __xfs_dir3_data_check(struct xfs_inode *dp,
struct xfs_buf *bp); struct xfs_buf *bp);
int xfs_dir3_data_read(struct xfs_trans *tp, struct xfs_inode *dp, int xfs_dir3_data_read(struct xfs_trans *tp, struct xfs_inode *dp,
xfs_dablk_t bno, unsigned int flags, struct xfs_buf **bpp); xfs_ino_t owner, xfs_dablk_t bno, unsigned int flags,
struct xfs_buf **bpp);
int xfs_dir3_data_readahead(struct xfs_inode *dp, xfs_dablk_t bno, int xfs_dir3_data_readahead(struct xfs_inode *dp, xfs_dablk_t bno,
unsigned int flags); unsigned int flags);
......
...@@ -196,8 +196,8 @@ xchk_dir_rec( ...@@ -196,8 +196,8 @@ xchk_dir_rec(
xchk_da_set_corrupt(ds, level); xchk_da_set_corrupt(ds, level);
goto out; goto out;
} }
error = xfs_dir3_data_read(ds->dargs.trans, dp, rec_bno, error = xfs_dir3_data_read(ds->dargs.trans, dp, ds->dargs.owner,
XFS_DABUF_MAP_HOLE_OK, &bp); rec_bno, XFS_DABUF_MAP_HOLE_OK, &bp);
if (!xchk_fblock_process_error(ds->sc, XFS_DATA_FORK, rec_bno, if (!xchk_fblock_process_error(ds->sc, XFS_DATA_FORK, rec_bno,
&error)) &error))
goto out; goto out;
...@@ -318,7 +318,8 @@ xchk_directory_data_bestfree( ...@@ -318,7 +318,8 @@ xchk_directory_data_bestfree(
error = xfs_dir3_block_read(sc->tp, sc->ip, &bp); error = xfs_dir3_block_read(sc->tp, sc->ip, &bp);
} else { } else {
/* dir data format */ /* dir data format */
error = xfs_dir3_data_read(sc->tp, sc->ip, lblk, 0, &bp); error = xfs_dir3_data_read(sc->tp, sc->ip, sc->ip->i_ino, lblk,
0, &bp);
} }
if (!xchk_fblock_process_error(sc, XFS_DATA_FORK, lblk, &error)) if (!xchk_fblock_process_error(sc, XFS_DATA_FORK, lblk, &error))
goto out; goto out;
...@@ -531,10 +532,9 @@ xchk_directory_leaf1_bestfree( ...@@ -531,10 +532,9 @@ xchk_directory_leaf1_bestfree(
/* Check all the bestfree entries. */ /* Check all the bestfree entries. */
for (i = 0; i < bestcount; i++, bestp++) { for (i = 0; i < bestcount; i++, bestp++) {
best = be16_to_cpu(*bestp); best = be16_to_cpu(*bestp);
error = xfs_dir3_data_read(sc->tp, sc->ip, error = xfs_dir3_data_read(sc->tp, sc->ip, args->owner,
xfs_dir2_db_to_da(args->geo, i), xfs_dir2_db_to_da(args->geo, i),
XFS_DABUF_MAP_HOLE_OK, XFS_DABUF_MAP_HOLE_OK, &dbp);
&dbp);
if (!xchk_fblock_process_error(sc, XFS_DATA_FORK, lblk, if (!xchk_fblock_process_error(sc, XFS_DATA_FORK, lblk,
&error)) &error))
break; break;
...@@ -597,7 +597,7 @@ xchk_directory_free_bestfree( ...@@ -597,7 +597,7 @@ xchk_directory_free_bestfree(
stale++; stale++;
continue; continue;
} }
error = xfs_dir3_data_read(sc->tp, sc->ip, error = xfs_dir3_data_read(sc->tp, sc->ip, args->owner,
(freehdr.firstdb + i) * args->geo->fsbcount, (freehdr.firstdb + i) * args->geo->fsbcount,
0, &dbp); 0, &dbp);
if (!xchk_fblock_process_error(sc, XFS_DATA_FORK, lblk, if (!xchk_fblock_process_error(sc, XFS_DATA_FORK, lblk,
......
...@@ -175,7 +175,7 @@ xchk_read_leaf_dir_buf( ...@@ -175,7 +175,7 @@ xchk_read_leaf_dir_buf(
if (new_off > *curoff) if (new_off > *curoff)
*curoff = new_off; *curoff = new_off;
return xfs_dir3_data_read(tp, dp, map.br_startoff, 0, bpp); return xfs_dir3_data_read(tp, dp, dp->i_ino, map.br_startoff, 0, bpp);
} }
/* Call a function for every entry in a leaf directory. */ /* Call a function for every entry in a leaf directory. */
......
...@@ -282,7 +282,8 @@ xfs_dir2_leaf_readbuf( ...@@ -282,7 +282,8 @@ xfs_dir2_leaf_readbuf(
new_off = xfs_dir2_da_to_byte(geo, map.br_startoff); new_off = xfs_dir2_da_to_byte(geo, map.br_startoff);
if (new_off > *cur_off) if (new_off > *cur_off)
*cur_off = new_off; *cur_off = new_off;
error = xfs_dir3_data_read(args->trans, dp, map.br_startoff, 0, &bp); error = xfs_dir3_data_read(args->trans, dp, args->owner,
map.br_startoff, 0, &bp);
if (error) if (error)
goto out; goto out;
......
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