| 12345678910111213141516171819202122232425262728293031323334353637383940414243444546474849505152535455565758596061626364656667686970717273747576777879808182838485868788899091 |
- // SPDX-License-Identifier: GPL-2.0
- #include <linux/kernel.h>
- #include <linux/errno.h>
- #include <linux/fs.h>
- #include <linux/file.h>
- #include <linux/io_uring.h>
- #include <uapi/linux/io_uring.h>
- #include "io_uring.h"
- #include "rsrc.h"
- #include "nop.h"
- struct io_nop {
- /* NOTE: kiocb has the file as the first member, so don't do it here */
- struct file *file;
- int result;
- int fd;
- unsigned int flags;
- __u64 extra1;
- __u64 extra2;
- };
- #define NOP_FLAGS (IORING_NOP_INJECT_RESULT | IORING_NOP_FIXED_FILE | \
- IORING_NOP_FIXED_BUFFER | IORING_NOP_FILE | \
- IORING_NOP_TW | IORING_NOP_CQE32)
- int io_nop_prep(struct io_kiocb *req, const struct io_uring_sqe *sqe)
- {
- struct io_nop *nop = io_kiocb_to_cmd(req, struct io_nop);
- nop->flags = READ_ONCE(sqe->nop_flags);
- if (nop->flags & ~NOP_FLAGS)
- return -EINVAL;
- if (nop->flags & IORING_NOP_INJECT_RESULT)
- nop->result = READ_ONCE(sqe->len);
- else
- nop->result = 0;
- if (nop->flags & IORING_NOP_FILE)
- nop->fd = READ_ONCE(sqe->fd);
- else
- nop->fd = -1;
- if (nop->flags & IORING_NOP_FIXED_BUFFER)
- req->buf_index = READ_ONCE(sqe->buf_index);
- if (nop->flags & IORING_NOP_CQE32) {
- struct io_ring_ctx *ctx = req->ctx;
- if (!(ctx->flags & (IORING_SETUP_CQE32|IORING_SETUP_CQE_MIXED)))
- return -EINVAL;
- nop->extra1 = READ_ONCE(sqe->off);
- nop->extra2 = READ_ONCE(sqe->addr);
- }
- return 0;
- }
- int io_nop(struct io_kiocb *req, unsigned int issue_flags)
- {
- struct io_nop *nop = io_kiocb_to_cmd(req, struct io_nop);
- int ret = nop->result;
- if (nop->flags & IORING_NOP_FILE) {
- if (nop->flags & IORING_NOP_FIXED_FILE) {
- req->file = io_file_get_fixed(req, nop->fd, issue_flags);
- req->flags |= REQ_F_FIXED_FILE;
- } else {
- req->file = io_file_get_normal(req, nop->fd);
- }
- if (!req->file) {
- ret = -EBADF;
- goto done;
- }
- }
- if (nop->flags & IORING_NOP_FIXED_BUFFER) {
- if (!io_find_buf_node(req, issue_flags))
- ret = -EFAULT;
- }
- done:
- if (ret < 0)
- req_set_fail(req);
- if (nop->flags & IORING_NOP_CQE32)
- io_req_set_res32(req, nop->result, 0, nop->extra1, nop->extra2);
- else
- io_req_set_res(req, nop->result, 0);
- if (nop->flags & IORING_NOP_TW) {
- req->io_task_work.func = io_req_task_complete;
- io_req_task_work_add(req);
- return IOU_ISSUE_SKIP_COMPLETE;
- }
- return IOU_COMPLETE;
- }
|