Re: [PATCH 12/12] xfs: create a new buf_ops pointer to verify structure metadata

[Date Prev][Date Next][Thread Prev][Thread Next][Date Index][Thread Index]

 



On Mon, Aug 28, 2017 at 11:17:45AM -0700, Darrick J. Wong wrote:
> From: Darrick J. Wong <darrick.wong@xxxxxxxxxx>
> 
> Expose all metadata structure buffer verifier functions via buf_ops.
> These will be used by the online scrub mechanism to look for problems
> with buffers that are already sitting around in memory.
> 

As it is, I think it probably makes sense to post this patch along with
whatever supporting code uses it. One quick thought in the meantime...

> Signed-off-by: Darrick J. Wong <darrick.wong@xxxxxxxxxx>
> ---
>  fs/xfs/libxfs/xfs_alloc.c          |   14 ++++++++------
>  fs/xfs/libxfs/xfs_alloc_btree.c    |    1 +
>  fs/xfs/libxfs/xfs_attr_leaf.c      |    1 +
>  fs/xfs/libxfs/xfs_attr_remote.c    |   32 ++++++++++++++++++++++++++++++++
>  fs/xfs/libxfs/xfs_bmap_btree.c     |    1 +
>  fs/xfs/libxfs/xfs_da_btree.c       |   25 +++++++++++++++++++++++++
>  fs/xfs/libxfs/xfs_dir2_block.c     |    1 +
>  fs/xfs/libxfs/xfs_dir2_data.c      |    1 +
>  fs/xfs/libxfs/xfs_dir2_leaf.c      |   16 ++++++++++++++++
>  fs/xfs/libxfs/xfs_dir2_node.c      |    1 +
>  fs/xfs/libxfs/xfs_dquot_buf.c      |   12 ++++++++++++
>  fs/xfs/libxfs/xfs_ialloc.c         |    1 +
>  fs/xfs/libxfs/xfs_ialloc_btree.c   |    1 +
>  fs/xfs/libxfs/xfs_refcount_btree.c |    1 +
>  fs/xfs/libxfs/xfs_rmap_btree.c     |    1 +
>  fs/xfs/libxfs/xfs_symlink_remote.c |    1 +
>  fs/xfs/xfs_buf.h                   |    1 +
>  17 files changed, 105 insertions(+), 6 deletions(-)
> 
> 
> diff --git a/fs/xfs/libxfs/xfs_alloc.c b/fs/xfs/libxfs/xfs_alloc.c
> index d49b1e3..21a1c21 100644
> --- a/fs/xfs/libxfs/xfs_alloc.c
> +++ b/fs/xfs/libxfs/xfs_alloc.c
> @@ -604,6 +604,7 @@ const struct xfs_buf_ops xfs_agfl_buf_ops = {
>  	.name = "xfs_agfl",
>  	.verify_read = xfs_agfl_read_verify,
>  	.verify_write = xfs_agfl_write_verify,
> +	.verify_struct = xfs_agfl_verify,
>  };
>  
>  /*
> @@ -2393,10 +2394,10 @@ xfs_alloc_put_freelist(
>  
>  static void *
>  xfs_agf_verify(
> -	struct xfs_mount *mp,
> -	struct xfs_buf	*bp)
> - {
> -	struct xfs_agf	*agf = XFS_BUF_TO_AGF(bp);
> +	struct xfs_buf		*bp)
> +{
> +	struct xfs_mount	*mp = bp->b_target->bt_mount;
> +	struct xfs_agf		*agf = XFS_BUF_TO_AGF(bp);
>  
>  	if (xfs_sb_version_hascrc(&mp->m_sb)) {
>  		if (!uuid_equal(&agf->agf_uuid, &mp->m_sb.sb_meta_uuid))
> @@ -2457,7 +2458,7 @@ xfs_agf_read_verify(
>  	if (xfs_sb_version_hascrc(&mp->m_sb) &&
>  	    !xfs_buf_verify_cksum(bp, XFS_AGF_CRC_OFF))
>  		xfs_buf_ioerror(bp, -EFSBADCRC);
> -	else if (XFS_TEST_ERROR((failed_at = xfs_agf_verify(mp, bp)), mp,
> +	else if (XFS_TEST_ERROR((failed_at = xfs_agf_verify(bp)), mp,
>  				XFS_ERRTAG_ALLOC_READ_AGF))
>  		xfs_buf_ioerror(bp, -EFSCORRUPTED);
>  
> @@ -2473,7 +2474,7 @@ xfs_agf_write_verify(
>  	struct xfs_buf_log_item	*bip = bp->b_fspriv;
>  	void			*failed_at;
>  
> -	if ((failed_at = xfs_agf_verify(mp, bp))) {
> +	if ((failed_at = xfs_agf_verify(bp))) {
>  		xfs_buf_ioerror(bp, -EFSCORRUPTED);
>  		xfs_verifier_error(bp, failed_at);
>  		return;
> @@ -2492,6 +2493,7 @@ const struct xfs_buf_ops xfs_agf_buf_ops = {
>  	.name = "xfs_agf",
>  	.verify_read = xfs_agf_read_verify,
>  	.verify_write = xfs_agf_write_verify,
> +	.verify_struct = xfs_agf_verify,
>  };

If we expose the core verifier function of each structure through the
ops table and the read/write verifiers mostly call that function and do
some CRC processing based on read or write, I'm wondering how many of
the unique read/write verifiers could be condensed into a generic buffer
read and generic buffer write function that invokes the
->verify_struct() function and does the CRC/LSN check or updates
appropriately. We'd have to at least include the CRC offset in the
buf_ops for read verifiers and additionally the lsn offset for write
verifiers. Thoughts?

Brian

>  
>  /*
> diff --git a/fs/xfs/libxfs/xfs_alloc_btree.c b/fs/xfs/libxfs/xfs_alloc_btree.c
> index 8d4c004..eb79f6c 100644
> --- a/fs/xfs/libxfs/xfs_alloc_btree.c
> +++ b/fs/xfs/libxfs/xfs_alloc_btree.c
> @@ -397,6 +397,7 @@ const struct xfs_buf_ops xfs_allocbt_buf_ops = {
>  	.name = "xfs_allocbt",
>  	.verify_read = xfs_allocbt_read_verify,
>  	.verify_write = xfs_allocbt_write_verify,
> +	.verify_struct = xfs_allocbt_verify,
>  };
>  
>  
> diff --git a/fs/xfs/libxfs/xfs_attr_leaf.c b/fs/xfs/libxfs/xfs_attr_leaf.c
> index bde5269..3a563c5 100644
> --- a/fs/xfs/libxfs/xfs_attr_leaf.c
> +++ b/fs/xfs/libxfs/xfs_attr_leaf.c
> @@ -339,6 +339,7 @@ const struct xfs_buf_ops xfs_attr3_leaf_buf_ops = {
>  	.name = "xfs_attr3_leaf",
>  	.verify_read = xfs_attr3_leaf_read_verify,
>  	.verify_write = xfs_attr3_leaf_write_verify,
> +	.verify_struct = xfs_attr3_leaf_verify,
>  };
>  
>  int
> diff --git a/fs/xfs/libxfs/xfs_attr_remote.c b/fs/xfs/libxfs/xfs_attr_remote.c
> index d33a4d3..164c366 100644
> --- a/fs/xfs/libxfs/xfs_attr_remote.c
> +++ b/fs/xfs/libxfs/xfs_attr_remote.c
> @@ -203,10 +203,42 @@ xfs_attr3_rmt_write_verify(
>  	ASSERT(len == 0);
>  }
>  
> +static void *
> +xfs_attr3_rmt_verify_struct(
> +	struct xfs_buf	*bp)
> +{
> +	struct xfs_mount *mp = bp->b_target->bt_mount;
> +	char		*ptr;
> +	void		*failed_at;
> +	int		len;
> +	xfs_daddr_t	bno;
> +	int		blksize = mp->m_attr_geo->blksize;
> +
> +	/* no verification of non-crc buffers */
> +	if (!xfs_sb_version_hascrc(&mp->m_sb))
> +		return NULL;
> +
> +	ptr = bp->b_addr;
> +	bno = bp->b_bn;
> +	len = BBTOB(bp->b_length);
> +	ASSERT(len >= blksize);
> +
> +	while (len > 0) {
> +		if ((failed_at = xfs_attr3_rmt_verify(mp, ptr, blksize, bno)))
> +			return failed_at;
> +		len -= blksize;
> +		ptr += blksize;
> +		bno += BTOBB(blksize);
> +	}
> +
> +	return NULL;
> +}
> +
>  const struct xfs_buf_ops xfs_attr3_rmt_buf_ops = {
>  	.name = "xfs_attr3_rmt",
>  	.verify_read = xfs_attr3_rmt_read_verify,
>  	.verify_write = xfs_attr3_rmt_write_verify,
> +	.verify_struct = xfs_attr3_rmt_verify_struct,
>  };
>  
>  STATIC int
> diff --git a/fs/xfs/libxfs/xfs_bmap_btree.c b/fs/xfs/libxfs/xfs_bmap_btree.c
> index d613080..0d5dd7b 100644
> --- a/fs/xfs/libxfs/xfs_bmap_btree.c
> +++ b/fs/xfs/libxfs/xfs_bmap_btree.c
> @@ -692,6 +692,7 @@ const struct xfs_buf_ops xfs_bmbt_buf_ops = {
>  	.name = "xfs_bmbt",
>  	.verify_read = xfs_bmbt_read_verify,
>  	.verify_write = xfs_bmbt_write_verify,
> +	.verify_struct = xfs_bmbt_verify,
>  };
>  
>  
> diff --git a/fs/xfs/libxfs/xfs_da_btree.c b/fs/xfs/libxfs/xfs_da_btree.c
> index f3544fb..af738bf 100644
> --- a/fs/xfs/libxfs/xfs_da_btree.c
> +++ b/fs/xfs/libxfs/xfs_da_btree.c
> @@ -247,10 +247,35 @@ xfs_da3_node_read_verify(
>  	xfs_verifier_error(bp, failed_at);
>  }
>  
> +/* Verify the structure of a da3 block. */
> +static void *
> +xfs_da3_node_verify_struct(
> +	struct xfs_buf		*bp)
> +{
> +	struct xfs_da_blkinfo	*info = bp->b_addr;
> +
> +	switch (be16_to_cpu(info->magic)) {
> +	case XFS_DA3_NODE_MAGIC:
> +	case XFS_DA_NODE_MAGIC:
> +		return xfs_da3_node_verify(bp);
> +	case XFS_ATTR_LEAF_MAGIC:
> +	case XFS_ATTR3_LEAF_MAGIC:
> +		bp->b_ops = &xfs_attr3_leaf_buf_ops;
> +		return bp->b_ops->verify_struct(bp);
> +	case XFS_DIR2_LEAFN_MAGIC:
> +	case XFS_DIR3_LEAFN_MAGIC:
> +		bp->b_ops = &xfs_dir3_leafn_buf_ops;
> +		return bp->b_ops->verify_struct(bp);
> +	default:
> +		return __this_address;
> +	}
> +}
> +
>  const struct xfs_buf_ops xfs_da3_node_buf_ops = {
>  	.name = "xfs_da3_node",
>  	.verify_read = xfs_da3_node_read_verify,
>  	.verify_write = xfs_da3_node_write_verify,
> +	.verify_struct = xfs_da3_node_verify_struct,
>  };
>  
>  int
> diff --git a/fs/xfs/libxfs/xfs_dir2_block.c b/fs/xfs/libxfs/xfs_dir2_block.c
> index 6c54d03..6dce935 100644
> --- a/fs/xfs/libxfs/xfs_dir2_block.c
> +++ b/fs/xfs/libxfs/xfs_dir2_block.c
> @@ -126,6 +126,7 @@ const struct xfs_buf_ops xfs_dir3_block_buf_ops = {
>  	.name = "xfs_dir3_block",
>  	.verify_read = xfs_dir3_block_read_verify,
>  	.verify_write = xfs_dir3_block_write_verify,
> +	.verify_struct = xfs_dir3_block_verify,
>  };
>  
>  int
> diff --git a/fs/xfs/libxfs/xfs_dir2_data.c b/fs/xfs/libxfs/xfs_dir2_data.c
> index 5e27b71..d08a7ac 100644
> --- a/fs/xfs/libxfs/xfs_dir2_data.c
> +++ b/fs/xfs/libxfs/xfs_dir2_data.c
> @@ -318,6 +318,7 @@ const struct xfs_buf_ops xfs_dir3_data_buf_ops = {
>  	.name = "xfs_dir3_data",
>  	.verify_read = xfs_dir3_data_read_verify,
>  	.verify_write = xfs_dir3_data_write_verify,
> +	.verify_struct = xfs_dir3_data_verify,
>  };
>  
>  static const struct xfs_buf_ops xfs_dir3_data_reada_buf_ops = {
> diff --git a/fs/xfs/libxfs/xfs_dir2_leaf.c b/fs/xfs/libxfs/xfs_dir2_leaf.c
> index 00dcfef..792fd98 100644
> --- a/fs/xfs/libxfs/xfs_dir2_leaf.c
> +++ b/fs/xfs/libxfs/xfs_dir2_leaf.c
> @@ -218,6 +218,13 @@ __write_verify(
>  	xfs_buf_update_cksum(bp, XFS_DIR3_LEAF_CRC_OFF);
>  }
>  
> +static void *
> +xfs_dir3_leaf1_verify(
> +	struct xfs_buf	*bp)
> +{
> +	return xfs_dir3_leaf_verify(bp, XFS_DIR2_LEAF1_MAGIC);
> +}
> +
>  static void
>  xfs_dir3_leaf1_read_verify(
>  	struct xfs_buf	*bp)
> @@ -232,6 +239,13 @@ xfs_dir3_leaf1_write_verify(
>  	__write_verify(bp, XFS_DIR2_LEAF1_MAGIC);
>  }
>  
> +static void *
> +xfs_dir3_leafn_verify(
> +	struct xfs_buf	*bp)
> +{
> +	return xfs_dir3_leaf_verify(bp, XFS_DIR2_LEAFN_MAGIC);
> +}
> +
>  static void
>  xfs_dir3_leafn_read_verify(
>  	struct xfs_buf	*bp)
> @@ -250,12 +264,14 @@ const struct xfs_buf_ops xfs_dir3_leaf1_buf_ops = {
>  	.name = "xfs_dir3_leaf1",
>  	.verify_read = xfs_dir3_leaf1_read_verify,
>  	.verify_write = xfs_dir3_leaf1_write_verify,
> +	.verify_struct = xfs_dir3_leaf1_verify,
>  };
>  
>  const struct xfs_buf_ops xfs_dir3_leafn_buf_ops = {
>  	.name = "xfs_dir3_leafn",
>  	.verify_read = xfs_dir3_leafn_read_verify,
>  	.verify_write = xfs_dir3_leafn_write_verify,
> +	.verify_struct = xfs_dir3_leafn_verify,
>  };
>  
>  int
> diff --git a/fs/xfs/libxfs/xfs_dir2_node.c b/fs/xfs/libxfs/xfs_dir2_node.c
> index a11585b..1a5e383 100644
> --- a/fs/xfs/libxfs/xfs_dir2_node.c
> +++ b/fs/xfs/libxfs/xfs_dir2_node.c
> @@ -155,6 +155,7 @@ const struct xfs_buf_ops xfs_dir3_free_buf_ops = {
>  	.name = "xfs_dir3_free",
>  	.verify_read = xfs_dir3_free_read_verify,
>  	.verify_write = xfs_dir3_free_write_verify,
> +	.verify_struct = xfs_dir3_free_verify,
>  };
>  
>  /* Everything ok in the free block header? */
> diff --git a/fs/xfs/libxfs/xfs_dquot_buf.c b/fs/xfs/libxfs/xfs_dquot_buf.c
> index 5561011..8ce5239 100644
> --- a/fs/xfs/libxfs/xfs_dquot_buf.c
> +++ b/fs/xfs/libxfs/xfs_dquot_buf.c
> @@ -242,6 +242,17 @@ xfs_dquot_buf_verify(
>  	return true;
>  }
>  
> +static void *
> +xfs_dquot_buf_verify_struct(
> +	struct xfs_buf	*bp)
> +{
> +	struct xfs_mount	*mp = bp->b_target->bt_mount;
> +
> +	if (!xfs_dquot_buf_verify(mp, bp, 0))
> +		return __this_address;
> +	return NULL;
> +}
> +
>  static void
>  xfs_dquot_buf_read_verify(
>  	struct xfs_buf	*bp)
> @@ -298,6 +309,7 @@ const struct xfs_buf_ops xfs_dquot_buf_ops = {
>  	.name = "xfs_dquot",
>  	.verify_read = xfs_dquot_buf_read_verify,
>  	.verify_write = xfs_dquot_buf_write_verify,
> +	.verify_struct = xfs_dquot_buf_verify_struct,
>  };
>  
>  const struct xfs_buf_ops xfs_dquot_buf_ra_ops = {
> diff --git a/fs/xfs/libxfs/xfs_ialloc.c b/fs/xfs/libxfs/xfs_ialloc.c
> index c63b708..a2c8c1d 100644
> --- a/fs/xfs/libxfs/xfs_ialloc.c
> +++ b/fs/xfs/libxfs/xfs_ialloc.c
> @@ -2586,6 +2586,7 @@ const struct xfs_buf_ops xfs_agi_buf_ops = {
>  	.name = "xfs_agi",
>  	.verify_read = xfs_agi_read_verify,
>  	.verify_write = xfs_agi_write_verify,
> +	.verify_struct = xfs_agi_verify,
>  };
>  
>  /*
> diff --git a/fs/xfs/libxfs/xfs_ialloc_btree.c b/fs/xfs/libxfs/xfs_ialloc_btree.c
> index 38d6a50..f920ca6 100644
> --- a/fs/xfs/libxfs/xfs_ialloc_btree.c
> +++ b/fs/xfs/libxfs/xfs_ialloc_btree.c
> @@ -327,6 +327,7 @@ const struct xfs_buf_ops xfs_inobt_buf_ops = {
>  	.name = "xfs_inobt",
>  	.verify_read = xfs_inobt_read_verify,
>  	.verify_write = xfs_inobt_write_verify,
> +	.verify_struct = xfs_inobt_verify,
>  };
>  
>  STATIC int
> diff --git a/fs/xfs/libxfs/xfs_refcount_btree.c b/fs/xfs/libxfs/xfs_refcount_btree.c
> index 8d6c0fc..a151b4d 100644
> --- a/fs/xfs/libxfs/xfs_refcount_btree.c
> +++ b/fs/xfs/libxfs/xfs_refcount_btree.c
> @@ -288,6 +288,7 @@ const struct xfs_buf_ops xfs_refcountbt_buf_ops = {
>  	.name			= "xfs_refcountbt",
>  	.verify_read		= xfs_refcountbt_read_verify,
>  	.verify_write		= xfs_refcountbt_write_verify,
> +	.verify_struct		= xfs_refcountbt_verify,
>  };
>  
>  STATIC int
> diff --git a/fs/xfs/libxfs/xfs_rmap_btree.c b/fs/xfs/libxfs/xfs_rmap_btree.c
> index 4fd77e7..ec7b216 100644
> --- a/fs/xfs/libxfs/xfs_rmap_btree.c
> +++ b/fs/xfs/libxfs/xfs_rmap_btree.c
> @@ -380,6 +380,7 @@ const struct xfs_buf_ops xfs_rmapbt_buf_ops = {
>  	.name			= "xfs_rmapbt",
>  	.verify_read		= xfs_rmapbt_read_verify,
>  	.verify_write		= xfs_rmapbt_write_verify,
> +	.verify_struct		= xfs_rmapbt_verify,
>  };
>  
>  STATIC int
> diff --git a/fs/xfs/libxfs/xfs_symlink_remote.c b/fs/xfs/libxfs/xfs_symlink_remote.c
> index ef5e754..53732f4 100644
> --- a/fs/xfs/libxfs/xfs_symlink_remote.c
> +++ b/fs/xfs/libxfs/xfs_symlink_remote.c
> @@ -173,6 +173,7 @@ const struct xfs_buf_ops xfs_symlink_buf_ops = {
>  	.name = "xfs_symlink",
>  	.verify_read = xfs_symlink_read_verify,
>  	.verify_write = xfs_symlink_write_verify,
> +	.verify_struct = xfs_symlink_verify,
>  };
>  
>  void
> diff --git a/fs/xfs/xfs_buf.h b/fs/xfs/xfs_buf.h
> index 2072126..094e3e7 100644
> --- a/fs/xfs/xfs_buf.h
> +++ b/fs/xfs/xfs_buf.h
> @@ -139,6 +139,7 @@ struct xfs_buf_ops {
>  	char *name;
>  	void (*verify_read)(struct xfs_buf *);
>  	void (*verify_write)(struct xfs_buf *);
> +	void *(*verify_struct)(struct xfs_buf *);
>  };
>  
>  typedef struct xfs_buf {
> 
> --
> To unsubscribe from this list: send the line "unsubscribe linux-xfs" in
> the body of a message to majordomo@xxxxxxxxxxxxxxx
> More majordomo info at  http://vger.kernel.org/majordomo-info.html
--
To unsubscribe from this list: send the line "unsubscribe linux-xfs" in
the body of a message to majordomo@xxxxxxxxxxxxxxx
More majordomo info at  http://vger.kernel.org/majordomo-info.html



[Index of Archives]     [XFS Filesystem Development (older mail)]     [Linux Filesystem Development]     [Linux Audio Users]     [Yosemite Trails]     [Linux Kernel]     [Linux RAID]     [Linux SCSI]


  Powered by Linux