Commit b2cb09b1 authored by Jiri Pirko's avatar Jiri Pirko Committed by David S. Miller

igb: do vlan cleanup

- unify vlan and nonvlan rx path
- kill adapter->vlgrp and igb_vlan_rx_register
- allow to turn on/off rx/tx vlan accel via ethtool (set_features)
Signed-off-by: default avatarJiri Pirko <jpirko@redhat.com>
Signed-off-by: default avatarDavid S. Miller <davem@davemloft.net>
parent 3326c784
...@@ -37,6 +37,8 @@ ...@@ -37,6 +37,8 @@
#include <linux/clocksource.h> #include <linux/clocksource.h>
#include <linux/timecompare.h> #include <linux/timecompare.h>
#include <linux/net_tstamp.h> #include <linux/net_tstamp.h>
#include <linux/bitops.h>
#include <linux/if_vlan.h>
struct igb_adapter; struct igb_adapter;
...@@ -252,7 +254,7 @@ static inline int igb_desc_unused(struct igb_ring *ring) ...@@ -252,7 +254,7 @@ static inline int igb_desc_unused(struct igb_ring *ring)
struct igb_adapter { struct igb_adapter {
struct timer_list watchdog_timer; struct timer_list watchdog_timer;
struct timer_list phy_info_timer; struct timer_list phy_info_timer;
struct vlan_group *vlgrp; unsigned long active_vlans[BITS_TO_LONGS(VLAN_N_VID)];
u16 mng_vlan_id; u16 mng_vlan_id;
u32 bd_number; u32 bd_number;
u32 wol; u32 wol;
......
...@@ -28,6 +28,7 @@ ...@@ -28,6 +28,7 @@
#include <linux/module.h> #include <linux/module.h>
#include <linux/types.h> #include <linux/types.h>
#include <linux/init.h> #include <linux/init.h>
#include <linux/bitops.h>
#include <linux/vmalloc.h> #include <linux/vmalloc.h>
#include <linux/pagemap.h> #include <linux/pagemap.h>
#include <linux/netdevice.h> #include <linux/netdevice.h>
...@@ -46,6 +47,7 @@ ...@@ -46,6 +47,7 @@
#include <linux/if_ether.h> #include <linux/if_ether.h>
#include <linux/aer.h> #include <linux/aer.h>
#include <linux/prefetch.h> #include <linux/prefetch.h>
#include <linux/if_vlan.h>
#ifdef CONFIG_IGB_DCA #ifdef CONFIG_IGB_DCA
#include <linux/dca.h> #include <linux/dca.h>
#endif #endif
...@@ -140,7 +142,7 @@ static bool igb_clean_rx_irq_adv(struct igb_q_vector *, int *, int); ...@@ -140,7 +142,7 @@ static bool igb_clean_rx_irq_adv(struct igb_q_vector *, int *, int);
static int igb_ioctl(struct net_device *, struct ifreq *, int cmd); static int igb_ioctl(struct net_device *, struct ifreq *, int cmd);
static void igb_tx_timeout(struct net_device *); static void igb_tx_timeout(struct net_device *);
static void igb_reset_task(struct work_struct *); static void igb_reset_task(struct work_struct *);
static void igb_vlan_rx_register(struct net_device *, struct vlan_group *); static void igb_vlan_mode(struct net_device *netdev, u32 features);
static void igb_vlan_rx_add_vid(struct net_device *, u16); static void igb_vlan_rx_add_vid(struct net_device *, u16);
static void igb_vlan_rx_kill_vid(struct net_device *, u16); static void igb_vlan_rx_kill_vid(struct net_device *, u16);
static void igb_restore_vlan(struct igb_adapter *); static void igb_restore_vlan(struct igb_adapter *);
...@@ -1362,7 +1364,7 @@ static void igb_update_mng_vlan(struct igb_adapter *adapter) ...@@ -1362,7 +1364,7 @@ static void igb_update_mng_vlan(struct igb_adapter *adapter)
if ((old_vid != (u16)IGB_MNG_VLAN_NONE) && if ((old_vid != (u16)IGB_MNG_VLAN_NONE) &&
(vid != old_vid) && (vid != old_vid) &&
!vlan_group_get_device(adapter->vlgrp, old_vid)) { !test_bit(old_vid, adapter->active_vlans)) {
/* remove VID from filter table */ /* remove VID from filter table */
igb_vfta_set(hw, old_vid, false); igb_vfta_set(hw, old_vid, false);
} }
...@@ -1748,10 +1750,25 @@ void igb_reset(struct igb_adapter *adapter) ...@@ -1748,10 +1750,25 @@ void igb_reset(struct igb_adapter *adapter)
igb_get_phy_info(hw); igb_get_phy_info(hw);
} }
static u32 igb_fix_features(struct net_device *netdev, u32 features)
{
/*
* Since there is no support for separate rx/tx vlan accel
* enable/disable make sure tx flag is always in same state as rx.
*/
if (features & NETIF_F_HW_VLAN_RX)
features |= NETIF_F_HW_VLAN_TX;
else
features &= ~NETIF_F_HW_VLAN_TX;
return features;
}
static int igb_set_features(struct net_device *netdev, u32 features) static int igb_set_features(struct net_device *netdev, u32 features)
{ {
struct igb_adapter *adapter = netdev_priv(netdev); struct igb_adapter *adapter = netdev_priv(netdev);
int i; int i;
u32 changed = netdev->features ^ features;
for (i = 0; i < adapter->num_rx_queues; i++) { for (i = 0; i < adapter->num_rx_queues; i++) {
if (features & NETIF_F_RXCSUM) if (features & NETIF_F_RXCSUM)
...@@ -1760,6 +1777,9 @@ static int igb_set_features(struct net_device *netdev, u32 features) ...@@ -1760,6 +1777,9 @@ static int igb_set_features(struct net_device *netdev, u32 features)
adapter->rx_ring[i]->flags &= ~IGB_RING_FLAG_RX_CSUM; adapter->rx_ring[i]->flags &= ~IGB_RING_FLAG_RX_CSUM;
} }
if (changed & NETIF_F_HW_VLAN_RX)
igb_vlan_mode(netdev, features);
return 0; return 0;
} }
...@@ -1775,7 +1795,6 @@ static const struct net_device_ops igb_netdev_ops = { ...@@ -1775,7 +1795,6 @@ static const struct net_device_ops igb_netdev_ops = {
.ndo_do_ioctl = igb_ioctl, .ndo_do_ioctl = igb_ioctl,
.ndo_tx_timeout = igb_tx_timeout, .ndo_tx_timeout = igb_tx_timeout,
.ndo_validate_addr = eth_validate_addr, .ndo_validate_addr = eth_validate_addr,
.ndo_vlan_rx_register = igb_vlan_rx_register,
.ndo_vlan_rx_add_vid = igb_vlan_rx_add_vid, .ndo_vlan_rx_add_vid = igb_vlan_rx_add_vid,
.ndo_vlan_rx_kill_vid = igb_vlan_rx_kill_vid, .ndo_vlan_rx_kill_vid = igb_vlan_rx_kill_vid,
.ndo_set_vf_mac = igb_ndo_set_vf_mac, .ndo_set_vf_mac = igb_ndo_set_vf_mac,
...@@ -1785,6 +1804,7 @@ static const struct net_device_ops igb_netdev_ops = { ...@@ -1785,6 +1804,7 @@ static const struct net_device_ops igb_netdev_ops = {
#ifdef CONFIG_NET_POLL_CONTROLLER #ifdef CONFIG_NET_POLL_CONTROLLER
.ndo_poll_controller = igb_netpoll, .ndo_poll_controller = igb_netpoll,
#endif #endif
.ndo_fix_features = igb_fix_features,
.ndo_set_features = igb_set_features, .ndo_set_features = igb_set_features,
}; };
...@@ -1930,11 +1950,11 @@ static int __devinit igb_probe(struct pci_dev *pdev, ...@@ -1930,11 +1950,11 @@ static int __devinit igb_probe(struct pci_dev *pdev,
NETIF_F_IPV6_CSUM | NETIF_F_IPV6_CSUM |
NETIF_F_TSO | NETIF_F_TSO |
NETIF_F_TSO6 | NETIF_F_TSO6 |
NETIF_F_RXCSUM; NETIF_F_RXCSUM |
NETIF_F_HW_VLAN_RX;
netdev->features = netdev->hw_features | netdev->features = netdev->hw_features |
NETIF_F_HW_VLAN_TX | NETIF_F_HW_VLAN_TX |
NETIF_F_HW_VLAN_RX |
NETIF_F_HW_VLAN_FILTER; NETIF_F_HW_VLAN_FILTER;
netdev->vlan_features |= NETIF_F_TSO; netdev->vlan_features |= NETIF_F_TSO;
...@@ -2057,6 +2077,8 @@ static int __devinit igb_probe(struct pci_dev *pdev, ...@@ -2057,6 +2077,8 @@ static int __devinit igb_probe(struct pci_dev *pdev,
if (err) if (err)
goto err_register; goto err_register;
igb_vlan_mode(netdev, netdev->features);
/* carrier off reporting is important to ethtool even BEFORE open */ /* carrier off reporting is important to ethtool even BEFORE open */
netif_carrier_off(netdev); netif_carrier_off(netdev);
...@@ -2939,12 +2961,11 @@ static inline int igb_set_vf_rlpml(struct igb_adapter *adapter, int size, ...@@ -2939,12 +2961,11 @@ static inline int igb_set_vf_rlpml(struct igb_adapter *adapter, int size,
**/ **/
static void igb_rlpml_set(struct igb_adapter *adapter) static void igb_rlpml_set(struct igb_adapter *adapter)
{ {
u32 max_frame_size = adapter->max_frame_size; u32 max_frame_size;
struct e1000_hw *hw = &adapter->hw; struct e1000_hw *hw = &adapter->hw;
u16 pf_id = adapter->vfs_allocated_count; u16 pf_id = adapter->vfs_allocated_count;
if (adapter->vlgrp) max_frame_size = adapter->max_frame_size + VLAN_TAG_SIZE;
max_frame_size += VLAN_TAG_SIZE;
/* if vfs are enabled we set RLPML to the largest possible request /* if vfs are enabled we set RLPML to the largest possible request
* size and set the VMOLR RLPML to the size we need */ * size and set the VMOLR RLPML to the size we need */
...@@ -5693,25 +5714,6 @@ static bool igb_clean_tx_irq(struct igb_q_vector *q_vector) ...@@ -5693,25 +5714,6 @@ static bool igb_clean_tx_irq(struct igb_q_vector *q_vector)
return count < tx_ring->count; return count < tx_ring->count;
} }
/**
* igb_receive_skb - helper function to handle rx indications
* @q_vector: structure containing interrupt and ring information
* @skb: packet to send up
* @vlan_tag: vlan tag for packet
**/
static void igb_receive_skb(struct igb_q_vector *q_vector,
struct sk_buff *skb,
u16 vlan_tag)
{
struct igb_adapter *adapter = q_vector->adapter;
if (vlan_tag && adapter->vlgrp)
vlan_gro_receive(&q_vector->napi, adapter->vlgrp,
vlan_tag, skb);
else
napi_gro_receive(&q_vector->napi, skb);
}
static inline void igb_rx_checksum_adv(struct igb_ring *ring, static inline void igb_rx_checksum_adv(struct igb_ring *ring,
u32 status_err, struct sk_buff *skb) u32 status_err, struct sk_buff *skb)
{ {
...@@ -5809,7 +5811,6 @@ static bool igb_clean_rx_irq_adv(struct igb_q_vector *q_vector, ...@@ -5809,7 +5811,6 @@ static bool igb_clean_rx_irq_adv(struct igb_q_vector *q_vector,
unsigned int i; unsigned int i;
u32 staterr; u32 staterr;
u16 length; u16 length;
u16 vlan_tag;
i = rx_ring->next_to_clean; i = rx_ring->next_to_clean;
buffer_info = &rx_ring->buffer_info[i]; buffer_info = &rx_ring->buffer_info[i];
...@@ -5894,10 +5895,12 @@ static bool igb_clean_rx_irq_adv(struct igb_q_vector *q_vector, ...@@ -5894,10 +5895,12 @@ static bool igb_clean_rx_irq_adv(struct igb_q_vector *q_vector,
skb->protocol = eth_type_trans(skb, netdev); skb->protocol = eth_type_trans(skb, netdev);
skb_record_rx_queue(skb, rx_ring->queue_index); skb_record_rx_queue(skb, rx_ring->queue_index);
vlan_tag = ((staterr & E1000_RXD_STAT_VP) ? if (staterr & E1000_RXD_STAT_VP) {
le16_to_cpu(rx_desc->wb.upper.vlan) : 0); u16 vid = le16_to_cpu(rx_desc->wb.upper.vlan);
igb_receive_skb(q_vector, skb, vlan_tag); __vlan_hwaccel_put_tag(skb, vid);
}
napi_gro_receive(&q_vector->napi, skb);
next_desc: next_desc:
rx_desc->wb.upper.status_error = 0; rx_desc->wb.upper.status_error = 0;
...@@ -6290,17 +6293,15 @@ s32 igb_write_pcie_cap_reg(struct e1000_hw *hw, u32 reg, u16 *value) ...@@ -6290,17 +6293,15 @@ s32 igb_write_pcie_cap_reg(struct e1000_hw *hw, u32 reg, u16 *value)
return 0; return 0;
} }
static void igb_vlan_rx_register(struct net_device *netdev, static void igb_vlan_mode(struct net_device *netdev, u32 features)
struct vlan_group *grp)
{ {
struct igb_adapter *adapter = netdev_priv(netdev); struct igb_adapter *adapter = netdev_priv(netdev);
struct e1000_hw *hw = &adapter->hw; struct e1000_hw *hw = &adapter->hw;
u32 ctrl, rctl; u32 ctrl, rctl;
igb_irq_disable(adapter); igb_irq_disable(adapter);
adapter->vlgrp = grp;
if (grp) { if (features & NETIF_F_HW_VLAN_RX) {
/* enable VLAN tag insert/strip */ /* enable VLAN tag insert/strip */
ctrl = rd32(E1000_CTRL); ctrl = rd32(E1000_CTRL);
ctrl |= E1000_CTRL_VME; ctrl |= E1000_CTRL_VME;
...@@ -6334,6 +6335,8 @@ static void igb_vlan_rx_add_vid(struct net_device *netdev, u16 vid) ...@@ -6334,6 +6335,8 @@ static void igb_vlan_rx_add_vid(struct net_device *netdev, u16 vid)
/* add the filter since PF can receive vlans w/o entry in vlvf */ /* add the filter since PF can receive vlans w/o entry in vlvf */
igb_vfta_set(hw, vid, true); igb_vfta_set(hw, vid, true);
set_bit(vid, adapter->active_vlans);
} }
static void igb_vlan_rx_kill_vid(struct net_device *netdev, u16 vid) static void igb_vlan_rx_kill_vid(struct net_device *netdev, u16 vid)
...@@ -6344,7 +6347,6 @@ static void igb_vlan_rx_kill_vid(struct net_device *netdev, u16 vid) ...@@ -6344,7 +6347,6 @@ static void igb_vlan_rx_kill_vid(struct net_device *netdev, u16 vid)
s32 err; s32 err;
igb_irq_disable(adapter); igb_irq_disable(adapter);
vlan_group_set_device(adapter->vlgrp, vid, NULL);
if (!test_bit(__IGB_DOWN, &adapter->state)) if (!test_bit(__IGB_DOWN, &adapter->state))
igb_irq_enable(adapter); igb_irq_enable(adapter);
...@@ -6355,20 +6357,16 @@ static void igb_vlan_rx_kill_vid(struct net_device *netdev, u16 vid) ...@@ -6355,20 +6357,16 @@ static void igb_vlan_rx_kill_vid(struct net_device *netdev, u16 vid)
/* if vid was not present in VLVF just remove it from table */ /* if vid was not present in VLVF just remove it from table */
if (err) if (err)
igb_vfta_set(hw, vid, false); igb_vfta_set(hw, vid, false);
clear_bit(vid, adapter->active_vlans);
} }
static void igb_restore_vlan(struct igb_adapter *adapter) static void igb_restore_vlan(struct igb_adapter *adapter)
{ {
igb_vlan_rx_register(adapter->netdev, adapter->vlgrp);
if (adapter->vlgrp) {
u16 vid; u16 vid;
for (vid = 0; vid < VLAN_N_VID; vid++) {
if (!vlan_group_get_device(adapter->vlgrp, vid)) for_each_set_bit(vid, adapter->active_vlans, VLAN_N_VID)
continue;
igb_vlan_rx_add_vid(adapter->netdev, vid); igb_vlan_rx_add_vid(adapter->netdev, vid);
}
}
} }
int igb_set_spd_dplx(struct igb_adapter *adapter, u32 spd, u8 dplx) int igb_set_spd_dplx(struct igb_adapter *adapter, u32 spd, u8 dplx)
......
Markdown is supported
0%
or
You are about to add 0 people to the discussion. Proceed with caution.
Finish editing this message first!
Please register or to comment