Message ID | 20241211085636.1380516-24-hch@lst.de (mailing list archive) |
---|---|
State | New |
Headers | show |
Series | [01/43] xfs: constify feature checks | expand |
On Wed, Dec 11, 2024 at 09:54:48AM +0100, Christoph Hellwig wrote: > Add support to validate and parse reported hardware zone state. > > Co-developed-by: Hans Holmberg <hans.holmberg@wdc.com> > Signed-off-by: Hans Holmberg <hans.holmberg@wdc.com> > Signed-off-by: Christoph Hellwig <hch@lst.de> > --- > fs/xfs/Makefile | 1 + > fs/xfs/libxfs/xfs_zones.c | 169 ++++++++++++++++++++++++++++++++++++++ > fs/xfs/libxfs/xfs_zones.h | 33 ++++++++ > 3 files changed, 203 insertions(+) > create mode 100644 fs/xfs/libxfs/xfs_zones.c > create mode 100644 fs/xfs/libxfs/xfs_zones.h > > diff --git a/fs/xfs/Makefile b/fs/xfs/Makefile > index 7afa51e41427..ea8e66c1e969 100644 > --- a/fs/xfs/Makefile > +++ b/fs/xfs/Makefile > @@ -64,6 +64,7 @@ xfs-y += $(addprefix libxfs/, \ > xfs-$(CONFIG_XFS_RT) += $(addprefix libxfs/, \ > xfs_rtbitmap.o \ > xfs_rtgroup.o \ > + xfs_zones.o \ > ) > > # highlevel code > diff --git a/fs/xfs/libxfs/xfs_zones.c b/fs/xfs/libxfs/xfs_zones.c > new file mode 100644 > index 000000000000..e170d7c13533 > --- /dev/null > +++ b/fs/xfs/libxfs/xfs_zones.c > @@ -0,0 +1,169 @@ > +// SPDX-License-Identifier: GPL-2.0 > +/* > + * Copyright (c) 2023-2024 Christoph Hellwig. > + * Copyright (c) 2024, Western Digital Corporation or its affiliates. > + */ > +#include "xfs.h" > +#include "xfs_fs.h" > +#include "xfs_shared.h" > +#include "xfs_format.h" > +#include "xfs_log_format.h" > +#include "xfs_trans_resv.h" > +#include "xfs_mount.h" > +#include "xfs_inode.h" > +#include "xfs_rtgroup.h" > +#include "xfs_zones.h" > + > +static int > +xfs_zone_validate_empty( > + struct blk_zone *zone, > + struct xfs_rtgroup *rtg, > + xfs_rgblock_t *write_pointer) > +{ > + struct xfs_mount *mp = rtg_mount(rtg); > + > + if (rtg_rmap(rtg)->i_used_blocks > 0) { > + xfs_warn(mp, "empty zone %u has non-zero used counter (0x%x).", > + rtg_rgno(rtg), rtg_rmap(rtg)->i_used_blocks); > + return -EIO; Why do some of these validation failures return EIO vs. EFSCORRUPTED? Is "EIO" used for "filesystem metadata out of sync with storage device" whereas "EFSCORRUPTED" is used for "filesystem metadata inconsistent with itself"? Do the _validate_{empty,full} functions need to validate zone->wp is zero/rtg_extents, respectively? --D > + } > + *write_pointer = 0; > + return 0; > +} > + > +static int > +xfs_zone_validate_wp( > + struct blk_zone *zone, > + struct xfs_rtgroup *rtg, > + xfs_rgblock_t *write_pointer) > +{ > + struct xfs_mount *mp = rtg_mount(rtg); > + xfs_rtblock_t wp_fsb = xfs_daddr_to_rtb(mp, zone->wp); > + > + if (rtg_rmap(rtg)->i_used_blocks > rtg->rtg_extents) { > + xfs_warn(mp, "zone %u has too large used counter (0x%x).", > + rtg_rgno(rtg), rtg_rmap(rtg)->i_used_blocks); > + return -EIO; > + } > + > + if (xfs_rtb_to_rgno(mp, wp_fsb) != rtg_rgno(rtg)) { > + xfs_warn(mp, "zone %u write pointer (0x%llx) outside of zone.", > + rtg_rgno(rtg), wp_fsb); > + return -EFSCORRUPTED; > + } > + > + *write_pointer = xfs_rtb_to_rgbno(mp, wp_fsb); > + if (*write_pointer >= rtg->rtg_extents) { > + xfs_warn(mp, "zone %u has invalid write pointer (0x%x).", > + rtg_rgno(rtg), *write_pointer); > + return -EFSCORRUPTED; > + } > + return 0; > +} > + > +static int > +xfs_zone_validate_full( > + struct blk_zone *zone, > + struct xfs_rtgroup *rtg, > + xfs_rgblock_t *write_pointer) > +{ > + struct xfs_mount *mp = rtg_mount(rtg); > + > + if (rtg_rmap(rtg)->i_used_blocks > rtg->rtg_extents) { > + xfs_warn(mp, "zone %u has too large used counter (0x%x).", > + rtg_rgno(rtg), rtg_rmap(rtg)->i_used_blocks); > + return -EIO; > + } > + *write_pointer = rtg->rtg_extents; > + > + return 0; > +} > + > +static int > +xfs_zone_validate_seq( > + struct blk_zone *zone, > + struct xfs_rtgroup *rtg, > + xfs_rgblock_t *write_pointer) > +{ > + struct xfs_mount *mp = rtg_mount(rtg); > + > + switch (zone->cond) { > + case BLK_ZONE_COND_EMPTY: > + return xfs_zone_validate_empty(zone, rtg, write_pointer); > + case BLK_ZONE_COND_IMP_OPEN: > + case BLK_ZONE_COND_EXP_OPEN: > + case BLK_ZONE_COND_CLOSED: > + return xfs_zone_validate_wp(zone, rtg, write_pointer); > + case BLK_ZONE_COND_FULL: > + return xfs_zone_validate_full(zone, rtg, write_pointer); > + case BLK_ZONE_COND_NOT_WP: > + case BLK_ZONE_COND_OFFLINE: > + case BLK_ZONE_COND_READONLY: > + xfs_warn(mp, "zone %u has unsupported zone condition 0x%x.", > + rtg_rgno(rtg), zone->cond); > + return -EIO; > + default: > + xfs_warn(mp, "zone %u has unknown zone condition 0x%x.", > + rtg_rgno(rtg), zone->cond); > + return -EIO; > + } > +} > + > +static int > +xfs_zone_validate_conv( > + struct blk_zone *zone, > + struct xfs_rtgroup *rtg) > +{ > + struct xfs_mount *mp = rtg_mount(rtg); > + > + switch (zone->cond) { > + case BLK_ZONE_COND_NOT_WP: > + return 0; > + default: > + xfs_warn(mp, > +"conventional zone %u has unsupported zone condition 0x%x.", > + rtg_rgno(rtg), zone->cond); > + return -EIO; > + } > +} > + > +int > +xfs_zone_validate( > + struct blk_zone *zone, > + struct xfs_rtgroup *rtg, > + xfs_rgblock_t *write_pointer) > +{ > + struct xfs_mount *mp = rtg_mount(rtg); > + struct xfs_groups *g = &mp->m_groups[XG_TYPE_RTG]; > + > + /* > + * Check that the zone capacity matches the rtgroup size stored in the > + * superblock. Note that all zones including the last one must have a > + * uniform capacity. > + */ > + if (XFS_BB_TO_FSB(mp, zone->capacity) != g->blocks) { > + xfs_warn(mp, > +"zone %u capacity (0x%llx) does not match RT group size (0x%x).", > + rtg_rgno(rtg), XFS_BB_TO_FSB(mp, zone->capacity), > + g->blocks); > + return -EIO; > + } > + > + if (XFS_BB_TO_FSB(mp, zone->len) != 1 << g->blklog) { > + xfs_warn(mp, > +"zone %u length (0x%llx) does match geometry (0x%x).", > + rtg_rgno(rtg), XFS_BB_TO_FSB(mp, zone->len), > + 1 << g->blklog); > + } > + > + switch (zone->type) { > + case BLK_ZONE_TYPE_CONVENTIONAL: > + return xfs_zone_validate_conv(zone, rtg); > + case BLK_ZONE_TYPE_SEQWRITE_REQ: > + return xfs_zone_validate_seq(zone, rtg, write_pointer); > + default: > + xfs_warn(mp, "zoned %u has unsupported type 0x%x.", > + rtg_rgno(rtg), zone->type); > + return -EFSCORRUPTED; > + } > +} > diff --git a/fs/xfs/libxfs/xfs_zones.h b/fs/xfs/libxfs/xfs_zones.h > new file mode 100644 > index 000000000000..4d3e53585654 > --- /dev/null > +++ b/fs/xfs/libxfs/xfs_zones.h > @@ -0,0 +1,33 @@ > +/* SPDX-License-Identifier: GPL-2.0 */ > +#ifndef _LIBXFS_ZONES_H > +#define _LIBXFS_ZONES_H > + > +/* > + * In order to guarantee forward progress for GC we need to reserve at least > + * two zones: one that will be used for moving data into and one spare zone > + * making sure that we have enough space to relocate a nearly-full zone. > + * To allow for slightly sloppy accounting for when we need to reserve the > + * second zone, we actually reserve three as that is easier than doing fully > + * accurate bookkeeping. > + */ > +#define XFS_GC_ZONES 3U > + > +/* > + * In addition we need two zones for user writes, one open zone for writing > + * and one to still have available blocks without resetting the open zone > + * when data in the open zone has been freed. > + */ > +#define XFS_RESERVED_ZONES (XFS_GC_ZONES + 1) > +#define XFS_MIN_ZONES (XFS_RESERVED_ZONES + 1) > + > +/* > + * Always keep one zone out of the general open zone pool to allow for GC to > + * happen while other writers are waiting for free space. > + */ > +#define XFS_OPEN_GC_ZONES 1U > +#define XFS_MIN_OPEN_ZONES (XFS_OPEN_GC_ZONES + 1U) > + > +int xfs_zone_validate(struct blk_zone *zone, struct xfs_rtgroup *rtg, > + xfs_rgblock_t *write_pointer); > + > +#endif /* _LIBXFS_ZONES_H */ > -- > 2.45.2 > >
On Fri, Dec 13, 2024 at 09:31:32AM -0800, Darrick J. Wong wrote: > > + xfs_rgblock_t *write_pointer) > > +{ > > + struct xfs_mount *mp = rtg_mount(rtg); > > + > > + if (rtg_rmap(rtg)->i_used_blocks > 0) { > > + xfs_warn(mp, "empty zone %u has non-zero used counter (0x%x).", > > + rtg_rgno(rtg), rtg_rmap(rtg)->i_used_blocks); > > + return -EIO; > > Why do some of these validation failures return EIO vs. EFSCORRUPTED? > Is "EIO" used for "filesystem metadata out of sync with storage device" > whereas "EFSCORRUPTED" is used for "filesystem metadata inconsistent > with itself"? If there was a rule I forgot about it :) This should be changed to return the same error everywhere, and that should probably be EFSCORRUPTED, or maybe the whole code should be changed to return a bool. > Do the _validate_{empty,full} functions need to validate zone->wp is > zero/rtg_extents, respectively? zone->wp is not defined for them in the hardware specs, so the only thing we'd validate is what the block layer / drivers put into it.
diff --git a/fs/xfs/Makefile b/fs/xfs/Makefile index 7afa51e41427..ea8e66c1e969 100644 --- a/fs/xfs/Makefile +++ b/fs/xfs/Makefile @@ -64,6 +64,7 @@ xfs-y += $(addprefix libxfs/, \ xfs-$(CONFIG_XFS_RT) += $(addprefix libxfs/, \ xfs_rtbitmap.o \ xfs_rtgroup.o \ + xfs_zones.o \ ) # highlevel code diff --git a/fs/xfs/libxfs/xfs_zones.c b/fs/xfs/libxfs/xfs_zones.c new file mode 100644 index 000000000000..e170d7c13533 --- /dev/null +++ b/fs/xfs/libxfs/xfs_zones.c @@ -0,0 +1,169 @@ +// SPDX-License-Identifier: GPL-2.0 +/* + * Copyright (c) 2023-2024 Christoph Hellwig. + * Copyright (c) 2024, Western Digital Corporation or its affiliates. + */ +#include "xfs.h" +#include "xfs_fs.h" +#include "xfs_shared.h" +#include "xfs_format.h" +#include "xfs_log_format.h" +#include "xfs_trans_resv.h" +#include "xfs_mount.h" +#include "xfs_inode.h" +#include "xfs_rtgroup.h" +#include "xfs_zones.h" + +static int +xfs_zone_validate_empty( + struct blk_zone *zone, + struct xfs_rtgroup *rtg, + xfs_rgblock_t *write_pointer) +{ + struct xfs_mount *mp = rtg_mount(rtg); + + if (rtg_rmap(rtg)->i_used_blocks > 0) { + xfs_warn(mp, "empty zone %u has non-zero used counter (0x%x).", + rtg_rgno(rtg), rtg_rmap(rtg)->i_used_blocks); + return -EIO; + } + *write_pointer = 0; + return 0; +} + +static int +xfs_zone_validate_wp( + struct blk_zone *zone, + struct xfs_rtgroup *rtg, + xfs_rgblock_t *write_pointer) +{ + struct xfs_mount *mp = rtg_mount(rtg); + xfs_rtblock_t wp_fsb = xfs_daddr_to_rtb(mp, zone->wp); + + if (rtg_rmap(rtg)->i_used_blocks > rtg->rtg_extents) { + xfs_warn(mp, "zone %u has too large used counter (0x%x).", + rtg_rgno(rtg), rtg_rmap(rtg)->i_used_blocks); + return -EIO; + } + + if (xfs_rtb_to_rgno(mp, wp_fsb) != rtg_rgno(rtg)) { + xfs_warn(mp, "zone %u write pointer (0x%llx) outside of zone.", + rtg_rgno(rtg), wp_fsb); + return -EFSCORRUPTED; + } + + *write_pointer = xfs_rtb_to_rgbno(mp, wp_fsb); + if (*write_pointer >= rtg->rtg_extents) { + xfs_warn(mp, "zone %u has invalid write pointer (0x%x).", + rtg_rgno(rtg), *write_pointer); + return -EFSCORRUPTED; + } + return 0; +} + +static int +xfs_zone_validate_full( + struct blk_zone *zone, + struct xfs_rtgroup *rtg, + xfs_rgblock_t *write_pointer) +{ + struct xfs_mount *mp = rtg_mount(rtg); + + if (rtg_rmap(rtg)->i_used_blocks > rtg->rtg_extents) { + xfs_warn(mp, "zone %u has too large used counter (0x%x).", + rtg_rgno(rtg), rtg_rmap(rtg)->i_used_blocks); + return -EIO; + } + *write_pointer = rtg->rtg_extents; + + return 0; +} + +static int +xfs_zone_validate_seq( + struct blk_zone *zone, + struct xfs_rtgroup *rtg, + xfs_rgblock_t *write_pointer) +{ + struct xfs_mount *mp = rtg_mount(rtg); + + switch (zone->cond) { + case BLK_ZONE_COND_EMPTY: + return xfs_zone_validate_empty(zone, rtg, write_pointer); + case BLK_ZONE_COND_IMP_OPEN: + case BLK_ZONE_COND_EXP_OPEN: + case BLK_ZONE_COND_CLOSED: + return xfs_zone_validate_wp(zone, rtg, write_pointer); + case BLK_ZONE_COND_FULL: + return xfs_zone_validate_full(zone, rtg, write_pointer); + case BLK_ZONE_COND_NOT_WP: + case BLK_ZONE_COND_OFFLINE: + case BLK_ZONE_COND_READONLY: + xfs_warn(mp, "zone %u has unsupported zone condition 0x%x.", + rtg_rgno(rtg), zone->cond); + return -EIO; + default: + xfs_warn(mp, "zone %u has unknown zone condition 0x%x.", + rtg_rgno(rtg), zone->cond); + return -EIO; + } +} + +static int +xfs_zone_validate_conv( + struct blk_zone *zone, + struct xfs_rtgroup *rtg) +{ + struct xfs_mount *mp = rtg_mount(rtg); + + switch (zone->cond) { + case BLK_ZONE_COND_NOT_WP: + return 0; + default: + xfs_warn(mp, +"conventional zone %u has unsupported zone condition 0x%x.", + rtg_rgno(rtg), zone->cond); + return -EIO; + } +} + +int +xfs_zone_validate( + struct blk_zone *zone, + struct xfs_rtgroup *rtg, + xfs_rgblock_t *write_pointer) +{ + struct xfs_mount *mp = rtg_mount(rtg); + struct xfs_groups *g = &mp->m_groups[XG_TYPE_RTG]; + + /* + * Check that the zone capacity matches the rtgroup size stored in the + * superblock. Note that all zones including the last one must have a + * uniform capacity. + */ + if (XFS_BB_TO_FSB(mp, zone->capacity) != g->blocks) { + xfs_warn(mp, +"zone %u capacity (0x%llx) does not match RT group size (0x%x).", + rtg_rgno(rtg), XFS_BB_TO_FSB(mp, zone->capacity), + g->blocks); + return -EIO; + } + + if (XFS_BB_TO_FSB(mp, zone->len) != 1 << g->blklog) { + xfs_warn(mp, +"zone %u length (0x%llx) does match geometry (0x%x).", + rtg_rgno(rtg), XFS_BB_TO_FSB(mp, zone->len), + 1 << g->blklog); + } + + switch (zone->type) { + case BLK_ZONE_TYPE_CONVENTIONAL: + return xfs_zone_validate_conv(zone, rtg); + case BLK_ZONE_TYPE_SEQWRITE_REQ: + return xfs_zone_validate_seq(zone, rtg, write_pointer); + default: + xfs_warn(mp, "zoned %u has unsupported type 0x%x.", + rtg_rgno(rtg), zone->type); + return -EFSCORRUPTED; + } +} diff --git a/fs/xfs/libxfs/xfs_zones.h b/fs/xfs/libxfs/xfs_zones.h new file mode 100644 index 000000000000..4d3e53585654 --- /dev/null +++ b/fs/xfs/libxfs/xfs_zones.h @@ -0,0 +1,33 @@ +/* SPDX-License-Identifier: GPL-2.0 */ +#ifndef _LIBXFS_ZONES_H +#define _LIBXFS_ZONES_H + +/* + * In order to guarantee forward progress for GC we need to reserve at least + * two zones: one that will be used for moving data into and one spare zone + * making sure that we have enough space to relocate a nearly-full zone. + * To allow for slightly sloppy accounting for when we need to reserve the + * second zone, we actually reserve three as that is easier than doing fully + * accurate bookkeeping. + */ +#define XFS_GC_ZONES 3U + +/* + * In addition we need two zones for user writes, one open zone for writing + * and one to still have available blocks without resetting the open zone + * when data in the open zone has been freed. + */ +#define XFS_RESERVED_ZONES (XFS_GC_ZONES + 1) +#define XFS_MIN_ZONES (XFS_RESERVED_ZONES + 1) + +/* + * Always keep one zone out of the general open zone pool to allow for GC to + * happen while other writers are waiting for free space. + */ +#define XFS_OPEN_GC_ZONES 1U +#define XFS_MIN_OPEN_ZONES (XFS_OPEN_GC_ZONES + 1U) + +int xfs_zone_validate(struct blk_zone *zone, struct xfs_rtgroup *rtg, + xfs_rgblock_t *write_pointer); + +#endif /* _LIBXFS_ZONES_H */