Source-Changes-HG archive
[Date Prev][Date Next][Thread Prev][Thread Next][Date Index][Thread Index][Old Index]
[src/trunk]: src/sys/net Do pppoe_timeout() in thread context
details: https://anonhg.NetBSD.org/src/rev/7746ae8e86d8
branches: trunk
changeset: 1014347:7746ae8e86d8
user: yamaguchi <yamaguchi%NetBSD.org@localhost>
date: Fri Sep 18 09:53:50 2020 +0000
description:
Do pppoe_timeout() in thread context
OKed by knakahara@n.o
fix port-amd64/55661
diffstat:
sys/net/if_pppoe.c | 60 +++++++++++++++++++++++++++++++++++++++++++++++------
1 files changed, 53 insertions(+), 7 deletions(-)
diffs (140 lines):
diff -r 436ae686ab8a -r 7746ae8e86d8 sys/net/if_pppoe.c
--- a/sys/net/if_pppoe.c Fri Sep 18 09:48:56 2020 +0000
+++ b/sys/net/if_pppoe.c Fri Sep 18 09:53:50 2020 +0000
@@ -1,4 +1,4 @@
-/* $NetBSD: if_pppoe.c,v 1.150 2020/09/18 09:48:56 yamaguchi Exp $ */
+/* $NetBSD: if_pppoe.c,v 1.151 2020/09/18 09:53:50 yamaguchi Exp $ */
/*
* Copyright (c) 2002, 2008 The NetBSD Foundation, Inc.
@@ -30,7 +30,7 @@
*/
#include <sys/cdefs.h>
-__KERNEL_RCSID(0, "$NetBSD: if_pppoe.c,v 1.150 2020/09/18 09:48:56 yamaguchi Exp $");
+__KERNEL_RCSID(0, "$NetBSD: if_pppoe.c,v 1.151 2020/09/18 09:53:50 yamaguchi Exp $");
#ifdef _KERNEL_OPT
#include "pppoe.h"
@@ -41,6 +41,7 @@
#include <sys/param.h>
#include <sys/systm.h>
#include <sys/kernel.h>
+#include <sys/atomic.h>
#include <sys/callout.h>
#include <sys/malloc.h>
#include <sys/mbuf.h>
@@ -57,6 +58,7 @@
#include <sys/mutex.h>
#include <sys/psref.h>
#include <sys/cprng.h>
+#include <sys/workqueue.h>
#include <net/if.h>
#include <net/if_types.h>
@@ -173,7 +175,10 @@
uint8_t *sc_hunique; /* content of host unique we must echo back */
size_t sc_hunique_len; /* length of host unique */
#endif
- callout_t sc_timeout; /* timeout while not in session state */
+ callout_t sc_timeout; /* timeout while not in session state */
+ struct workqueue *sc_timeout_wq; /* workqueue for timeout */
+ struct work sc_timeout_wk;
+ u_int sc_timeout_scheduled;
int sc_padi_retried; /* number of PADI retries already done */
int sc_padr_retried; /* number of PADR retries already done */
krwlock_t sc_lock; /* lock of sc_state, sc_session, and sc_eth_if */
@@ -209,7 +214,10 @@
static void pppoe_clear_softc(struct pppoe_softc *, const char *);
/* internal timeout handling */
-static void pppoe_timeout(void *);
+static void pppoe_timeout_co(void *);
+static void pppoe_timeout_co_halt(void *);
+static void pppoe_timeout_wk(struct work *, void *);
+static void pppoe_timeout(struct pppoe_softc *);
/* sending actual protocol controll packets */
static int pppoe_send_padi(struct pppoe_softc *);
@@ -349,8 +357,16 @@
/* changed to real address later */
memcpy(&sc->sc_dest, etherbroadcastaddr, sizeof(sc->sc_dest));
+ rv = workqueue_create(&sc->sc_timeout_wq,
+ sc->sc_sppp.pp_if.if_xname, pppoe_timeout_wk, sc,
+ PRI_SOFTNET, IPL_SOFTNET, 0);
+ if (rv != 0) {
+ free(sc, M_DEVBUF);
+ return rv;
+ }
+
callout_init(&sc->sc_timeout, CALLOUT_MPSAFE);
- callout_setfunc(&sc->sc_timeout, pppoe_timeout, sc);
+ callout_setfunc(&sc->sc_timeout, pppoe_timeout_co, sc);
sc->sc_sppp.pp_if.if_start = pppoe_start;
#ifdef PPPOE_MPSAFE
@@ -362,6 +378,7 @@
rv = if_initialize(&sc->sc_sppp.pp_if);
if (rv != 0) {
+ workqueue_destroy(sc->sc_timeout_wq);
callout_halt(&sc->sc_timeout, NULL);
callout_destroy(&sc->sc_timeout);
free(sc, M_DEVBUF);
@@ -395,6 +412,8 @@
rw_enter(&pppoe_softc_list_lock, RW_WRITER);
PPPOE_LOCK(sc, RW_WRITER);
+ callout_setfunc(&sc->sc_timeout, pppoe_timeout_co_halt, sc);
+ workqueue_wait(sc->sc_timeout_wq, &sc->sc_timeout_wk);
callout_halt(&sc->sc_timeout, NULL);
LIST_REMOVE(sc, sc_list);
@@ -416,6 +435,7 @@
if (sc->sc_relay_sid)
free(sc->sc_relay_sid, M_DEVBUF);
callout_destroy(&sc->sc_timeout);
+ workqueue_destroy(sc->sc_timeout_wq);
PPPOE_UNLOCK(sc);
rw_destroy(&sc->sc_lock);
@@ -1411,10 +1431,36 @@
}
static void
-pppoe_timeout(void *arg)
+pppoe_timeout_co(void *arg)
+{
+ struct pppoe_softc *sc = (struct pppoe_softc *)arg;
+
+ if (atomic_swap_uint(&sc->sc_timeout_scheduled, 1) != 0)
+ return;
+
+ workqueue_enqueue(sc->sc_timeout_wq, &sc->sc_timeout_wk, NULL);
+}
+
+static void
+pppoe_timeout_co_halt(void *unused __unused)
+{
+
+ /* do nothing to halt callout safely */
+}
+
+static void
+pppoe_timeout_wk(struct work *wk __unused, void *arg)
+{
+ struct pppoe_softc *sc = (struct pppoe_softc *)arg;
+
+ atomic_swap_uint(&sc->sc_timeout_scheduled, 0);
+ pppoe_timeout(sc);
+}
+
+static void
+pppoe_timeout(struct pppoe_softc *sc)
{
int retry_wait, err;
- struct pppoe_softc *sc = (struct pppoe_softc*)arg;
DECLARE_SPLNET_VARIABLE;
#ifdef PPPOE_DEBUG
Home |
Main Index |
Thread Index |
Old Index