On 2018-12-10, Brian Foster <bfoster@xxxxxxxxxx> wrote: > On Mon, Dec 10, 2018 at 08:50:20AM -0800, Darrick J. Wong wrote: >> On Mon, Dec 10, 2018 at 11:11:22AM -0500, Brian Foster wrote: >> > 0x6e corresponds to the GROWFSDATA[_32] cmd and I think 0x10 is the >> > size, which is 16 bytes as opposed to the 12 bytes expected for >> > GROWFSDATA_32 for struct compat_xfs_growfs_data: >> > >> > typedef struct compat_xfs_growfs_data { >> > __u64 newblocks; /* new data subvol size, >> > fsblocks */ >> > __u32 imaxpct; /* new inode space percentage >> > limit */ >> > } __attribute__((packed)) compat_xfs_growfs_data_t; >> > >> > On a 64-bit kernel, that packed attribute is the difference between >> > expecting a padded 16 byte struct vs. a 12 byte version presumably from >> > a 32-bit application. So if you are calling into the ->compat_ioctl() >> > path I think the question is why is your xfsprogs sending the 16 byte >> > structure? >> >> ...because the x32 ABI is weird in that pointers are 4 bytes like on >> x86, but the registers are 64 bits wide like on x64, and (except for >> pointers being 4 bytes wide) the structure alignment rules follow x64. [...] > Yeah, it seems to me that fundamentally conflicts with the whole > BROKEN_X86_ALIGNMENT thing we have now. IIUC, compat_ioctl() on an > x86_64 kernel needs to account for x86 userspace via all of the > associated _32 ioctl commands as it already does, but at the same time > x32 means we could get any of the traditional x86_64 commands through > that path as well. In the specific case of this one ioctl on this one architecture since the only problem is unused padding at the end of the structure, the fix might be simple: just accept both ioctl numbers in the compat path. The packed compat struct layout looks like it should match what x32 userspace sends just fine. (I didn't realize x32 syscalls would go through compat_ioctl). i.e., perhaps we just do something like this? (TOTALLY UNTESTED) diff --git a/fs/xfs/xfs_ioctl32.c b/fs/xfs/xfs_ioctl32.c index fba115f4103a..b5a02f36d568 100644 --- a/fs/xfs/xfs_ioctl32.c +++ b/fs/xfs/xfs_ioctl32.c @@ -581,6 +581,9 @@ xfs_file_compat_ioctl( } case XFS_IOC_FSGEOMETRY_V1_32: return xfs_compat_ioc_fsgeometry_v1(mp, arg); +#ifdef CONFIG_X86_X32 + case XFS_IOC_FSGROWFSDATA: +#endif case XFS_IOC_FSGROWFSDATA_32: { struct xfs_growfs_data in; I can have a go at fixing the FSGEOMETRY ioctl too (and submit it properly) if this approach seems reasonable. Possibly other things may be broken too but I haven't hit any other issues yet in my XFS adventure. Cheers, Nick