]> Git Repo - linux.git/commitdiff
netfilter: reject: skip csum verification for protocols that don't support it
authorAlin Nastac <[email protected]>
Wed, 13 Feb 2019 08:14:53 +0000 (09:14 +0100)
committerPablo Neira Ayuso <[email protected]>
Wed, 13 Feb 2019 09:03:53 +0000 (10:03 +0100)
Some protocols have other means to verify the payload integrity
(AH, ESP, SCTP) while others are incompatible with nf_ip(6)_checksum
implementation because checksum is either optional or might be
partial (UDPLITE, DCCP, GRE). Because nf_ip(6)_checksum was used
to validate the packets, ip(6)tables REJECT rules were not capable
to generate ICMP(v6) errors for the protocols mentioned above.

This commit also fixes the incorrect pseudo-header protocol used
for IPv4 packets that carry other transport protocols than TCP or
UDP (pseudo-header used protocol 0 iso the proper value).

Signed-off-by: Alin Nastac <[email protected]>
Signed-off-by: Pablo Neira Ayuso <[email protected]>
include/net/netfilter/ipv4/nf_reject.h
include/net/netfilter/ipv6/nf_reject.h
include/net/netfilter/nf_reject.h [new file with mode: 0644]
net/bridge/netfilter/nft_reject_bridge.c
net/ipv4/netfilter/nf_reject_ipv4.c
net/ipv6/netfilter/nf_reject_ipv6.c

index 2eb43fcefc5057102db0c2ff2dbf0f3e23a772b1..40e0e0623f46130a4646ab027f85579244e98745 100644 (file)
@@ -5,6 +5,7 @@
 #include <linux/skbuff.h>
 #include <net/ip.h>
 #include <net/icmp.h>
+#include <net/netfilter/nf_reject.h>
 
 void nf_send_unreach(struct sk_buff *skb_in, int code, int hook);
 void nf_send_reset(struct net *net, struct sk_buff *oldskb, int hook);
index 3a5a9a36a0b2d3eadeb4f39dbc57bdd5e7566429..4a3ef9ebdf6f6e3f1fe9412afd5f7fe072b99176 100644 (file)
@@ -3,6 +3,7 @@
 #define _IPV6_NF_REJECT_H
 
 #include <linux/icmpv6.h>
+#include <net/netfilter/nf_reject.h>
 
 void nf_send_unreach6(struct net *net, struct sk_buff *skb_in, unsigned char code,
                      unsigned int hooknum);
diff --git a/include/net/netfilter/nf_reject.h b/include/net/netfilter/nf_reject.h
new file mode 100644 (file)
index 0000000..221f877
--- /dev/null
@@ -0,0 +1,27 @@
+/* SPDX-License-Identifier: GPL-2.0 */
+#ifndef _NF_REJECT_H
+#define _NF_REJECT_H
+
+static inline bool nf_reject_verify_csum(__u8 proto)
+{
+       /* Skip protocols that don't use 16-bit one's complement checksum
+        * of the entire payload.
+        */
+       switch (proto) {
+               /* Protocols with other integrity checks. */
+               case IPPROTO_AH:
+               case IPPROTO_ESP:
+               case IPPROTO_SCTP:
+
+               /* Protocols with partial checksums. */
+               case IPPROTO_UDPLITE:
+               case IPPROTO_DCCP:
+
+               /* Protocols with optional checksums. */
+               case IPPROTO_GRE:
+                       return false;
+       }
+       return true;
+}
+
+#endif /* _NF_REJECT_H */
index 419e8edf23ba8bb5f7626529bc873f81b0f4c63c..1b1856744c80a9dadb2748c9f16218fecbe113fc 100644 (file)
@@ -125,13 +125,10 @@ static void nft_reject_br_send_v4_unreach(struct net *net,
        if (pskb_trim_rcsum(oldskb, ntohs(ip_hdr(oldskb)->tot_len)))
                return;
 
-       if (ip_hdr(oldskb)->protocol == IPPROTO_TCP ||
-           ip_hdr(oldskb)->protocol == IPPROTO_UDP)
-               proto = ip_hdr(oldskb)->protocol;
-       else
-               proto = 0;
+       proto = ip_hdr(oldskb)->protocol;
 
        if (!skb_csum_unnecessary(oldskb) &&
+           nf_reject_verify_csum(proto) &&
            nf_ip_checksum(oldskb, hook, ip_hdrlen(oldskb), proto))
                return;
 
@@ -234,6 +231,9 @@ static bool reject6_br_csum_ok(struct sk_buff *skb, int hook)
        if (thoff < 0 || thoff >= skb->len || (fo & htons(~0x7)) != 0)
                return false;
 
+       if (!nf_reject_verify_csum(proto))
+               return true;
+
        return nf_ip6_checksum(skb, hook, thoff, proto) == 0;
 }
 
index aa8304c618b80f85d16166a00e0b93266bd9b232..7dc3c324b911bd4299aad2e7523b66d96f2a48f3 100644 (file)
@@ -173,21 +173,16 @@ EXPORT_SYMBOL_GPL(nf_send_reset);
 void nf_send_unreach(struct sk_buff *skb_in, int code, int hook)
 {
        struct iphdr *iph = ip_hdr(skb_in);
-       u8 proto;
+       u8 proto = iph->protocol;
 
        if (iph->frag_off & htons(IP_OFFSET))
                return;
 
-       if (skb_csum_unnecessary(skb_in)) {
+       if (skb_csum_unnecessary(skb_in) || !nf_reject_verify_csum(proto)) {
                icmp_send(skb_in, ICMP_DEST_UNREACH, code, 0);
                return;
        }
 
-       if (iph->protocol == IPPROTO_TCP || iph->protocol == IPPROTO_UDP)
-               proto = iph->protocol;
-       else
-               proto = 0;
-
        if (nf_ip_checksum(skb_in, hook, ip_hdrlen(skb_in), proto) == 0)
                icmp_send(skb_in, ICMP_DEST_UNREACH, code, 0);
 }
index b9c8a763c863cb720575b324670b11a60bc70bb4..02e9228641e0804b0ec95dd3efaced06843779bc 100644 (file)
@@ -233,6 +233,9 @@ static bool reject6_csum_ok(struct sk_buff *skb, int hook)
        if (thoff < 0 || thoff >= skb->len || (fo & htons(~0x7)) != 0)
                return false;
 
+       if (!nf_reject_verify_csum(proto))
+               return true;
+
        return nf_ip6_checksum(skb, hook, thoff, proto) == 0;
 }
 
This page took 0.067289 seconds and 4 git commands to generate.