Source-Changes-HG archive

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

[src/netbsd-9]: src/sys/net Pull up following revision(s) (requested by msait...



details:   https://anonhg.NetBSD.org/src/rev/0bd1ebde0a15
branches:  netbsd-9
changeset: 1000768:0bd1ebde0a15
user:      martin <martin%NetBSD.org@localhost>
date:      Sun Sep 01 11:07:05 2019 +0000

description:
Pull up following revision(s) (requested by msaitoh in ticket #133):

        sys/dev/pci/ixgbe/ixgbe.c: revision 1.200
        sys/dev/pci/ixgbe/ixgbe.c: revision 1.201
        sys/dev/pci/ixgbe/ixv.c: revision 1.126
        sys/dev/pci/ixgbe/ixv.c: revision 1.127
        sys/net/if_vlan.c: revision 1.142
        sys/net/if_vlan.c: revision 1.143
        sys/net/if_vlan.c: revision 1.144
        sys/net/if_vlan.c: revision 1.145
        sys/net/if_vlan.c: revision 1.146

 Check ec_capenable instead of ec_capabilities to control TX side of VLAN HW
tagging correctly.
XXX pullup-9

 Add missing IFNET_LOCK() and IFNET_UNLOCK() in vlan_config().
XXX pullup-9

 Fix a bug that VLAN HW "tagging" enable/disable may not reflect correctly.
  - Always call ec_vlan_cb() if it exists.
  - Some (or all?) ethernet drivers don't enable HW tagging if no any vlan is
    attached. ixgbe is one of them. Check the the transition and update
    VLAN HW tagging function.
XXX pullup-9

 Use ETHER_LOCK()/ETHER_UNLOCK() suggested by knakahara.
- kmem_alloc(,KM_SLEEP) never return NULL, so remove NULL check.
- VLAN ID is never duplicated, so break the loop when found. Also move
  kmen_free() outside of ETHER_LOCK(ec)/ETHER_UNLOCK(ec) to reduce the hold
  time. suggested by ozaki-r.
- Whitespace fix.

diffstat:

 sys/dev/pci/ixgbe/ixgbe.c |  70 +++++++++++++++++++++++++++++++++-----------
 sys/dev/pci/ixgbe/ixv.c   |  71 ++++++++++++++++++++++++++++++++-------------
 sys/net/if_vlan.c         |  73 +++++++++++++++++++++++-----------------------
 3 files changed, 139 insertions(+), 75 deletions(-)

diffs (truncated from 408 to 300 lines):

diff -r 7025609f5c31 -r 0bd1ebde0a15 sys/dev/pci/ixgbe/ixgbe.c
--- a/sys/dev/pci/ixgbe/ixgbe.c Sun Sep 01 11:02:27 2019 +0000
+++ b/sys/dev/pci/ixgbe/ixgbe.c Sun Sep 01 11:07:05 2019 +0000
@@ -1,4 +1,4 @@
-/* $NetBSD: ixgbe.c,v 1.199 2019/07/30 08:44:28 msaitoh Exp $ */
+/* $NetBSD: ixgbe.c,v 1.199.2.1 2019/09/01 11:07:05 martin Exp $ */
 
 /******************************************************************************
 
@@ -219,6 +219,7 @@
 static u8 *    ixgbe_mc_array_itr(struct ixgbe_hw *, u8 **, u32 *);
 static void    ixgbe_eitr_write(struct adapter *, uint32_t, uint32_t);
 
+static void    ixgbe_setup_vlan_hw_tagging(struct adapter *);
 static void    ixgbe_setup_vlan_hw_support(struct adapter *);
 static int     ixgbe_vlan_cb(struct ethercom *, uint16_t, bool);
 static int     ixgbe_register_vlan(void *, struct ifnet *, u16);
@@ -2305,6 +2306,7 @@
 ixgbe_vlan_cb(struct ethercom *ec, uint16_t vid, bool set)
 {
        struct ifnet *ifp = &ec->ec_if;
+       struct adapter *adapter = ifp->if_softc;
        int rv;
 
        if (set)
@@ -2312,6 +2314,16 @@
        else
                rv = ixgbe_unregister_vlan(ifp->if_softc, ifp, vid);
 
+       if (rv != 0)
+               return rv;
+
+       /*
+        * Control VLAN HW tagging when ec_nvlan is changed from 1 to 0
+        * or 0 to 1.
+        */
+       if ((set && (ec->ec_nvlans == 1)) || (!set && (ec->ec_nvlans == 0)))
+               ixgbe_setup_vlan_hw_tagging(adapter);
+
        return rv;
 }
 
@@ -2381,21 +2393,15 @@
 } /* ixgbe_unregister_vlan */
 
 static void
