]> git.ipfire.org Git - thirdparty/kernel/stable.git/commitdiff
io_uring/kbuf: pass in struct io_buffer_list to commit/recycle helpers
authorJens Axboe <axboe@kernel.dk>
Thu, 21 Aug 2025 02:03:33 +0000 (20:03 -0600)
committerJens Axboe <axboe@kernel.dk>
Sun, 24 Aug 2025 17:41:12 +0000 (11:41 -0600)
Rather than have this implied being in the io_kiocb, pass it in directly
so it's immediately obvious where these users of ->buf_list are coming
from.

Link: https://lore.kernel.org/r/20250821020750.598432-6-axboe@kernel.dk
Signed-off-by: Jens Axboe <axboe@kernel.dk>
io_uring/io_uring.c
io_uring/kbuf.c
io_uring/kbuf.h
io_uring/net.c
io_uring/poll.c
io_uring/rw.c

index ee484a0e2c4fcc3964267120f57545c9e84e43cd..27bc1486f07b6f1f6d1e645eabc2f88fd3e8f02c 100644 (file)
@@ -1007,7 +1007,7 @@ void io_req_defer_failed(struct io_kiocb *req, s32 res)
        lockdep_assert_held(&req->ctx->uring_lock);
 
        req_set_fail(req);
-       io_req_set_res(req, res, io_put_kbuf(req, res));
+       io_req_set_res(req, res, io_put_kbuf(req, res, req->buf_list));
        if (def->fail)
                def->fail(req);
        io_req_complete_defer(req);
@@ -2025,11 +2025,11 @@ fail:
 
        switch (io_arm_poll_handler(req, 0)) {
        case IO_APOLL_READY:
-               io_kbuf_recycle(req, 0);
+               io_kbuf_recycle(req, req->buf_list, 0);
                io_req_task_queue(req);
                break;
        case IO_APOLL_ABORTED:
-               io_kbuf_recycle(req, 0);
+               io_kbuf_recycle(req, req->buf_list, 0);
                io_queue_iowq(req);
                break;
        case IO_APOLL_OK:
index f2d2cc319faac5f5045d055c2bfff5999872dd1d..b8b2f6dee7547ffb80cae991df32ce9aeb830c87 100644 (file)
@@ -354,9 +354,9 @@ int io_buffers_peek(struct io_kiocb *req, struct buf_sel_arg *arg)
        return io_provided_buffers_select(req, &arg->max_len, bl, arg->iovs);
 }
 
-static inline bool __io_put_kbuf_ring(struct io_kiocb *req, int len, int nr)
+static inline bool __io_put_kbuf_ring(struct io_kiocb *req,
+                                     struct io_buffer_list *bl, int len, int nr)
 {
-       struct io_buffer_list *bl = req->buf_list;
        bool ret = true;
 
        if (bl)
@@ -366,7 +366,8 @@ static inline bool __io_put_kbuf_ring(struct io_kiocb *req, int len, int nr)
        return ret;
 }
 
-unsigned int __io_put_kbufs(struct io_kiocb *req, int len, int nbufs)
+unsigned int __io_put_kbufs(struct io_kiocb *req, struct io_buffer_list *bl,
+                           int len, int nbufs)
 {
        unsigned int ret;
 
@@ -377,7 +378,7 @@ unsigned int __io_put_kbufs(struct io_kiocb *req, int len, int nbufs)
                return ret;
        }
 
-       if (!__io_put_kbuf_ring(req, len, nbufs))
+       if (!__io_put_kbuf_ring(req, bl, len, nbufs))
                ret |= IORING_CQE_F_BUF_MORE;
        return ret;
 }
index 3d01778f378b1caa38f12ace3f2595926f02c991..58451ab4ab8ac9987dc3be45ce4b255f4059766d 100644 (file)
@@ -80,14 +80,16 @@ int io_register_pbuf_status(struct io_ring_ctx *ctx, void __user *arg);
 bool io_kbuf_recycle_legacy(struct io_kiocb *req, unsigned issue_flags);
 void io_kbuf_drop_legacy(struct io_kiocb *req);
 
