From: Darrick J. Wong <djwong@xxxxxxxxxx> Move our private copy of the GETFSMAP definition into a libfrog header file that (preferentially) uses the system header files. We have no business shipping kernel headers in the xfslibs package, but this shim is still needed to build fully functional xfsprogs on old userspace. Signed-off-by: Darrick J. Wong <djwong@xxxxxxxxxx> --- include/linux.h | 105 ------------------------------------------------ io/fsmap.c | 1 io/io.h | 5 -- libfrog/Makefile | 1 libfrog/fsmap.h | 117 +++++++++++++++++++++++++++++++++++++++++++++++++++++ scrub/phase6.c | 1 scrub/phase7.c | 1 scrub/spacemap.c | 1 spaceman/freesp.c | 1 spaceman/space.h | 4 -- 10 files changed, 123 insertions(+), 114 deletions(-) create mode 100644 libfrog/fsmap.h diff --git a/include/linux.h b/include/linux.h index de8a7122..3d9f4e3d 100644 --- a/include/linux.h +++ b/include/linux.h @@ -251,111 +251,6 @@ struct fsxattr { #define FS_XFLAG_COWEXTSIZE 0x00010000 /* CoW extent size allocator hint */ #endif -#ifdef HAVE_GETFSMAP -# include <linux/fsmap.h> -#else -/* - * Structure for FS_IOC_GETFSMAP. - * - * The memory layout for this call are the scalar values defined in - * struct fsmap_head, followed by two struct fsmap that describe - * the lower and upper bound of mappings to return, followed by an - * array of struct fsmap mappings. - * - * fmh_iflags control the output of the call, whereas fmh_oflags report - * on the overall record output. fmh_count should be set to the - * length of the fmh_recs array, and fmh_entries will be set to the - * number of entries filled out during each call. If fmh_count is - * zero, the number of reverse mappings will be returned in - * fmh_entries, though no mappings will be returned. fmh_reserved - * must be set to zero. - * - * The two elements in the fmh_keys array are used to constrain the - * output. The first element in the array should represent the - * lowest disk mapping ("low key") that the user wants to learn - * about. If this value is all zeroes, the filesystem will return - * the first entry it knows about. For a subsequent call, the - * contents of fsmap_head.fmh_recs[fsmap_head.fmh_count - 1] should be - * copied into fmh_keys[0] to have the kernel start where it left off. - * - * The second element in the fmh_keys array should represent the - * highest disk mapping ("high key") that the user wants to learn - * about. If this value is all ones, the filesystem will not stop - * until it runs out of mapping to return or runs out of space in - * fmh_recs. - * - * fmr_device can be either a 32-bit cookie representing a device, or - * a 32-bit dev_t if the FMH_OF_DEV_T flag is set. fmr_physical, - * fmr_offset, and fmr_length are expressed in units of bytes. - * fmr_owner is either an inode number, or a special value if - * FMR_OF_SPECIAL_OWNER is set in fmr_flags. - */ -struct fsmap { - __u32 fmr_device; /* device id */ - __u32 fmr_flags; /* mapping flags */ - __u64 fmr_physical; /* device offset of segment */ - __u64 fmr_owner; /* owner id */ - __u64 fmr_offset; /* file offset of segment */ - __u64 fmr_length; /* length of segment */ - __u64 fmr_reserved[3]; /* must be zero */ -}; - -struct fsmap_head { - __u32 fmh_iflags; /* control flags */ - __u32 fmh_oflags; /* output flags */ - __u32 fmh_count; /* # of entries in array incl. input */ - __u32 fmh_entries; /* # of entries filled in (output). */ - __u64 fmh_reserved[6]; /* must be zero */ - - struct fsmap fmh_keys[2]; /* low and high keys for the mapping search */ - struct fsmap fmh_recs[]; /* returned records */ -}; - -/* Size of an fsmap_head with room for nr records. */ -static inline size_t -fsmap_sizeof( - unsigned int nr) -{ - return sizeof(struct fsmap_head) + nr * sizeof(struct fsmap); -} - -/* Start the next fsmap query at the end of the current query results. */ -static inline void -fsmap_advance( - struct fsmap_head *head) -{ - head->fmh_keys[0] = head->fmh_recs[head->fmh_entries - 1]; -} - -/* fmh_iflags values - set by XFS_IOC_GETFSMAP caller in the header. */ -/* no flags defined yet */ -#define FMH_IF_VALID 0 - -/* fmh_oflags values - returned in the header segment only. */ -#define FMH_OF_DEV_T 0x1 /* fmr_device values will be dev_t */ - -/* fmr_flags values - returned for each non-header segment */ -#define FMR_OF_PREALLOC 0x1 /* segment = unwritten pre-allocation */ -#define FMR_OF_ATTR_FORK 0x2 /* segment = attribute fork */ -#define FMR_OF_EXTENT_MAP 0x4 /* segment = extent map */ -#define FMR_OF_SHARED 0x8 /* segment = shared with another file */ -#define FMR_OF_SPECIAL_OWNER 0x10 /* owner is a special value */ -#define FMR_OF_LAST 0x20 /* segment is the last in the FS */ - -/* Each FS gets to define its own special owner codes. */ -#define FMR_OWNER(type, code) (((__u64)type << 32) | \ - ((__u64)code & 0xFFFFFFFFULL)) -#define FMR_OWNER_TYPE(owner) ((__u32)((__u64)owner >> 32)) -#define FMR_OWNER_CODE(owner) ((__u32)(((__u64)owner & 0xFFFFFFFFULL))) -#define FMR_OWN_FREE FMR_OWNER(0, 1) /* free space */ -#define FMR_OWN_UNKNOWN FMR_OWNER(0, 2) /* unknown owner */ -#define FMR_OWN_METADATA FMR_OWNER(0, 3) /* metadata */ - -#define FS_IOC_GETFSMAP _IOWR('X', 59, struct fsmap_head) - -#define HAVE_GETFSMAP -#endif /* HAVE_GETFSMAP */ - #ifndef HAVE_MAP_SYNC #define MAP_SYNC 0 #define MAP_SHARED_VALIDATE 0 diff --git a/io/fsmap.c b/io/fsmap.c index f540a7c0..9ff36bf4 100644 --- a/io/fsmap.c +++ b/io/fsmap.c @@ -10,6 +10,7 @@ #include "io.h" #include "input.h" #include "libfrog/fsgeom.h" +#include "libfrog/fsmap.h" static cmdinfo_t fsmap_cmd; static dev_t xfs_data_dev; diff --git a/io/io.h b/io/io.h index 49db902f..39fb5878 100644 --- a/io/io.h +++ b/io/io.h @@ -167,12 +167,7 @@ extern void readdir_init(void); extern void reflink_init(void); extern void cowextsize_init(void); - -#ifdef HAVE_GETFSMAP extern void fsmap_init(void); -#else -# define fsmap_init() do { } while (0) -#endif #ifdef HAVE_DEVMAPPER extern void log_writes_init(void); diff --git a/libfrog/Makefile b/libfrog/Makefile index 01107082..d6044455 100644 --- a/libfrog/Makefile +++ b/libfrog/Makefile @@ -40,6 +40,7 @@ crc32cselftest.h \ crc32defs.h \ crc32table.h \ fsgeom.h \ +fsmap.h \ logging.h \ paths.h \ projects.h \ diff --git a/libfrog/fsmap.h b/libfrog/fsmap.h new file mode 100644 index 00000000..dc290962 --- /dev/null +++ b/libfrog/fsmap.h @@ -0,0 +1,117 @@ +#ifdef HAVE_GETFSMAP +# include <linux/fsmap.h> +#endif + +/* SPDX-License-Identifier: GPL-2.0 WITH Linux-syscall-note */ +/* + * FS_IOC_GETFSMAP ioctl infrastructure. + * + * Copyright (C) 2017 Oracle. All Rights Reserved. + * + * Author: Darrick J. Wong <djwong@xxxxxxxxxx> + */ +#ifndef _LINUX_FSMAP_H +#define _LINUX_FSMAP_H + +#include <linux/types.h> + +/* + * Structure for FS_IOC_GETFSMAP. + * + * The memory layout for this call are the scalar values defined in + * struct fsmap_head, followed by two struct fsmap that describe + * the lower and upper bound of mappings to return, followed by an + * array of struct fsmap mappings. + * + * fmh_iflags control the output of the call, whereas fmh_oflags report + * on the overall record output. fmh_count should be set to the + * length of the fmh_recs array, and fmh_entries will be set to the + * number of entries filled out during each call. If fmh_count is + * zero, the number of reverse mappings will be returned in + * fmh_entries, though no mappings will be returned. fmh_reserved + * must be set to zero. + * + * The two elements in the fmh_keys array are used to constrain the + * output. The first element in the array should represent the + * lowest disk mapping ("low key") that the user wants to learn + * about. If this value is all zeroes, the filesystem will return + * the first entry it knows about. For a subsequent call, the + * contents of fsmap_head.fmh_recs[fsmap_head.fmh_count - 1] should be + * copied into fmh_keys[0] to have the kernel start where it left off. + * + * The second element in the fmh_keys array should represent the + * highest disk mapping ("high key") that the user wants to learn + * about. If this value is all ones, the filesystem will not stop + * until it runs out of mapping to return or runs out of space in + * fmh_recs. + * + * fmr_device can be either a 32-bit cookie representing a device, or + * a 32-bit dev_t if the FMH_OF_DEV_T flag is set. fmr_physical, + * fmr_offset, and fmr_length are expressed in units of bytes. + * fmr_owner is either an inode number, or a special value if + * FMR_OF_SPECIAL_OWNER is set in fmr_flags. + */ +struct fsmap { + __u32 fmr_device; /* device id */ + __u32 fmr_flags; /* mapping flags */ + __u64 fmr_physical; /* device offset of segment */ + __u64 fmr_owner; /* owner id */ + __u64 fmr_offset; /* file offset of segment */ + __u64 fmr_length; /* length of segment */ + __u64 fmr_reserved[3]; /* must be zero */ +}; + +struct fsmap_head { + __u32 fmh_iflags; /* control flags */ + __u32 fmh_oflags; /* output flags */ + __u32 fmh_count; /* # of entries in array incl. input */ + __u32 fmh_entries; /* # of entries filled in (output). */ + __u64 fmh_reserved[6]; /* must be zero */ + + struct fsmap fmh_keys[2]; /* low and high keys for the mapping search */ + struct fsmap fmh_recs[]; /* returned records */ +}; + +/* Size of an fsmap_head with room for nr records. */ +static __inline__ size_t +fsmap_sizeof( + unsigned int nr) +{ + return sizeof(struct fsmap_head) + nr * sizeof(struct fsmap); +} + +/* Start the next fsmap query at the end of the current query results. */ +static __inline__ void +fsmap_advance( + struct fsmap_head *head) +{ + head->fmh_keys[0] = head->fmh_recs[head->fmh_entries - 1]; +} + +/* fmh_iflags values - set by FS_IOC_GETFSMAP caller in the header. */ +/* no flags defined yet */ +#define FMH_IF_VALID 0 + +/* fmh_oflags values - returned in the header segment only. */ +#define FMH_OF_DEV_T 0x1 /* fmr_device values will be dev_t */ + +/* fmr_flags values - returned for each non-header segment */ +#define FMR_OF_PREALLOC 0x1 /* segment = unwritten pre-allocation */ +#define FMR_OF_ATTR_FORK 0x2 /* segment = attribute fork */ +#define FMR_OF_EXTENT_MAP 0x4 /* segment = extent map */ +#define FMR_OF_SHARED 0x8 /* segment = shared with another file */ +#define FMR_OF_SPECIAL_OWNER 0x10 /* owner is a special value */ +#define FMR_OF_LAST 0x20 /* segment is the last in the dataset */ + +/* Each FS gets to define its own special owner codes. */ +#define FMR_OWNER(type, code) (((__u64)type << 32) | \ + ((__u64)code & 0xFFFFFFFFULL)) +#define FMR_OWNER_TYPE(owner) ((__u32)((__u64)owner >> 32)) +#define FMR_OWNER_CODE(owner) ((__u32)(((__u64)owner & 0xFFFFFFFFULL))) +#define FMR_OWN_FREE FMR_OWNER(0, 1) /* free space */ +#define FMR_OWN_UNKNOWN FMR_OWNER(0, 2) /* unknown owner */ +#define FMR_OWN_METADATA FMR_OWNER(0, 3) /* metadata */ + +#define FS_IOC_GETFSMAP _IOWR('X', 59, struct fsmap_head) + +#endif /* _LINUX_FSMAP_H */ diff --git a/scrub/phase6.c b/scrub/phase6.c index 87828b60..dd42b66c 100644 --- a/scrub/phase6.c +++ b/scrub/phase6.c @@ -10,6 +10,7 @@ #include "handle.h" #include "libfrog/paths.h" #include "libfrog/workqueue.h" +#include "libfrog/fsmap.h" #include "xfs_scrub.h" #include "common.h" #include "libfrog/bitmap.h" diff --git a/scrub/phase7.c b/scrub/phase7.c index bc652ab6..e24906d1 100644 --- a/scrub/phase7.c +++ b/scrub/phase7.c @@ -9,6 +9,7 @@ #include <sys/statvfs.h> #include "libfrog/paths.h" #include "libfrog/ptvar.h" +#include "libfrog/fsmap.h" #include "list.h" #include "xfs_scrub.h" #include "common.h" diff --git a/scrub/spacemap.c b/scrub/spacemap.c index a5508d56..b7f17e57 100644 --- a/scrub/spacemap.c +++ b/scrub/spacemap.c @@ -10,6 +10,7 @@ #include <sys/statvfs.h> #include "libfrog/workqueue.h" #include "libfrog/paths.h" +#include "libfrog/fsmap.h" #include "xfs_scrub.h" #include "common.h" #include "spacemap.h" diff --git a/spaceman/freesp.c b/spaceman/freesp.c index de301c19..4e46ab26 100644 --- a/spaceman/freesp.c +++ b/spaceman/freesp.c @@ -9,6 +9,7 @@ #include "libxfs.h" #include <linux/fiemap.h> #include "libfrog/fsgeom.h" +#include "libfrog/fsmap.h" #include "command.h" #include "init.h" #include "libfrog/paths.h" diff --git a/spaceman/space.h b/spaceman/space.h index 723209ed..a8055535 100644 --- a/spaceman/space.h +++ b/spaceman/space.h @@ -26,11 +26,7 @@ extern void help_init(void); extern void prealloc_init(void); extern void quit_init(void); extern void trim_init(void); -#ifdef HAVE_GETFSMAP extern void freesp_init(void); -#else -# define freesp_init() do { } while (0) -#endif extern void info_init(void); extern void health_init(void);