All of lore.kernel.org
 help / color / mirror / Atom feed
From: "Darrick J. Wong" <djwong@kernel.org>
To: miklos@szeredi.hu
Cc: joannelkoong@gmail.com, neal@gompa.dev,
	linux-fsdevel@vger.kernel.org, bernd@bsbernd.com,
	fuse-devel@lists.linux.dev
Subject: Re: [PATCH 14/33] fuse: implement direct IO with iomap
Date: Wed, 13 May 2026 11:44:51 -0700	[thread overview]
Message-ID: <20260513184451.GM9544@frogsfrogsfrogs> (raw)
In-Reply-To: <177747205450.4101881.8473164564364220576.stgit@frogsfrogsfrogs>

On Wed, Apr 29, 2026 at 07:27:20AM -0700, Darrick J. Wong wrote:
> From: Darrick J. Wong <djwong@kernel.org>
> 
> Start implementing the fuse-iomap file I/O paths by adding direct I/O
> support and all the signalling flags that come with it.  Buffered I/O
> is much more complicated, so we leave that to a subsequent patch.
> 
> Signed-off-by: "Darrick J. Wong" <djwong@kernel.org>
> ---
>  fs/fuse/fuse_i.h          |   20 +++
>  fs/fuse/fuse_iomap.h      |   18 ++
>  include/uapi/linux/fuse.h |   27 ++++
>  fs/fuse/dir.c             |   13 ++
>  fs/fuse/file.c            |   29 +++-
>  fs/fuse/fuse_iomap.c      |  335 +++++++++++++++++++++++++++++++++++++++++++++
>  fs/fuse/inode.c           |    3 
>  fs/fuse/iomode.c          |    2 
>  8 files changed, 441 insertions(+), 6 deletions(-)
> 
> 
> diff --git a/fs/fuse/fuse_i.h b/fs/fuse/fuse_i.h
> index a4d64fd2837778..ea95615fbff598 100644
> --- a/fs/fuse/fuse_i.h
> +++ b/fs/fuse/fuse_i.h
> @@ -188,6 +188,11 @@ struct fuse_inode {
>  
>  			/* waitq for direct-io completion */
>  			wait_queue_head_t direct_io_waitq;
> +
> +#ifdef CONFIG_FUSE_IOMAP
> +			/* file size as reported by fuse server */
> +			loff_t i_disk_size;
> +#endif
>  		};
>  
>  		/* readdir cache (directory only) */
> @@ -658,6 +663,16 @@ struct fuse_sync_bucket {
>  	struct rcu_head rcu;
>  };
>  
> +#ifdef CONFIG_FUSE_IOMAP
> +struct fuse_iomap_conn {
> +	/* fuse server doesn't implement iomap_end */
> +	unsigned int no_end:1;
> +
> +	/* fuse server doesn't implement iomap_ioend */
> +	unsigned int no_ioend:1;
> +};
> +#endif
> +
>  /**
>   * A Fuse connection.
>   *
> @@ -1006,6 +1021,11 @@ struct fuse_conn {
>  	struct idr backing_files_map;
>  #endif
>  
> +#ifdef CONFIG_FUSE_IOMAP
> +	/** iomap information */
> +	struct fuse_iomap_conn iomap_conn;
> +#endif
> +
>  #ifdef CONFIG_FUSE_IO_URING
>  	/**  uring connection information*/
>  	struct fuse_ring *ring;
> diff --git a/fs/fuse/fuse_iomap.h b/fs/fuse/fuse_iomap.h
> index 8ba30a496545f5..476e1b869d1906 100644
> --- a/fs/fuse/fuse_iomap.h
> +++ b/fs/fuse/fuse_iomap.h
> @@ -38,6 +38,17 @@ int fuse_iomap_fiemap(struct inode *inode, struct fiemap_extent_info *fieinfo,
>  		      u64 start, u64 length);
>  loff_t fuse_iomap_lseek(struct file *file, loff_t offset, int whence);
>  sector_t fuse_iomap_bmap(struct address_space *mapping, sector_t block);
> +
> +void fuse_iomap_open(struct inode *inode, struct file *file);
> +int fuse_iomap_finish_open(const struct fuse_file *ff,
> +			   const struct inode *inode);
> +void fuse_iomap_open_truncate(struct inode *inode);
> +
> +void fuse_iomap_set_disk_size(struct fuse_inode *fi, loff_t newsize);
> +int fuse_iomap_setsize_finish(struct inode *inode, loff_t newsize);
> +
> +ssize_t fuse_iomap_read_iter(struct kiocb *iocb, struct iov_iter *to);
> +ssize_t fuse_iomap_write_iter(struct kiocb *iocb, struct iov_iter *from);
>  #else
>  # define fuse_iomap_enabled(...)		(false)
>  # define fuse_has_iomap(...)			(false)
> @@ -49,6 +60,13 @@ sector_t fuse_iomap_bmap(struct address_space *mapping, sector_t block);
>  # define fuse_iomap_fiemap			NULL
>  # define fuse_iomap_lseek(...)			(-ENOSYS)
>  # define fuse_iomap_bmap(...)			(-ENOSYS)
> +# define fuse_iomap_open(...)			((void)0)
> +# define fuse_iomap_finish_open(...)		(-ENOSYS)
> +# define fuse_iomap_open_truncate(...)		((void)0)
> +# define fuse_iomap_set_disk_size(...)		((void)0)
> +# define fuse_iomap_setsize_finish(...)		(-ENOSYS)
> +# define fuse_iomap_read_iter(...)		(-ENOSYS)
> +# define fuse_iomap_write_iter(...)		(-ENOSYS)
>  #endif /* CONFIG_FUSE_IOMAP */
>  
>  #endif /* _FS_FUSE_IOMAP_H */
> diff --git a/include/uapi/linux/fuse.h b/include/uapi/linux/fuse.h
> index 88f76f4be749a7..543965b2f8fb37 100644
> --- a/include/uapi/linux/fuse.h
> +++ b/include/uapi/linux/fuse.h
> @@ -677,6 +677,7 @@ enum fuse_opcode {
>  	FUSE_STATX		= 52,
>  	FUSE_COPY_FILE_RANGE_64	= 53,
>  
> +	FUSE_IOMAP_IOEND	= 4093,
>  	FUSE_IOMAP_BEGIN	= 4094,
>  	FUSE_IOMAP_END		= 4095,
>  
> @@ -1411,4 +1412,30 @@ struct fuse_iomap_end_in {
>  	struct fuse_iomap_io	map;
>  };
>  
> +/* out of place write extent */
> +#define FUSE_IOMAP_IOEND_SHARED		(1U << 0)
> +/* unwritten extent */
> +#define FUSE_IOMAP_IOEND_UNWRITTEN	(1U << 1)
> +/* don't merge into previous ioend */
> +#define FUSE_IOMAP_IOEND_BOUNDARY	(1U << 2)
> +/* is direct I/O */
> +#define FUSE_IOMAP_IOEND_DIRECT		(1U << 3)
> +/* is append ioend */
> +#define FUSE_IOMAP_IOEND_APPEND		(1U << 4)
> +
> +struct fuse_iomap_ioend_in {
> +	uint32_t flags;		/* FUSE_IOMAP_IOEND_* */
> +	int32_t error;		/* negative errno or 0 */
> +	uint64_t attr_ino;	/* matches fuse_attr:ino */
> +	uint64_t pos;		/* file position, in bytes */
> +	uint64_t new_addr;	/* disk offset of new mapping, in bytes */
> +	uint64_t written;	/* bytes processed */
> +	uint32_t dev;		/* device cookie */
> +	uint32_t pad;		/* zero */
> +};
> +
> +struct fuse_iomap_ioend_out {
> +	uint64_t newsize;	/* new ondisk size */
> +};
> +
>  #endif /* _LINUX_FUSE_H */
> diff --git a/fs/fuse/dir.c b/fs/fuse/dir.c
> index 2ed19e21d90702..944ed53852444c 100644
> --- a/fs/fuse/dir.c
> +++ b/fs/fuse/dir.c
> @@ -911,6 +911,10 @@ static int fuse_create_open(struct mnt_idmap *idmap, struct inode *dir,
>  		goto out_acl_release;
>  
>  	fuse_dir_changed(dir);
> +
> +	if (fuse_inode_has_iomap(inode))
> +		fuse_iomap_open(inode, file);
> +
>  	err = generic_file_open(inode, file);
>  	if (!err) {
>  		file->private_data = ff;
> @@ -1952,6 +1956,9 @@ static int fuse_dir_open(struct inode *inode, struct file *file)
>  	if (fuse_is_bad(inode))
>  		return -EIO;
>  
> +	if (fuse_inode_has_iomap(inode))
> +		fuse_iomap_open(inode, file);
> +
>  	err = generic_file_open(inode, file);
>  	if (err)
>  		return err;
> @@ -2312,6 +2319,12 @@ int fuse_do_setattr(struct mnt_idmap *idmap, struct dentry *dentry,
>  		goto error;
>  	}
>  
> +	if (is_iomap && is_truncate) {
> +		err = fuse_iomap_setsize_finish(inode, outarg.attr.size);
> +		if (err)
> +			goto error;
> +	}
> +
>  	spin_lock(&fi->lock);
>  	/* the kernel maintains i_mtime locally */
>  	if (trust_local_cmtime) {
> diff --git a/fs/fuse/file.c b/fs/fuse/file.c
> index 67cb0844181851..ad28bcd36d15e6 100644
> --- a/fs/fuse/file.c
> +++ b/fs/fuse/file.c
> @@ -222,7 +222,10 @@ int fuse_finish_open(struct inode *inode, struct file *file)
>  	struct fuse_conn *fc = get_fuse_conn(inode);
>  	int err;
>  
> -	err = fuse_file_io_open(file, inode);
> +	if (fuse_inode_has_iomap(inode))
> +		err = fuse_iomap_finish_open(ff, inode);
> +	else
> +		err = fuse_file_io_open(file, inode);
>  	if (err)
>  		return err;
>  
> @@ -265,6 +268,9 @@ static int fuse_open(struct inode *inode, struct file *file)
>  	if (fuse_is_bad(inode))
>  		return -EIO;
>  
> +	if (is_iomap)
> +		fuse_iomap_open(inode, file);
> +
>  	err = generic_file_open(inode, file);
>  	if (err)
>  		return err;
> @@ -295,9 +301,11 @@ static int fuse_open(struct inode *inode, struct file *file)
>  	if ((is_wb_truncate || dax_truncate) && !is_iomap)
>  		fuse_release_nowrite(inode);
>  	if (!err) {
> -		if (is_truncate)
> +		if (is_truncate) {
>  			truncate_pagecache(inode, 0);
> -		else if (!(ff->open_flags & FOPEN_KEEP_CACHE))
> +			if (is_iomap)
> +				fuse_iomap_open_truncate(inode);
> +		} else if (!(ff->open_flags & FOPEN_KEEP_CACHE))
>  			invalidate_inode_pages2(inode->i_mapping);
>  	}
>  	if (dax_truncate)
> @@ -1827,6 +1835,9 @@ static ssize_t fuse_file_read_iter(struct kiocb *iocb, struct iov_iter *to)
>  	if (fuse_is_bad(inode))
>  		return -EIO;
>  
> +	if (fuse_inode_has_iomap(inode))
> +		return fuse_iomap_read_iter(iocb, to);
> +
>  	if (FUSE_IS_DAX(inode))
>  		return fuse_dax_read_iter(iocb, to);
>  
> @@ -1848,6 +1859,9 @@ static ssize_t fuse_file_write_iter(struct kiocb *iocb, struct iov_iter *from)
>  	if (fuse_is_bad(inode))
>  		return -EIO;
>  
> +	if (fuse_inode_has_iomap(inode))
> +		return fuse_iomap_write_iter(iocb, from);
> +
>  	if (FUSE_IS_DAX(inode))
>  		return fuse_dax_write_iter(iocb, from);
>  
> @@ -2960,7 +2974,9 @@ static long fuse_file_fallocate(struct file *file, int mode, loff_t offset,
>  		.length = length,
>  		.mode = mode
>  	};
> +	loff_t newsize = 0;
>  	int err;
> +	const bool is_iomap = fuse_inode_has_iomap(inode);
>  	bool block_faults = FUSE_IS_DAX(inode) &&
>  		(!(mode & FALLOC_FL_KEEP_SIZE) ||
>  		 (mode & (FALLOC_FL_PUNCH_HOLE | FALLOC_FL_ZERO_RANGE)));
> @@ -2993,6 +3009,7 @@ static long fuse_file_fallocate(struct file *file, int mode, loff_t offset,
>  		err = inode_newsize_ok(inode, offset + length);
>  		if (err)
>  			goto out;
> +		newsize = offset + length;
>  	}
>  
>  	err = file_modified(file);
> @@ -3017,6 +3034,12 @@ static long fuse_file_fallocate(struct file *file, int mode, loff_t offset,
>  
>  	/* we could have extended the file */
>  	if (!(mode & FALLOC_FL_KEEP_SIZE)) {
> +		if (is_iomap && newsize > 0) {
> +			err = fuse_iomap_setsize_finish(inode, newsize);
> +			if (err)
> +				goto out;
> +		}
> +
>  		if (fuse_write_update_attr(inode, offset + length, length))
>  			file_update_time(file);
>  	}
> diff --git a/fs/fuse/fuse_iomap.c b/fs/fuse/fuse_iomap.c
> index be922888ae9e8a..833ac0ce682838 100644
> --- a/fs/fuse/fuse_iomap.c
> +++ b/fs/fuse/fuse_iomap.c
> @@ -476,10 +476,15 @@ static int fuse_iomap_begin(struct inode *inode, loff_t pos, loff_t count,
>  }
>  
>  /* Decide if we send FUSE_IOMAP_END to the fuse server */
> -static bool fuse_should_send_iomap_end(const struct iomap *iomap,
> +static bool fuse_should_send_iomap_end(const struct fuse_mount *fm,
> +				       const struct iomap *iomap,
>  				       unsigned int opflags, loff_t count,
>  				       ssize_t written)
>  {
> +	/* Not implemented on fuse server */
> +	if (fm->fc->iomap_conn.no_end)
> +		return false;
> +
>  	/* fuse server demanded an iomap_end call. */
>  	if (iomap->flags & FUSE_IOMAP_F_WANT_IOMAP_END)
>  		return true;
> @@ -504,7 +509,7 @@ static int fuse_iomap_end(struct inode *inode, loff_t pos, loff_t count,
>  	struct fuse_mount *fm = get_fuse_mount(inode);
>  	int err;
>  
> -	if (fuse_should_send_iomap_end(iomap, opflags, count, written)) {
> +	if (fuse_should_send_iomap_end(fm, iomap, opflags, count, written)) {
>  		struct fuse_iomap_end_in inarg = {
>  			.opflags = fuse_iomap_op_to_server(opflags),
>  			.attr_ino = fi->orig_ino,
> @@ -529,6 +534,7 @@ static int fuse_iomap_end(struct inode *inode, loff_t pos, loff_t count,
>  			 * libfuse returns ENOSYS for servers that don't
>  			 * implement iomap_end
>  			 */
> +			fm->fc->iomap_conn.no_end = 1;
>  			err = 0;
>  		}
>  		if (err) {
> @@ -545,6 +551,119 @@ static const struct iomap_ops fuse_iomap_ops = {
>  	.iomap_end		= fuse_iomap_end,
>  };
>  
> +static inline bool
> +fuse_should_send_iomap_ioend(const struct fuse_mount *fm,
> +			     const struct fuse_iomap_ioend_in *inarg)
> +{
> +	/* Not implemented on fuse server */
> +	if (fm->fc->iomap_conn.no_ioend)
> +		return false;
> +
> +	/* Always send an ioend for errors. */
> +	if (inarg->error)
> +		return true;
> +
> +	/* Send an ioend if we performed an IO involving metadata changes. */
> +	return inarg->written > 0 &&
> +	       (inarg->flags & (FUSE_IOMAP_IOEND_SHARED |
> +				FUSE_IOMAP_IOEND_UNWRITTEN |
> +				FUSE_IOMAP_IOEND_APPEND));
> +}
> +
> +int
> +fuse_iomap_setsize_finish(
> +	struct inode		*inode,
> +	loff_t			newsize)
> +{
> +	struct fuse_inode *fi = get_fuse_inode(inode);
> +
> +	ASSERT(fuse_inode_has_iomap(inode));
> +
> +	spin_lock(&fi->lock);
> +	fi->i_disk_size = newsize;
> +	spin_unlock(&fi->lock);
> +	return 0;
> +}
> +
> +static int fuse_iomap_ioend(struct inode *inode, loff_t pos, size_t written,
> +			    int error, unsigned ioendflags,
> +			    struct block_device *bdev, sector_t new_addr)
> +{
> +	struct fuse_inode *fi = get_fuse_inode(inode);
> +	struct fuse_mount *fm = get_fuse_mount(inode);
> +	struct fuse_iomap_ioend_in inarg = {
> +		.flags = ioendflags,
> +		.error = error,
> +		.attr_ino = fi->orig_ino,
> +		.pos = pos,
> +		.written = written,
> +		.dev = fuse_iomap_find_backing_id(fm->fc, bdev),
> +		.new_addr = new_addr,
> +	};
> +	struct fuse_iomap_ioend_out outarg = { };
> +
> +	spin_lock(&fi->lock);
> +	if (pos + written > fi->i_disk_size)
> +		inarg.flags |= FUSE_IOMAP_IOEND_APPEND;
> +
> +	outarg.newsize = max_t(loff_t, fi->i_disk_size, pos + written),
> +	spin_unlock(&fi->lock);
> +
> +	if (fuse_should_send_iomap_ioend(fm, &inarg)) {
> +		FUSE_ARGS(args);
> +		int iomap_error;
> +
> +		args.opcode = FUSE_IOMAP_IOEND;
> +		args.nodeid = get_node_id(inode);
> +		args.in_numargs = 1;
> +		args.in_args[0].size = sizeof(inarg);
> +		args.in_args[0].value = &inarg;
> +		args.out_numargs = 1;
> +		args.out_args[0].size = sizeof(outarg);
> +		args.out_args[0].value = &outarg;
> +		iomap_error = fuse_simple_request(fm, &args);
> +		switch (iomap_error) {
> +		case -ENOSYS:
> +			/*
> +			 * fuse servers can return ENOSYS if ioend processing
> +			 * is never needed for this filesystem.  Don't pass
> +			 * that up to iomap.
> +			 */
> +			fm->fc->iomap_conn.no_ioend = 1;
> +			break;
> +		case 0:
> +			break;
> +		default:
> +			/*
> +			 * If the write IO failed, return the failure code to
> +			 * the caller no matter what happens with the ioend.
> +			 * If the write IO succeeded but the ioend did not,
> +			 * pass the new error up to the caller.
> +			 */
> +			if (!error)
> +				error = iomap_error;
> +			break;
> +		}
> +	}
> +
> +	/*
> +	 * Pass whatever error iomap gave us (or any new errors since then)
> +	 * back to iomap.
> +	 */
> +	if (error)
> +		return error;
> +
> +	/*
> +	 * If there weren't any ioend errors, update the incore isize, which
> +	 * confusingly takes the new i_size as "pos".
> +	 */
> +	spin_lock(&fi->lock);
> +	fi->i_disk_size = outarg.newsize;

Codex points out that multiple directio bios can race to ioend
completion, and that they can complete out of order.  Hence this should
be
	fi->i_disk_size = max_t(loff_t, fi->i_disk_size, outarg.newsize);

to account for a high-offset direct write completing before a low-offset
direct write.

> +	spin_unlock(&fi->lock);
> +	fuse_write_update_attr(inode, pos + written, written);
> +	return 0;
> +}
> +
>  static int fuse_iomap_may_admin(struct fuse_conn *fc, unsigned int flags)
>  {
>  	if (!fc->iomap)
> @@ -739,3 +858,215 @@ loff_t fuse_iomap_lseek(struct file *file, loff_t offset, int whence)
>  		return offset;
>  	return vfs_setpos(file, offset, inode->i_sb->s_maxbytes);
>  }
> +
> +void fuse_iomap_open(struct inode *inode, struct file *file)
> +{
> +	ASSERT(fuse_inode_has_iomap(inode));
> +
> +	file->f_mode |= FMODE_NOWAIT | FMODE_CAN_ODIRECT;

HAH while grepping for other FMODE_ flag usage whilst implementing the
backing dev file access mode checks, I saw this and thought that looked
totally wrong.  FUSE is not a good nowait candidate because we can block
on userspace(!)

Maybe later after we implement iomap caching?  Or probably just never.

--D

> +}
> +
> +int fuse_iomap_finish_open(const struct fuse_file *ff,
> +			   const struct inode *inode)
> +{
> +	ASSERT(fuse_inode_has_iomap(inode));
> +
> +	/* no weird modes, iomap only handles seekable regular files */
> +	if (ff->open_flags & (FOPEN_PASSTHROUGH |
> +			      FOPEN_STREAM |
> +			      FOPEN_NONSEEKABLE))
> +		return -EINVAL;
> +
> +	return 0;
> +}
> +
> +enum fuse_ilock_type {
> +	SHARED,
> +	EXCL,
> +};
> +
> +static int fuse_iomap_ilock_iocb(const struct kiocb *iocb,
> +				 enum fuse_ilock_type type)
> +{
> +	struct inode *inode = file_inode(iocb->ki_filp);
> +
> +	if (iocb->ki_flags & IOCB_NOWAIT) {
> +		switch (type) {
> +		case SHARED:
> +			return inode_trylock_shared(inode) ? 0 : -EAGAIN;
> +		case EXCL:
> +			return inode_trylock(inode) ? 0 : -EAGAIN;
> +		default:
> +			ASSERT(0);
> +			return -EIO;
> +		}
> +
> +		/* shut up gcc */
> +		return 0;
> +	}
> +
> +	switch (type) {
> +	case SHARED:
> +		inode_lock_shared(inode);
> +		break;
> +	case EXCL:
> +		inode_lock(inode);
> +		break;
> +	default:
> +		ASSERT(0);
> +		return -EIO;
> +	}
> +
> +	return 0;
> +}
> +
> +static ssize_t fuse_iomap_direct_read(struct kiocb *iocb, struct iov_iter *to)
> +{
> +	struct inode *inode = file_inode(iocb->ki_filp);
> +	ssize_t ret;
> +
> +	if (!iov_iter_count(to))
> +		return 0; /* skip atime */
> +
> +	ret = fuse_iomap_ilock_iocb(iocb, SHARED);
> +	if (ret)
> +		return ret;
> +	ret = iomap_dio_rw(iocb, to, &fuse_iomap_ops, NULL, 0, NULL, 0);
> +	if (ret > 0)
> +		file_accessed(iocb->ki_filp);
> +	inode_unlock_shared(inode);
> +
> +	return ret;
> +}
> +
> +static int fuse_iomap_dio_write_end_io(struct kiocb *iocb, ssize_t written,
> +				       int error, unsigned dioflags)
> +{
> +	struct inode *inode = file_inode(iocb->ki_filp);
> +	unsigned int ioendflags = FUSE_IOMAP_IOEND_DIRECT;
> +
> +	if (fuse_is_bad(inode))
> +		return -EIO;
> +
> +	ASSERT(fuse_inode_has_iomap(inode));
> +
> +	if (dioflags & IOMAP_DIO_COW)
> +		ioendflags |= FUSE_IOMAP_IOEND_SHARED;
> +	if (dioflags & IOMAP_DIO_UNWRITTEN)
> +		ioendflags |= FUSE_IOMAP_IOEND_UNWRITTEN;
> +
> +	return fuse_iomap_ioend(inode, iocb->ki_pos, written, error,
> +				ioendflags, NULL, FUSE_IOMAP_NULL_ADDR);
> +}
> +
> +static const struct iomap_dio_ops fuse_iomap_dio_write_ops = {
> +	.end_io		= fuse_iomap_dio_write_end_io,
> +};
> +
> +static ssize_t
> +fuse_iomap_write_checks(
> +	struct kiocb		*iocb,
> +	struct iov_iter		*from)
> +{
> +	ssize_t			error;
> +
> +	error = generic_write_checks(iocb, from);
> +	if (error <= 0)
> +		return error;
> +
> +	return kiocb_modified(iocb);
> +}
> +
> +static ssize_t fuse_iomap_direct_write(struct kiocb *iocb,
> +				       struct iov_iter *from)
> +{
> +	struct inode *inode = file_inode(iocb->ki_filp);
> +	loff_t blockmask = i_blocksize(inode) - 1;
> +	size_t count = iov_iter_count(from);
> +	unsigned int flags = IOMAP_DIO_COMP_WORK;
> +	ssize_t ret;
> +
> +	if (!count)
> +		return 0;
> +
> +	/*
> +	 * Unaligned direct writes require zeroing of unwritten head and tail
> +	 * blocks.  Extending writes require zeroing of post-EOF tail blocks.
> +	 * The zeroing writes must complete before we return the direct write
> +	 * to userspace.  Don't even bother trying the fast path.
> +	 */
> +	if ((iocb->ki_pos | count) & blockmask)
> +		flags |= IOMAP_DIO_FORCE_WAIT;
> +
> +	ret = fuse_iomap_ilock_iocb(iocb, EXCL);
> +	if (ret)
> +		goto out_dsync;
> +
> +	ret = fuse_iomap_write_checks(iocb, from);
> +	if (ret)
> +		goto out_unlock;
> +
> +	/*
> +	 * If we are doing exclusive unaligned I/O, this must be the only I/O
> +	 * in-flight.  Otherwise we risk data corruption due to unwritten
> +	 * extent conversions from the AIO end_io handler.  Wait for all other
> +	 * I/O to drain first.
> +	 */
> +	if (flags & IOMAP_DIO_FORCE_WAIT)
> +		inode_dio_wait(inode);
> +
> +	ret = iomap_dio_rw(iocb, from, &fuse_iomap_ops,
> +			   &fuse_iomap_dio_write_ops, flags, NULL, 0);
> +out_unlock:
> +	inode_unlock(inode);
> +out_dsync:
> +	return ret;
> +}
> +
> +void fuse_iomap_set_disk_size(struct fuse_inode *fi, loff_t newsize)
> +{
> +	lockdep_assert_held(&fi->lock);
> +
> +	if (fuse_inode_has_iomap(&fi->inode))
> +		fi->i_disk_size = newsize;
> +}
> +
> +void fuse_iomap_open_truncate(struct inode *inode)
> +{
> +	struct fuse_inode *fi = get_fuse_inode(inode);
> +
> +	ASSERT(fuse_inode_has_iomap(inode));
> +
> +	spin_lock(&fi->lock);
> +	fi->i_disk_size = 0;
> +	spin_unlock(&fi->lock);
> +}
> +
> +static inline bool fuse_iomap_force_directio(const struct kiocb *iocb)
> +{
> +	struct fuse_file *ff = iocb->ki_filp->private_data;
> +
> +	return ff->open_flags & FOPEN_DIRECT_IO;
> +}
> +
> +ssize_t fuse_iomap_read_iter(struct kiocb *iocb, struct iov_iter *to)
> +{
> +	const bool force_directio = fuse_iomap_force_directio(iocb);
> +
> +	ASSERT(fuse_inode_has_iomap(file_inode(iocb->ki_filp)));
> +
> +	if ((iocb->ki_flags & IOCB_DIRECT) || force_directio)
> +		return fuse_iomap_direct_read(iocb, to);
> +	return -EIO;
> +}
> +
> +ssize_t fuse_iomap_write_iter(struct kiocb *iocb, struct iov_iter *from)
> +{
> +	const bool force_directio = fuse_iomap_force_directio(iocb);
> +
> +	ASSERT(fuse_inode_has_iomap(file_inode(iocb->ki_filp)));
> +
> +	if ((iocb->ki_flags & IOCB_DIRECT) || force_directio)
> +		return fuse_iomap_direct_write(iocb, from);
> +	return -EIO;
> +}
> diff --git a/fs/fuse/inode.c b/fs/fuse/inode.c
> index 3cb2ef161ef7c5..23ca401a3e08e6 100644
> --- a/fs/fuse/inode.c
> +++ b/fs/fuse/inode.c
> @@ -303,6 +303,8 @@ void fuse_change_attributes_common(struct inode *inode, struct fuse_attr *attr,
>  	else
>  		fi->cached_i_blkbits = inode->i_sb->s_blocksize_bits;
>  
> +	fuse_iomap_set_disk_size(fi, attr->size);
> +
>  	/*
>  	 * Don't set the sticky bit in i_mode, unless we want the VFS
>  	 * to check permissions.  This prevents failures due to the
> @@ -352,6 +354,7 @@ static void fuse_change_attributes_i(struct inode *inode, struct fuse_attr *attr
>  	 * inode.
>  	 */
>  	cache_mask = fuse_get_cache_mask(inode);
> +	fuse_iomap_set_disk_size(fi, attr->size);
>  	if (cache_mask & STATX_SIZE)
>  		attr->size = i_size_read(inode);
>  
> diff --git a/fs/fuse/iomode.c b/fs/fuse/iomode.c
> index 9be9ae3520003e..ed666b39cf8af4 100644
> --- a/fs/fuse/iomode.c
> +++ b/fs/fuse/iomode.c
> @@ -204,7 +204,7 @@ int fuse_file_io_open(struct file *file, struct inode *inode)
>  	 * io modes are not relevant with DAX and with server that does not
>  	 * implement open.
>  	 */
> -	if (fuse_inode_has_iomap(inode) || FUSE_IS_DAX(inode) || !ff->args)
> +	if (FUSE_IS_DAX(inode) || !ff->args)
>  		return 0;
>  
>  	/*
> 
> 

  reply	other threads:[~2026-05-13 18:44 UTC|newest]

Thread overview: 226+ messages / expand[flat|nested]  mbox.gz  Atom feed  top
2026-04-29 14:12 [PATCHBLIZZARD v8] fuse/libfuse/e2fsprogs: faster file IO for containerized ext4 servers Darrick J. Wong
2026-04-29 14:16 ` [PATCHSET v8 1/8] fuse: general bug fixes Darrick J. Wong
2026-04-29 14:21   ` [PATCH 1/4] fuse: flush pending FUSE_RELEASE requests before sending FUSE_DESTROY Darrick J. Wong
2026-04-29 14:22   ` [PATCH 2/4] fuse: implement file attributes mask for statx Darrick J. Wong
2026-04-29 14:22   ` [PATCH 3/4] fuse: update file mode when updating acls Darrick J. Wong
2026-04-30 13:48     ` Joanne Koong
2026-04-30 20:57       ` Darrick J. Wong
2026-05-01  9:53         ` Joanne Koong
2026-05-01 16:15           ` Darrick J. Wong
2026-04-29 14:22   ` [PATCH 4/4] fuse: propagate default and file acls on creation Darrick J. Wong
2026-05-01 11:11     ` Joanne Koong
2026-05-01 16:57       ` Darrick J. Wong
2026-04-29 14:16 ` [PATCHSET v8 2/8] iomap: cleanups ahead of adding fuse support Darrick J. Wong
2026-04-29 14:22   ` [PATCH 1/2] iomap: allow directio callers to supply _COMP_WORK Darrick J. Wong
2026-04-29 14:23   ` [PATCH 2/2] iomap: allow NULL swap info bdev when activating swapfile Darrick J. Wong
2026-05-08  9:06     ` Christoph Hellwig
2026-05-08 23:41       ` Darrick J. Wong
2026-04-29 14:17 ` [PATCHSET v8 3/8] fuse: cleanups ahead of adding fuse support Darrick J. Wong
2026-04-29 14:23   ` [PATCH 1/2] fuse: move the passthrough-specific code back to passthrough.c Darrick J. Wong
2026-05-14 18:38     ` Darrick J. Wong
2026-05-14 19:48       ` Amir Goldstein
2026-05-14 22:21         ` Darrick J. Wong
2026-04-29 14:23   ` [PATCH 2/2] fuse_trace: " Darrick J. Wong
2026-05-14 18:41     ` Darrick J. Wong
2026-04-29 14:17 ` [PATCHSET v8 4/8] fuse: allow servers to use iomap for better file IO performance Darrick J. Wong
2026-04-29 14:23   ` [PATCH 01/33] fuse: implement the basic iomap mechanisms Darrick J. Wong
2026-05-13 17:38     ` Darrick J. Wong
2026-04-29 14:24   ` [PATCH 02/33] fuse_trace: " Darrick J. Wong
2026-04-29 14:24   ` [PATCH 03/33] fuse: make debugging configurable at runtime Darrick J. Wong
2026-05-13 17:41     ` Darrick J. Wong
2026-05-13 17:46       ` Bernd Schubert
2026-05-13 18:35         ` Darrick J. Wong
2026-05-13 19:44           ` Bernd Schubert
2026-05-14 14:35             ` Darrick J. Wong
2026-04-29 14:24   ` [PATCH 04/33] fuse: adapt FUSE_DEV_IOC_BACKING_{OPEN,CLOSE} to add new iomap devices Darrick J. Wong
2026-05-13 18:04     ` Darrick J. Wong
2026-04-29 14:24   ` [PATCH 05/33] fuse_trace: " Darrick J. Wong
2026-04-29 14:25   ` [PATCH 06/33] fuse: enable SYNCFS and ensure we flush everything before sending DESTROY Darrick J. Wong
2026-04-29 14:25   ` [PATCH 07/33] fuse: clean up per-file type inode initialization Darrick J. Wong
2026-04-29 14:25   ` [PATCH 08/33] fuse: create a per-inode flag for setting exclusive mode Darrick J. Wong
2026-05-13 18:09     ` Darrick J. Wong
2026-04-29 14:26   ` [PATCH 09/33] fuse: create a per-inode flag for toggling iomap Darrick J. Wong
2026-05-14 19:52     ` Darrick J. Wong
2026-04-29 14:26   ` [PATCH 10/33] fuse_trace: " Darrick J. Wong
2026-04-29 14:26   ` [PATCH 11/33] fuse: isolate the other regular file IO paths from iomap Darrick J. Wong
2026-05-13 18:18     ` Darrick J. Wong
2026-04-29 14:26   ` [PATCH 12/33] fuse: implement basic iomap reporting such as FIEMAP and SEEK_{DATA,HOLE} Darrick J. Wong
2026-05-13 18:30     ` Darrick J. Wong
2026-04-29 14:27   ` [PATCH 13/33] fuse_trace: " Darrick J. Wong
2026-04-29 14:27   ` [PATCH 14/33] fuse: implement direct IO with iomap Darrick J. Wong
2026-05-13 18:44     ` Darrick J. Wong [this message]
2026-04-29 14:27   ` [PATCH 15/33] fuse_trace: " Darrick J. Wong
2026-04-29 14:27   ` [PATCH 16/33] fuse: implement buffered " Darrick J. Wong
2026-05-13 19:18     ` Darrick J. Wong
2026-04-29 14:28   ` [PATCH 17/33] fuse_trace: " Darrick J. Wong
2026-05-13 19:25     ` Darrick J. Wong
2026-04-29 14:28   ` [PATCH 18/33] fuse: use an unrestricted backing device with iomap pagecache io Darrick J. Wong
2026-05-13 21:24     ` Darrick J. Wong
2026-04-29 14:28   ` [PATCH 19/33] fuse: implement large folios for iomap pagecache files Darrick J. Wong
2026-04-29 14:28   ` [PATCH 20/33] fuse: advertise support for iomap Darrick J. Wong
2026-04-29 14:29   ` [PATCH 21/33] fuse: query filesystem geometry when using iomap Darrick J. Wong
2026-05-13 21:05     ` Darrick J. Wong
2026-04-29 14:29   ` [PATCH 22/33] fuse_trace: " Darrick J. Wong
2026-05-14 20:54     ` Darrick J. Wong
2026-04-29 14:29   ` [PATCH 23/33] fuse: implement fadvise for iomap files Darrick J. Wong
2026-04-29 14:29   ` [PATCH 24/33] fuse: invalidate ranges of block devices being used for iomap Darrick J. Wong
2026-05-13 22:13     ` Darrick J. Wong
2026-04-29 14:30   ` [PATCH 25/33] fuse_trace: " Darrick J. Wong
2026-04-29 14:30   ` [PATCH 26/33] fuse: implement inline data file IO via iomap Darrick J. Wong
2026-05-13 20:03     ` Darrick J. Wong
2026-04-29 14:30   ` [PATCH 27/33] fuse_trace: " Darrick J. Wong
2026-05-13 20:09     ` Darrick J. Wong
2026-04-29 14:31   ` [PATCH 28/33] fuse: allow more statx fields Darrick J. Wong
2026-04-29 14:31   ` [PATCH 29/33] fuse: support atomic writes with iomap Darrick J. Wong
2026-04-29 14:31   ` [PATCH 30/33] fuse_trace: " Darrick J. Wong
2026-04-29 14:31   ` [PATCH 31/33] fuse: disable direct fs reclaim for any fuse server that uses iomap Darrick J. Wong
2026-05-13 21:27     ` Darrick J. Wong
2026-04-29 14:32   ` [PATCH 32/33] fuse: enable swapfile activation on iomap Darrick J. Wong
2026-05-14 14:42     ` Darrick J. Wong
2026-04-29 14:32   ` [PATCH 33/33] fuse: implement freeze and shutdowns for iomap filesystems Darrick J. Wong
2026-04-29 14:17 ` [PATCHSET v8 5/8] fuse: allow servers to specify root node id Darrick J. Wong
2026-04-29 14:32   ` [PATCH 1/3] fuse: make the root nodeid dynamic Darrick J. Wong
2026-05-14 21:38     ` Darrick J. Wong
2026-04-29 14:32   ` [PATCH 2/3] fuse_trace: " Darrick J. Wong
2026-04-29 14:33   ` [PATCH 3/3] fuse: allow setting of root nodeid Darrick J. Wong
2026-05-14 21:37     ` Darrick J. Wong
2026-04-29 14:17 ` [PATCHSET v8 6/8] fuse: handle timestamps and ACLs correctly when iomap is enabled Darrick J. Wong
2026-04-29 14:33   ` [PATCH 1/9] fuse: enable caching of timestamps Darrick J. Wong
2026-05-14 22:09     ` Darrick J. Wong
2026-04-29 14:33   ` [PATCH 2/9] fuse: force a ctime update after a fileattr_set call when in iomap mode Darrick J. Wong
2026-05-14 21:44     ` Darrick J. Wong
2026-04-29 14:33   ` [PATCH 3/9] fuse: allow local filesystems to set some VFS iflags Darrick J. Wong
2026-05-14 22:14     ` Darrick J. Wong
2026-04-29 14:34   ` [PATCH 4/9] fuse_trace: " Darrick J. Wong
2026-04-29 14:34   ` [PATCH 5/9] fuse: cache atime when in iomap mode Darrick J. Wong
2026-05-14 22:16     ` Darrick J. Wong
2026-04-29 14:34   ` [PATCH 6/9] fuse: let the kernel handle KILL_SUID/KILL_SGID for iomap filesystems Darrick J. Wong
2026-05-14 22:18     ` Darrick J. Wong
2026-04-29 14:34   ` [PATCH 7/9] fuse_trace: " Darrick J. Wong
2026-05-14 22:19     ` Darrick J. Wong
2026-04-29 14:35   ` [PATCH 8/9] fuse: update ctime when updating acls on an iomap inode Darrick J. Wong
2026-05-14 22:20     ` Darrick J. Wong
2026-04-29 14:35   ` [PATCH 9/9] fuse: always cache ACLs when using iomap Darrick J. Wong
2026-04-29 14:18 ` [PATCHSET v8 7/8] fuse: cache iomap mappings for even better file IO performance Darrick J. Wong
2026-04-29 14:35   ` [PATCH 01/12] fuse: cache iomaps Darrick J. Wong
2026-05-14 22:28     ` Darrick J. Wong
2026-04-29 14:35   ` [PATCH 02/12] fuse_trace: " Darrick J. Wong
2026-04-29 14:36   ` [PATCH 03/12] fuse: use the iomap cache for iomap_begin Darrick J. Wong
2026-04-29 14:36   ` [PATCH 04/12] fuse_trace: " Darrick J. Wong
2026-04-29 14:36   ` [PATCH 05/12] fuse: invalidate iomap cache after file updates Darrick J. Wong
2026-04-29 14:36   ` [PATCH 06/12] fuse_trace: " Darrick J. Wong
2026-04-29 14:37   ` [PATCH 07/12] fuse: enable iomap cache management Darrick J. Wong
2026-05-14 22:59     ` Darrick J. Wong
2026-04-29 14:37   ` [PATCH 08/12] fuse_trace: " Darrick J. Wong
2026-04-29 14:37   ` [PATCH 09/12] fuse: overlay iomap inode info in struct fuse_inode Darrick J. Wong
2026-04-29 14:38   ` [PATCH 10/12] fuse: constrain iomap mapping cache size Darrick J. Wong
2026-05-14 23:00     ` Darrick J. Wong
2026-04-29 14:38   ` [PATCH 11/12] fuse_trace: " Darrick J. Wong
2026-05-14 23:05     ` Darrick J. Wong
2026-04-29 14:38   ` [PATCH 12/12] fuse: enable iomap Darrick J. Wong
2026-04-29 14:18 ` [PATCHSET v8 8/8] fuse: run fuse-iomap servers as a contained service Darrick J. Wong
2026-04-29 14:38   ` [PATCH 1/2] fuse: allow privileged mount helpers to pre-approve iomap usage Darrick J. Wong
2026-05-13 21:33     ` Darrick J. Wong
2026-04-29 14:39   ` [PATCH 2/2] fuse: set iomap backing device block size Darrick J. Wong
2026-04-29 14:18 ` [PATCHSET v8 1/6] libfuse: allow servers to use iomap for better file IO performance Darrick J. Wong
2026-04-29 14:39   ` [PATCH 01/25] libfuse: bump kernel and library ABI versions Darrick J. Wong
2026-04-29 14:39   ` [PATCH 02/25] libfuse: wait in do_destroy until all open files are closed Darrick J. Wong
2026-04-29 14:39   ` [PATCH 03/25] libfuse: add kernel gates for FUSE_IOMAP Darrick J. Wong
2026-04-29 14:40   ` [PATCH 04/25] libfuse: add fuse commands for iomap_begin and end Darrick J. Wong
2026-04-29 14:40   ` [PATCH 05/25] libfuse: add upper level iomap commands Darrick J. Wong
2026-04-29 14:40   ` [PATCH 06/25] libfuse: add a lowlevel notification to add a new device to iomap Darrick J. Wong
2026-04-29 14:40   ` [PATCH 07/25] libfuse: add upper-level iomap add device function Darrick J. Wong
2026-04-29 14:41   ` [PATCH 08/25] libfuse: add iomap ioend low level handler Darrick J. Wong
2026-04-29 14:41   ` [PATCH 09/25] libfuse: add upper level iomap ioend commands Darrick J. Wong
2026-04-29 14:41   ` [PATCH 10/25] libfuse: add a reply function to send FUSE_ATTR_* to the kernel Darrick J. Wong
2026-04-29 14:41   ` [PATCH 11/25] libfuse: connect high level fuse library to fuse_reply_attr_iflags Darrick J. Wong
2026-04-29 14:42   ` [PATCH 12/25] libfuse: support enabling exclusive mode for files Darrick J. Wong
2026-04-29 14:42   ` [PATCH 13/25] libfuse: support direct I/O through iomap Darrick J. Wong
2026-04-29 14:42   ` [PATCH 14/25] libfuse: don't allow hardlinking of iomap files in the upper level fuse library Darrick J. Wong
2026-04-29 14:42   ` [PATCH 15/25] libfuse: allow discovery of the kernel's iomap capabilities Darrick J. Wong
2026-04-29 14:43   ` [PATCH 16/25] libfuse: add lower level iomap_config implementation Darrick J. Wong
2026-04-29 14:43   ` [PATCH 17/25] libfuse: add upper " Darrick J. Wong
2026-04-29 14:43   ` [PATCH 18/25] libfuse: add low level code to invalidate iomap block device ranges Darrick J. Wong
2026-04-29 14:44   ` [PATCH 19/25] libfuse: add upper-level API to invalidate parts of an iomap block device Darrick J. Wong
2026-04-29 14:44   ` [PATCH 20/25] libfuse: add atomic write support Darrick J. Wong
2026-04-29 14:44   ` [PATCH 21/25] libfuse: allow disabling of fs memory reclaim and write throttling Darrick J. Wong
2026-04-29 14:44   ` [PATCH 22/25] libfuse: create a helper to transform an open regular file into an open loopdev Darrick J. Wong
2026-04-29 14:45   ` [PATCH 23/25] libfuse: add swapfile support for iomap files Darrick J. Wong
2026-04-29 14:45   ` [PATCH 24/25] libfuse: add lower-level filesystem freeze, thaw, and shutdown requests Darrick J. Wong
2026-04-29 14:45   ` [PATCH 25/25] libfuse: add upper-level filesystem freeze, thaw, and shutdown events Darrick J. Wong
2026-04-29 14:19 ` [PATCHSET v8 2/6] libfuse: allow servers to specify root node id Darrick J. Wong
2026-04-29 14:45   ` [PATCH 1/1] libfuse: allow root_nodeid mount option Darrick J. Wong
2026-04-29 14:19 ` [PATCHSET v8 3/6] libfuse: implement syncfs Darrick J. Wong
2026-04-29 14:46   ` [PATCH 1/2] libfuse: add strictatime/lazytime mount options Darrick J. Wong
2026-04-29 14:46   ` [PATCH 2/2] libfuse: set sync, immutable, and append when loading files Darrick J. Wong
2026-04-29 14:19 ` [PATCHSET v8 4/6] libfuse: add some service helper commands for iomap Darrick J. Wong
2026-04-29 14:46   ` [PATCH 1/3] mount_service: delegate iomap privilege from mount.service to fuse services Darrick J. Wong
2026-04-29 14:46   ` [PATCH 2/3] libfuse: enable setting iomap block device block size Darrick J. Wong
2026-04-29 14:47   ` [PATCH 3/3] mount_service: create loop devices for regular files Darrick J. Wong
2026-04-29 14:19 ` [PATCHSET v8 5/6] fuse: add sample iomap fuse servers Darrick J. Wong
2026-04-29 14:47   ` [PATCH 1/7] example/iomap_ll: create a simple iomap server Darrick J. Wong
2026-04-29 14:47   ` [PATCH 2/7] example/iomap_ll: track block state Darrick J. Wong
2026-04-29 14:47   ` [PATCH 3/7] example/iomap_ll: implement atomic writes Darrick J. Wong
2026-04-29 14:48   ` [PATCH 4/7] example/iomap_inline_ll: create a simple server to test inlinedata Darrick J. Wong
2026-04-29 14:48   ` [PATCH 5/7] example/iomap_ow_ll: create a simple iomap out of place write server Darrick J. Wong
2026-04-29 14:48   ` [PATCH 6/7] example/iomap_ow_ll: implement atomic writes Darrick J. Wong
2026-04-29 14:48   ` [PATCH 7/7] example/iomap_service_ll: create a sample systemd service fuse server Darrick J. Wong
2026-04-29 14:20 ` [PATCHSET v8 6/6] libfuse: cache iomap mappings for even better file IO performance Darrick J. Wong
2026-04-29 14:49   ` [PATCH 1/9] libfuse: enable iomap cache management for lowlevel fuse Darrick J. Wong
2026-04-29 14:49   ` [PATCH 2/9] libfuse: add upper-level iomap cache management Darrick J. Wong
2026-04-29 14:49   ` [PATCH 3/9] libfuse: allow constraining of iomap mapping cache size Darrick J. Wong
2026-04-29 14:50   ` [PATCH 4/9] libfuse: add upper-level iomap mapping cache constraint code Darrick J. Wong
2026-04-29 14:50   ` [PATCH 5/9] libfuse: enable iomap Darrick J. Wong
2026-04-29 14:50   ` [PATCH 6/9] example/iomap_ll: cache mappings for later Darrick J. Wong
2026-04-29 14:50   ` [PATCH 7/9] example/iomap_inline_ll: cache iomappings in the kernel Darrick J. Wong
2026-04-29 14:51   ` [PATCH 8/9] example/iomap_ow_ll: " Darrick J. Wong
2026-04-29 14:51   ` [PATCH 9/9] example/iomap_service_ll: " Darrick J. Wong
2026-04-29 14:20 ` [PATCHSET v8 1/6] libext2fs: refactoring for fuse2fs iomap support Darrick J. Wong
2026-04-29 14:51   ` [PATCH 1/5] libext2fs: invalidate cached blocks when freeing them Darrick J. Wong
2026-04-29 14:51   ` [PATCH 2/5] libext2fs: only flush affected blocks in unix_write_byte Darrick J. Wong
2026-04-29 14:52   ` [PATCH 3/5] libext2fs: allow unix_write_byte when the write would be aligned Darrick J. Wong
2026-04-29 14:52   ` [PATCH 4/5] libext2fs: allow clients to ask to write full superblocks Darrick J. Wong
2026-04-29 14:52   ` [PATCH 5/5] libext2fs: allow callers to disallow I/O to file data blocks Darrick J. Wong
2026-04-29 14:20 ` [PATCHSET v8 2/6] fuse2fs: use fuse iomap data paths for better file I/O performance Darrick J. Wong
2026-04-29 14:52   ` [PATCH 01/19] fuse2fs: implement bare minimum iomap for file mapping reporting Darrick J. Wong
2026-04-29 14:53   ` [PATCH 02/19] fuse2fs: add iomap= mount option Darrick J. Wong
2026-04-29 14:53   ` [PATCH 03/19] fuse2fs: implement iomap configuration Darrick J. Wong
2026-04-29 14:53   ` [PATCH 04/19] fuse2fs: register block devices for use with iomap Darrick J. Wong
2026-04-29 14:53   ` [PATCH 05/19] fuse2fs: implement directio file reads Darrick J. Wong
2026-04-29 14:54   ` [PATCH 06/19] fuse2fs: add extent dump function for debugging Darrick J. Wong
2026-04-29 14:54   ` [PATCH 07/19] fuse2fs: implement direct write support Darrick J. Wong
2026-04-29 14:54   ` [PATCH 08/19] fuse2fs: turn on iomap for pagecache IO Darrick J. Wong
2026-04-29 14:54   ` [PATCH 09/19] fuse2fs: don't zero bytes in punch hole Darrick J. Wong
2026-04-29 14:55   ` [PATCH 10/19] fuse2fs: don't do file data block IO when iomap is enabled Darrick J. Wong
2026-04-29 14:55   ` [PATCH 11/19] fuse2fs: try to create loop device when ext4 device is a regular file Darrick J. Wong
2026-04-29 14:55   ` [PATCH 12/19] fuse2fs: enable file IO to inline data files Darrick J. Wong
2026-04-29 14:56   ` [PATCH 13/19] fuse2fs: set iomap-related inode flags Darrick J. Wong
2026-04-29 14:56   ` [PATCH 14/19] fuse2fs: configure block device block size Darrick J. Wong
2026-04-29 14:56   ` [PATCH 15/19] fuse4fs: separate invalidation Darrick J. Wong
2026-04-29 14:56   ` [PATCH 16/19] fuse2fs: implement statx Darrick J. Wong
2026-04-29 14:57   ` [PATCH 17/19] fuse2fs: enable atomic writes Darrick J. Wong
2026-04-29 14:57   ` [PATCH 18/19] fuse4fs: disable fs reclaim and write throttling Darrick J. Wong
2026-04-29 14:57   ` [PATCH 19/19] fuse2fs: implement freeze and shutdown requests Darrick J. Wong
2026-04-29 14:20 ` [PATCHSET v8 3/6] fuse4fs: adapt iomap for fuse services Darrick J. Wong
2026-04-29 14:57   ` [PATCH 1/3] fuse4fs: configure iomap when running as a service Darrick J. Wong
2026-04-29 14:58   ` [PATCH 2/3] fuse4fs: set iomap backing device blocksize Darrick J. Wong
2026-04-29 14:58   ` [PATCH 3/3] fuse4fs: ask for loop devices when opening via fuservicemount Darrick J. Wong
2026-04-29 14:21 ` [PATCHSET v8 4/6] fuse4fs: specify the root node id Darrick J. Wong
2026-04-29 14:58   ` [PATCH 1/1] fuse4fs: don't use inode number translation when possible Darrick J. Wong
2026-04-29 14:21 ` [PATCHSET v8 5/6] fuse2fs: handle timestamps and ACLs correctly when iomap is enabled Darrick J. Wong
2026-04-29 14:58   ` [PATCH 01/10] fuse2fs: add strictatime/lazytime mount options Darrick J. Wong
2026-04-29 14:59   ` [PATCH 02/10] fuse2fs: skip permission checking on utimens when iomap is enabled Darrick J. Wong
2026-04-29 14:59   ` [PATCH 03/10] fuse2fs: let the kernel tell us about acl/mode updates Darrick J. Wong
2026-04-29 14:59   ` [PATCH 04/10] fuse2fs: better debugging for file mode updates Darrick J. Wong
2026-04-29 14:59   ` [PATCH 05/10] fuse2fs: debug timestamp updates Darrick J. Wong
2026-04-29 15:00   ` [PATCH 06/10] fuse2fs: use coarse timestamps for iomap mode Darrick J. Wong
2026-04-29 15:00   ` [PATCH 07/10] fuse2fs: add tracing for retrieving timestamps Darrick J. Wong
2026-04-29 15:00   ` [PATCH 08/10] fuse2fs: enable syncfs Darrick J. Wong
2026-04-29 15:00   ` [PATCH 09/10] fuse2fs: set sync, immutable, and append at file load time Darrick J. Wong
2026-04-29 15:01   ` [PATCH 10/10] fuse4fs: increase attribute timeout in iomap mode Darrick J. Wong
2026-04-29 14:21 ` [PATCHSET v8 6/6] fuse2fs: cache iomap mappings for even better file IO performance Darrick J. Wong
2026-04-29 15:01   ` [PATCH 1/4] fuse2fs: enable caching of iomaps Darrick J. Wong
2026-04-29 15:01   ` [PATCH 2/4] fuse2fs: constrain iomap mapping cache size Darrick J. Wong
2026-04-29 15:02   ` [PATCH 3/4] fuse4fs: upsert first file mapping to kernel on open Darrick J. Wong
2026-04-29 15:02   ` [PATCH 4/4] fuse2fs: enable iomap Darrick J. Wong
  -- strict thread matches above, loose matches on Subject: below --
2026-02-23 23:01 [PATCHSET v7 4/9] fuse: allow servers to use iomap for better file IO performance Darrick J. Wong
2026-02-23 23:12 ` [PATCH 14/33] fuse: implement direct IO with iomap Darrick J. Wong

Reply instructions:

You may reply publicly to this message via plain-text email
using any one of the following methods:

* Save the following mbox file, import it into your mail client,
  and reply-to-all from there: mbox

  Avoid top-posting and favor interleaved quoting:
  https://en.wikipedia.org/wiki/Posting_style#Interleaved_style

* Reply using the --to, --cc, and --in-reply-to
  switches of git-send-email(1):

  git send-email \
    --in-reply-to=20260513184451.GM9544@frogsfrogsfrogs \
    --to=djwong@kernel.org \
    --cc=bernd@bsbernd.com \
    --cc=fuse-devel@lists.linux.dev \
    --cc=joannelkoong@gmail.com \
    --cc=linux-fsdevel@vger.kernel.org \
    --cc=miklos@szeredi.hu \
    --cc=neal@gompa.dev \
    /path/to/YOUR_REPLY

  https://kernel.org/pub/software/scm/git/docs/git-send-email.html

* If your mail client supports setting the In-Reply-To header
  via mailto: links, try the mailto: link
Be sure your reply has a Subject: header at the top and a blank line before the message body.
This is an external index of several public inboxes,
see mirroring instructions on how to clone and mirror
all data and code used by this external index.