mirror of
https://github.com/openwrt/openwrt.git
synced 2025-01-25 21:59:32 +00:00
c0cb86e1d5
Rather than using the clunky, old, slower wireguard-linux-compat out of tree module, this commit does a patch-by-patch backport of upstream's wireguard to 5.4. This specific backport is in widespread use, being part of SUSE's enterprise kernel, Oracle's enterprise kernel, Google's Android kernel, Gentoo's distro kernel, and probably more I've forgotten about. It's definately the "more proper" way of adding wireguard to a kernel than the ugly compat.h hell of the wireguard-linux-compat repo. And most importantly for OpenWRT, it allows using the same module configuration code for 5.10 as for 5.4, with no need for bifurcation. These patches are from the backport tree which is maintained in the open here: https://git.zx2c4.com/wireguard-linux/log/?h=backport-5.4.y I'll be sending PRs to update this as needed. Signed-off-by: Jason A. Donenfeld <Jason@zx2c4.com> (cherry picked from commit 3888fa78802354ab7bbd19b7d061fd80a16ce06b) (cherry picked from commit d54072587146dd0db9bb52b513234d944edabda3) (cherry picked from commit 196f3d586f11d96ba4ab60068cfb12420bcd20fd) (cherry picked from commit 3500fd7938a6d0c0e320295f0aa2fa34b1ebc08d) (cherry picked from commit 23b801d3ba57e34cc609ea40982c7fbed08164e9) (cherry picked from commit 0c0cb97da7f5cc06919449131dd57ed805f8f78d) (cherry picked from commit 2a27f6f90a430342cdbe84806e8b10acff446a2d) Signed-off-by: Ilya Lipnitskiy <ilya.lipnitskiy@gmail.com>
163 lines
6.9 KiB
Diff
163 lines
6.9 KiB
Diff
From 0000000000000000000000000000000000000000 Mon Sep 17 00:00:00 2001
|
|
From: "Jason A. Donenfeld" <Jason@zx2c4.com>
|
|
Date: Wed, 6 May 2020 15:33:03 -0600
|
|
Subject: [PATCH] wireguard: socket: remove errant restriction on looping to
|
|
self
|
|
|
|
commit b673e24aad36981f327a6570412ffa7754de8911 upstream.
|
|
|
|
It's already possible to create two different interfaces and loop
|
|
packets between them. This has always been possible with tunnels in the
|
|
kernel, and isn't specific to wireguard. Therefore, the networking stack
|
|
already needs to deal with that. At the very least, the packet winds up
|
|
exceeding the MTU and is discarded at that point. So, since this is
|
|
already something that happens, there's no need to forbid the not very
|
|
exceptional case of routing a packet back to the same interface; this
|
|
loop is no different than others, and we shouldn't special case it, but
|
|
rather rely on generic handling of loops in general. This also makes it
|
|
easier to do interesting things with wireguard such as onion routing.
|
|
|
|
At the same time, we add a selftest for this, ensuring that both onion
|
|
routing works and infinite routing loops do not crash the kernel. We
|
|
also add a test case for wireguard interfaces nesting packets and
|
|
sending traffic between each other, as well as the loop in this case
|
|
too. We make sure to send some throughput-heavy traffic for this use
|
|
case, to stress out any possible recursion issues with the locks around
|
|
workqueues.
|
|
|
|
Fixes: e7096c131e51 ("net: WireGuard secure network tunnel")
|
|
Signed-off-by: Jason A. Donenfeld <Jason@zx2c4.com>
|
|
Signed-off-by: David S. Miller <davem@davemloft.net>
|
|
Signed-off-by: Jason A. Donenfeld <Jason@zx2c4.com>
|
|
---
|
|
drivers/net/wireguard/socket.c | 12 -----
|
|
tools/testing/selftests/wireguard/netns.sh | 54 ++++++++++++++++++++--
|
|
2 files changed, 51 insertions(+), 15 deletions(-)
|
|
|
|
--- a/drivers/net/wireguard/socket.c
|
|
+++ b/drivers/net/wireguard/socket.c
|
|
@@ -76,12 +76,6 @@ static int send4(struct wg_device *wg, s
|
|
net_dbg_ratelimited("%s: No route to %pISpfsc, error %d\n",
|
|
wg->dev->name, &endpoint->addr, ret);
|
|
goto err;
|
|
- } else if (unlikely(rt->dst.dev == skb->dev)) {
|
|
- ip_rt_put(rt);
|
|
- ret = -ELOOP;
|
|
- net_dbg_ratelimited("%s: Avoiding routing loop to %pISpfsc\n",
|
|
- wg->dev->name, &endpoint->addr);
|
|
- goto err;
|
|
}
|
|
if (cache)
|
|
dst_cache_set_ip4(cache, &rt->dst, fl.saddr);
|
|
@@ -149,12 +143,6 @@ static int send6(struct wg_device *wg, s
|
|
net_dbg_ratelimited("%s: No route to %pISpfsc, error %d\n",
|
|
wg->dev->name, &endpoint->addr, ret);
|
|
goto err;
|
|
- } else if (unlikely(dst->dev == skb->dev)) {
|
|
- dst_release(dst);
|
|
- ret = -ELOOP;
|
|
- net_dbg_ratelimited("%s: Avoiding routing loop to %pISpfsc\n",
|
|
- wg->dev->name, &endpoint->addr);
|
|
- goto err;
|
|
}
|
|
if (cache)
|
|
dst_cache_set_ip6(cache, dst, &fl.saddr);
|
|
--- a/tools/testing/selftests/wireguard/netns.sh
|
|
+++ b/tools/testing/selftests/wireguard/netns.sh
|
|
@@ -48,8 +48,11 @@ cleanup() {
|
|
exec 2>/dev/null
|
|
printf "$orig_message_cost" > /proc/sys/net/core/message_cost
|
|
ip0 link del dev wg0
|
|
+ ip0 link del dev wg1
|
|
ip1 link del dev wg0
|
|
+ ip1 link del dev wg1
|
|
ip2 link del dev wg0
|
|
+ ip2 link del dev wg1
|
|
local to_kill="$(ip netns pids $netns0) $(ip netns pids $netns1) $(ip netns pids $netns2)"
|
|
[[ -n $to_kill ]] && kill $to_kill
|
|
pp ip netns del $netns1
|
|
@@ -77,18 +80,20 @@ ip0 link set wg0 netns $netns2
|
|
key1="$(pp wg genkey)"
|
|
key2="$(pp wg genkey)"
|
|
key3="$(pp wg genkey)"
|
|
+key4="$(pp wg genkey)"
|
|
pub1="$(pp wg pubkey <<<"$key1")"
|
|
pub2="$(pp wg pubkey <<<"$key2")"
|
|
pub3="$(pp wg pubkey <<<"$key3")"
|
|
+pub4="$(pp wg pubkey <<<"$key4")"
|
|
psk="$(pp wg genpsk)"
|
|
[[ -n $key1 && -n $key2 && -n $psk ]]
|
|
|
|
configure_peers() {
|
|
ip1 addr add 192.168.241.1/24 dev wg0
|
|
- ip1 addr add fd00::1/24 dev wg0
|
|
+ ip1 addr add fd00::1/112 dev wg0
|
|
|
|
ip2 addr add 192.168.241.2/24 dev wg0
|
|
- ip2 addr add fd00::2/24 dev wg0
|
|
+ ip2 addr add fd00::2/112 dev wg0
|
|
|
|
n1 wg set wg0 \
|
|
private-key <(echo "$key1") \
|
|
@@ -230,9 +235,38 @@ n1 ping -W 1 -c 1 192.168.241.2
|
|
n1 wg set wg0 private-key <(echo "$key3")
|
|
n2 wg set wg0 peer "$pub3" preshared-key <(echo "$psk") allowed-ips 192.168.241.1/32 peer "$pub1" remove
|
|
n1 ping -W 1 -c 1 192.168.241.2
|
|
+n2 wg set wg0 peer "$pub3" remove
|
|
|
|
-ip1 link del wg0
|
|
+# Test that we can route wg through wg
|
|
+ip1 addr flush dev wg0
|
|
+ip2 addr flush dev wg0
|
|
+ip1 addr add fd00::5:1/112 dev wg0
|
|
+ip2 addr add fd00::5:2/112 dev wg0
|
|
+n1 wg set wg0 private-key <(echo "$key1") peer "$pub2" preshared-key <(echo "$psk") allowed-ips fd00::5:2/128 endpoint 127.0.0.1:2
|
|
+n2 wg set wg0 private-key <(echo "$key2") listen-port 2 peer "$pub1" preshared-key <(echo "$psk") allowed-ips fd00::5:1/128 endpoint 127.212.121.99:9998
|
|
+ip1 link add wg1 type wireguard
|
|
+ip2 link add wg1 type wireguard
|
|
+ip1 addr add 192.168.241.1/24 dev wg1
|
|
+ip1 addr add fd00::1/112 dev wg1
|
|
+ip2 addr add 192.168.241.2/24 dev wg1
|
|
+ip2 addr add fd00::2/112 dev wg1
|
|
+ip1 link set mtu 1340 up dev wg1
|
|
+ip2 link set mtu 1340 up dev wg1
|
|
+n1 wg set wg1 listen-port 5 private-key <(echo "$key3") peer "$pub4" allowed-ips 192.168.241.2/32,fd00::2/128 endpoint [fd00::5:2]:5
|
|
+n2 wg set wg1 listen-port 5 private-key <(echo "$key4") peer "$pub3" allowed-ips 192.168.241.1/32,fd00::1/128 endpoint [fd00::5:1]:5
|
|
+tests
|
|
+# Try to set up a routing loop between the two namespaces
|
|
+ip1 link set netns $netns0 dev wg1
|
|
+ip0 addr add 192.168.241.1/24 dev wg1
|
|
+ip0 link set up dev wg1
|
|
+n0 ping -W 1 -c 1 192.168.241.2
|
|
+n1 wg set wg0 peer "$pub2" endpoint 192.168.241.2:7
|
|
ip2 link del wg0
|
|
+ip2 link del wg1
|
|
+! n0 ping -W 1 -c 10 -f 192.168.241.2 || false # Should not crash kernel
|
|
+
|
|
+ip0 link del wg1
|
|
+ip1 link del wg0
|
|
|
|
# Test using NAT. We now change the topology to this:
|
|
# ┌────────────────────────────────────────┐ ┌────────────────────────────────────────────────┐ ┌────────────────────────────────────────┐
|
|
@@ -282,6 +316,20 @@ pp sleep 3
|
|
n2 ping -W 1 -c 1 192.168.241.1
|
|
n1 wg set wg0 peer "$pub2" persistent-keepalive 0
|
|
|
|
+# Test that onion routing works, even when it loops
|
|
+n1 wg set wg0 peer "$pub3" allowed-ips 192.168.242.2/32 endpoint 192.168.241.2:5
|
|
+ip1 addr add 192.168.242.1/24 dev wg0
|
|
+ip2 link add wg1 type wireguard
|
|
+ip2 addr add 192.168.242.2/24 dev wg1
|
|
+n2 wg set wg1 private-key <(echo "$key3") listen-port 5 peer "$pub1" allowed-ips 192.168.242.1/32
|
|
+ip2 link set wg1 up
|
|
+n1 ping -W 1 -c 1 192.168.242.2
|
|
+ip2 link del wg1
|
|
+n1 wg set wg0 peer "$pub3" endpoint 192.168.242.2:5
|
|
+! n1 ping -W 1 -c 1 192.168.242.2 || false # Should not crash kernel
|
|
+n1 wg set wg0 peer "$pub3" remove
|
|
+ip1 addr del 192.168.242.1/24 dev wg0
|
|
+
|
|
# Do a wg-quick(8)-style policy routing for the default route, making sure vethc has a v6 address to tease out bugs.
|
|
ip1 -6 addr add fc00::9/96 dev vethc
|
|
ip1 -6 route add default via fc00::1
|