-ixgbe_setup_vlan_hw_support(struct adapter *adapter)
+ixgbe_setup_vlan_hw_tagging(struct adapter *adapter)
 {
        struct ethercom *ec = &adapter->osdep.ec;
        struct ixgbe_hw *hw = &adapter->hw;
        struct rx_ring  *rxr;
+       u32             ctrl;
        int             i;
-       u32             ctrl;
-       struct vlanid_list *vlanidp;
        bool            hwtagging;
 
-       /*
-        *  This function is called from both if_init and ifflags_cb()
-        * on NetBSD.
-        */
-
        /* Enable HW tagging only if any vlan is attached */
        hwtagging = (ec->ec_capenable & ETHERCAP_VLAN_HWTAGGING)
            && VLAN_ATTACHED(ec);
@@ -2417,11 +2423,46 @@
                rxr->vtag_strip = hwtagging ? TRUE : FALSE;
        }
 
+       /* VLAN hw tagging for 82598 */
+       if (hw->mac.type == ixgbe_mac_82598EB) {
+               ctrl = IXGBE_READ_REG(hw, IXGBE_VLNCTRL);
+               if (hwtagging)
+                       ctrl |= IXGBE_VLNCTRL_VME;
+               else
+                       ctrl &= ~IXGBE_VLNCTRL_VME;
+               IXGBE_WRITE_REG(hw, IXGBE_VLNCTRL, ctrl);
+       }
+} /* ixgbe_setup_vlan_hw_tagging */
+
+static void
+ixgbe_setup_vlan_hw_support(struct adapter *adapter)
+{
+       struct ethercom *ec = &adapter->osdep.ec;
+       struct ixgbe_hw *hw = &adapter->hw;
+       int             i;
+       u32             ctrl;
+       struct vlanid_list *vlanidp;
+
+       /*
+        *  This function is called from both if_init and ifflags_cb()
+        * on NetBSD.
+        */
+
+       /*
+        * Part 1:
+        * Setup VLAN HW tagging
+        */
+       ixgbe_setup_vlan_hw_tagging(adapter);
+
+       /*
+        * Part 2:
+        * Setup VLAN HW filter
+        */
        /* Cleanup shadow_vfta */
        for (i = 0; i < IXGBE_VFTA_SIZE; i++)
                adapter->shadow_vfta[i] = 0;
        /* Generate shadow_vfta from ec_vids */
-       mutex_enter(ec->ec_lock);
+       ETHER_LOCK(ec);
        SIMPLEQ_FOREACH(vlanidp, &ec->ec_vids, vid_list) {
                uint32_t idx;
 
@@ -2429,7 +2470,7 @@
                KASSERT(idx < IXGBE_VFTA_SIZE);
                adapter->shadow_vfta[idx] |= (u32)1 << (vlanidp->vid % 32);
        }
-       mutex_exit(ec->ec_lock);
+       ETHER_UNLOCK(ec);
        for (i = 0; i < IXGBE_VFTA_SIZE; i++)
                IXGBE_WRITE_REG(hw, IXGBE_VFTA(i), adapter->shadow_vfta[i]);
 
@@ -2439,13 +2480,6 @@
                ctrl |= IXGBE_VLNCTRL_VFE;
        else
                ctrl &= ~IXGBE_VLNCTRL_VFE;
-       /* VLAN hw tagging for 82598 */
-       if (hw->mac.type == ixgbe_mac_82598EB) {
-               if (hwtagging)
-                       ctrl |= IXGBE_VLNCTRL_VME;
-               else
-                       ctrl &= ~IXGBE_VLNCTRL_VME;
-       }
        IXGBE_WRITE_REG(hw, IXGBE_VLNCTRL, ctrl);
 } /* ixgbe_setup_vlan_hw_support */
 
diff -r 7025609f5c31 -r 0bd1ebde0a15 sys/dev/pci/ixgbe/ixv.c
--- a/sys/dev/pci/ixgbe/ixv.c   Sun Sep 01 11:02:27 2019 +0000
+++ b/sys/dev/pci/ixgbe/ixv.c   Sun Sep 01 11:07:05 2019 +0000
@@ -1,4 +1,4 @@
-/*$NetBSD: ixv.c,v 1.125 2019/07/30 08:38:03 msaitoh Exp $*/
+/*$NetBSD: ixv.c,v 1.125.2.1 2019/09/01 11:07:05 martin Exp $*/
 
 /******************************************************************************
 
@@ -120,6 +120,7 @@
 static u8 *    ixv_mc_array_itr(struct ixgbe_hw *, u8 **, u32 *);
 static void    ixv_eitr_write(struct adapter *, uint32_t, uint32_t);
 
+static void    ixv_setup_vlan_tagging(struct adapter *);
 static int     ixv_setup_vlan_support(struct adapter *);
 static int     ixv_vlan_cb(struct ethercom *, uint16_t, bool);
 static int     ixv_register_vlan(void *, struct ifnet *, u16);
@@ -1935,33 +1936,22 @@
        return sysctl_lookup(SYSCTLFN_CALL(&node));
 } /* ixv_sysctl_rdt_handler */
 
