]> www.infradead.org Git - users/hch/misc.git/commitdiff
netpoll: Optimize skb refilling on critical path
authorBreno Leitao <leitao@debian.org>
Tue, 4 Mar 2025 15:50:41 +0000 (07:50 -0800)
committerJakub Kicinski <kuba@kernel.org>
Sat, 8 Mar 2025 03:55:40 +0000 (19:55 -0800)
netpoll tries to refill the skb queue on every packet send, independently
if packets are being consumed from the pool or not. This was
particularly problematic while being called from printk(), where the
operation would be done while holding the console lock.

Introduce a more intelligent approach to skb queue management. Instead
of constantly attempting to refill the queue, the system now defers
refilling to a work queue and only triggers the workqueue when a buffer
is actually dequeued. This change significantly reduces operations with
the lock held.

Add a work_struct to the netpoll structure for asynchronous refilling,
updating find_skb() to schedule refill work only when necessary (skb is
dequeued).

These changes have demonstrated a 15% reduction in time spent during
netpoll_send_msg operations, especially when no SKBs are not consumed
from consumed from pool.

When SKBs are being dequeued, the improvement is even better, around
70%, mainly because refilling the SKB pool is now happening outside of
the critical patch (with console_owner lock held).

Signed-off-by: Breno Leitao <leitao@debian.org>
Reviewed-by: Simon Horman <horms@kernel.org>
Link: https://patch.msgid.link/20250304-netpoll_refill_v2-v1-1-06e2916a4642@debian.org
Signed-off-by: Jakub Kicinski <kuba@kernel.org>
include/linux/netpoll.h
net/core/netpoll.c

index f91e50a76efd4b016381c632456397eea1ea877f..f6e8abe0b1f19a34a35ec976c127be468d863028 100644 (file)
@@ -33,6 +33,7 @@ struct netpoll {
        u16 local_port, remote_port;
        u8 remote_mac[ETH_ALEN];
        struct sk_buff_head skb_pool;
+       struct work_struct refill_wq;
 };
 
 struct netpoll_info {
index 62b4041aae1ae8c7dc47c89fb40b14bbd4ad0e0e..8a0df2b274a88213f4ca4632ecbb66d74ac960a9 100644 (file)
@@ -284,12 +284,13 @@ static struct sk_buff *find_skb(struct netpoll *np, int len, int reserve)
        struct sk_buff *skb;
 
        zap_completion_queue();
-       refill_skbs(np);
 repeat:
 
        skb = alloc_skb(len, GFP_ATOMIC);
-       if (!skb)
+       if (!skb) {
                skb = skb_dequeue(&np->skb_pool);
+               schedule_work(&np->refill_wq);
+       }
 
        if (!skb) {
                if (++count < 10) {
@@ -535,6 +536,7 @@ static void skb_pool_flush(struct netpoll *np)
 {
        struct sk_buff_head *skb_pool;
 
+       cancel_work_sync(&np->refill_wq);
        skb_pool = &np->skb_pool;
        skb_queue_purge_reason(skb_pool, SKB_CONSUMED);
 }
@@ -621,6 +623,14 @@ int netpoll_parse_options(struct netpoll *np, char *opt)
 }
 EXPORT_SYMBOL(netpoll_parse_options);
 
+static void refill_skbs_work_handler(struct work_struct *work)
+{
+       struct netpoll *np =
+               container_of(work, struct netpoll, refill_wq);
+
+       refill_skbs(np);
+}
+
 int __netpoll_setup(struct netpoll *np, struct net_device *ndev)
 {
        struct netpoll_info *npinfo;
@@ -666,6 +676,7 @@ int __netpoll_setup(struct netpoll *np, struct net_device *ndev)
 
        /* fill up the skb queue */
        refill_skbs(np);
+       INIT_WORK(&np->refill_wq, refill_skbs_work_handler);
 
        /* last thing to do is link it to the net device structure */
        rcu_assign_pointer(ndev->npinfo, npinfo);