mirror of
https://github.com/openwrt/openwrt.git
synced 2025-01-07 06:18:54 +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 commit3888fa7880
) (cherry picked from commitd540725871
) (cherry picked from commit196f3d586f
) (cherry picked from commit3500fd7938
) (cherry picked from commit23b801d3ba
) (cherry picked from commit0c0cb97da7
) (cherry picked from commit2a27f6f90a
) Signed-off-by: Ilya Lipnitskiy <ilya.lipnitskiy@gmail.com>
297 lines
9.2 KiB
Diff
297 lines
9.2 KiB
Diff
From 0000000000000000000000000000000000000000 Mon Sep 17 00:00:00 2001
|
|
From: "Jason A. Donenfeld" <Jason@zx2c4.com>
|
|
Date: Tue, 23 Jun 2020 03:59:45 -0600
|
|
Subject: [PATCH] wireguard: device: avoid circular netns references
|
|
|
|
commit 900575aa33a3eaaef802b31de187a85c4a4b4bd0 upstream.
|
|
|
|
Before, we took a reference to the creating netns if the new netns was
|
|
different. This caused issues with circular references, with two
|
|
wireguard interfaces swapping namespaces. The solution is to rather not
|
|
take any extra references at all, but instead simply invalidate the
|
|
creating netns pointer when that netns is deleted.
|
|
|
|
In order to prevent this from happening again, this commit improves the
|
|
rough object leak tracking by allowing it to account for created and
|
|
destroyed interfaces, aside from just peers and keys. That then makes it
|
|
possible to check for the object leak when having two interfaces take a
|
|
reference to each others' namespaces.
|
|
|
|
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/device.c | 58 ++++++++++------------
|
|
drivers/net/wireguard/device.h | 3 +-
|
|
drivers/net/wireguard/netlink.c | 14 ++++--
|
|
drivers/net/wireguard/socket.c | 25 +++++++---
|
|
tools/testing/selftests/wireguard/netns.sh | 13 ++++-
|
|
5 files changed, 67 insertions(+), 46 deletions(-)
|
|
|
|
--- a/drivers/net/wireguard/device.c
|
|
+++ b/drivers/net/wireguard/device.c
|
|
@@ -45,17 +45,18 @@ static int wg_open(struct net_device *de
|
|
if (dev_v6)
|
|
dev_v6->cnf.addr_gen_mode = IN6_ADDR_GEN_MODE_NONE;
|
|
|
|
+ mutex_lock(&wg->device_update_lock);
|
|
ret = wg_socket_init(wg, wg->incoming_port);
|
|
if (ret < 0)
|
|
- return ret;
|
|
- mutex_lock(&wg->device_update_lock);
|
|
+ goto out;
|
|
list_for_each_entry(peer, &wg->peer_list, peer_list) {
|
|
wg_packet_send_staged_packets(peer);
|
|
if (peer->persistent_keepalive_interval)
|
|
wg_packet_send_keepalive(peer);
|
|
}
|
|
+out:
|
|
mutex_unlock(&wg->device_update_lock);
|
|
- return 0;
|
|
+ return ret;
|
|
}
|
|
|
|
#ifdef CONFIG_PM_SLEEP
|
|
@@ -225,6 +226,7 @@ static void wg_destruct(struct net_devic
|
|
list_del(&wg->device_list);
|
|
rtnl_unlock();
|
|
mutex_lock(&wg->device_update_lock);
|
|
+ rcu_assign_pointer(wg->creating_net, NULL);
|
|
wg->incoming_port = 0;
|
|
wg_socket_reinit(wg, NULL, NULL);
|
|
/* The final references are cleared in the below calls to destroy_workqueue. */
|
|
@@ -240,13 +242,11 @@ static void wg_destruct(struct net_devic
|
|
skb_queue_purge(&wg->incoming_handshakes);
|
|
free_percpu(dev->tstats);
|
|
free_percpu(wg->incoming_handshakes_worker);
|
|
- if (wg->have_creating_net_ref)
|
|
- put_net(wg->creating_net);
|
|
kvfree(wg->index_hashtable);
|
|
kvfree(wg->peer_hashtable);
|
|
mutex_unlock(&wg->device_update_lock);
|
|
|
|
- pr_debug("%s: Interface deleted\n", dev->name);
|
|
+ pr_debug("%s: Interface destroyed\n", dev->name);
|
|
free_netdev(dev);
|
|
}
|
|
|
|
@@ -292,7 +292,7 @@ static int wg_newlink(struct net *src_ne
|
|
struct wg_device *wg = netdev_priv(dev);
|
|
int ret = -ENOMEM;
|
|
|
|
- wg->creating_net = src_net;
|
|
+ rcu_assign_pointer(wg->creating_net, src_net);
|
|
init_rwsem(&wg->static_identity.lock);
|
|
mutex_init(&wg->socket_update_lock);
|
|
mutex_init(&wg->device_update_lock);
|
|
@@ -393,30 +393,26 @@ static struct rtnl_link_ops link_ops __r
|
|
.newlink = wg_newlink,
|
|
};
|
|
|
|
-static int wg_netdevice_notification(struct notifier_block *nb,
|
|
- unsigned long action, void *data)
|
|
+static void wg_netns_pre_exit(struct net *net)
|
|
{
|
|
- struct net_device *dev = ((struct netdev_notifier_info *)data)->dev;
|
|
- struct wg_device *wg = netdev_priv(dev);
|
|
-
|
|
- ASSERT_RTNL();
|
|
-
|
|
- if (action != NETDEV_REGISTER || dev->netdev_ops != &netdev_ops)
|
|
- return 0;
|
|
+ struct wg_device *wg;
|
|
|
|
- if (dev_net(dev) == wg->creating_net && wg->have_creating_net_ref) {
|
|
- put_net(wg->creating_net);
|
|
- wg->have_creating_net_ref = false;
|
|
- } else if (dev_net(dev) != wg->creating_net &&
|
|
- !wg->have_creating_net_ref) {
|
|
- wg->have_creating_net_ref = true;
|
|
- get_net(wg->creating_net);
|
|
+ rtnl_lock();
|
|
+ list_for_each_entry(wg, &device_list, device_list) {
|
|
+ if (rcu_access_pointer(wg->creating_net) == net) {
|
|
+ pr_debug("%s: Creating namespace exiting\n", wg->dev->name);
|
|
+ netif_carrier_off(wg->dev);
|
|
+ mutex_lock(&wg->device_update_lock);
|
|
+ rcu_assign_pointer(wg->creating_net, NULL);
|
|
+ wg_socket_reinit(wg, NULL, NULL);
|
|
+ mutex_unlock(&wg->device_update_lock);
|
|
+ }
|
|
}
|
|
- return 0;
|
|
+ rtnl_unlock();
|
|
}
|
|
|
|
-static struct notifier_block netdevice_notifier = {
|
|
- .notifier_call = wg_netdevice_notification
|
|
+static struct pernet_operations pernet_ops = {
|
|
+ .pre_exit = wg_netns_pre_exit
|
|
};
|
|
|
|
int __init wg_device_init(void)
|
|
@@ -429,18 +425,18 @@ int __init wg_device_init(void)
|
|
return ret;
|
|
#endif
|
|
|
|
- ret = register_netdevice_notifier(&netdevice_notifier);
|
|
+ ret = register_pernet_device(&pernet_ops);
|
|
if (ret)
|
|
goto error_pm;
|
|
|
|
ret = rtnl_link_register(&link_ops);
|
|
if (ret)
|
|
- goto error_netdevice;
|
|
+ goto error_pernet;
|
|
|
|
return 0;
|
|
|
|
-error_netdevice:
|
|
- unregister_netdevice_notifier(&netdevice_notifier);
|
|
+error_pernet:
|
|
+ unregister_pernet_device(&pernet_ops);
|
|
error_pm:
|
|
#ifdef CONFIG_PM_SLEEP
|
|
unregister_pm_notifier(&pm_notifier);
|
|
@@ -451,7 +447,7 @@ error_pm:
|
|
void wg_device_uninit(void)
|
|
{
|
|
rtnl_link_unregister(&link_ops);
|
|
- unregister_netdevice_notifier(&netdevice_notifier);
|
|
+ unregister_pernet_device(&pernet_ops);
|
|
#ifdef CONFIG_PM_SLEEP
|
|
unregister_pm_notifier(&pm_notifier);
|
|
#endif
|
|
--- a/drivers/net/wireguard/device.h
|
|
+++ b/drivers/net/wireguard/device.h
|
|
@@ -40,7 +40,7 @@ struct wg_device {
|
|
struct net_device *dev;
|
|
struct crypt_queue encrypt_queue, decrypt_queue;
|
|
struct sock __rcu *sock4, *sock6;
|
|
- struct net *creating_net;
|
|
+ struct net __rcu *creating_net;
|
|
struct noise_static_identity static_identity;
|
|
struct workqueue_struct *handshake_receive_wq, *handshake_send_wq;
|
|
struct workqueue_struct *packet_crypt_wq;
|
|
@@ -56,7 +56,6 @@ struct wg_device {
|
|
unsigned int num_peers, device_update_gen;
|
|
u32 fwmark;
|
|
u16 incoming_port;
|
|
- bool have_creating_net_ref;
|
|
};
|
|
|
|
int wg_device_init(void);
|
|
--- a/drivers/net/wireguard/netlink.c
|
|
+++ b/drivers/net/wireguard/netlink.c
|
|
@@ -517,11 +517,15 @@ static int wg_set_device(struct sk_buff
|
|
if (flags & ~__WGDEVICE_F_ALL)
|
|
goto out;
|
|
|
|
- ret = -EPERM;
|
|
- if ((info->attrs[WGDEVICE_A_LISTEN_PORT] ||
|
|
- info->attrs[WGDEVICE_A_FWMARK]) &&
|
|
- !ns_capable(wg->creating_net->user_ns, CAP_NET_ADMIN))
|
|
- goto out;
|
|
+ if (info->attrs[WGDEVICE_A_LISTEN_PORT] || info->attrs[WGDEVICE_A_FWMARK]) {
|
|
+ struct net *net;
|
|
+ rcu_read_lock();
|
|
+ net = rcu_dereference(wg->creating_net);
|
|
+ ret = !net || !ns_capable(net->user_ns, CAP_NET_ADMIN) ? -EPERM : 0;
|
|
+ rcu_read_unlock();
|
|
+ if (ret)
|
|
+ goto out;
|
|
+ }
|
|
|
|
++wg->device_update_gen;
|
|
|
|
--- a/drivers/net/wireguard/socket.c
|
|
+++ b/drivers/net/wireguard/socket.c
|
|
@@ -347,6 +347,7 @@ static void set_sock_opts(struct socket
|
|
|
|
int wg_socket_init(struct wg_device *wg, u16 port)
|
|
{
|
|
+ struct net *net;
|
|
int ret;
|
|
struct udp_tunnel_sock_cfg cfg = {
|
|
.sk_user_data = wg,
|
|
@@ -371,37 +372,47 @@ int wg_socket_init(struct wg_device *wg,
|
|
};
|
|
#endif
|
|
|
|
+ rcu_read_lock();
|
|
+ net = rcu_dereference(wg->creating_net);
|
|
+ net = net ? maybe_get_net(net) : NULL;
|
|
+ rcu_read_unlock();
|
|
+ if (unlikely(!net))
|
|
+ return -ENONET;
|
|
+
|
|
#if IS_ENABLED(CONFIG_IPV6)
|
|
retry:
|
|
#endif
|
|
|
|
- ret = udp_sock_create(wg->creating_net, &port4, &new4);
|
|
+ ret = udp_sock_create(net, &port4, &new4);
|
|
if (ret < 0) {
|
|
pr_err("%s: Could not create IPv4 socket\n", wg->dev->name);
|
|
- return ret;
|
|
+ goto out;
|
|
}
|
|
set_sock_opts(new4);
|
|
- setup_udp_tunnel_sock(wg->creating_net, new4, &cfg);
|
|
+ setup_udp_tunnel_sock(net, new4, &cfg);
|
|
|
|
#if IS_ENABLED(CONFIG_IPV6)
|
|
if (ipv6_mod_enabled()) {
|
|
port6.local_udp_port = inet_sk(new4->sk)->inet_sport;
|
|
- ret = udp_sock_create(wg->creating_net, &port6, &new6);
|
|
+ ret = udp_sock_create(net, &port6, &new6);
|
|
if (ret < 0) {
|
|
udp_tunnel_sock_release(new4);
|
|
if (ret == -EADDRINUSE && !port && retries++ < 100)
|
|
goto retry;
|
|
pr_err("%s: Could not create IPv6 socket\n",
|
|
wg->dev->name);
|
|
- return ret;
|
|
+ goto out;
|
|
}
|
|
set_sock_opts(new6);
|
|
- setup_udp_tunnel_sock(wg->creating_net, new6, &cfg);
|
|
+ setup_udp_tunnel_sock(net, new6, &cfg);
|
|
}
|
|
#endif
|
|
|
|
wg_socket_reinit(wg, new4->sk, new6 ? new6->sk : NULL);
|
|
- return 0;
|
|
+ ret = 0;
|
|
+out:
|
|
+ put_net(net);
|
|
+ return ret;
|
|
}
|
|
|
|
void wg_socket_reinit(struct wg_device *wg, struct sock *new4,
|
|
--- a/tools/testing/selftests/wireguard/netns.sh
|
|
+++ b/tools/testing/selftests/wireguard/netns.sh
|
|
@@ -587,9 +587,20 @@ ip0 link set wg0 up
|
|
kill $ncat_pid
|
|
ip0 link del wg0
|
|
|
|
+# Ensure there aren't circular reference loops
|
|
+ip1 link add wg1 type wireguard
|
|
+ip2 link add wg2 type wireguard
|
|
+ip1 link set wg1 netns $netns2
|
|
+ip2 link set wg2 netns $netns1
|
|
+pp ip netns delete $netns1
|
|
+pp ip netns delete $netns2
|
|
+pp ip netns add $netns1
|
|
+pp ip netns add $netns2
|
|
+
|
|
+sleep 2 # Wait for cleanup and grace periods
|
|
declare -A objects
|
|
while read -t 0.1 -r line 2>/dev/null || [[ $? -ne 142 ]]; do
|
|
- [[ $line =~ .*(wg[0-9]+:\ [A-Z][a-z]+\ [0-9]+)\ .*(created|destroyed).* ]] || continue
|
|
+ [[ $line =~ .*(wg[0-9]+:\ [A-Z][a-z]+\ ?[0-9]*)\ .*(created|destroyed).* ]] || continue
|
|
objects["${BASH_REMATCH[1]}"]+="${BASH_REMATCH[2]}"
|
|
done < /dev/kmsg
|
|
alldeleted=1
|