-/************************************************************************
- * ixv_setup_vlan_support
- ************************************************************************/
-static int
-ixv_setup_vlan_support(struct adapter *adapter)
+static void
+ixv_setup_vlan_tagging(struct adapter *adapter)
 {
        struct ethercom *ec = &adapter->osdep.ec;
        struct ixgbe_hw *hw = &adapter->hw;
        struct rx_ring  *rxr;
-       u32             ctrl, vid, vfta, retry;
-       struct vlanid_list *vlanidp;
-       int rv, error = 0;
-       bool usevlan;
+       u32             ctrl;
+       int             i;
        bool            hwtagging;
 
-       /*
-        *  This function is called from both if_init and ifflags_cb()
-        * on NetBSD.
-        */
-       usevlan = VLAN_ATTACHED(ec);
-
        /* Enable HW tagging only if any vlan is attached */
        hwtagging = (ec->ec_capenable & ETHERCAP_VLAN_HWTAGGING)
            && VLAN_ATTACHED(ec);
 
        /* Enable the queues */
-       for (int i = 0; i < adapter->num_queues; i++) {
+       for (i = 0; i < adapter->num_queues; i++) {
                rxr = &adapter->rx_rings[i];
                ctrl = IXGBE_READ_REG(hw, IXGBE_VFRXDCTL(rxr->me));
                if (hwtagging)
@@ -1975,15 +1965,43 @@
                 */
                rxr->vtag_strip = hwtagging ? TRUE : FALSE;
        }
-
-       if (!usevlan)
+} /* ixv_setup_vlan_tagging */
+
+/************************************************************************
+ * ixv_setup_vlan_support
+ ************************************************************************/
+static int
+ixv_setup_vlan_support(struct adapter *adapter)
+{
+       struct ethercom *ec = &adapter->osdep.ec;
+       struct ixgbe_hw *hw = &adapter->hw;
+       u32             vid, vfta, retry;
+       struct vlanid_list *vlanidp;
+       int rv, error = 0;
+
+       /*
+        *  This function is called from both if_init and ifflags_cb()
+        * on NetBSD.
+        */
+
+       /*
+        * Part 1:
+        * Setup VLAN HW tagging
+        */
+       ixv_setup_vlan_tagging(adapter);
+
+       if (!VLAN_ATTACHED(ec))
                return 0;
 
+       /*
+        * Part 2:
+        * Setup VLAN HW filter
+        */
        /* Cleanup shadow_vfta */
        for (int i = 0; i < IXGBE_VFTA_SIZE; i++)
                adapter->shadow_vfta[i] = 0;
        /* Generate shadow_vfta from ec_vids */
-       mutex_enter(ec->ec_lock);
+       ETHER_LOCK(ec);
        SIMPLEQ_FOREACH(vlanidp, &ec->ec_vids, vid_list) {
                uint32_t idx;
 
@@ -1991,7 +2009,7 @@
                KASSERT(idx < IXGBE_VFTA_SIZE);
                adapter->shadow_vfta[idx] |= (u32)1 << (vlanidp->vid % 32);
        }
-       mutex_exit(ec->ec_lock);
+       ETHER_UNLOCK(ec);
        
        /*
         * A soft reset zero's out the VFTA, so
@@ -2036,6 +2054,7 @@
 ixv_vlan_cb(struct ethercom *ec, uint16_t vid, bool set)
 {
        struct ifnet *ifp = &ec->ec_if;
+       struct adapter *adapter = ifp->if_softc;
        int rv;
 
        if (set)
@@ -2043,6 +2062,16 @@
        else
                rv = ixv_unregister_vlan(ifp->if_softc, ifp, vid);
 
+       if (rv != 0)
+               return rv;
+
+       /*
+        * Control VLAN HW tagging when ec_nvlan is changed from 1 to 0
+        * or 0 to 1.
+        */
+       if ((set && (ec->ec_nvlans == 1)) || (!set && (ec->ec_nvlans == 0)))
+               ixv_setup_vlan_tagging(adapter);
+
        return rv;
 }
 
diff -r 7025609f5c31 -r 0bd1ebde0a15 sys/net/if_vlan.c
--- a/sys/net/if_vlan.c Sun Sep 01 11:02:27 2019 +0000
+++ b/sys/net/if_vlan.c Sun Sep 01 11:07:05 2019 +0000
@@ -1,4 +1,4 @@
-/*     $NetBSD: if_vlan.c,v 1.141 2019/07/17 03:26:24 msaitoh Exp $    */
+/*     $NetBSD: if_vlan.c,v 1.141.2.1 2019/09/01 11:07:06 martin Exp $ */
 
 /*
  * Copyright (c) 2000, 2001 The NetBSD Foundation, Inc.
@@ -78,7 +78,7 @@
  */
 
 #include <sys/cdefs.h>
-__KERNEL_RCSID(0, "$NetBSD: if_vlan.c,v 1.141 2019/07/17 03:26:24 msaitoh Exp $");
+__KERNEL_RCSID(0, "$NetBSD: if_vlan.c,v 1.141.2.1 2019/09/01 11:07:06 martin Exp $");
 
 #ifdef _KERNEL_OPT
 #include "opt_inet.h"
@@ -485,35 +485,28 @@
                        }
                        error = 0;
                }
-               /*
-                * Add a vid to the list even if it's not enabled in case
-                * it's enabled later.
-                */



Home | Main Index | Thread Index | Old Index