This patch series adds support for async buffered writes when using both xfs and io-uring. Currently io-uring only supports buffered writes in the slow path, by processing them in the io workers. With this patch series it is now possible to support buffered writes in the fast path. To be able to use the fast path the required pages must be in the page cache, the required locks in xfs can be granted immediately and no additional blocks need to be read form disk. Updating the inode can take time. An optimization has been implemented for the time update. Time updates will be processed in the slow path. While there is already a time update in process, other write requests for the same file, can skip the update of the modification time. Performance results: For fio the following results have been obtained with a queue depth of 1 and 4k block size (runtime 600 secs): sequential writes: without patch with patch libaio psync iops: 77k 209k 195K 233K bw: 314MB/s 854MB/s 790MB/s 953MB/s clat: 9600ns 120ns 540ns 3000ns For an io depth of 1, the new patch improves throughput by over three times (compared to the exiting behavior, where buffered writes are processed by an io-worker process) and also the latency is considerably reduced. To achieve the same or better performance with the exisiting code an io depth of 4 is required. Increasing the iodepth further does not lead to improvements. In addition the latency of buffered write operations is reduced considerably. Support for async buffered writes: To support async buffered writes the flag FMODE_BUF_WASYNC is introduced. In addition the check in generic_write_checks is modified to allow for async buffered writes that have this flag set. Changes to the iomap page create function to allow the caller to specify the gfp flags. Sets the IOMAP_NOWAIT flag in iomap if IOCB_NOWAIT has been set and specifies the requested gfp flags. Adds the iomap async buffered write support to the xfs iomap layer. Adds async buffered write support to the xfs iomap layer. Support for async buffered write support and inode time modification Splits the functions for checking if the file privileges need to be removed in two functions: check function and a function for the removal of file privileges. The same split is also done for the function to update the file modification time. Implement an optimization that while a file modification time is pending other requests for the same file don't need to wait for the file modification update. This avoids that a considerable number of buffered async write requests get punted. Take the ilock in nowait mode if async buffered writes are enabled and enable the async buffered writes optimization in io_uring. Support for write throttling of async buffered writes: Split of a _balance_dirty_pages() function that tells wether to sleep or not. The function is called by the async buffered writes in iomap_write_iter() to determine if write throttling is required or not. In case write throttling is necessary, return -EAGAIN to the caller in io-uring. The request will then be handled in io-uring io-worker. The existing function balance_dirty_pages() will call _balance_dirty_pages(). Enable async buffered write support in xfs Patch 18: xfs: enable async buffered write support This enables the flag that enables async buffered writes for xfs. Testing: This patch has been tested with xfstests and fio. Changes: V2: - Remove atomic allocation - Use direct write in xfs_buffered_write_iomap_begin() - Use xfs_ilock_for_iomap() in xfs_buffered_write_iomap_begin() - Remove no_wait check at the end of xfs_buffered_write_iomap_begin() for the COW path. - Pass xfs_inode pointer to xfs_ilock_iocb and rename function to xfs_lock_xfs_inode - Replace exisitng uses of xfs_ilock_iocb with xfs_ilock_xfs_inode - Use xfs_ilock_xfs_inode in xfs_file_buffered_write() - Callers of xfs_ilock_for_iomap need to initialize lock mode. This is required so writes use an exclusive lock - Split of _balance_dirty_pages() from balance_dirty_pages() and return sleep time - Call _balance_dirty_pages() in balance_dirty_pages_ratelimited_flags() - Move call to balance_dirty_pages_ratelimited_flags() in iomap_write_iter() to the beginning of the loop Stefan Roesch (16): block: add check for async buffered writes to generic_write_checks iomap: add iomap_page_create_gfp to allocate iomap_pages iomap: use iomap_page_create_gfp() in __iomap_write_begin iomap: add async buffered write support xfs: add iomap async buffered write support fs: split off need_remove_file_privs() do_remove_file_privs() fs: split off need_file_update_time and do_file_update_time fs: add pending file update time flag. xfs: enable async write file modification handling. xfs: add async buffered write support io_uring: add support for async buffered writes mm: factor out _balance_dirty_pages() from balance_dirty_pages() mm: add balance_dirty_pages_ratelimited_flags() function iomap: use balance_dirty_pages_ratelimited_flags in iomap_write_iter io_uring: add tracepoint for short writes xfs: enable async buffered write support fs/inode.c | 146 +++++++--- fs/io_uring.c | 32 +- fs/iomap/buffered-io.c | 67 ++++- fs/read_write.c | 3 +- fs/xfs/xfs_file.c | 36 ++- fs/xfs/xfs_iomap.c | 14 +- include/linux/fs.h | 7 + include/linux/writeback.h | 1 + include/trace/events/io_uring.h | 25 ++ mm/page-writeback.c | 500 +++++++++++++++++--------------- 10 files changed, 520 insertions(+), 311 deletions(-) base-commit: 0cdd776ec92c0fec768c7079331804d3e52d4b27 -- 2.30.2