]> git.ipfire.org Git - thirdparty/kernel/linux.git/commitdiff
netfilter: nf_dup_netdev: Move the recursion counter struct netdev_xmit
authorSebastian Andrzej Siewior <bigeasy@linutronix.de>
Mon, 12 May 2025 10:28:46 +0000 (12:28 +0200)
committerPablo Neira Ayuso <pablo@netfilter.org>
Fri, 23 May 2025 11:57:12 +0000 (13:57 +0200)
nf_dup_skb_recursion is a per-CPU variable and relies on disabled BH for its
locking. Without per-CPU locking in local_bh_disable() on PREEMPT_RT
this data structure requires explicit locking.

Move nf_dup_skb_recursion to struct netdev_xmit, provide wrappers.

Signed-off-by: Sebastian Andrzej Siewior <bigeasy@linutronix.de>
Signed-off-by: Pablo Neira Ayuso <pablo@netfilter.org>
include/linux/netdevice_xmit.h
net/netfilter/nf_dup_netdev.c

index 848735b3a7c02d934bb037f89100eeecd1b5466e..813a19122ebbb2c6a04176330b1055b7c2b9c902 100644 (file)
@@ -11,6 +11,9 @@ struct netdev_xmit {
 #if IS_ENABLED(CONFIG_NET_ACT_MIRRED)
        u8 sched_mirred_nest;
 #endif
+#if IS_ENABLED(CONFIG_NF_DUP_NETDEV)
+       u8 nf_dup_skb_recursion;
+#endif
 };
 
 #endif
index a8e2425e43b0d755cc5cfc5a14c0e42785dc3849..fab8b9011098f46a00580d2206e9ddb5bb47cbbc 100644 (file)
 
 #define NF_RECURSION_LIMIT     2
 
-static DEFINE_PER_CPU(u8, nf_dup_skb_recursion);
+#ifndef CONFIG_PREEMPT_RT
+static u8 *nf_get_nf_dup_skb_recursion(void)
+{
+       return this_cpu_ptr(&softnet_data.xmit.nf_dup_skb_recursion);
+}
+#else
+
+static u8 *nf_get_nf_dup_skb_recursion(void)
+{
+       return &current->net_xmit.nf_dup_skb_recursion;
+}
+
+#endif
 
 static void nf_do_netdev_egress(struct sk_buff *skb, struct net_device *dev,
                                enum nf_dev_hooks hook)
 {
-       if (__this_cpu_read(nf_dup_skb_recursion) > NF_RECURSION_LIMIT)
+       u8 *nf_dup_skb_recursion = nf_get_nf_dup_skb_recursion();
+
+       if (*nf_dup_skb_recursion > NF_RECURSION_LIMIT)
                goto err;
 
        if (hook == NF_NETDEV_INGRESS && skb_mac_header_was_set(skb)) {
@@ -32,9 +46,9 @@ static void nf_do_netdev_egress(struct sk_buff *skb, struct net_device *dev,
 
        skb->dev = dev;
        skb_clear_tstamp(skb);
-       __this_cpu_inc(nf_dup_skb_recursion);
+       (*nf_dup_skb_recursion)++;
        dev_queue_xmit(skb);
-       __this_cpu_dec(nf_dup_skb_recursion);
+       (*nf_dup_skb_recursion)--;
        return;
 err:
        kfree_skb(skb);