]> git.proxmox.com Git - mirror_ubuntu-artful-kernel.git/commitdiff
netfilter: add and use nf_fwd_netdev_egress
authorFlorian Westphal <fw@strlen.de>
Mon, 28 Nov 2016 10:40:06 +0000 (11:40 +0100)
committerPablo Neira Ayuso <pablo@netfilter.org>
Tue, 6 Dec 2016 20:48:22 +0000 (21:48 +0100)
... so we can use current skb instead of working with a clone.

Signed-off-by: Florian Westphal <fw@strlen.de>
Signed-off-by: Pablo Neira Ayuso <pablo@netfilter.org>
include/net/netfilter/nf_dup_netdev.h
net/netfilter/nf_dup_netdev.c
net/netfilter/nft_fwd_netdev.c

index 397dcae349f9cd02c9e1c773bd0126fae3fbd5db..3e919356bedf00aeabd74e331d5c3070c14e74eb 100644 (file)
@@ -2,5 +2,6 @@
 #define _NF_DUP_NETDEV_H_
 
 void nf_dup_netdev_egress(const struct nft_pktinfo *pkt, int oif);
+void nf_fwd_netdev_egress(const struct nft_pktinfo *pkt, int oif);
 
 #endif
index 44ae986c383f1312c4f17f0d27e7d05470aced27..c9d7f95768ab35e26bc88c4c2350d792e78da9ba 100644 (file)
 #include <linux/netfilter/nf_tables.h>
 #include <net/netfilter/nf_tables.h>
 
+static void nf_do_netdev_egress(struct sk_buff *skb, struct net_device *dev)
+{
+       if (skb_mac_header_was_set(skb))
+               skb_push(skb, skb->mac_len);
+
+       skb->dev = dev;
+       dev_queue_xmit(skb);
+}
+
+void nf_fwd_netdev_egress(const struct nft_pktinfo *pkt, int oif)
+{
+       struct net_device *dev;
+
+       dev = dev_get_by_index_rcu(nft_net(pkt), oif);
+       if (!dev) {
+               kfree_skb(pkt->skb);
+               return;
+       }
+
+       nf_do_netdev_egress(pkt->skb, dev);
+}
+EXPORT_SYMBOL_GPL(nf_fwd_netdev_egress);
+
 void nf_dup_netdev_egress(const struct nft_pktinfo *pkt, int oif)
 {
        struct net_device *dev;
@@ -24,14 +47,8 @@ void nf_dup_netdev_egress(const struct nft_pktinfo *pkt, int oif)
                return;
 
        skb = skb_clone(pkt->skb, GFP_ATOMIC);
-       if (skb == NULL)
-               return;
-
-       if (skb_mac_header_was_set(skb))
-               skb_push(skb, skb->mac_len);
-
-       skb->dev = dev;
-       dev_queue_xmit(skb);
+       if (skb)
+               nf_do_netdev_egress(skb, dev);
 }
 EXPORT_SYMBOL_GPL(nf_dup_netdev_egress);
 
index 763ebc3e0b2b58d823518eb5036092a4030db47f..ce13a50b91893fd67a9c8586f7cca53278b1067d 100644 (file)
@@ -26,8 +26,8 @@ static void nft_fwd_netdev_eval(const struct nft_expr *expr,
        struct nft_fwd_netdev *priv = nft_expr_priv(expr);
        int oif = regs->data[priv->sreg_dev];
 
-       nf_dup_netdev_egress(pkt, oif);
-       regs->verdict.code = NF_DROP;
+       nf_fwd_netdev_egress(pkt, oif);
+       regs->verdict.code = NF_STOLEN;
 }
 
 static const struct nla_policy nft_fwd_netdev_policy[NFTA_FWD_MAX + 1] = {