forked from mirrors/linux
		
	 8e9fad0e70
			
		
	
	
		8e9fad0e70
		
	
	
	
	
		
			
			Enable io_uring commands on network sockets. Create two new SOCKET_URING_OP commands that will operate on sockets. In order to call ioctl on sockets, use the file_operations->io_uring_cmd callbacks, and map it to a uring socket function, which handles the SOCKET_URING_OP accordingly, and calls socket ioctls. This patches was tested by creating a new test case in liburing. Link: https://github.com/leitao/liburing/tree/io_uring_cmd Signed-off-by: Breno Leitao <leitao@debian.org> Acked-by: Jakub Kicinski <kuba@kernel.org> Link: https://lore.kernel.org/r/20230627134424.2784797-1-leitao@debian.org Signed-off-by: Jens Axboe <axboe@kernel.dk>
		
			
				
	
	
		
			127 lines
		
	
	
	
		
			3.4 KiB
		
	
	
	
		
			C
		
	
	
	
	
	
			
		
		
	
	
			127 lines
		
	
	
	
		
			3.4 KiB
		
	
	
	
		
			C
		
	
	
	
	
	
| /* SPDX-License-Identifier: GPL-2.0-or-later */
 | |
| #ifndef _LINUX_IO_URING_H
 | |
| #define _LINUX_IO_URING_H
 | |
| 
 | |
| #include <linux/sched.h>
 | |
| #include <linux/xarray.h>
 | |
| #include <uapi/linux/io_uring.h>
 | |
| 
 | |
| enum io_uring_cmd_flags {
 | |
| 	IO_URING_F_COMPLETE_DEFER	= 1,
 | |
| 	IO_URING_F_UNLOCKED		= 2,
 | |
| 	/* the request is executed from poll, it should not be freed */
 | |
| 	IO_URING_F_MULTISHOT		= 4,
 | |
| 	/* executed by io-wq */
 | |
| 	IO_URING_F_IOWQ			= 8,
 | |
| 	/* int's last bit, sign checks are usually faster than a bit test */
 | |
| 	IO_URING_F_NONBLOCK		= INT_MIN,
 | |
| 
 | |
| 	/* ctx state flags, for URING_CMD */
 | |
| 	IO_URING_F_SQE128		= (1 << 8),
 | |
| 	IO_URING_F_CQE32		= (1 << 9),
 | |
| 	IO_URING_F_IOPOLL		= (1 << 10),
 | |
| };
 | |
| 
 | |
| struct io_uring_cmd {
 | |
| 	struct file	*file;
 | |
| 	const struct io_uring_sqe *sqe;
 | |
| 	union {
 | |
| 		/* callback to defer completions to task context */
 | |
| 		void (*task_work_cb)(struct io_uring_cmd *cmd, unsigned);
 | |
| 		/* used for polled completion */
 | |
| 		void *cookie;
 | |
| 	};
 | |
| 	u32		cmd_op;
 | |
| 	u32		flags;
 | |
| 	u8		pdu[32]; /* available inline for free use */
 | |
| };
 | |
| 
 | |
| static inline const void *io_uring_sqe_cmd(const struct io_uring_sqe *sqe)
 | |
| {
 | |
| 	return sqe->cmd;
 | |
| }
 | |
| 
 | |
| #if defined(CONFIG_IO_URING)
 | |
| int io_uring_cmd_import_fixed(u64 ubuf, unsigned long len, int rw,
 | |
| 			      struct iov_iter *iter, void *ioucmd);
 | |
| void io_uring_cmd_done(struct io_uring_cmd *cmd, ssize_t ret, ssize_t res2,
 | |
| 			unsigned issue_flags);
 | |
| struct sock *io_uring_get_socket(struct file *file);
 | |
| void __io_uring_cancel(bool cancel_all);
 | |
| void __io_uring_free(struct task_struct *tsk);
 | |
| void io_uring_unreg_ringfd(void);
 | |
| const char *io_uring_get_opcode(u8 opcode);
 | |
| void __io_uring_cmd_do_in_task(struct io_uring_cmd *ioucmd,
 | |
| 			    void (*task_work_cb)(struct io_uring_cmd *, unsigned),
 | |
| 			    unsigned flags);
 | |
| /* users should follow semantics of IOU_F_TWQ_LAZY_WAKE */
 | |
| void io_uring_cmd_do_in_task_lazy(struct io_uring_cmd *ioucmd,
 | |
| 			void (*task_work_cb)(struct io_uring_cmd *, unsigned));
 | |
| 
 | |
| static inline void io_uring_cmd_complete_in_task(struct io_uring_cmd *ioucmd,
 | |
| 			void (*task_work_cb)(struct io_uring_cmd *, unsigned))
 | |
| {
 | |
| 	__io_uring_cmd_do_in_task(ioucmd, task_work_cb, 0);
 | |
| }
 | |
| 
 | |
| static inline void io_uring_files_cancel(void)
 | |
| {
 | |
| 	if (current->io_uring) {
 | |
| 		io_uring_unreg_ringfd();
 | |
| 		__io_uring_cancel(false);
 | |
| 	}
 | |
| }
 | |
| static inline void io_uring_task_cancel(void)
 | |
| {
 | |
| 	if (current->io_uring)
 | |
| 		__io_uring_cancel(true);
 | |
| }
 | |
| static inline void io_uring_free(struct task_struct *tsk)
 | |
| {
 | |
| 	if (tsk->io_uring)
 | |
| 		__io_uring_free(tsk);
 | |
| }
 | |
| int io_uring_cmd_sock(struct io_uring_cmd *cmd, unsigned int issue_flags);
 | |
| #else
 | |
| static inline int io_uring_cmd_import_fixed(u64 ubuf, unsigned long len, int rw,
 | |
| 			      struct iov_iter *iter, void *ioucmd)
 | |
| {
 | |
| 	return -EOPNOTSUPP;
 | |
| }
 | |
| static inline void io_uring_cmd_done(struct io_uring_cmd *cmd, ssize_t ret,
 | |
| 		ssize_t ret2, unsigned issue_flags)
 | |
| {
 | |
| }
 | |
| static inline void io_uring_cmd_complete_in_task(struct io_uring_cmd *ioucmd,
 | |
| 			void (*task_work_cb)(struct io_uring_cmd *, unsigned))
 | |
| {
 | |
| }
 | |
| static inline void io_uring_cmd_do_in_task_lazy(struct io_uring_cmd *ioucmd,
 | |
| 			void (*task_work_cb)(struct io_uring_cmd *, unsigned))
 | |
| {
 | |
| }
 | |
| static inline struct sock *io_uring_get_socket(struct file *file)
 | |
| {
 | |
| 	return NULL;
 | |
| }
 | |
| static inline void io_uring_task_cancel(void)
 | |
| {
 | |
| }
 | |
| static inline void io_uring_files_cancel(void)
 | |
| {
 | |
| }
 | |
| static inline void io_uring_free(struct task_struct *tsk)
 | |
| {
 | |
| }
 | |
| static inline const char *io_uring_get_opcode(u8 opcode)
 | |
| {
 | |
| 	return "";
 | |
| }
 | |
| static inline int io_uring_cmd_sock(struct io_uring_cmd *cmd,
 | |
| 				    unsigned int issue_flags)
 | |
| {
 | |
| 	return -EOPNOTSUPP;
 | |
| }
 | |
| #endif
 | |
| 
 | |
| #endif
 |