Source-Changes-HG archive

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

[src/trunk]: src/sys/dev/pci Fix IFF_ALLMULTI handling.



details:   https://anonhg.NetBSD.org/src/rev/08600850d654
branches:  trunk
changeset: 465517:08600850d654
user:      jmcneill <jmcneill%NetBSD.org@localhost>
date:      Tue Nov 26 10:33:19 2019 +0000

description:
Fix IFF_ALLMULTI handling.

diffstat:

 sys/dev/pci/if_mcx.c |  38 +++++++++++++++++++++++++++++---------
 1 files changed, 29 insertions(+), 9 deletions(-)

diffs (83 lines):

diff -r a9ad92f74851 -r 08600850d654 sys/dev/pci/if_mcx.c
--- a/sys/dev/pci/if_mcx.c      Tue Nov 26 10:24:17 2019 +0000
+++ b/sys/dev/pci/if_mcx.c      Tue Nov 26 10:33:19 2019 +0000
@@ -1,4 +1,4 @@
-/*     $NetBSD: if_mcx.c,v 1.6 2019/11/18 04:40:05 nonaka Exp $ */
+/*     $NetBSD: if_mcx.c,v 1.7 2019/11/26 10:33:19 jmcneill Exp $ */
 /*     $OpenBSD: if_mcx.c,v 1.33 2019/09/12 04:23:59 jmatthew Exp $ */
 
 /*
@@ -6205,8 +6205,11 @@
 {
        struct mcx_softc *sc = (struct mcx_softc *)ifp->if_softc;
        struct ifreq *ifr = (struct ifreq *)data;
+       struct ethercom *ec = &sc->sc_ec;
        uint8_t addrhi[ETHER_ADDR_LEN], addrlo[ETHER_ADDR_LEN];
-       int s, i, error = 0;
+       struct ether_multi *enm;
+       struct ether_multistep step;
+       int s, i, flags, error = 0;
 
        s = splnet();
        switch (cmd) {
@@ -6214,8 +6217,10 @@
        case SIOCADDMULTI:
                if (ether_addmulti(ifreq_getaddr(cmd, ifr), &sc->sc_ec) == ENETRESET) {
                        error = ether_multiaddr(&ifr->ifr_addr, addrlo, addrhi);
-                       if (error != 0)
+                       if (error != 0) {
+                               splx(s);
                                return (error);
+                       }
 
                        for (i = 0; i < MCX_NUM_MCAST_FLOWS; i++) {
                                if (sc->sc_mcast_flows[i][0] == 0) {
@@ -6238,7 +6243,7 @@
                                        error = ENETRESET;
                                }
 
-                               if (sc->sc_ec.ec_multicnt > 0) {
+                               if (memcmp(addrlo, addrhi, ETHER_ADDR_LEN)) {
                                        SET(ifp->if_flags, IFF_ALLMULTI);
                                        error = ENETRESET;
                                }
@@ -6249,8 +6254,10 @@
        case SIOCDELMULTI:
                if (ether_delmulti(ifreq_getaddr(cmd, ifr), &sc->sc_ec) == ENETRESET) {
                        error = ether_multiaddr(&ifr->ifr_addr, addrlo, addrhi);
-                       if (error != 0)
+                       if (error != 0) {
+                               splx(s);
                                return (error);
+                       }
 
                        for (i = 0; i < MCX_NUM_MCAST_FLOWS; i++) {
                                if (memcmp(sc->sc_mcast_flows[i], addrlo,
@@ -6269,10 +6276,23 @@
                                sc->sc_extra_mcast--;
 
                        if (ISSET(ifp->if_flags, IFF_ALLMULTI) &&
-                           (sc->sc_extra_mcast == 0) &&
-                           (sc->sc_ec.ec_multicnt == 0)) {
-                               CLR(ifp->if_flags, IFF_ALLMULTI);
-                               error = ENETRESET;
+                           sc->sc_extra_mcast == 0) {
+                               flags = 0;
+                               ETHER_LOCK(ec);
+                               ETHER_FIRST_MULTI(step, ec, enm);
+                               while (enm != NULL) {
+                                       if (memcmp(enm->enm_addrlo,
+                                           enm->enm_addrhi, ETHER_ADDR_LEN)) {
+                                               SET(flags, IFF_ALLMULTI);
+                                               break;
+                                       }
+                                       ETHER_NEXT_MULTI(step, enm);
+                               }
+                               ETHER_UNLOCK(ec);
+                               if (!ISSET(flags, IFF_ALLMULTI)) {
+                                       CLR(ifp->if_flags, IFF_ALLMULTI);
+                                       error = ENETRESET;
+                               }
                        }
                }
                break;



Home | Main Index | Thread Index | Old Index