Page Menu
Home
FreeBSD
Search
Configure Global Search
Log In
Files
F143747289
D50448.diff
No One
Temporary
Actions
View File
Edit File
Delete File
View Transforms
Subscribe
Mute Notifications
Flag For Later
Award Token
Size
3 KB
Referenced Files
None
Subscribers
None
D50448.diff
View Options
diff --git a/sys/dev/wg/if_wg.h b/sys/dev/wg/if_wg.h
--- a/sys/dev/wg/if_wg.h
+++ b/sys/dev/wg/if_wg.h
@@ -32,4 +32,10 @@
#define SIOCSWG _IOWR('i', 210, struct wg_data_io)
#define SIOCGWG _IOWR('i', 211, struct wg_data_io)
+
+/* Keep these in sync with wireguard-tools:containers.h */
+#define WGALLOWEDIP_REMOVE_ME 0x0001
+
+#define WGALLOWEDIP_VALID_FLAGS WGALLOWEDIP_REMOVE_ME
+
#endif /* __IF_WG_H__ */
diff --git a/sys/dev/wg/if_wg.c b/sys/dev/wg/if_wg.c
--- a/sys/dev/wg/if_wg.c
+++ b/sys/dev/wg/if_wg.c
@@ -314,6 +314,8 @@
static void wg_timers_run_persistent_keepalive(void *);
static int wg_aip_add(struct wg_softc *, struct wg_peer *, sa_family_t,
const void *, uint8_t);
+static int wg_aip_del(struct wg_softc *, struct wg_peer *, sa_family_t,
+ const void *, uint8_t);
static struct wg_peer *wg_aip_lookup(struct wg_softc *, sa_family_t, void *);
static void wg_aip_remove_all(struct wg_softc *, struct wg_peer *);
static struct wg_peer *wg_peer_create(struct wg_softc *,
@@ -608,6 +610,58 @@
return (ret);
}
+static int
+wg_aip_del(struct wg_softc *sc, struct wg_peer *peer, sa_family_t af,
+ const void *baddr, uint8_t cidr)
+{
+ struct radix_node_head *root = NULL;
+ struct radix_node *dnode __diagused, *node;
+ struct wg_aip *aip, addr;
+ int ret = 0;
+
+ /*
+ * We need to be sure that all padding is cleared, as it is above when
+ * new AllowedIPs are added, since we want to do a direct comparison.
+ */
+ memset(&addr, 0, sizeof(addr));
+ addr.a_af = af;
+
+ ret = wg_aip_addrinfo(&addr, baddr, cidr);
+ if (ret != 0)
+ return (ret);
+
+ root = af == AF_INET ? sc->sc_aip4 : sc->sc_aip6;
+
+ MPASS(root != NULL);
+ RADIX_NODE_HEAD_LOCK(root);
+
+ node = root->rnh_lookup(&addr.a_addr, &addr.a_mask, &root->rh);
+ if (node == NULL) {
+ RADIX_NODE_HEAD_UNLOCK(root);
+ return (0);
+ }
+
+ aip = (struct wg_aip *)node;
+ if (aip->a_peer != peer) {
+ /*
+ * They could have specified an allowed-ip that belonged to a
+ * different peer, in which case our job is done because the
+ * AllowedIP has been removed.
+ */
+ RADIX_NODE_HEAD_UNLOCK(root);
+ return (0);
+ }
+
+ dnode = root->rnh_deladdr(&aip->a_addr, &aip->a_mask, &root->rh);
+ MPASS(dnode == node);
+ RADIX_NODE_HEAD_UNLOCK(root);
+
+ LIST_REMOVE(aip, a_entry);
+ peer->p_aips_num--;
+ free(aip, M_WG);
+ return (0);
+}
+
static struct wg_peer *
wg_aip_lookup(struct wg_softc *sc, sa_family_t af, void *a)
{
@@ -2479,11 +2533,19 @@
aipl = nvlist_get_nvlist_array(nvl, "allowed-ips", &allowedip_count);
for (size_t idx = 0; idx < allowedip_count; idx++) {
sa_family_t ipaf;
+ int ipflags;
if (!nvlist_exists_number(aipl[idx], "cidr"))
continue;
ipaf = AF_UNSPEC;
+ ipflags = 0;
+ if (nvlist_exists_number(aipl[idx], "flags"))
+ ipflags = nvlist_get_number(aipl[idx], "flags");
+ if ((ipflags & ~WGALLOWEDIP_VALID_FLAGS) != 0) {
+ err = EOPNOTSUPP;
+ goto out;
+ }
cidr = nvlist_get_number(aipl[idx], "cidr");
if (nvlist_exists_binary(aipl[idx], "ipv4")) {
addr = nvlist_get_binary(aipl[idx], "ipv4", &size);
@@ -2506,7 +2568,13 @@
}
MPASS(ipaf != AF_UNSPEC);
- if ((err = wg_aip_add(sc, peer, ipaf, addr, cidr)) != 0)
+ if ((ipflags & WGALLOWEDIP_REMOVE_ME) != 0) {
+ err = wg_aip_del(sc, peer, ipaf, addr, cidr);
+ } else {
+ err = wg_aip_add(sc, peer, ipaf, addr, cidr);
+ }
+
+ if (err != 0)
goto out;
}
}
File Metadata
Details
Attached
Mime Type
text/plain
Expires
Sun, Feb 1, 10:39 AM (21 h, 43 m)
Storage Engine
blob
Storage Format
Raw Data
Storage Handle
28345942
Default Alt Text
D50448.diff (3 KB)
Attached To
Mode
D50448: kern: wg: add support for removing Allowed-IPs
Attached
Detach File
Event Timeline
Log In to Comment