Commit 51d8b1a6 authored by Patrick McHardy's avatar Patrick McHardy Committed by David S. Miller

[NETFILTER]: Fix ip6_tables protocol bypass bug

As reported by Mark Dowd <Mark_Dowd@McAfee.com>, ip6_tables is susceptible
to a fragmentation attack causing false negatives on protocol matches.

When the protocol header doesn't follow the fragment header immediately,
the fragment header contains the protocol number of the next extension
header. When the extension header and the protocol header are sent in
a second fragment a rule like "ip6tables .. -p udp -j DROP" will never
match.

Drop fragments that are at offset 0 and don't contain the final protocol
header regardless of the ruleset, since this should not happen normally.

With help from Yasuyuki KOZAKAI <yasuyuki.kozakai@toshiba.co.jp>.
Signed-off-by: default avatarPatrick McHardy <kaber@trash.net>
Signed-off-by: default avatarDavid S. Miller <davem@davemloft.net>
parent 2fab22f2
...@@ -111,7 +111,7 @@ ip6_packet_match(const struct sk_buff *skb, ...@@ -111,7 +111,7 @@ ip6_packet_match(const struct sk_buff *skb,
const char *outdev, const char *outdev,
const struct ip6t_ip6 *ip6info, const struct ip6t_ip6 *ip6info,
unsigned int *protoff, unsigned int *protoff,
int *fragoff) int *fragoff, int *hotdrop)
{ {
size_t i; size_t i;
unsigned long ret; unsigned long ret;
...@@ -169,9 +169,11 @@ ip6_packet_match(const struct sk_buff *skb, ...@@ -169,9 +169,11 @@ ip6_packet_match(const struct sk_buff *skb,
unsigned short _frag_off; unsigned short _frag_off;
protohdr = ipv6_find_hdr(skb, protoff, -1, &_frag_off); protohdr = ipv6_find_hdr(skb, protoff, -1, &_frag_off);
if (protohdr < 0) if (protohdr < 0) {
if (_frag_off == 0)
*hotdrop = 1;
return 0; return 0;
}
*fragoff = _frag_off; *fragoff = _frag_off;
dprintf("Packet protocol %hi ?= %s%hi.\n", dprintf("Packet protocol %hi ?= %s%hi.\n",
...@@ -290,7 +292,7 @@ ip6t_do_table(struct sk_buff **pskb, ...@@ -290,7 +292,7 @@ ip6t_do_table(struct sk_buff **pskb,
IP_NF_ASSERT(e); IP_NF_ASSERT(e);
IP_NF_ASSERT(back); IP_NF_ASSERT(back);
if (ip6_packet_match(*pskb, indev, outdev, &e->ipv6, if (ip6_packet_match(*pskb, indev, outdev, &e->ipv6,
&protoff, &offset)) { &protoff, &offset, &hotdrop)) {
struct ip6t_entry_target *t; struct ip6t_entry_target *t;
if (IP6T_MATCH_ITERATE(e, do_match, if (IP6T_MATCH_ITERATE(e, do_match,
......
Markdown is supported
0%
or
You are about to add 0 people to the discussion. Proceed with caution.
Finish editing this message first!
Please register or to comment