mirror of
				https://github.com/torvalds/linux.git
				synced 2025-11-01 00:58:39 +02:00 
			
		
		
		
	Add io_uring IO interface
The submission queue (SQ) and completion queue (CQ) rings are shared between the application and the kernel. This eliminates the need to copy data back and forth to submit and complete IO. IO submissions use the io_uring_sqe data structure, and completions are generated in the form of io_uring_cqe data structures. The SQ ring is an index into the io_uring_sqe array, which makes it possible to submit a batch of IOs without them being contiguous in the ring. The CQ ring is always contiguous, as completion events are inherently unordered, and hence any io_uring_cqe entry can point back to an arbitrary submission. Two new system calls are added for this: io_uring_setup(entries, params) Sets up an io_uring instance for doing async IO. On success, returns a file descriptor that the application can mmap to gain access to the SQ ring, CQ ring, and io_uring_sqes. io_uring_enter(fd, to_submit, min_complete, flags, sigset, sigsetsize) Initiates IO against the rings mapped to this fd, or waits for them to complete, or both. The behavior is controlled by the parameters passed in. If 'to_submit' is non-zero, then we'll try and submit new IO. If IORING_ENTER_GETEVENTS is set, the kernel will wait for 'min_complete' events, if they aren't already available. It's valid to set IORING_ENTER_GETEVENTS and 'min_complete' == 0 at the same time, this allows the kernel to return already completed events without waiting for them. This is useful only for polling, as for IRQ driven IO, the application can just check the CQ ring without entering the kernel. With this setup, it's possible to do async IO with a single system call. Future developments will enable polled IO with this interface, and polled submission as well. The latter will enable an application to do IO without doing ANY system calls at all. For IRQ driven IO, an application only needs to enter the kernel for completions if it wants to wait for them to occur. Each io_uring is backed by a workqueue, to support buffered async IO as well. We will only punt to an async context if the command would need to wait for IO on the device side. Any data that can be accessed directly in the page cache is done inline. This avoids the slowness issue of usual threadpools, since cached data is accessed as quickly as a sync interface. Sample application: http://git.kernel.dk/cgit/fio/plain/t/io_uring.c Reviewed-by: Hannes Reinecke <hare@suse.com> Signed-off-by: Jens Axboe <axboe@kernel.dk>
This commit is contained in:
		
							parent
							
								
									594b9a89af
								
							
						
					
					
						commit
						2b188cc1bb
					
				
					 12 changed files with 1390 additions and 2 deletions
				
			
		|  | @ -398,3 +398,5 @@ | |||
| 384	i386	arch_prctl		sys_arch_prctl			__ia32_compat_sys_arch_prctl | ||||
| 385	i386	io_pgetevents		sys_io_pgetevents		__ia32_compat_sys_io_pgetevents | ||||
| 386	i386	rseq			sys_rseq			__ia32_sys_rseq | ||||
| 425	i386	io_uring_setup		sys_io_uring_setup		__ia32_sys_io_uring_setup | ||||
| 426	i386	io_uring_enter		sys_io_uring_enter		__ia32_sys_io_uring_enter | ||||
|  |  | |||
|  | @ -343,6 +343,8 @@ | |||
| 332	common	statx			__x64_sys_statx | ||||
| 333	common	io_pgetevents		__x64_sys_io_pgetevents | ||||
| 334	common	rseq			__x64_sys_rseq | ||||
| 425	common	io_uring_setup		__x64_sys_io_uring_setup | ||||
| 426	common	io_uring_enter		__x64_sys_io_uring_enter | ||||
| 
 | ||||
| # | ||||
| # x32-specific system call numbers start at 512 to avoid cache impact | ||||
|  |  | |||
|  | @ -30,6 +30,7 @@ obj-$(CONFIG_TIMERFD)		+= timerfd.o | |||
| obj-$(CONFIG_EVENTFD)		+= eventfd.o | ||||
| obj-$(CONFIG_USERFAULTFD)	+= userfaultfd.o | ||||
| obj-$(CONFIG_AIO)               += aio.o | ||||
| obj-$(CONFIG_IO_URING)		+= io_uring.o | ||||
| obj-$(CONFIG_FS_DAX)		+= dax.o | ||||
| obj-$(CONFIG_FS_ENCRYPTION)	+= crypto/ | ||||
| obj-$(CONFIG_FILE_LOCKING)      += locks.o | ||||
|  |  | |||
							
								
								
									
										1255
									
								
								fs/io_uring.c
									
									
									
									
									
										Normal file
									
								
							
							
						
						
									
										1255
									
								
								fs/io_uring.c
									
									
									
									
									
										Normal file
									
								
							
										
											
												File diff suppressed because it is too large
												Load diff
											
										
									
								
							|  | @ -3517,4 +3517,13 @@ extern void inode_nohighmem(struct inode *inode); | |||
