Source-Changes-HG archive

[Date Prev][Date Next][Thread Prev][Thread Next][Date Index][Thread Index][Old Index]

[src/trunk]: src/sys/net/lagg Fix to acquire LAGG_LOCK without psref



details:   https://anonhg.NetBSD.org/src/rev/04759e3edcd4
branches:  trunk
changeset: 987515:04759e3edcd4
user:      yamaguchi <yamaguchi%NetBSD.org@localhost>
date:      Thu Sep 30 04:20:14 2021 +0000

description:
Fix to acquire LAGG_LOCK without psref
to remove possibility of deadlock

the deadlock maybe happened between lagg_ifdetach()
and lagg_delport()

1. lagg_ifdetach calls psref_target_acquire()
2. lagg_delport calls LAGG_LOCK()
3. lagg_ifdetach calls LAGG_LOCK()
   - wait for lagg_delport
4. lagg_delport calls psref_target_destroy()
   - wait for lagg_ifdetach

diffstat:

 sys/net/lagg/if_lagg.c |  43 +++++++++++++++++++++++--------------------
 1 files changed, 23 insertions(+), 20 deletions(-)

diffs (117 lines):

diff -r e4d25c79d266 -r 04759e3edcd4 sys/net/lagg/if_lagg.c
--- a/sys/net/lagg/if_lagg.c    Thu Sep 30 04:13:42 2021 +0000
+++ b/sys/net/lagg/if_lagg.c    Thu Sep 30 04:20:14 2021 +0000
@@ -1,4 +1,4 @@
-/*     $NetBSD: if_lagg.c,v 1.7 2021/09/30 03:39:39 yamaguchi Exp $    */
+/*     $NetBSD: if_lagg.c,v 1.8 2021/09/30 04:20:14 yamaguchi Exp $    */
 
 /*
  * Copyright (c) 2005, 2006 Reyk Floeter <reyk%openbsd.org@localhost>
@@ -20,7 +20,7 @@
  */
 
 #include <sys/cdefs.h>
-__KERNEL_RCSID(0, "$NetBSD: if_lagg.c,v 1.7 2021/09/30 03:39:39 yamaguchi Exp $");
+__KERNEL_RCSID(0, "$NetBSD: if_lagg.c,v 1.8 2021/09/30 04:20:14 yamaguchi Exp $");
 
 #ifdef _KERNEL_OPT
 #include "opt_inet.h"
@@ -2051,15 +2051,19 @@
 }
 
 static void
-lagg_delport_locked(struct lagg_softc *sc, struct lagg_port *lp)
+lagg_delport_locked(struct lagg_softc *sc, struct lagg_port *lp,
+    bool is_ifdetach)
 {
        struct ifnet *ifp_port;
        struct ifreq ifr;
        u_long if_type;
-       bool stopped, detaching;
+       bool stopped;
 
        KASSERT(LAGG_LOCKED(sc));
 
+       if (lp == NULL)
+               return;
+
        ifp_port = lp->lp_ifp;
        stopped = false;
 
@@ -2093,7 +2097,6 @@
        lagg_port_purgemulti(sc, lp);
        lagg_port_purgevlan(sc, lp);
 
-       detaching = lp->lp_detaching;
        IFNET_LOCK(ifp_port);
        if_type = ifp_port->if_type;
        ifp_port->if_type = lp->lp_iftype;
@@ -2111,7 +2114,7 @@
                ifp_port->if_init(ifp_port);
        }
 
-       if (!detaching) {
+       if (is_ifdetach == false) {
                IFNET_LOCK(ifp_port);
                lagg_in6_ifattach(ifp_port);
                IFNET_UNLOCK(ifp_port);
@@ -2125,7 +2128,7 @@
 
        LAGG_LOCK(sc);
        LAGG_PORTS_FOREACH_SAFE(sc, lp, lp0) {
-               lagg_delport_locked(sc, lp);
+               lagg_delport_locked(sc, lp, false);
        }
        LAGG_UNLOCK(sc);
 }
@@ -2142,7 +2145,7 @@
                return ENOENT;
 
        LAGG_LOCK(sc);
-       lagg_delport_locked(sc, lp);
+       lagg_delport_locked(sc, lp, false);
        LAGG_UNLOCK(sc);
 
        return 0;
@@ -2315,7 +2318,6 @@
 {
        struct lagg_port *lp;
        struct lagg_softc *sc;
-       struct psref psref;
        int s;
 
        IFNET_ASSERT_UNLOCKED(ifp_port);
@@ -2326,21 +2328,22 @@
                pserialize_read_exit(s);
                return;
        }
-       lagg_port_getref(lp, &psref);
-       pserialize_read_exit(s);
 
        sc = lp->lp_softc;
-       if (sc != NULL) {
-               IFNET_LOCK(&sc->sc_if);
-               LAGG_LOCK(sc);
-               lagg_port_putref(lp, &psref);
-       } else {
-               lagg_port_putref(lp, &psref);
+       if (sc == NULL) {
+               pserialize_read_exit(s);
                return;
        }
-
-       lp->lp_detaching = true;
-       lagg_delport_locked(sc, lp);
+       pserialize_read_exit(s);
+
+       IFNET_LOCK(&sc->sc_if);
+       LAGG_LOCK(sc);
+       /*
+        * reload if_lagg because it may write
+        * after pserialize_read_exit.
+        */
+       lp = ifp_port->if_lagg;
+       lagg_delport_locked(sc, lp, true);
        LAGG_UNLOCK(sc);
        IFNET_UNLOCK(&sc->sc_if);
 }



Home | Main Index | Thread Index | Old Index