mirror of
https://git.kernel.org/pub/scm/linux/kernel/git/torvalds/linux.git
synced 2024-12-29 17:23:36 +00:00
6fe4220912
The network stack allows only one ubuf_info per skb, and unlike MSG_ZEROCOPY, each io_uring zerocopy send will carry a separate ubuf_info. That means that send requests can't reuse a previosly allocated skb and need to get one more or more of new ones. That's fine for large sends, but otherwise it would spam the stack with lots of skbs carrying just a little data each. To help with that implement linking notification (i.e. an io_uring wrapper around ubuf_info) into a list. Each is refcounted by skbs and the stack as usual. additionally all non head entries keep a reference to the head, which they put down when their refcount hits 0. When the head have no more users, it'll efficiently put all notifications in a batch. As mentioned previously about ->io_link_skb, the callback implementation always allows to bind to an skb without a ubuf_info. Reviewed-by: Jens Axboe <axboe@kernel.dk> Signed-off-by: Pavel Begunkov <asml.silence@gmail.com> Link: https://lore.kernel.org/r/bf1e7f9b72f9ecc99999fdc0d2cded5eea87fd0b.1713369317.git.asml.silence@gmail.com Signed-off-by: Jens Axboe <axboe@kernel.dk>
58 lines
1.3 KiB
C
58 lines
1.3 KiB
C
// SPDX-License-Identifier: GPL-2.0
|
|
|
|
#include <linux/net.h>
|
|
#include <linux/uio.h>
|
|
#include <net/sock.h>
|
|
#include <linux/nospec.h>
|
|
|
|
#include "rsrc.h"
|
|
|
|
#define IO_NOTIF_UBUF_FLAGS (SKBFL_ZEROCOPY_FRAG | SKBFL_DONT_ORPHAN)
|
|
#define IO_NOTIF_SPLICE_BATCH 32
|
|
|
|
struct io_notif_data {
|
|
struct file *file;
|
|
struct ubuf_info uarg;
|
|
|
|
struct io_notif_data *next;
|
|
struct io_notif_data *head;
|
|
|
|
unsigned account_pages;
|
|
bool zc_report;
|
|
bool zc_used;
|
|
bool zc_copied;
|
|
};
|
|
|
|
struct io_kiocb *io_alloc_notif(struct io_ring_ctx *ctx);
|
|
void io_tx_ubuf_complete(struct sk_buff *skb, struct ubuf_info *uarg,
|
|
bool success);
|
|
|
|
static inline struct io_notif_data *io_notif_to_data(struct io_kiocb *notif)
|
|
{
|
|
return io_kiocb_to_cmd(notif, struct io_notif_data);
|
|
}
|
|
|
|
static inline void io_notif_flush(struct io_kiocb *notif)
|
|
__must_hold(¬if->ctx->uring_lock)
|
|
{
|
|
struct io_notif_data *nd = io_notif_to_data(notif);
|
|
|
|
io_tx_ubuf_complete(NULL, &nd->uarg, true);
|
|
}
|
|
|
|
static inline int io_notif_account_mem(struct io_kiocb *notif, unsigned len)
|
|
{
|
|
struct io_ring_ctx *ctx = notif->ctx;
|
|
struct io_notif_data *nd = io_notif_to_data(notif);
|
|
unsigned nr_pages = (len >> PAGE_SHIFT) + 2;
|
|
int ret;
|
|
|
|
if (ctx->user) {
|
|
ret = __io_account_mem(ctx->user, nr_pages);
|
|
if (ret)
|
|
return ret;
|
|
nd->account_pages += nr_pages;
|
|
}
|
|
return 0;
|
|
}
|