-unsigned int __io_put_kbufs(struct io_kiocb *req, int len, int nbufs);
+unsigned int __io_put_kbufs(struct io_kiocb *req, struct io_buffer_list *bl,
+                           int len, int nbufs);
 bool io_kbuf_commit(struct io_kiocb *req,
                    struct io_buffer_list *bl, int len, int nr);
 
 struct io_mapped_region *io_pbuf_get_region(struct io_ring_ctx *ctx,
                                            unsigned int bgid);
 
-static inline bool io_kbuf_recycle_ring(struct io_kiocb *req)
+static inline bool io_kbuf_recycle_ring(struct io_kiocb *req,
+                                       struct io_buffer_list *bl)
 {
        /*
         * We don't need to recycle for REQ_F_BUFFER_RING, we can just clear
@@ -96,7 +98,7 @@ static inline bool io_kbuf_recycle_ring(struct io_kiocb *req)
         * The exception is partial io, that case we should increment bl->head
         * to monopolize the buffer.
         */
-       if (req->buf_list) {
+       if (bl) {
                req->flags &= ~(REQ_F_BUFFER_RING|REQ_F_BUFFERS_COMMIT);
                return true;
        }
@@ -110,29 +112,31 @@ static inline bool io_do_buffer_select(struct io_kiocb *req)
        return !(req->flags & (REQ_F_BUFFER_SELECTED|REQ_F_BUFFER_RING));
 }
 
-static inline bool io_kbuf_recycle(struct io_kiocb *req, unsigned issue_flags)
+static inline bool io_kbuf_recycle(struct io_kiocb *req, struct io_buffer_list *bl,
+                                  unsigned issue_flags)
 {
        if (req->flags & REQ_F_BL_NO_RECYCLE)
                return false;
        if (req->flags & REQ_F_BUFFER_SELECTED)
                return io_kbuf_recycle_legacy(req, issue_flags);
        if (req->flags & REQ_F_BUFFER_RING)
-               return io_kbuf_recycle_ring(req);
+               return io_kbuf_recycle_ring(req, bl);
        return false;
 }
 
-static inline unsigned int io_put_kbuf(struct io_kiocb *req, int len)
+static inline unsigned int io_put_kbuf(struct io_kiocb *req, int len,
+                                      struct io_buffer_list *bl)
 {
        if (!(req->flags & (REQ_F_BUFFER_RING | REQ_F_BUFFER_SELECTED)))
                return 0;
-       return __io_put_kbufs(req, len, 1);
+       return __io_put_kbufs(req, bl, len, 1);
 }
 
 static inline unsigned int io_put_kbufs(struct io_kiocb *req, int len,
-                                       int nbufs)
+                                       struct io_buffer_list *bl, int nbufs)
 {
        if (!(req->flags & (REQ_F_BUFFER_RING | REQ_F_BUFFER_SELECTED)))
                return 0;
-       return __io_put_kbufs(req, len, nbufs);
+       return __io_put_kbufs(req, bl, len, nbufs);
 }
 #endif
index 381454ed41c00bb64f9667271ac05f96b5920384..54e13d1b0d0b30e3697dec29cb4f5daafccef2e9 100644 (file)
@@ -494,12 +494,12 @@ static int io_bundle_nbufs(struct io_async_msghdr *kmsg, int ret)
        return nbufs;
 }
 
-static int io_net_kbuf_recyle(struct io_kiocb *req,
+static int io_net_kbuf_recyle(struct io_kiocb *req, struct io_buffer_list *bl,
                              struct io_async_msghdr *kmsg, int len)
 {
        req->flags |= REQ_F_BL_NO_RECYCLE;
        if (req->flags & REQ_F_BUFFERS_COMMIT)
-               io_kbuf_commit(req, req->buf_list, len, io_bundle_nbufs(kmsg, len));
+               io_kbuf_commit(req, bl, len, io_bundle_nbufs(kmsg, len));
        return IOU_RETRY;
 }
 
@@ -511,11 +511,11 @@ static inline bool io_send_finish(struct io_kiocb *req, int *ret,
        unsigned int cflags;
 
        if (!(sr->flags & IORING_RECVSEND_BUNDLE)) {
-               cflags = io_put_kbuf(req, *ret);
+               cflags = io_put_kbuf(req, *ret, req->buf_list);
                goto finish;
        }
 
-       cflags = io_put_kbufs(req, *ret, io_bundle_nbufs(kmsg, *ret));
+       cflags = io_put_kbufs(req, *ret, req->buf_list, io_bundle_nbufs(kmsg, *ret));
 
        if (bundle_finished || req->flags & REQ_F_BL_EMPTY)
                goto finish;
@@ -681,7 +681,7 @@ retry_bundle:
                        sr->len -= ret;
                        sr->buf += ret;
                        sr->done_io += ret;
-                       return io_net_kbuf_recyle(req, kmsg, ret);
+                       return io_net_kbuf_recyle(req, req->buf_list, kmsg, ret);
                }
                if (ret == -ERESTARTSYS)
                        ret = -EINTR;
@@ -871,7 +871,7 @@ static inline bool io_recv_finish(struct io_kiocb *req, int *ret,
        if (sr->flags & IORING_RECVSEND_BUNDLE) {
                size_t this_ret = *ret - sr->done_io;
 
-               cflags |= io_put_kbufs(req, this_ret, io_bundle_nbufs(kmsg, this_ret));
+               cflags |= io_put_kbufs(req, this_ret, req->buf_list, io_bundle_nbufs(kmsg, this_ret));
                if (sr->flags & IORING_RECV_RETRY)
                        cflags = req->cqe.flags | (cflags & CQE_F_MASK);
                if (sr->mshot_len && *ret >= sr->mshot_len)
@@ -893,7 +893,7 @@ static inline bool io_recv_finish(struct io_kiocb *req, int *ret,
                        return false;
                }
        } else {
-               cflags |= io_put_kbuf(req, *ret);
+               cflags |= io_put_kbuf(req, *ret, req->buf_list);
        }
 
        /*
@@ -1045,7 +1045,7 @@ retry_multishot:
                if (req->flags & REQ_F_APOLL_MULTISHOT) {
                        ret = io_recvmsg_prep_multishot(kmsg, sr, &buf, &len);
                        if (ret) {
-                               io_kbuf_recycle(req, issue_flags);
+                               io_kbuf_recycle(req, req->buf_list, issue_flags);
                                return ret;
                        }
                }
@@ -1070,13 +1070,13 @@ retry_multishot:
        if (ret < min_ret) {
                if (ret == -EAGAIN && force_nonblock) {
                        if (issue_flags & IO_URING_F_MULTISHOT)
-                               io_kbuf_recycle(req, issue_flags);
+                               io_kbuf_recycle(req, req->buf_list, issue_flags);
 
                        return IOU_RETRY;
                }
                if (ret > 0 && io_net_retry(sock, flags)) {
                        sr->done_io += ret;
-                       return io_net_kbuf_recyle(req, kmsg, ret);
+                       return io_net_kbuf_recyle(req, req->buf_list, kmsg, ret);
                }
                if (ret == -ERESTARTSYS)
                        ret = -EINTR;
@@ -1090,7 +1090,7 @@ retry_multishot:
        else if (sr->done_io)
                ret = sr->done_io;
        else
-               io_kbuf_recycle(req, issue_flags);
+               io_kbuf_recycle(req, req->buf_list, issue_flags);
 
        if (!io_recv_finish(req, &ret, kmsg, mshot_finished, issue_flags))
                goto retry_multishot;
@@ -1214,7 +1214,7 @@ retry_multishot:
        if (ret < min_ret) {
                if (ret == -EAGAIN && force_nonblock) {
                        if (issue_flags & IO_URING_F_MULTISHOT)
-                               io_kbuf_recycle(req, issue_flags);
+                               io_kbuf_recycle(req, req->buf_list, issue_flags);
 
                        return IOU_RETRY;
                }
@@ -1222,7 +1222,7 @@ retry_multishot:
                        sr->len -= ret;
                        sr->buf += ret;
                        sr->done_io += ret;
-                       return io_net_kbuf_recyle(req, kmsg, ret);
+                       return io_net_kbuf_recyle(req, req->buf_list, kmsg, ret);
                }
                if (ret == -ERESTARTSYS)
                        ret = -EINTR;
@@ -1238,7 +1238,7 @@ out_free:
        else if (sr->done_io)
                ret = sr->done_io;
        else
-               io_kbuf_recycle(req, issue_flags);
+               io_kbuf_recycle(req, req->buf_list, issue_flags);
 
        if (!io_recv_finish(req, &ret, kmsg, mshot_finished, issue_flags))
                goto retry_multishot;
index c786e587563b0598228a2eb74a423485cf0b2063..07ab22380c7857c442a250a43c6bab081bc04a49 100644 (file)
@@ -316,10 +316,10 @@ void io_poll_task_func(struct io_kiocb *req, io_tw_token_t tw)
 
        ret = io_poll_check_events(req, tw);
        if (ret == IOU_POLL_NO_ACTION) {
-               io_kbuf_recycle(req, 0);
+               io_kbuf_recycle(req, req->buf_list, 0);
                return;
        } else if (ret == IOU_POLL_REQUEUE) {
-               io_kbuf_recycle(req, 0);
+               io_kbuf_recycle(req, req->buf_list, 0);
                __io_poll_execute(req, 0);
                return;
        }
@@ -686,7 +686,7 @@ int io_arm_apoll(struct io_kiocb *req, unsigned issue_flags, __poll_t mask)
        req->flags |= REQ_F_POLLED;
        ipt.pt._qproc = io_async_queue_proc;
 
-       io_kbuf_recycle(req, issue_flags);
+       io_kbuf_recycle(req, req->buf_list, issue_flags);
 
        ret = __io_arm_poll_handler(req, &apoll->poll, &ipt, mask, issue_flags);
        if (ret)
index ae5229ae7dcae53964d1e3abb5ba47baa075b060..7fe188872279412fb4009b414cd181f2833c815c 100644 (file)
@@ -576,7 +576,7 @@ void io_req_rw_complete(struct io_kiocb *req, io_tw_token_t tw)
        io_req_io_end(req);
 
        if (req->flags & (REQ_F_BUFFER_SELECTED|REQ_F_BUFFER_RING))
-               req->cqe.flags |= io_put_kbuf(req, req->cqe.res);
+               req->cqe.flags |= io_put_kbuf(req, req->cqe.res, req->buf_list);
 
        io_req_rw_cleanup(req, 0);
        io_req_task_complete(req, tw);
@@ -659,7 +659,7 @@ static int kiocb_done(struct io_kiocb *req, ssize_t ret,
                 * from the submission path.
                 */
                io_req_io_end(req);
-               io_req_set_res(req, final_ret, io_put_kbuf(req, ret));
+               io_req_set_res(req, final_ret, io_put_kbuf(req, ret, req->buf_list));
                io_req_rw_cleanup(req, issue_flags);
                return IOU_COMPLETE;
        } else {
@@ -1049,15 +1049,15 @@ int io_read_mshot(struct io_kiocb *req, unsigned int issue_flags)
                 * Reset rw->len to 0 again to avoid clamping future mshot
                 * reads, in case the buffer size varies.
                 */
-               if (io_kbuf_recycle(req, issue_flags))
+               if (io_kbuf_recycle(req, req->buf_list, issue_flags))
                        rw->len = 0;
                return IOU_RETRY;
        } else if (ret <= 0) {
-               io_kbuf_recycle(req, issue_flags);
+               io_kbuf_recycle(req, req->buf_list, issue_flags);
                if (ret < 0)
                        req_set_fail(req);
        } else if (!(req->flags & REQ_F_APOLL_MULTISHOT)) {
-               cflags = io_put_kbuf(req, ret);
+               cflags = io_put_kbuf(req, ret, req->buf_list);
        } else {
                /*
                 * Any successful return value will keep the multishot read
@@ -1065,7 +1065,7 @@ int io_read_mshot(struct io_kiocb *req, unsigned int issue_flags)
                 * we fail to post a CQE, or multishot is no longer set, then
                 * jump to the termination path. This request is then done.
                 */
-               cflags = io_put_kbuf(req, ret);
+               cflags = io_put_kbuf(req, ret, req->buf_list);
                rw->len = 0; /* similarly to above, reset len to 0 */
 
                if (io_req_post_cqe(req, ret, cflags | IORING_CQE_F_MORE)) {
@@ -1362,7 +1362,7 @@ int io_do_iopoll(struct io_ring_ctx *ctx, bool force_nonspin)
                if (!smp_load_acquire(&req->iopoll_completed))
                        break;
                nr_events++;
-               req->cqe.flags = io_put_kbuf(req, req->cqe.res);
+               req->cqe.flags = io_put_kbuf(req, req->cqe.res, req->buf_list);
                if (req->opcode != IORING_OP_URING_CMD)
                        io_req_rw_cleanup(req, 0);
        }