| extern int vfs_fadvise(struct file *file, loff_t offset, loff_t len, | ||||
| 		       int advice); | ||||
| 
 | ||||
| #if defined(CONFIG_IO_URING) | ||||
| extern struct sock *io_uring_get_socket(struct file *file); | ||||
| #else | ||||
| static inline struct sock *io_uring_get_socket(struct file *file) | ||||
| { | ||||
| 	return NULL; | ||||
| } | ||||
| #endif | ||||
| 
 | ||||
| #endif /* _LINUX_FS_H */ | ||||
|  |  | |||
|  | @ -40,7 +40,7 @@ struct user_struct { | |||
| 	kuid_t uid; | ||||
| 
 | ||||
| #if defined(CONFIG_PERF_EVENTS) || defined(CONFIG_BPF_SYSCALL) || \ | ||||
|     defined(CONFIG_NET) | ||||
|     defined(CONFIG_NET) || defined(CONFIG_IO_URING) | ||||
| 	atomic_long_t locked_vm; | ||||
| #endif | ||||
| 
 | ||||
|  |  | |||
|  | @ -69,6 +69,7 @@ struct file_handle; | |||
| struct sigaltstack; | ||||
| struct rseq; | ||||
| union bpf_attr; | ||||
| struct io_uring_params; | ||||
| 
 | ||||
| #include <linux/types.h> | ||||
| #include <linux/aio_abi.h> | ||||
|  | @ -309,6 +310,11 @@ asmlinkage long sys_io_pgetevents_time32(aio_context_t ctx_id, | |||
| 				struct io_event __user *events, | ||||
| 				struct old_timespec32 __user *timeout, | ||||
| 				const struct __aio_sigset *sig); | ||||
| asmlinkage long sys_io_uring_setup(u32 entries, | ||||
| 				struct io_uring_params __user *p); | ||||
| asmlinkage long sys_io_uring_enter(unsigned int fd, u32 to_submit, | ||||
| 				u32 min_complete, u32 flags, | ||||
| 				const sigset_t __user *sig, size_t sigsz); | ||||
| 
 | ||||
| /* fs/xattr.c */ | ||||
| asmlinkage long sys_setxattr(const char __user *path, const char __user *name, | ||||
|  |  | |||
|  | @ -740,9 +740,13 @@ __SC_COMP(__NR_io_pgetevents, sys_io_pgetevents, compat_sys_io_pgetevents) | |||
| __SYSCALL(__NR_rseq, sys_rseq) | ||||
| #define __NR_kexec_file_load 294 | ||||
| __SYSCALL(__NR_kexec_file_load,     sys_kexec_file_load) | ||||
| #define __NR_io_uring_setup 425 | ||||
| __SYSCALL(__NR_io_uring_setup, sys_io_uring_setup) | ||||
| #define __NR_io_uring_enter 426 | ||||
| __SYSCALL(__NR_io_uring_enter, sys_io_uring_enter) | ||||
| 
 | ||||
| #undef __NR_syscalls | ||||
| #define __NR_syscalls 295 | ||||
| #define __NR_syscalls 427 | ||||
| 
 | ||||
| /*
 | ||||
|  * 32 bit systems traditionally used different | ||||
|  |  | |||
							
								
								
									
										95
									
								
								include/uapi/linux/io_uring.h
									
									
									
									
									
										Normal file
									
								
							
							
						
						
									
										95
									
								
								include/uapi/linux/io_uring.h
									
									
									
									
									
										Normal file
									
								
							|  | @ -0,0 +1,95 @@ | |||
| /* SPDX-License-Identifier: GPL-2.0 WITH Linux-syscall-note */ | ||||
| /*
 | ||||
|  * Header file for the io_uring interface. | ||||
|  * | ||||
|  * Copyright (C) 2019 Jens Axboe | ||||
|  * Copyright (C) 2019 Christoph Hellwig | ||||
|  */ | ||||
| #ifndef LINUX_IO_URING_H | ||||
| #define LINUX_IO_URING_H | ||||
| 
 | ||||
| #include <linux/fs.h> | ||||
| #include <linux/types.h> | ||||
| 
 | ||||
| /*
 | ||||
|  * IO submission data structure (Submission Queue Entry) | ||||
|  */ | ||||
| struct io_uring_sqe { | ||||
| 	__u8	opcode;		/* type of operation for this sqe */ | ||||
| 	__u8	flags;		/* as of now unused */ | ||||
| 	__u16	ioprio;		/* ioprio for the request */ | ||||
| 	__s32	fd;		/* file descriptor to do IO on */ | ||||
| 	__u64	off;		/* offset into file */ | ||||
| 	__u64	addr;		/* pointer to buffer or iovecs */ | ||||
| 	__u32	len;		/* buffer size or number of iovecs */ | ||||
| 	union { | ||||
| 		__kernel_rwf_t	rw_flags; | ||||
| 		__u32		__resv; | ||||
| 	}; | ||||
| 	__u64	user_data;	/* data to be passed back at completion time */ | ||||
| 	__u64	__pad2[3]; | ||||
| }; | ||||
| 
 | ||||
| #define IORING_OP_NOP		0 | ||||
| #define IORING_OP_READV		1 | ||||
| #define IORING_OP_WRITEV	2 | ||||
| 
 | ||||
| /*
 | ||||
|  * IO completion data structure (Completion Queue Entry) | ||||
|  */ | ||||
| struct io_uring_cqe { | ||||
| 	__u64	user_data;	/* sqe->data submission passed back */ | ||||
| 	__s32	res;		/* result code for this event */ | ||||
| 	__u32	flags; | ||||
| }; | ||||
| 
 | ||||
| /*
 | ||||
|  * Magic offsets for the application to mmap the data it needs | ||||
|  */ | ||||
| #define IORING_OFF_SQ_RING		0ULL | ||||
| #define IORING_OFF_CQ_RING		0x8000000ULL | ||||
| #define IORING_OFF_SQES			0x10000000ULL | ||||
| 
 | ||||
| /*
 | ||||
|  * Filled with the offset for mmap(2) | ||||
|  */ | ||||
| struct io_sqring_offsets { | ||||
| 	__u32 head; | ||||
| 	__u32 tail; | ||||
| 	__u32 ring_mask; | ||||
| 	__u32 ring_entries; | ||||
| 	__u32 flags; | ||||
| 	__u32 dropped; | ||||
| 	__u32 array; | ||||
| 	__u32 resv1; | ||||
| 	__u64 resv2; | ||||
| }; | ||||
| 
 | ||||
| struct io_cqring_offsets { | ||||
| 	__u32 head; | ||||
| 	__u32 tail; | ||||
| 	__u32 ring_mask; | ||||
| 	__u32 ring_entries; | ||||
| 	__u32 overflow; | ||||
| 	__u32 cqes; | ||||
| 	__u64 resv[2]; | ||||
| }; | ||||
| 
 | ||||
| /*
 | ||||
|  * io_uring_enter(2) flags | ||||
|  */ | ||||
| #define IORING_ENTER_GETEVENTS	(1U << 0) | ||||
| 
 | ||||
| /*
 | ||||
|  * Passed in for io_uring_setup(2). Copied back with updated info on success | ||||
|  */ | ||||
| struct io_uring_params { | ||||
| 	__u32 sq_entries; | ||||
| 	__u32 cq_entries; | ||||
| 	__u32 flags; | ||||
| 	__u32 resv[7]; | ||||
| 	struct io_sqring_offsets sq_off; | ||||
| 	struct io_cqring_offsets cq_off; | ||||
| }; | ||||
| 
 | ||||
| #endif | ||||
|  | @ -1414,6 +1414,15 @@ config AIO | |||
| 	  by some high performance threaded applications. Disabling | ||||
| 	  this option saves about 7k. | ||||
| 
 | ||||
| config IO_URING | ||||
| 	bool "Enable IO uring support" if EXPERT | ||||
| 	select ANON_INODES | ||||
| 	default y | ||||
| 	help | ||||
| 	  This option enables support for the io_uring interface, enabling | ||||
| 	  applications to submit and complete IO through submission and | ||||
| 	  completion rings that are shared between the kernel and application. | ||||
| 
 | ||||
| config ADVISE_SYSCALLS | ||||
| 	bool "Enable madvise/fadvise syscalls" if EXPERT | ||||
| 	default y | ||||
|  |  | |||
|  | @ -46,6 +46,8 @@ COND_SYSCALL(io_getevents); | |||
| COND_SYSCALL(io_pgetevents); | ||||
| COND_SYSCALL_COMPAT(io_getevents); | ||||
| COND_SYSCALL_COMPAT(io_pgetevents); | ||||
| COND_SYSCALL(io_uring_setup); | ||||
| COND_SYSCALL(io_uring_enter); | ||||
| 
 | ||||
| /* fs/xattr.c */ | ||||
| 
 | ||||
|  |  | |||
|  | @ -108,6 +108,9 @@ struct sock *unix_get_socket(struct file *filp) | |||
| 		/* PF_UNIX ? */ | ||||
| 		if (s && sock->ops && sock->ops->family == PF_UNIX) | ||||
| 			u_sock = s; | ||||
| 	} else { | ||||
| 		/* Could be an io_uring instance */ | ||||
| 		u_sock = io_uring_get_socket(filp); | ||||
| 	} | ||||
| 	return u_sock; | ||||
| } | ||||
|  |  | |||
		Loading…
	
		Reference in a new issue
	
	 Jens Axboe
						Jens Axboe