diff mbox

ext4: create an ioctl report fs geometry

Message ID 20170218012012.GB6804@birch.djwong.org
State Superseded, archived
Headers show

Commit Message

Darrick Wong Feb. 18, 2017, 1:20 a.m. UTC
From: Darrick J. Wong <darrick.wong@oracle.com>

Add an ioctl to report the geometry of a mounted filesystem.

Signed-off-by: Darrick J. Wong <darrick.wong@oracle.com>
---
 fs/ext4/ext4.h  |   36 ++++++++++++++++++++++++++++++++++
 fs/ext4/ioctl.c |   58 +++++++++++++++++++++++++++++++++++++++++++++++++++++++
 2 files changed, 94 insertions(+)

Comments

Andreas Dilger Feb. 27, 2017, 8:43 p.m. UTC | #1
> On Feb 17, 2017, at 6:20 PM, Darrick J. Wong <darrick.wong@oracle.com> wrote:
> 
> From: Darrick J. Wong <darrick.wong@oracle.com>
> 
> Add an ioctl to report the geometry of a mounted filesystem.
> 
> Signed-off-by: Darrick J. Wong <darrick.wong@oracle.com>
> ---
> fs/ext4/ext4.h  |   36 ++++++++++++++++++++++++++++++++++
> fs/ext4/ioctl.c |   58 +++++++++++++++++++++++++++++++++++++++++++++++++++++++
> 2 files changed, 94 insertions(+)
> 
> diff --git a/fs/ext4/ext4.h b/fs/ext4/ext4.h
> index 2163c1e..bee511d 100644
> --- a/fs/ext4/ext4.h
> +++ b/fs/ext4/ext4.h
> @@ -614,6 +614,41 @@ enum {
> #define EXT4_FREE_BLOCKS_NOFREE_FIRST_CLUSTER	0x0010
> #define EXT4_FREE_BLOCKS_NOFREE_LAST_CLUSTER	0x0020
> 
> +/* ext4 fs geometry. */
> +struct ext4_fsop_geom {
> +	__u32		blocksize;	/* filesystem (data) block size */
> +	__u32		inodecount;	/* inode count			*/

Should this be a __u64, even though we don't currently support 64-bit inode
numbers today?  Easy to change the struct now, much harder to do later.  Would
need to align it properly though.

> +	__u32		agblocks;	/* fsblocks in an AG		*/
> +	__u32		agcount;	/* number of allocation groups	*/

Please don't introduce XFS terminology into ext4.  In ext4 these are called
"block groups", and use "bg" as the prefix.

(style) please add a unique prefix to the struct names, like "efg_" or similar
("fg_" if this is being hoisted to the VFS at some point, to avoid churn).

> +	__u32		logblocks;	/* fsblocks in the log		*/
> +	__u32		resvblocks;	/* number of reserved blocks	*/
> +	__u32		inodesize;	/* inode size in bytes		*/
> +	__u32		agiblocks;	/* inode blocks per AG		*/

Similarly, "bgi_blocks      /* inode blocks per block group */

> +	__u64		datablocks;	/* fsblocks in data subvolume	*/
> +	unsigned char	uuid[16];	/* unique id of the filesystem	*/
> +	__u32		sunit;		/* stripe unit, fsblocks	*/
> +	__u32		swidth;		/* stripe width, fsblocks	*/
> +	__s32		version;	/* structure version		*/

Why would you want to allow a negative version?  Also, feature flags are so
much better than version numbers, let's just stick with flags.

> +	__u32		flags;		/* superblock version flags	*/

s/version/feature/?

> +	__u32		clustersize;	/* fs cluster size		*/
> +	__u32		flexbgsize;	/* number of bg's in a flexbg	*/
> +	__u64		resv[6];
> +};
> +
> +#define EXT4_FSOP_GEOM_VERSION	0
> +
> +#define EXT4_FSOP_GEOM_FLAGS_ATTR	0x00001	/* attributes in use	 */

Sorry, it isn't clear what this means?

> +#define EXT4_FSOP_GEOM_FLAGS_NLINK	0x00002	/* 32-bit nlink values	 */
> +#define EXT4_FSOP_GEOM_FLAGS_QUOTA	0x00004	/* quotas enabled	 */
> +#define EXT4_FSOP_GEOM_FLAGS_PROJQ	0x00008	/* project quotas	 */
> +#define EXT4_FSOP_GEOM_FLAGS_METACRC	0x00010	/* metadata checksums	 */

"METACKSUM" would be better?

> +#define EXT4_FSOP_GEOM_FLAGS_FTYPE	0x00020	/* inode directory types */

Are there any filesystems in existence that don't have this feature?

> +#define EXT4_FSOP_GEOM_FLAGS_64BIT	0x00040	/* 64-bit support	 */
> +#define EXT4_FSOP_GEOM_FLAGS_INLINEDATA	0x00080	/* inline data		 */
> +#define EXT4_FSOP_GEOM_FLAGS_ENCRYPT	0x00100	/* encrypted files	 */
> +#define EXT4_FSOP_GEOM_FLAGS_LARGEDIR	0x00200	/* large directories	 */
> +#define EXT4_FSOP_GEOM_FLAGS_EXTENTS	0x00400	/* extents		 */

Sigh, this isn't quite the same as the existing feature flags, but doesn't
actually encompass all of the available features, nor could it ever fit all
three __u32 into one __u32.  It isn't clear why this duplicate information
is needed in this struct?  If needed, why not return all three feature flags?

> +
> /*
>  * ioctl commands
>  */
> @@ -638,6 +673,7 @@ enum {
> #define EXT4_IOC_SET_ENCRYPTION_POLICY	FS_IOC_SET_ENCRYPTION_POLICY
> #define EXT4_IOC_GET_ENCRYPTION_PWSALT	FS_IOC_GET_ENCRYPTION_PWSALT
> #define EXT4_IOC_GET_ENCRYPTION_POLICY	FS_IOC_GET_ENCRYPTION_POLICY
> +#define EXT4_IOC_FSGEOMETRY		_IOR ('f', 19, struct ext4_fsop_geom)
> 
> #ifndef FS_IOC_FSGETXATTR
> /* Until the uapi changes get merged for project quota... */
> diff --git a/fs/ext4/ioctl.c b/fs/ext4/ioctl.c
> index ed62465..e0b695b 100644
> --- a/fs/ext4/ioctl.c
> +++ b/fs/ext4/ioctl.c
> @@ -547,6 +547,62 @@ ext4_ioc_getfsmap(
> 	return 0;
> }
> 
> +static int
> +ext4_ioc_fsgeometry(
> +	struct super_block	*sb,
> +	void			__user *arg)

Should pack arguments on the declaration line like:

static int ext4_ioc_fsgeometry(struct super_block *sb, void __user *arg)

> +{
> +	struct ext4_sb_info	*sbi = EXT4_SB(sb);
> +	journal_t		*journal = sbi->s_journal;
> +	struct ext4_fsop_geom	geom;

I thought kernel style frowned upon aligning local variable declarations
(at least we were told that for the Lustre code and have had to remove it).

> +
> +	memset(&geom, 0, sizeof(geom));

You could avoid the explicit memset by initializing the struct at declaration.

> +	geom.version = EXT4_FSOP_GEOM_VERSION;
> +	geom.blocksize = EXT4_BLOCK_SIZE(sb);
> +	geom.inodecount = le32_to_cpu(sbi->s_es->s_inodes_count);
> +	geom.agblocks = EXT4_BLOCKS_PER_GROUP(sb);
> +	geom.agcount = sbi->s_groups_count;
> +	geom.logblocks = journal ? journal->j_maxlen : 0;
> +	geom.resvblocks = ext4_r_blocks_count(sbi->s_es);
> +	geom.inodesize = EXT4_INODE_SIZE(sb);
> +	geom.agiblocks = sbi->s_itb_per_group;
> +	geom.datablocks = ext4_blocks_count(sbi->s_es);
> +	memcpy(geom.uuid, sbi->s_es->s_uuid, sizeof(sbi->s_es->s_uuid));
> +	geom.sunit = le16_to_cpu(sbi->s_es->s_raid_stride);
> +	geom.swidth = le16_to_cpu(sbi->s_es->s_raid_stripe_width);
> +	geom.flags = 0;
> +	if (ext4_has_feature_xattr(sb))
> +		geom.flags |= EXT4_FSOP_GEOM_FLAGS_ATTR;
> +	if (ext4_has_feature_dir_nlink(sb))
> +		geom.flags |= EXT4_FSOP_GEOM_FLAGS_NLINK;
> +	if (ext4_has_feature_quota(sb))
> +		geom.flags |= EXT4_FSOP_GEOM_FLAGS_QUOTA;
> +	if (ext4_has_feature_project(sb))
> +		geom.flags |= EXT4_FSOP_GEOM_FLAGS_PROJQ;
> +	if (ext4_has_metadata_csum(sb))
> +		geom.flags |= EXT4_FSOP_GEOM_FLAGS_METACRC;
> +	if (ext4_has_feature_filetype(sb))
> +		geom.flags |= EXT4_FSOP_GEOM_FLAGS_FTYPE;
> +	if (ext4_has_feature_64bit(sb))
> +		geom.flags |= EXT4_FSOP_GEOM_FLAGS_64BIT;
> +	if (ext4_has_feature_inline_data(sb))
> +		geom.flags |= EXT4_FSOP_GEOM_FLAGS_INLINEDATA;
> +	if (ext4_has_feature_encrypt(sb))
> +		geom.flags |= EXT4_FSOP_GEOM_FLAGS_ENCRYPT;
> +	if (ext4_has_feature_largedir(sb))
> +		geom.flags |= EXT4_FSOP_GEOM_FLAGS_LARGEDIR;
> +	if (ext4_has_feature_extents(sb))
> +		geom.flags |= EXT4_FSOP_GEOM_FLAGS_EXTENTS;
> +	if (ext4_has_feature_bigalloc(sb))
> +		geom.clustersize = EXT4_C2B(sbi, 1);
> +	if (ext4_has_feature_flex_bg(sb))
> +		geom.flexbgsize = ext4_flex_bg_size(sbi);
> +
> +	if (copy_to_user(arg, &geom, sizeof(geom)))
> +		return -EFAULT;
> +	return 0;
> +}
> +
> long ext4_ioctl(struct file *filp, unsigned int cmd, unsigned long arg)
> {
> 	struct inode *inode = file_inode(filp);
> @@ -557,6 +613,8 @@ long ext4_ioctl(struct file *filp, unsigned int cmd, unsigned long arg)
> 	ext4_debug("cmd = %u, arg = %lu\n", cmd, arg);
> 
> 	switch (cmd) {
> +	case EXT4_IOC_FSGEOMETRY:
> +		return ext4_ioc_fsgeometry(sb, (void __user *)arg);

> 	case FS_IOC_GETFSMAP:
> 		return ext4_ioc_getfsmap(sb, (void __user *)arg);
> 	case EXT4_IOC_GETFLAGS:


Cheers, Andreas
Darrick Wong Feb. 27, 2017, 9:30 p.m. UTC | #2
On Mon, Feb 27, 2017 at 01:43:37PM -0700, Andreas Dilger wrote:
> 
> > On Feb 17, 2017, at 6:20 PM, Darrick J. Wong <darrick.wong@oracle.com> wrote:
> > 
> > From: Darrick J. Wong <darrick.wong@oracle.com>
> > 
> > Add an ioctl to report the geometry of a mounted filesystem.
> > 
> > Signed-off-by: Darrick J. Wong <darrick.wong@oracle.com>
> > ---
> > fs/ext4/ext4.h  |   36 ++++++++++++++++++++++++++++++++++
> > fs/ext4/ioctl.c |   58 +++++++++++++++++++++++++++++++++++++++++++++++++++++++
> > 2 files changed, 94 insertions(+)
> > 
> > diff --git a/fs/ext4/ext4.h b/fs/ext4/ext4.h
> > index 2163c1e..bee511d 100644
> > --- a/fs/ext4/ext4.h
> > +++ b/fs/ext4/ext4.h
> > @@ -614,6 +614,41 @@ enum {
> > #define EXT4_FREE_BLOCKS_NOFREE_FIRST_CLUSTER	0x0010
> > #define EXT4_FREE_BLOCKS_NOFREE_LAST_CLUSTER	0x0020
> > 
> > +/* ext4 fs geometry. */
> > +struct ext4_fsop_geom {
> > +	__u32		blocksize;	/* filesystem (data) block size */
> > +	__u32		inodecount;	/* inode count			*/
> 
> Should this be a __u64, even though we don't currently support 64-bit inode
> numbers today?  Easy to change the struct now, much harder to do later.  Would
> need to align it properly though.

Sure.

> > +	__u32		agblocks;	/* fsblocks in an AG		*/
> > +	__u32		agcount;	/* number of allocation groups	*/
> 
> Please don't introduce XFS terminology into ext4.  In ext4 these are called
> "block groups", and use "bg" as the prefix.

Ugh, I forgot, sorry. :/

> (style) please add a unique prefix to the struct names, like "efg_" or similar
> ("fg_" if this is being hoisted to the VFS at some point, to avoid churn).

efg_, then.  I have no intention of ever hoisting fs geometry ioctls
to the VFS; each fs has its own distinct design.

FYI Ted also suggested trying to find a common VFS geometry definition,
but I don't think that's really possible beyond what stat{,v}fs returns.

> > +	__u32		logblocks;	/* fsblocks in the log		*/
> > +	__u32		resvblocks;	/* number of reserved blocks	*/
> > +	__u32		inodesize;	/* inode size in bytes		*/
> > +	__u32		agiblocks;	/* inode blocks per AG		*/
> 
> Similarly, "bgi_blocks      /* inode blocks per block group */

Fixed.

> > +	__u64		datablocks;	/* fsblocks in data subvolume	*/
> > +	unsigned char	uuid[16];	/* unique id of the filesystem	*/
> > +	__u32		sunit;		/* stripe unit, fsblocks	*/
> > +	__u32		swidth;		/* stripe width, fsblocks	*/
> > +	__s32		version;	/* structure version		*/
> 
> Why would you want to allow a negative version?  Also, feature flags are so
> much better than version numbers, let's just stick with flags.

It was for the structure version, but meh, let's get rid of it.
It'll free up 32 bytes for the expanded inodecount.

> > +	__u32		flags;		/* superblock version flags	*/
> 
> s/version/feature/?

Oops, yes.

> > +	__u32		clustersize;	/* fs cluster size		*/
> > +	__u32		flexbgsize;	/* number of bg's in a flexbg	*/
> > +	__u64		resv[6];
> > +};
> > +
> > +#define EXT4_FSOP_GEOM_VERSION	0
> > +
> > +#define EXT4_FSOP_GEOM_FLAGS_ATTR	0x00001	/* attributes in use	 */
> 
> Sorry, it isn't clear what this means?

"extended attributes in use".  This means that the COMPAT_EXT_ATTR flag is set.

> > +#define EXT4_FSOP_GEOM_FLAGS_NLINK	0x00002	/* 32-bit nlink values	 */
> > +#define EXT4_FSOP_GEOM_FLAGS_QUOTA	0x00004	/* quotas enabled	 */
> > +#define EXT4_FSOP_GEOM_FLAGS_PROJQ	0x00008	/* project quotas	 */
> > +#define EXT4_FSOP_GEOM_FLAGS_METACRC	0x00010	/* metadata checksums	 */
> 
> "METACKSUM" would be better?

META_CSUM?

> > +#define EXT4_FSOP_GEOM_FLAGS_FTYPE	0x00020	/* inode directory types */
> 
> Are there any filesystems in existence that don't have this feature?

I dunno, but it's a superblock feature flag...

> > +#define EXT4_FSOP_GEOM_FLAGS_64BIT	0x00040	/* 64-bit support	 */
> > +#define EXT4_FSOP_GEOM_FLAGS_INLINEDATA	0x00080	/* inline data		 */
> > +#define EXT4_FSOP_GEOM_FLAGS_ENCRYPT	0x00100	/* encrypted files	 */
> > +#define EXT4_FSOP_GEOM_FLAGS_LARGEDIR	0x00200	/* large directories	 */
> > +#define EXT4_FSOP_GEOM_FLAGS_EXTENTS	0x00400	/* extents		 */
> 
> Sigh, this isn't quite the same as the existing feature flags, but doesn't
> actually encompass all of the available features, nor could it ever fit all
> three __u32 into one __u32.  It isn't clear why this duplicate information
> is needed in this struct?  If needed, why not return all three feature flags?

I was trying to capture all of the feature flags that have userspace-visible
behavioral changes and seem to be in use by the current ext4 codebase.

Any thoughts, Ted?

> 
> > +
> > /*
> >  * ioctl commands
> >  */
> > @@ -638,6 +673,7 @@ enum {
> > #define EXT4_IOC_SET_ENCRYPTION_POLICY	FS_IOC_SET_ENCRYPTION_POLICY
> > #define EXT4_IOC_GET_ENCRYPTION_PWSALT	FS_IOC_GET_ENCRYPTION_PWSALT
> > #define EXT4_IOC_GET_ENCRYPTION_POLICY	FS_IOC_GET_ENCRYPTION_POLICY
> > +#define EXT4_IOC_FSGEOMETRY		_IOR ('f', 19, struct ext4_fsop_geom)
> > 
> > #ifndef FS_IOC_FSGETXATTR
> > /* Until the uapi changes get merged for project quota... */
> > diff --git a/fs/ext4/ioctl.c b/fs/ext4/ioctl.c
> > index ed62465..e0b695b 100644
> > --- a/fs/ext4/ioctl.c
> > +++ b/fs/ext4/ioctl.c
> > @@ -547,6 +547,62 @@ ext4_ioc_getfsmap(
> > 	return 0;
> > }
> > 
> > +static int
> > +ext4_ioc_fsgeometry(
> > +	struct super_block	*sb,
> > +	void			__user *arg)
> 
> Should pack arguments on the declaration line like:
> 
> static int ext4_ioc_fsgeometry(struct super_block *sb, void __user *arg)
> 
> > +{
> > +	struct ext4_sb_info	*sbi = EXT4_SB(sb);
> > +	journal_t		*journal = sbi->s_journal;
> > +	struct ext4_fsop_geom	geom;
> 
> I thought kernel style frowned upon aligning local variable declarations
> (at least we were told that for the Lustre code and have had to remove it).

<shrug> Personally I think it makes the code much easier to scan with one's
eyeballs when both arguments and local variables line up like that.

(But you're probably right that it's frowned upon.  But the entire
XFS codebase does that.)

> > +
> > +	memset(&geom, 0, sizeof(geom));
> 
> You could avoid the explicit memset by initializing the struct at declaration.

<nod>

Thanks for the review!

--D

> > +	geom.version = EXT4_FSOP_GEOM_VERSION;
> > +	geom.blocksize = EXT4_BLOCK_SIZE(sb);
> > +	geom.inodecount = le32_to_cpu(sbi->s_es->s_inodes_count);
> > +	geom.agblocks = EXT4_BLOCKS_PER_GROUP(sb);
> > +	geom.agcount = sbi->s_groups_count;
> > +	geom.logblocks = journal ? journal->j_maxlen : 0;
> > +	geom.resvblocks = ext4_r_blocks_count(sbi->s_es);
> > +	geom.inodesize = EXT4_INODE_SIZE(sb);
> > +	geom.agiblocks = sbi->s_itb_per_group;
> > +	geom.datablocks = ext4_blocks_count(sbi->s_es);
> > +	memcpy(geom.uuid, sbi->s_es->s_uuid, sizeof(sbi->s_es->s_uuid));
> > +	geom.sunit = le16_to_cpu(sbi->s_es->s_raid_stride);
> > +	geom.swidth = le16_to_cpu(sbi->s_es->s_raid_stripe_width);
> > +	geom.flags = 0;
> > +	if (ext4_has_feature_xattr(sb))
> > +		geom.flags |= EXT4_FSOP_GEOM_FLAGS_ATTR;
> > +	if (ext4_has_feature_dir_nlink(sb))
> > +		geom.flags |= EXT4_FSOP_GEOM_FLAGS_NLINK;
> > +	if (ext4_has_feature_quota(sb))
> > +		geom.flags |= EXT4_FSOP_GEOM_FLAGS_QUOTA;
> > +	if (ext4_has_feature_project(sb))
> > +		geom.flags |= EXT4_FSOP_GEOM_FLAGS_PROJQ;
> > +	if (ext4_has_metadata_csum(sb))
> > +		geom.flags |= EXT4_FSOP_GEOM_FLAGS_METACRC;
> > +	if (ext4_has_feature_filetype(sb))
> > +		geom.flags |= EXT4_FSOP_GEOM_FLAGS_FTYPE;
> > +	if (ext4_has_feature_64bit(sb))
> > +		geom.flags |= EXT4_FSOP_GEOM_FLAGS_64BIT;
> > +	if (ext4_has_feature_inline_data(sb))
> > +		geom.flags |= EXT4_FSOP_GEOM_FLAGS_INLINEDATA;
> > +	if (ext4_has_feature_encrypt(sb))
> > +		geom.flags |= EXT4_FSOP_GEOM_FLAGS_ENCRYPT;
> > +	if (ext4_has_feature_largedir(sb))
> > +		geom.flags |= EXT4_FSOP_GEOM_FLAGS_LARGEDIR;
> > +	if (ext4_has_feature_extents(sb))
> > +		geom.flags |= EXT4_FSOP_GEOM_FLAGS_EXTENTS;
> > +	if (ext4_has_feature_bigalloc(sb))
> > +		geom.clustersize = EXT4_C2B(sbi, 1);
> > +	if (ext4_has_feature_flex_bg(sb))
> > +		geom.flexbgsize = ext4_flex_bg_size(sbi);
> > +
> > +	if (copy_to_user(arg, &geom, sizeof(geom)))
> > +		return -EFAULT;
> > +	return 0;
> > +}
> > +
> > long ext4_ioctl(struct file *filp, unsigned int cmd, unsigned long arg)
> > {
> > 	struct inode *inode = file_inode(filp);
> > @@ -557,6 +613,8 @@ long ext4_ioctl(struct file *filp, unsigned int cmd, unsigned long arg)
> > 	ext4_debug("cmd = %u, arg = %lu\n", cmd, arg);
> > 
> > 	switch (cmd) {
> > +	case EXT4_IOC_FSGEOMETRY:
> > +		return ext4_ioc_fsgeometry(sb, (void __user *)arg);
> 
> > 	case FS_IOC_GETFSMAP:
> > 		return ext4_ioc_getfsmap(sb, (void __user *)arg);
> > 	case EXT4_IOC_GETFLAGS:
> 
> 
> Cheers, Andreas
> 
> 
> 
> 
>
diff mbox

Patch

diff --git a/fs/ext4/ext4.h b/fs/ext4/ext4.h
index 2163c1e..bee511d 100644
--- a/fs/ext4/ext4.h
+++ b/fs/ext4/ext4.h
@@ -614,6 +614,41 @@  enum {
 #define EXT4_FREE_BLOCKS_NOFREE_FIRST_CLUSTER	0x0010
 #define EXT4_FREE_BLOCKS_NOFREE_LAST_CLUSTER	0x0020
 
+/* ext4 fs geometry. */
+struct ext4_fsop_geom {
+	__u32		blocksize;	/* filesystem (data) block size */
+	__u32		inodecount;	/* inode count			*/
+	__u32		agblocks;	/* fsblocks in an AG		*/
+	__u32		agcount;	/* number of allocation groups	*/
+	__u32		logblocks;	/* fsblocks in the log		*/
+	__u32		resvblocks;	/* number of reserved blocks	*/
+	__u32		inodesize;	/* inode size in bytes		*/
+	__u32		agiblocks;	/* inode blocks per AG		*/
+	__u64		datablocks;	/* fsblocks in data subvolume	*/
+	unsigned char	uuid[16];	/* unique id of the filesystem	*/
+	__u32		sunit;		/* stripe unit, fsblocks	*/
+	__u32		swidth;		/* stripe width, fsblocks	*/
+	__s32		version;	/* structure version		*/
+	__u32		flags;		/* superblock version flags	*/
+	__u32		clustersize;	/* fs cluster size		*/
+	__u32		flexbgsize;	/* number of bg's in a flexbg	*/
+	__u64		resv[6];
+};
+
+#define EXT4_FSOP_GEOM_VERSION	0
+
+#define EXT4_FSOP_GEOM_FLAGS_ATTR	0x00001	/* attributes in use	 */
+#define EXT4_FSOP_GEOM_FLAGS_NLINK	0x00002	/* 32-bit nlink values	 */
+#define EXT4_FSOP_GEOM_FLAGS_QUOTA	0x00004	/* quotas enabled	 */
+#define EXT4_FSOP_GEOM_FLAGS_PROJQ	0x00008	/* project quotas	 */
+#define EXT4_FSOP_GEOM_FLAGS_METACRC	0x00010	/* metadata checksums	 */
+#define EXT4_FSOP_GEOM_FLAGS_FTYPE	0x00020	/* inode directory types */
+#define EXT4_FSOP_GEOM_FLAGS_64BIT	0x00040	/* 64-bit support	 */
+#define EXT4_FSOP_GEOM_FLAGS_INLINEDATA	0x00080	/* inline data		 */
+#define EXT4_FSOP_GEOM_FLAGS_ENCRYPT	0x00100	/* encrypted files	 */
+#define EXT4_FSOP_GEOM_FLAGS_LARGEDIR	0x00200	/* large directories	 */
+#define EXT4_FSOP_GEOM_FLAGS_EXTENTS	0x00400	/* extents		 */
+
 /*
  * ioctl commands
  */
@@ -638,6 +673,7 @@  enum {
 #define EXT4_IOC_SET_ENCRYPTION_POLICY	FS_IOC_SET_ENCRYPTION_POLICY
 #define EXT4_IOC_GET_ENCRYPTION_PWSALT	FS_IOC_GET_ENCRYPTION_PWSALT
 #define EXT4_IOC_GET_ENCRYPTION_POLICY	FS_IOC_GET_ENCRYPTION_POLICY
+#define EXT4_IOC_FSGEOMETRY		_IOR ('f', 19, struct ext4_fsop_geom)
 
 #ifndef FS_IOC_FSGETXATTR
 /* Until the uapi changes get merged for project quota... */
diff --git a/fs/ext4/ioctl.c b/fs/ext4/ioctl.c
index ed62465..e0b695b 100644
--- a/fs/ext4/ioctl.c
+++ b/fs/ext4/ioctl.c
@@ -547,6 +547,62 @@  ext4_ioc_getfsmap(
 	return 0;
 }
 
+static int
+ext4_ioc_fsgeometry(
+	struct super_block	*sb,
+	void			__user *arg)
+{
+	struct ext4_sb_info	*sbi = EXT4_SB(sb);
+	journal_t		*journal = sbi->s_journal;
+	struct ext4_fsop_geom	geom;
+
+	memset(&geom, 0, sizeof(geom));
+	geom.version = EXT4_FSOP_GEOM_VERSION;
+	geom.blocksize = EXT4_BLOCK_SIZE(sb);
+	geom.inodecount = le32_to_cpu(sbi->s_es->s_inodes_count);
+	geom.agblocks = EXT4_BLOCKS_PER_GROUP(sb);
+	geom.agcount = sbi->s_groups_count;
+	geom.logblocks = journal ? journal->j_maxlen : 0;
+	geom.resvblocks = ext4_r_blocks_count(sbi->s_es);
+	geom.inodesize = EXT4_INODE_SIZE(sb);
+	geom.agiblocks = sbi->s_itb_per_group;
+	geom.datablocks = ext4_blocks_count(sbi->s_es);
+	memcpy(geom.uuid, sbi->s_es->s_uuid, sizeof(sbi->s_es->s_uuid));
+	geom.sunit = le16_to_cpu(sbi->s_es->s_raid_stride);
+	geom.swidth = le16_to_cpu(sbi->s_es->s_raid_stripe_width);
+	geom.flags = 0;
+	if (ext4_has_feature_xattr(sb))
+		geom.flags |= EXT4_FSOP_GEOM_FLAGS_ATTR;
+	if (ext4_has_feature_dir_nlink(sb))
+		geom.flags |= EXT4_FSOP_GEOM_FLAGS_NLINK;
+	if (ext4_has_feature_quota(sb))
+		geom.flags |= EXT4_FSOP_GEOM_FLAGS_QUOTA;
+	if (ext4_has_feature_project(sb))
+		geom.flags |= EXT4_FSOP_GEOM_FLAGS_PROJQ;
+	if (ext4_has_metadata_csum(sb))
+		geom.flags |= EXT4_FSOP_GEOM_FLAGS_METACRC;
+	if (ext4_has_feature_filetype(sb))
+		geom.flags |= EXT4_FSOP_GEOM_FLAGS_FTYPE;
+	if (ext4_has_feature_64bit(sb))
+		geom.flags |= EXT4_FSOP_GEOM_FLAGS_64BIT;
+	if (ext4_has_feature_inline_data(sb))
+		geom.flags |= EXT4_FSOP_GEOM_FLAGS_INLINEDATA;
+	if (ext4_has_feature_encrypt(sb))
+		geom.flags |= EXT4_FSOP_GEOM_FLAGS_ENCRYPT;
+	if (ext4_has_feature_largedir(sb))
+		geom.flags |= EXT4_FSOP_GEOM_FLAGS_LARGEDIR;
+	if (ext4_has_feature_extents(sb))
+		geom.flags |= EXT4_FSOP_GEOM_FLAGS_EXTENTS;
+	if (ext4_has_feature_bigalloc(sb))
+		geom.clustersize = EXT4_C2B(sbi, 1);
+	if (ext4_has_feature_flex_bg(sb))
+		geom.flexbgsize = ext4_flex_bg_size(sbi);
+
+	if (copy_to_user(arg, &geom, sizeof(geom)))
+		return -EFAULT;
+	return 0;
+}
+
 long ext4_ioctl(struct file *filp, unsigned int cmd, unsigned long arg)
 {
 	struct inode *inode = file_inode(filp);
@@ -557,6 +613,8 @@  long ext4_ioctl(struct file *filp, unsigned int cmd, unsigned long arg)
 	ext4_debug("cmd = %u, arg = %lu\n", cmd, arg);
 
 	switch (cmd) {
+	case EXT4_IOC_FSGEOMETRY:
+		return ext4_ioc_fsgeometry(sb, (void __user *)arg);
 	case FS_IOC_GETFSMAP:
 		return ext4_ioc_getfsmap(sb, (void __user *)arg);
 	case EXT4_IOC_GETFLAGS: