From mboxrd@z Thu Jan 1 00:00:00 1970 Return-Path: Received: from mail-wr1-f65.google.com ([209.85.221.65]:42906 "EHLO mail-wr1-f65.google.com" rhost-flags-OK-OK-OK-OK) by vger.kernel.org with ESMTP id S1725966AbeLGJJ0 (ORCPT ); Fri, 7 Dec 2018 04:09:26 -0500 Received: by mail-wr1-f65.google.com with SMTP id q18so3040936wrx.9 for ; Fri, 07 Dec 2018 01:09:24 -0800 (PST) Date: Fri, 7 Dec 2018 10:09:20 +0100 From: Carlos Maiolino To: "Darrick J. Wong" Cc: linux-fsdevel@vger.kernel.org, hch@lst.de, adilger@dilger.ca, sandeen@redhat.com, david@fromorbit.com Subject: Re: [PATCH 09/10 V2] Use FIEMAP for FIBMAP calls Message-ID: <20181207090920.jrnwwb35n6r5546c@hades.usersys.redhat.com> References: <20181205091728.29903-1-cmaiolino@redhat.com> <20181205091728.29903-10-cmaiolino@redhat.com> <20181205173650.GA8112@magnolia> MIME-Version: 1.0 Content-Type: text/plain; charset=us-ascii Content-Disposition: inline In-Reply-To: <20181205173650.GA8112@magnolia> Sender: linux-fsdevel-owner@vger.kernel.org List-ID: On Wed, Dec 05, 2018 at 09:36:50AM -0800, Darrick J. Wong wrote: > On Wed, Dec 05, 2018 at 10:17:27AM +0100, Carlos Maiolino wrote: > > Enables the usage of FIEMAP ioctl infrastructure to handle FIBMAP calls. > > From now on, ->bmap() methods can start to be removed from filesystems > > which already provides ->fiemap(). > > > > This adds a new helper - bmap_fiemap() - which is used to fill in the > > fiemap request, call into the underlying filesystem and check the flags > > set in the extent requested. > > > > Add a new fiemap fill extent callback to handl the in-kernel only > > fiemap_extent structure used for FIBMAP. > > > > V2: > > - Now based on the updated fiemap_extent_info, > > - move the fiemap call itself to a new helper > > > > Signed-off-by: Carlos Maiolino > > --- > > fs/inode.c | 42 ++++++++++++++++++++++++++++++++++++++++-- > > fs/ioctl.c | 32 ++++++++++++++++++++++++++++++++ > > include/linux/fs.h | 2 ++ > > 3 files changed, 74 insertions(+), 2 deletions(-) > > > > diff --git a/fs/inode.c b/fs/inode.c > > index db681d310465..f07cc183ddbd 100644 > > --- a/fs/inode.c > > +++ b/fs/inode.c > > @@ -1578,6 +1578,40 @@ void iput(struct inode *inode) > > } > > EXPORT_SYMBOL(iput); > > > > +static int bmap_fiemap(struct inode *inode, sector_t *block) > > +{ > > + struct fiemap_extent_info fieinfo = { 0, }; > > + struct fiemap_extent fextent; > > + u64 start = *block << inode->i_blkbits; > > + int error = -EINVAL; > > + > > + fextent.fe_logical = 0; > > + fextent.fe_physical = 0; > > + fieinfo.fi_extents_max = 1; > > + fieinfo.fi_extents_mapped = 0; > > + fieinfo.fi_extents_start = &fextent; > > + fieinfo.fi_start = start; > > + fieinfo.fi_len = 1 << inode->i_blkbits; > > + fieinfo.fi_flags = 0; > > + fieinfo.fi_cb = fiemap_fill_kernel_extent; > > + > > + error = inode->i_op->fiemap(inode, &fieinfo); > > + > > + if (error) > > + return error; > > + > > + if (fieinfo.fi_flags & (FIEMAP_EXTENT_UNKNOWN | > > + FIEMAP_EXTENT_ENCODED | > > + FIEMAP_EXTENT_DATA_INLINE | > > + FIEMAP_EXTENT_UNWRITTEN)) > > + return -EINVAL; > > AFAICT, three of the filesystems that support COW writes (xfs, ocfs2, > and btrfs) do not return bmap results for files with shared blocks. > This check here should include FIEMAP_EXTENT_SHARED since external > overwrites of a COW file block are bad news on btrfs (and ocfs2 and > xfs). Yes, it does need to check for FIEMAP_EXTENT_SHARED too, I had it on my plans but I forgot to add it when setting up the flags. Thanks for reminding me. > > > + > > + *block = (fextent.fe_physical + > > + (start - fextent.fe_logical)) >> inode->i_blkbits; > > Hmmm, so there's nothing here checking that the physical device fiemap > reports is the same device that was passed into the mount. This is > trivially true for most of the filesystems that implement bmap and > fiemap, but definitely not true for xfs or btrfs. I would bet most > userspace callers of bmap (since it's an ext2-era ioctl) make that > assumption and don't even know how to find the device. > > On xfs, the bmap implementation won't return any results for realtime > files, but it looks as though we suddenly will start doing that here, > because in the new bmap implementation we will use fiemap, and fiemap > reports extents without providing any context about which device they're > on, and that context-less extent gets passed back to bmap_fiemap. > > In any case, I think a better solution to the multi-device problem is to > start returning device information via struct fiemap_extent, at least > inside the kernel. Use one of the reserved fields to declare a new > '__u32 fe_device' field in struct fiemap_extent which can be the dev_t > device number, and then you can check that against inode->i_sb->s_bdev > to avoid returning results for the non-primary device of a multi-device > filesystem. Yes, you are right, I haven't thought about multi-dev filesystems. I checked btrfs code and it doesn't even support fibmap, exactly because of this problem, I wonder though, why it does support FIEMAP then, maybe because the fiemap idea isn't provide a way to userspace do IO directly to the device?! I'm not sure if crossing dev information is enough though, I did a quick read of btrfs code, and the assumption that the block/extent location won't change on time, could lead to a time bomb in the future. I wonder if it wouldn't maybe be better to add a flag, to identify the usage type, and the filesystem itself would define if it should return anything, or not. Like, for example, passing in fieinfo->fi_flags, something like FIEMAP_FIBMAP, and check inside the filesystem for it. >>From my shallow understanding of btrfs, looks like the location of the data, can be moved inside the same device, so, even if the devices are the same as you suggested, there is no guarantee the offset will be the same. Cheers. > > > + > > + return error; > > +} > > + > > /** > > * bmap - find a block number in a file > > * @inode: inode owning the block number being requested > > @@ -1594,10 +1628,14 @@ EXPORT_SYMBOL(iput); > > */ > > int bmap(struct inode *inode, sector_t *block) > > { > > - if (!inode->i_mapping->a_ops->bmap) > > + if (inode->i_op->fiemap) > > + return bmap_fiemap(inode, block); > > + else if (inode->i_mapping->a_ops->bmap) > > + *block = inode->i_mapping->a_ops->bmap(inode->i_mapping, > > + *block); > > + else > > return -EINVAL; > > Waitaminute. btrfs currently supports fiemap but not bmap, and now > suddenly it will support this legacy interface they've never supported > before. Are they on board with this? > > --D > > > > > - *block = inode->i_mapping->a_ops->bmap(inode->i_mapping, *block); > > return 0; > > } > > EXPORT_SYMBOL(bmap); > > diff --git a/fs/ioctl.c b/fs/ioctl.c > > index 6086978fe01e..bfa59df332bf 100644 > > --- a/fs/ioctl.c > > +++ b/fs/ioctl.c > > @@ -116,6 +116,38 @@ int fiemap_fill_user_extent(struct fiemap_extent_info *fieinfo, u64 logical, > > return (flags & FIEMAP_EXTENT_LAST) ? 1 : 0; > > } > > > > +int fiemap_fill_kernel_extent(struct fiemap_extent_info *fieinfo, u64 logical, > > + u64 phys, u64 len, u32 flags) > > +{ > > + struct fiemap_extent *extent = fieinfo->fi_extents_start; > > + > > + /* only count the extents */ > > + if (fieinfo->fi_extents_max == 0) { > > + fieinfo->fi_extents_mapped++; > > + return (flags & FIEMAP_EXTENT_LAST) ? 1 : 0; > > + } > > + > > + if (fieinfo->fi_extents_mapped >= fieinfo->fi_extents_max) > > + return 1; > > + > > + if (flags & SET_UNKNOWN_FLAGS) > > + flags |= FIEMAP_EXTENT_UNKNOWN; > > + if (flags & SET_NO_UNMOUNTED_IO_FLAGS) > > + flags |= FIEMAP_EXTENT_ENCODED; > > + if (flags & SET_NOT_ALIGNED_FLAGS) > > + flags |= FIEMAP_EXTENT_NOT_ALIGNED; > > + > > + extent->fe_logical = logical; > > + extent->fe_physical = phys; > > + extent->fe_length = len; > > + extent->fe_flags = flags; > > + > > + fieinfo->fi_extents_mapped++; > > + > > + if (fieinfo->fi_extents_mapped == fieinfo->fi_extents_max) > > + return 1; > > + return (flags & FIEMAP_EXTENT_LAST) ? 1 : 0; > > +} > > /** > > * fiemap_fill_next_extent - Fiemap helper function > > * @fieinfo: Fiemap context passed into ->fiemap > > diff --git a/include/linux/fs.h b/include/linux/fs.h > > index 7a434979201c..28bb523d532a 100644 > > --- a/include/linux/fs.h > > +++ b/include/linux/fs.h > > @@ -1711,6 +1711,8 @@ struct fiemap_extent_info { > > fiemap_fill_cb fi_cb; > > }; > > > > +int fiemap_fill_kernel_extent(struct fiemap_extent_info *info, u64 logical, > > + u64 phys, u64 len, u32 flags); > > int fiemap_fill_next_extent(struct fiemap_extent_info *info, u64 logical, > > u64 phys, u64 len, u32 flags); > > int fiemap_check_flags(struct fiemap_extent_info *fieinfo, u32 fs_flags); > > -- > > 2.17.2 > > -- Carlos