# HG changeset patch # User Taylor R Campbell # Date 1599923371 0 # Sat Sep 12 15:09:31 2020 +0000 # Branch trunk # Node ID 54e5251a4b67eaa215559741d772eaebb40a33f8 # Parent 43dbd167274e8c1a09862fb04b004b14bc73b7f1 # EXP-Topic riastradh-wg wg: Add altq hooks. While here, remove the IFQ_CLASSIFY bottleneck (takes the ifq lock, so it would serialize all transmission to all peers on a single wg(4) interface). altq can be disabled at compile-time or at run-time; even if included at comple-time the run-time impact should be negligible if disabled. diff -r 43dbd167274e -r 54e5251a4b67 sys/net/if_wg.c --- a/sys/net/if_wg.c Sun Sep 13 16:49:59 2020 +0000 +++ b/sys/net/if_wg.c Sat Sep 12 15:09:31 2020 +0000 @@ -44,6 +44,7 @@ __KERNEL_RCSID(0, "$NetBSD: if_wg.c,v 1.56 2020/09/08 16:39:57 riastradh Exp $"); #ifdef _KERNEL_OPT +#include "opt_altq_enabled.h" #include "opt_inet.h" #endif @@ -702,6 +703,9 @@ static void wg_input(struct ifnet *, str static int wg_ioctl(struct ifnet *, u_long, void *); static int wg_bind_port(struct wg_softc *, const uint16_t); static int wg_init(struct ifnet *); +#ifdef ALTQ +static void wg_start(struct ifnet *); +#endif static void wg_stop(struct ifnet *, int); static void wg_peer_work(struct work *, void *); @@ -3537,11 +3541,16 @@ wg_if_attach(struct wg_softc *wg) wg->wg_if.if_ioctl = wg_ioctl; wg->wg_if.if_output = wg_output; wg->wg_if.if_init = wg_init; +#ifdef ALTQ + wg->wg_if.if_start = wg_start; +#endif wg->wg_if.if_stop = wg_stop; wg->wg_if.if_type = IFT_OTHER; wg->wg_if.if_dlt = DLT_NULL; wg->wg_if.if_softc = wg; +#ifdef ALTQ IFQ_SET_READY(&wg->wg_if.if_snd); +#endif error = if_initialize(&wg->wg_if); if (error != 0) @@ -3780,7 +3789,12 @@ wg_output(struct ifnet *ifp, struct mbuf goto out0; } - IFQ_CLASSIFY(&ifp->if_snd, m, dst->sa_family); +#ifdef ALTQ + bool altq = atomic_load_relaxed(&ifp->if_snd.altq_flags) + & ALTQF_ENABLED; + if (altq) + IFQ_CLASSIFY(&ifp->if_snd, m, dst->sa_family); +#endif bpf_mtap_af(ifp, dst->sa_family, m, BPF_D_OUT); @@ -3819,18 +3833,36 @@ wg_output(struct ifnet *ifp, struct mbuf } /* There's an established session. Toss it in the queue. */ +#ifdef ALTQ + if (altq) { + mutex_enter(ifp->if_snd.ifq_lock); + if (ALTQ_IS_ENABLED(&ifp->if_snd)) { + M_SETCTX(m, wgp); + ALTQ_ENQUEUE(&ifp->if_snd, m, error); + m = NULL; /* consume */ + } + mutex_exit(ifp->if_snd.ifq_lock); + if (m == NULL) { + wg_start(ifp); + goto out2; + } + } +#endif kpreempt_disable(); const uint32_t h = curcpu()->ci_index; // pktq_rps_hash(m) M_SETCTX(m, wgp); if (__predict_false(!pktq_enqueue(wg_pktq, m, h))) { WGLOG(LOG_ERR, "pktq full, dropping\n"); error = ENOBUFS; - goto out2; + goto out3; } m = NULL; /* consumed */ error = 0; -out2: kpreempt_enable(); - +out3: kpreempt_enable(); + +#ifdef ALTQ +out2: +#endif wg_put_session(wgs, &wgs_psref); out1: wg_put_peer(wgp, &wgp_psref); out0: if (m) @@ -4688,6 +4720,28 @@ wg_init(struct ifnet *ifp) return 0; } +#ifdef ALTQ +static void +wg_start(struct ifnet *ifp) +{ + struct mbuf *m; + + for (;;) { + IFQ_DEQUEUE(&ifp->if_snd, m); + if (m == NULL) + break; + + kpreempt_disable(); + const uint32_t h = curcpu()->ci_index; // pktq_rps_hash(m) + if (__predict_false(!pktq_enqueue(wg_pktq, m, h))) { + WGLOG(LOG_ERR, "pktq full, dropping\n"); + m_freem(m); + } + kpreempt_enable(); + } +} +#endif + static void wg_stop(struct ifnet *ifp, int disable) {