diff options
author | Florian Westphal <fw@strlen.de> | 2018-07-25 15:38:43 -0400 |
---|---|---|
committer | Pablo Neira Ayuso <pablo@netfilter.org> | 2018-08-16 13:36:58 -0400 |
commit | da786717e0894886301ed2536843c13f9e8fd53e (patch) | |
tree | e5f1e5a4799f6381e5edf6b9932358caf71b7c1d | |
parent | b71ed54dc2a1dbb4328f7d7c98d712747aee92ba (diff) |
netfilter: ip6t_rpfilter: set F_IFACE for linklocal addresses
Roman reports that DHCPv6 client no longer sees replies from server
due to
ip6tables -t raw -A PREROUTING -m rpfilter --invert -j DROP
rule. We need to set the F_IFACE flag for linklocal addresses, they
are scoped per-device.
Fixes: 47b7e7f82802 ("netfilter: don't set F_IFACE on ipv6 fib lookups")
Reported-by: Roman Mamedov <rm@romanrm.net>
Tested-by: Roman Mamedov <rm@romanrm.net>
Signed-off-by: Florian Westphal <fw@strlen.de>
Signed-off-by: Pablo Neira Ayuso <pablo@netfilter.org>
-rw-r--r-- | net/ipv6/netfilter/ip6t_rpfilter.c | 12 |
1 files changed, 11 insertions, 1 deletions
diff --git a/net/ipv6/netfilter/ip6t_rpfilter.c b/net/ipv6/netfilter/ip6t_rpfilter.c index 0fe61ede77c6..c3c6b09acdc4 100644 --- a/net/ipv6/netfilter/ip6t_rpfilter.c +++ b/net/ipv6/netfilter/ip6t_rpfilter.c | |||
@@ -26,6 +26,12 @@ static bool rpfilter_addr_unicast(const struct in6_addr *addr) | |||
26 | return addr_type & IPV6_ADDR_UNICAST; | 26 | return addr_type & IPV6_ADDR_UNICAST; |
27 | } | 27 | } |
28 | 28 | ||
29 | static bool rpfilter_addr_linklocal(const struct in6_addr *addr) | ||
30 | { | ||
31 | int addr_type = ipv6_addr_type(addr); | ||
32 | return addr_type & IPV6_ADDR_LINKLOCAL; | ||
33 | } | ||
34 | |||
29 | static bool rpfilter_lookup_reverse6(struct net *net, const struct sk_buff *skb, | 35 | static bool rpfilter_lookup_reverse6(struct net *net, const struct sk_buff *skb, |
30 | const struct net_device *dev, u8 flags) | 36 | const struct net_device *dev, u8 flags) |
31 | { | 37 | { |
@@ -48,7 +54,11 @@ static bool rpfilter_lookup_reverse6(struct net *net, const struct sk_buff *skb, | |||
48 | } | 54 | } |
49 | 55 | ||
50 | fl6.flowi6_mark = flags & XT_RPFILTER_VALID_MARK ? skb->mark : 0; | 56 | fl6.flowi6_mark = flags & XT_RPFILTER_VALID_MARK ? skb->mark : 0; |
51 | if ((flags & XT_RPFILTER_LOOSE) == 0) | 57 | |
58 | if (rpfilter_addr_linklocal(&iph->saddr)) { | ||
59 | lookup_flags |= RT6_LOOKUP_F_IFACE; | ||
60 | fl6.flowi6_oif = dev->ifindex; | ||
61 | } else if ((flags & XT_RPFILTER_LOOSE) == 0) | ||
52 | fl6.flowi6_oif = dev->ifindex; | 62 | fl6.flowi6_oif = dev->ifindex; |
53 | 63 | ||
54 | rt = (void *)ip6_route_lookup(net, &fl6, skb, lookup_flags); | 64 | rt = (void *)ip6_route_lookup(net, &fl6, skb, lookup_flags); |