Skip to content
Snippets Groups Projects
igb_main.c 169 KiB
Newer Older
  • Learn to ignore specific revisions
  • /*******************************************************************************
    
      Intel(R) Gigabit Ethernet Linux driver
    
      Copyright(c) 2007-2009 Intel Corporation.
    
    
      This program is free software; you can redistribute it and/or modify it
      under the terms and conditions of the GNU General Public License,
      version 2, as published by the Free Software Foundation.
    
      This program is distributed in the hope it will be useful, but WITHOUT
      ANY WARRANTY; without even the implied warranty of MERCHANTABILITY or
      FITNESS FOR A PARTICULAR PURPOSE.  See the GNU General Public License for
      more details.
    
      You should have received a copy of the GNU General Public License along with
      this program; if not, write to the Free Software Foundation, Inc.,
      51 Franklin St - Fifth Floor, Boston, MA 02110-1301 USA.
    
      The full GNU General Public License is included in this distribution in
      the file called "COPYING".
    
      Contact Information:
      e1000-devel Mailing List <e1000-devel@lists.sourceforge.net>
      Intel Corporation, 5200 N.E. Elam Young Parkway, Hillsboro, OR 97124-6497
    
    *******************************************************************************/
    
    #include <linux/module.h>
    #include <linux/types.h>
    #include <linux/init.h>
    #include <linux/vmalloc.h>
    #include <linux/pagemap.h>
    #include <linux/netdevice.h>
    #include <linux/ipv6.h>
    #include <net/checksum.h>
    #include <net/ip6_checksum.h>
    
    #include <linux/net_tstamp.h>
    
    #include <linux/mii.h>
    #include <linux/ethtool.h>
    #include <linux/if_vlan.h>
    #include <linux/pci.h>
    
    #include <linux/pci-aspm.h>
    
    #include <linux/delay.h>
    #include <linux/interrupt.h>
    #include <linux/if_ether.h>
    
    #include <linux/aer.h>
    
    #ifdef CONFIG_IGB_DCA
    
    Jeb Cramer's avatar
    Jeb Cramer committed
    #include <linux/dca.h>
    #endif
    
    #define DRV_VERSION "2.1.0-k2"
    
    char igb_driver_name[] = "igb";
    char igb_driver_version[] = DRV_VERSION;
    static const char igb_driver_string[] =
    				"Intel(R) Gigabit Ethernet Network Driver";
    
    static const char igb_copyright[] = "Copyright (c) 2007-2009 Intel Corporation.";
    
    
    static const struct e1000_info *igb_info_tbl[] = {
    	[board_82575] = &e1000_82575_info,
    };
    
    
    static DEFINE_PCI_DEVICE_TABLE(igb_pci_tbl) = {
    
    	{ PCI_VDEVICE(INTEL, E1000_DEV_ID_82580_COPPER), board_82575 },
    	{ PCI_VDEVICE(INTEL, E1000_DEV_ID_82580_FIBER), board_82575 },
    	{ PCI_VDEVICE(INTEL, E1000_DEV_ID_82580_SERDES), board_82575 },
    	{ PCI_VDEVICE(INTEL, E1000_DEV_ID_82580_SGMII), board_82575 },
    	{ PCI_VDEVICE(INTEL, E1000_DEV_ID_82580_COPPER_DUAL), board_82575 },
    
    	{ PCI_VDEVICE(INTEL, E1000_DEV_ID_82576), board_82575 },
    
    	{ PCI_VDEVICE(INTEL, E1000_DEV_ID_82576_NS), board_82575 },
    
    	{ PCI_VDEVICE(INTEL, E1000_DEV_ID_82576_NS_SERDES), board_82575 },
    
    	{ PCI_VDEVICE(INTEL, E1000_DEV_ID_82576_FIBER), board_82575 },
    	{ PCI_VDEVICE(INTEL, E1000_DEV_ID_82576_SERDES), board_82575 },
    
    	{ PCI_VDEVICE(INTEL, E1000_DEV_ID_82576_SERDES_QUAD), board_82575 },
    
    	{ PCI_VDEVICE(INTEL, E1000_DEV_ID_82576_QUAD_COPPER), board_82575 },
    
    	{ PCI_VDEVICE(INTEL, E1000_DEV_ID_82575EB_COPPER), board_82575 },
    	{ PCI_VDEVICE(INTEL, E1000_DEV_ID_82575EB_FIBER_SERDES), board_82575 },
    	{ PCI_VDEVICE(INTEL, E1000_DEV_ID_82575GB_QUAD_COPPER), board_82575 },
    	/* required last entry */
    	{0, }
    };
    
    MODULE_DEVICE_TABLE(pci, igb_pci_tbl);
    
    void igb_reset(struct igb_adapter *);
    static int igb_setup_all_tx_resources(struct igb_adapter *);
    static int igb_setup_all_rx_resources(struct igb_adapter *);
    static void igb_free_all_tx_resources(struct igb_adapter *);
    static void igb_free_all_rx_resources(struct igb_adapter *);
    
    static void igb_setup_mrqc(struct igb_adapter *);
    
    void igb_update_stats(struct igb_adapter *);
    static int igb_probe(struct pci_dev *, const struct pci_device_id *);
    static void __devexit igb_remove(struct pci_dev *pdev);
    static int igb_sw_init(struct igb_adapter *);
    static int igb_open(struct net_device *);
    static int igb_close(struct net_device *);
    static void igb_configure_tx(struct igb_adapter *);
    static void igb_configure_rx(struct igb_adapter *);
    static void igb_clean_all_tx_rings(struct igb_adapter *);
    static void igb_clean_all_rx_rings(struct igb_adapter *);
    
    static void igb_clean_tx_ring(struct igb_ring *);
    static void igb_clean_rx_ring(struct igb_ring *);
    
    static void igb_set_rx_mode(struct net_device *);
    
    static void igb_update_phy_info(unsigned long);
    static void igb_watchdog(unsigned long);
    static void igb_watchdog_task(struct work_struct *);
    
    static netdev_tx_t igb_xmit_frame_adv(struct sk_buff *skb, struct net_device *);
    
    static struct net_device_stats *igb_get_stats(struct net_device *);
    static int igb_change_mtu(struct net_device *, int);
    static int igb_set_mac(struct net_device *, void *);
    
    static void igb_set_uta(struct igb_adapter *adapter);
    
    static irqreturn_t igb_intr(int irq, void *);
    static irqreturn_t igb_intr_msi(int irq, void *);
    static irqreturn_t igb_msix_other(int irq, void *);
    
    static irqreturn_t igb_msix_ring(int irq, void *);
    
    #ifdef CONFIG_IGB_DCA
    
    static void igb_update_dca(struct igb_q_vector *);
    
    Jeb Cramer's avatar
    Jeb Cramer committed
    static void igb_setup_dca(struct igb_adapter *);
    
    #endif /* CONFIG_IGB_DCA */
    
    static bool igb_clean_tx_irq(struct igb_q_vector *);
    
    static int igb_poll(struct napi_struct *, int);
    
    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 void igb_tx_timeout(struct net_device *);
    static void igb_reset_task(struct work_struct *);
    static void igb_vlan_rx_register(struct net_device *, struct vlan_group *);
    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_restore_vlan(struct igb_adapter *);
    
    static void igb_rar_set_qsel(struct igb_adapter *, u8 *, u32 , u8);
    
    static void igb_ping_all_vfs(struct igb_adapter *);
    static void igb_msg_task(struct igb_adapter *);
    static void igb_vmm_control(struct igb_adapter *);
    
    static int igb_set_vf_mac(struct igb_adapter *, int, unsigned char *);
    
    static void igb_restore_vf_multicasts(struct igb_adapter *adapter);
    
    static int igb_ndo_set_vf_mac(struct net_device *netdev, int vf, u8 *mac);
    static int igb_ndo_set_vf_vlan(struct net_device *netdev,
    			       int vf, u16 vlan, u8 qos);
    static int igb_ndo_set_vf_bw(struct net_device *netdev, int vf, int tx_rate);
    static int igb_ndo_get_vf_config(struct net_device *netdev, int vf,
    				 struct ifla_vf_info *ivi);
    
    static int igb_suspend(struct pci_dev *, pm_message_t);
    
    static int igb_resume(struct pci_dev *);
    #endif
    static void igb_shutdown(struct pci_dev *);
    
    #ifdef CONFIG_IGB_DCA
    
    Jeb Cramer's avatar
    Jeb Cramer committed
    static int igb_notify_dca(struct notifier_block *, unsigned long, void *);
    static struct notifier_block dca_notifier = {
    	.notifier_call	= igb_notify_dca,
    	.next		= NULL,
    	.priority	= 0
    };
    #endif
    
    #ifdef CONFIG_NET_POLL_CONTROLLER
    /* for netdump / net console */
    static void igb_netpoll(struct net_device *);
    #endif
    
    static unsigned int max_vfs = 0;
    module_param(max_vfs, uint, 0);
    MODULE_PARM_DESC(max_vfs, "Maximum number of virtual functions to allocate "
                     "per physical function");
    #endif /* CONFIG_PCI_IOV */
    
    
    static pci_ers_result_t igb_io_error_detected(struct pci_dev *,
    		     pci_channel_state_t);
    static pci_ers_result_t igb_io_slot_reset(struct pci_dev *);
    static void igb_io_resume(struct pci_dev *);
    
    static struct pci_error_handlers igb_err_handler = {
    	.error_detected = igb_io_error_detected,
    	.slot_reset = igb_io_slot_reset,
    	.resume = igb_io_resume,
    };
    
    
    static struct pci_driver igb_driver = {
    	.name     = igb_driver_name,
    	.id_table = igb_pci_tbl,
    	.probe    = igb_probe,
    	.remove   = __devexit_p(igb_remove),
    #ifdef CONFIG_PM
    	/* Power Managment Hooks */
    	.suspend  = igb_suspend,
    	.resume   = igb_resume,
    #endif
    	.shutdown = igb_shutdown,
    	.err_handler = &igb_err_handler
    };
    
    MODULE_AUTHOR("Intel Corporation, <e1000-devel@lists.sourceforge.net>");
    MODULE_DESCRIPTION("Intel(R) Gigabit Ethernet Network Driver");
    MODULE_LICENSE("GPL");
    MODULE_VERSION(DRV_VERSION);
    
    
    Patrick Ohly's avatar
    Patrick Ohly committed
    /**
     * igb_read_clock - read raw cycle counter (to be used by time counter)
     */
    static cycle_t igb_read_clock(const struct cyclecounter *tc)
    {
    	struct igb_adapter *adapter =
    		container_of(tc, struct igb_adapter, cycles);
    	struct e1000_hw *hw = &adapter->hw;
    
    	/*
    	 * The timestamp latches on lowest register read. For the 82580
    	 * the lowest register is SYSTIMR instead of SYSTIML.  However we never
    	 * adjusted TIMINCA so SYSTIMR will just read as all 0s so ignore it.
    	 */
    	if (hw->mac.type == e1000_82580) {
    		stamp = rd32(E1000_SYSTIMR) >> 8;
    		shift = IGB_82580_TSYNC_SHIFT;
    	}
    
    
    	stamp |= (u64)rd32(E1000_SYSTIML) << shift;
    	stamp |= (u64)rd32(E1000_SYSTIMH) << (shift + 32);
    
    Patrick Ohly's avatar
    Patrick Ohly committed
    	return stamp;
    }
    
    
    #ifdef DEBUG
    /**
     * igb_get_hw_dev_name - return device name string
     * used by hardware layer to print debugging information
     **/
    char *igb_get_hw_dev_name(struct e1000_hw *hw)
    {
    	struct igb_adapter *adapter = hw->back;
    	return adapter->netdev->name;
    }
    
    Patrick Ohly's avatar
    Patrick Ohly committed
    
    /**
     * igb_get_time_str - format current NIC and system time as string
     */
    static char *igb_get_time_str(struct igb_adapter *adapter,
    			      char buffer[160])
    {
    	cycle_t hw = adapter->cycles.read(&adapter->cycles);
    	struct timespec nic = ns_to_timespec(timecounter_read(&adapter->clock));
    	struct timespec sys;
    	struct timespec delta;
    	getnstimeofday(&sys);
    
    	delta = timespec_sub(nic, sys);
    
    	sprintf(buffer,
    
    		"HW %llu, NIC %ld.%09lus, SYS %ld.%09lus, NIC-SYS %lds + %09luns",
    		hw,
    
    Patrick Ohly's avatar
    Patrick Ohly committed
    		(long)nic.tv_sec, nic.tv_nsec,
    		(long)sys.tv_sec, sys.tv_nsec,
    		(long)delta.tv_sec, delta.tv_nsec);
    
    	return buffer;
    }
    
    #endif
    
    /**
     * igb_init_module - Driver Registration Routine
     *
     * igb_init_module is the first routine called when the driver is
     * loaded. All it does is register with the PCI subsystem.
     **/
    static int __init igb_init_module(void)
    {
    	int ret;
    	printk(KERN_INFO "%s - version %s\n",
    	       igb_driver_string, igb_driver_version);
    
    	printk(KERN_INFO "%s\n", igb_copyright);
    
    
    #ifdef CONFIG_IGB_DCA
    
    Jeb Cramer's avatar
    Jeb Cramer committed
    	dca_register_notify(&dca_notifier);
    #endif
    
    	ret = pci_register_driver(&igb_driver);
    
    	return ret;
    }
    
    module_init(igb_init_module);
    
    /**
     * igb_exit_module - Driver Exit Cleanup Routine
     *
     * igb_exit_module is called just before the driver is removed
     * from memory.
     **/
    static void __exit igb_exit_module(void)
    {
    
    #ifdef CONFIG_IGB_DCA
    
    Jeb Cramer's avatar
    Jeb Cramer committed
    	dca_unregister_notify(&dca_notifier);
    #endif
    
    	pci_unregister_driver(&igb_driver);
    }
    
    module_exit(igb_exit_module);
    
    
    #define Q_IDX_82576(i) (((i & 0x1) << 3) + (i >> 1))
    /**
     * igb_cache_ring_register - Descriptor ring to register mapping
     * @adapter: board private structure to initialize
     *
     * Once we know the feature-set enabled for the device, we'll cache
     * the register offset the descriptor ring is assigned to.
     **/
    static void igb_cache_ring_register(struct igb_adapter *adapter)
    {
    
    	u32 rbase_offset = adapter->vfs_allocated_count;
    
    
    	switch (adapter->hw.mac.type) {
    	case e1000_82576:
    		/* The queues are allocated for virtualization such that VF 0
    		 * is allocated queues 0 and 8, VF 1 queues 1 and 9, etc.
    		 * In order to avoid collision we start at the first free queue
    		 * and continue consuming queues in the same sequence
    		 */
    
    		if (adapter->vfs_allocated_count) {
    
    			for (; i < adapter->rss_queues; i++)
    
    				adapter->rx_ring[i].reg_idx = rbase_offset +
    				                              Q_IDX_82576(i);
    
    			for (; j < adapter->rss_queues; j++)
    
    				adapter->tx_ring[j].reg_idx = rbase_offset +
    				                              Q_IDX_82576(j);
    		}
    
    	case e1000_82580:
    
    		for (; i < adapter->num_rx_queues; i++)
    			adapter->rx_ring[i].reg_idx = rbase_offset + i;
    		for (; j < adapter->num_tx_queues; j++)
    			adapter->tx_ring[j].reg_idx = rbase_offset + j;
    
    static void igb_free_queues(struct igb_adapter *adapter)
    {
    	kfree(adapter->tx_ring);
    	kfree(adapter->rx_ring);
    
    	adapter->tx_ring = NULL;
    	adapter->rx_ring = NULL;
    
    	adapter->num_rx_queues = 0;
    	adapter->num_tx_queues = 0;
    }
    
    
    /**
     * igb_alloc_queues - Allocate memory for all rings
     * @adapter: board private structure to initialize
     *
     * We allocate one ring per queue at run-time since we don't know the
     * number of queues at compile-time.
     **/
    static int igb_alloc_queues(struct igb_adapter *adapter)
    {
    	int i;
    
    	adapter->tx_ring = kcalloc(adapter->num_tx_queues,
    				   sizeof(struct igb_ring), GFP_KERNEL);
    	if (!adapter->tx_ring)
    
    
    	adapter->rx_ring = kcalloc(adapter->num_rx_queues,
    				   sizeof(struct igb_ring), GFP_KERNEL);
    
    	if (!adapter->rx_ring)
    		goto err;
    
    	for (i = 0; i < adapter->num_tx_queues; i++) {
    		struct igb_ring *ring = &(adapter->tx_ring[i]);
    
    		ring->count = adapter->tx_ring_count;
    
    		ring->pdev = adapter->pdev;
    
    		ring->netdev = adapter->netdev;
    
    		/* For 82575, context index must be unique per ring. */
    		if (adapter->hw.mac.type == e1000_82575)
    			ring->flags = IGB_RING_FLAG_TX_CTX_IDX;
    
    	for (i = 0; i < adapter->num_rx_queues; i++) {
    		struct igb_ring *ring = &(adapter->rx_ring[i]);
    
    		ring->count = adapter->rx_ring_count;
    
    		ring->queue_index = i;
    
    		ring->pdev = adapter->pdev;
    
    		ring->netdev = adapter->netdev;
    
    		ring->rx_buffer_len = MAXIMUM_ETHERNET_VLAN_SIZE;
    
    		ring->flags = IGB_RING_FLAG_RX_CSUM; /* enable rx checksum */
    		/* set flag indicating ring supports SCTP checksum offload */
    		if (adapter->hw.mac.type >= e1000_82576)
    			ring->flags |= IGB_RING_FLAG_RX_SCTP_CSUM;
    
    
    	igb_cache_ring_register(adapter);
    
    #define IGB_N0_QUEUE -1
    
    static void igb_assign_vector(struct igb_q_vector *q_vector, int msix_vector)
    
    	struct igb_adapter *adapter = q_vector->adapter;
    
    	struct e1000_hw *hw = &adapter->hw;
    
    	u32 ivar, index;
    
    	int rx_queue = IGB_N0_QUEUE;
    	int tx_queue = IGB_N0_QUEUE;
    
    	if (q_vector->rx_ring)
    		rx_queue = q_vector->rx_ring->reg_idx;
    	if (q_vector->tx_ring)
    		tx_queue = q_vector->tx_ring->reg_idx;
    
    
    	switch (hw->mac.type) {
    	case e1000_82575:
    
    		/* The 82575 assigns vectors using a bitmask, which matches the
    		   bitmask for the EICR/EIMS/EIMC registers.  To assign one
    		   or more queues to a vector, we write the appropriate bits
    		   into the MSIXBM register for that vector. */
    
    		if (rx_queue > IGB_N0_QUEUE)
    
    			msixbm = E1000_EICR_RX_QUEUE0 << rx_queue;
    
    		if (tx_queue > IGB_N0_QUEUE)
    
    			msixbm |= E1000_EICR_TX_QUEUE0 << tx_queue;
    
    		if (!adapter->msix_entries && msix_vector == 0)
    			msixbm |= E1000_EIMS_OTHER;
    
    		array_wr32(E1000_MSIXBM(0), msix_vector, msixbm);
    
    		q_vector->eims_value = msixbm;
    
    		break;
    	case e1000_82576:
    
    		/* 82576 uses a table-based method for assigning vectors.
    
    		   Each queue has a single entry in the table to which we write
    		   a vector number along with a "valid" bit.  Sadly, the layout
    		   of the table is somewhat counterintuitive. */
    		if (rx_queue > IGB_N0_QUEUE) {
    
    			ivar = array_rd32(E1000_IVAR0, index);
    
    				/* vector goes into low byte of register */
    				ivar = ivar & 0xFFFFFF00;
    				ivar |= msix_vector | E1000_IVAR_VALID;
    
    			} else {
    				/* vector goes into third byte of register */
    				ivar = ivar & 0xFF00FFFF;
    				ivar |= (msix_vector | E1000_IVAR_VALID) << 16;
    
    			}
    			array_wr32(E1000_IVAR0, index, ivar);
    		}
    		if (tx_queue > IGB_N0_QUEUE) {
    
    			ivar = array_rd32(E1000_IVAR0, index);
    
    				/* vector goes into second byte of register */
    				ivar = ivar & 0xFFFF00FF;
    				ivar |= (msix_vector | E1000_IVAR_VALID) << 8;
    
    			} else {
    				/* vector goes into high byte of register */
    				ivar = ivar & 0x00FFFFFF;
    				ivar |= (msix_vector | E1000_IVAR_VALID) << 24;
    
    			}
    			array_wr32(E1000_IVAR0, index, ivar);
    		}
    
    		q_vector->eims_value = 1 << msix_vector;
    
    		break;
    
    	case e1000_82580:
    		/* 82580 uses the same table-based approach as 82576 but has fewer
    		   entries as a result we carry over for queues greater than 4. */
    		if (rx_queue > IGB_N0_QUEUE) {
    			index = (rx_queue >> 1);
    			ivar = array_rd32(E1000_IVAR0, index);
    			if (rx_queue & 0x1) {
    				/* vector goes into third byte of register */
    				ivar = ivar & 0xFF00FFFF;
    				ivar |= (msix_vector | E1000_IVAR_VALID) << 16;
    			} else {
    				/* vector goes into low byte of register */
    				ivar = ivar & 0xFFFFFF00;
    				ivar |= msix_vector | E1000_IVAR_VALID;
    			}
    			array_wr32(E1000_IVAR0, index, ivar);
    		}
    		if (tx_queue > IGB_N0_QUEUE) {
    			index = (tx_queue >> 1);
    			ivar = array_rd32(E1000_IVAR0, index);
    			if (tx_queue & 0x1) {
    				/* vector goes into high byte of register */
    				ivar = ivar & 0x00FFFFFF;
    				ivar |= (msix_vector | E1000_IVAR_VALID) << 24;
    			} else {
    				/* vector goes into second byte of register */
    				ivar = ivar & 0xFFFF00FF;
    				ivar |= (msix_vector | E1000_IVAR_VALID) << 8;
    			}
    			array_wr32(E1000_IVAR0, index, ivar);
    		}
    		q_vector->eims_value = 1 << msix_vector;
    		break;
    
    	default:
    		BUG();
    		break;
    	}
    
    
    	/* add q_vector eims value to global eims_enable_mask */
    	adapter->eims_enable_mask |= q_vector->eims_value;
    
    	/* configure q_vector to set itr on first interrupt */
    	q_vector->set_itr = 1;
    
    }
    
    /**
     * igb_configure_msix - Configure MSI-X hardware
     *
     * igb_configure_msix sets up the hardware to properly
     * generate MSI-X interrupts.
     **/
    static void igb_configure_msix(struct igb_adapter *adapter)
    {
    	u32 tmp;
    	int i, vector = 0;
    	struct e1000_hw *hw = &adapter->hw;
    
    	adapter->eims_enable_mask = 0;
    
    	/* set vector for other causes, i.e. link changes */
    
    	switch (hw->mac.type) {
    	case e1000_82575:
    
    		tmp = rd32(E1000_CTRL_EXT);
    		/* enable MSI-X PBA support*/
    		tmp |= E1000_CTRL_EXT_PBA_CLR;
    
    		/* Auto-Mask interrupts upon ICR read. */
    		tmp |= E1000_CTRL_EXT_EIAME;
    		tmp |= E1000_CTRL_EXT_IRCA;
    
    		wr32(E1000_CTRL_EXT, tmp);
    
    
    		/* enable msix_other interrupt */
    		array_wr32(E1000_MSIXBM(0), vector++,
    		                      E1000_EIMS_OTHER);
    
    		adapter->eims_other = E1000_EIMS_OTHER;
    
    		break;
    
    	case e1000_82576:
    
    	case e1000_82580:
    
    		/* Turn on MSI-X capability first, or our settings
    		 * won't stick.  And it will take days to debug. */
    		wr32(E1000_GPIE, E1000_GPIE_MSIX_MODE |
    		                E1000_GPIE_PBA | E1000_GPIE_EIAME |
    		                E1000_GPIE_NSICR);
    
    		/* enable msix_other interrupt */
    		adapter->eims_other = 1 << vector;
    
    		tmp = (vector++ | E1000_IVAR_VALID) << 8;
    
    
    		break;
    	default:
    		/* do nothing, since nothing else supports MSI-X */
    		break;
    	} /* switch (hw->mac.type) */
    
    
    	adapter->eims_enable_mask |= adapter->eims_other;
    
    
    	for (i = 0; i < adapter->num_q_vectors; i++)
    		igb_assign_vector(adapter->q_vector[i], vector++);
    
    	wrfl();
    }
    
    /**
     * igb_request_msix - Initialize MSI-X interrupts
     *
     * igb_request_msix allocates MSI-X vectors and requests interrupts from the
     * kernel.
     **/
    static int igb_request_msix(struct igb_adapter *adapter)
    {
    	struct net_device *netdev = adapter->netdev;
    
    	struct e1000_hw *hw = &adapter->hw;
    
    	int i, err = 0, vector = 0;
    
    
    	err = request_irq(adapter->msix_entries[vector].vector,
    
    	                  igb_msix_other, 0, netdev->name, adapter);
    
    	if (err)
    		goto out;
    	vector++;
    
    	for (i = 0; i < adapter->num_q_vectors; i++) {
    		struct igb_q_vector *q_vector = adapter->q_vector[i];
    
    		q_vector->itr_register = hw->hw_addr + E1000_EITR(vector);
    
    		if (q_vector->rx_ring && q_vector->tx_ring)
    			sprintf(q_vector->name, "%s-TxRx-%u", netdev->name,
    			        q_vector->rx_ring->queue_index);
    		else if (q_vector->tx_ring)
    			sprintf(q_vector->name, "%s-tx-%u", netdev->name,
    			        q_vector->tx_ring->queue_index);
    		else if (q_vector->rx_ring)
    			sprintf(q_vector->name, "%s-rx-%u", netdev->name,
    			        q_vector->rx_ring->queue_index);
    
    			sprintf(q_vector->name, "%s-unused", netdev->name);
    
    
    		err = request_irq(adapter->msix_entries[vector].vector,
    
    		                  igb_msix_ring, 0, q_vector->name,
    
    		if (err)
    			goto out;
    		vector++;
    	}
    
    	igb_configure_msix(adapter);
    	return 0;
    out:
    	return err;
    }
    
    static void igb_reset_interrupt_capability(struct igb_adapter *adapter)
    {
    	if (adapter->msix_entries) {
    		pci_disable_msix(adapter->pdev);
    		kfree(adapter->msix_entries);
    		adapter->msix_entries = NULL;
    
    	} else if (adapter->flags & IGB_FLAG_HAS_MSI) {
    
    		pci_disable_msi(adapter->pdev);
    
    /**
     * igb_free_q_vectors - Free memory allocated for interrupt vectors
     * @adapter: board private structure to initialize
     *
     * This function frees the memory allocated to the q_vectors.  In addition if
     * NAPI is enabled it will delete any references to the NAPI struct prior
     * to freeing the q_vector.
     **/
    static void igb_free_q_vectors(struct igb_adapter *adapter)
    {
    	int v_idx;
    
    	for (v_idx = 0; v_idx < adapter->num_q_vectors; v_idx++) {
    		struct igb_q_vector *q_vector = adapter->q_vector[v_idx];
    		adapter->q_vector[v_idx] = NULL;
    		netif_napi_del(&q_vector->napi);
    		kfree(q_vector);
    	}
    	adapter->num_q_vectors = 0;
    }
    
    /**
     * igb_clear_interrupt_scheme - reset the device to a state of no interrupts
     *
     * This function resets the device so that it has 0 rx queues, tx queues, and
     * MSI-X interrupts allocated.
     */
    static void igb_clear_interrupt_scheme(struct igb_adapter *adapter)
    {
    	igb_free_queues(adapter);
    	igb_free_q_vectors(adapter);
    	igb_reset_interrupt_capability(adapter);
    }
    
    
    /**
     * igb_set_interrupt_capability - set MSI or MSI-X if supported
     *
     * Attempt to configure interrupts using the best available
     * capabilities of the hardware and kernel.
     **/
    static void igb_set_interrupt_capability(struct igb_adapter *adapter)
    {
    	int err;
    	int numvecs, i;
    
    
    	adapter->num_rx_queues = adapter->rss_queues;
    	adapter->num_tx_queues = adapter->rss_queues;
    
    	/* start with one vector for every rx queue */
    	numvecs = adapter->num_rx_queues;
    
    	/* if tx handler is seperate add 1 for every tx queue */
    
    	if (!(adapter->flags & IGB_FLAG_QUEUE_PAIRS))
    		numvecs += adapter->num_tx_queues;
    
    
    	/* store the number of vectors reserved for queues */
    	adapter->num_q_vectors = numvecs;
    
    	/* add 1 vector for link status interrupts */
    	numvecs++;
    
    	adapter->msix_entries = kcalloc(numvecs, sizeof(struct msix_entry),
    					GFP_KERNEL);
    	if (!adapter->msix_entries)
    		goto msi_only;
    
    	for (i = 0; i < numvecs; i++)
    		adapter->msix_entries[i].entry = i;
    
    	err = pci_enable_msix(adapter->pdev,
    			      adapter->msix_entries,
    			      numvecs);
    	if (err == 0)
    
    
    	igb_reset_interrupt_capability(adapter);
    
    	/* If we can't do MSI-X, try MSI */
    msi_only:
    
    #ifdef CONFIG_PCI_IOV
    	/* disable SR-IOV for non MSI-X configurations */
    	if (adapter->vf_data) {
    		struct e1000_hw *hw = &adapter->hw;
    		/* disable iov and allow time for transactions to clear */
    		pci_disable_sriov(adapter->pdev);
    		msleep(500);
    
    		kfree(adapter->vf_data);
    		adapter->vf_data = NULL;
    		wr32(E1000_IOVCTL, E1000_IOVCTL_REUSE_VFQ);
    		msleep(100);
    		dev_info(&adapter->pdev->dev, "IOV Disabled\n");
    	}
    #endif
    
    	adapter->vfs_allocated_count = 0;
    
    	adapter->flags |= IGB_FLAG_QUEUE_PAIRS;
    
    	adapter->num_rx_queues = 1;
    
    	adapter->num_tx_queues = 1;
    
    	if (!pci_enable_msi(adapter->pdev))
    
    		adapter->flags |= IGB_FLAG_HAS_MSI;
    
    	/* Notify the stack of the (possibly) reduced Tx Queue count. */
    
    	adapter->netdev->real_num_tx_queues = adapter->num_tx_queues;
    
    /**
     * igb_alloc_q_vectors - Allocate memory for interrupt vectors
     * @adapter: board private structure to initialize
     *
     * We allocate one q_vector per queue interrupt.  If allocation fails we
     * return -ENOMEM.
     **/
    static int igb_alloc_q_vectors(struct igb_adapter *adapter)
    {
    	struct igb_q_vector *q_vector;
    	struct e1000_hw *hw = &adapter->hw;
    	int v_idx;
    
    	for (v_idx = 0; v_idx < adapter->num_q_vectors; v_idx++) {
    		q_vector = kzalloc(sizeof(struct igb_q_vector), GFP_KERNEL);
    		if (!q_vector)
    			goto err_out;
    		q_vector->adapter = adapter;
    		q_vector->itr_register = hw->hw_addr + E1000_EITR(0);
    		q_vector->itr_val = IGB_START_ITR;
    		netif_napi_add(adapter->netdev, &q_vector->napi, igb_poll, 64);
    		adapter->q_vector[v_idx] = q_vector;
    	}
    	return 0;
    
    err_out:
    	while (v_idx) {
    		v_idx--;
    		q_vector = adapter->q_vector[v_idx];
    		netif_napi_del(&q_vector->napi);
    		kfree(q_vector);
    		adapter->q_vector[v_idx] = NULL;
    	}
    	return -ENOMEM;
    }
    
    static void igb_map_rx_ring_to_vector(struct igb_adapter *adapter,
                                          int ring_idx, int v_idx)
    {
    	struct igb_q_vector *q_vector;
    
    	q_vector = adapter->q_vector[v_idx];
    	q_vector->rx_ring = &adapter->rx_ring[ring_idx];
    	q_vector->rx_ring->q_vector = q_vector;
    
    	q_vector->itr_val = adapter->rx_itr_setting;
    	if (q_vector->itr_val && q_vector->itr_val <= 3)
    		q_vector->itr_val = IGB_START_ITR;
    
    }
    
    static void igb_map_tx_ring_to_vector(struct igb_adapter *adapter,
                                          int ring_idx, int v_idx)
    {
    	struct igb_q_vector *q_vector;
    
    	q_vector = adapter->q_vector[v_idx];
    	q_vector->tx_ring = &adapter->tx_ring[ring_idx];
    	q_vector->tx_ring->q_vector = q_vector;
    
    	q_vector->itr_val = adapter->tx_itr_setting;
    	if (q_vector->itr_val && q_vector->itr_val <= 3)
    		q_vector->itr_val = IGB_START_ITR;
    
    }
    
    /**
     * igb_map_ring_to_vector - maps allocated queues to vectors
     *
     * This function maps the recently allocated queues to vectors.
     **/
    static int igb_map_ring_to_vector(struct igb_adapter *adapter)
    {
    	int i;
    	int v_idx = 0;
    
    	if ((adapter->num_q_vectors < adapter->num_rx_queues) ||
    	    (adapter->num_q_vectors < adapter->num_tx_queues))
    		return -ENOMEM;
    
    	if (adapter->num_q_vectors >=
    	    (adapter->num_rx_queues + adapter->num_tx_queues)) {
    		for (i = 0; i < adapter->num_rx_queues; i++)
    			igb_map_rx_ring_to_vector(adapter, i, v_idx++);
    		for (i = 0; i < adapter->num_tx_queues; i++)
    			igb_map_tx_ring_to_vector(adapter, i, v_idx++);
    	} else {
    		for (i = 0; i < adapter->num_rx_queues; i++) {
    			if (i < adapter->num_tx_queues)
    				igb_map_tx_ring_to_vector(adapter, i, v_idx);
    			igb_map_rx_ring_to_vector(adapter, i, v_idx++);
    		}
    		for (; i < adapter->num_tx_queues; i++)
    			igb_map_tx_ring_to_vector(adapter, i, v_idx++);
    	}
    	return 0;
    }
    
    /**
     * igb_init_interrupt_scheme - initialize interrupts, allocate queues/vectors
     *
     * This function initializes the interrupts and allocates all of the queues.
     **/
    static int igb_init_interrupt_scheme(struct igb_adapter *adapter)
    {
    	struct pci_dev *pdev = adapter->pdev;
    	int err;
    
    	igb_set_interrupt_capability(adapter);
    
    	err = igb_alloc_q_vectors(adapter);
    	if (err) {
    		dev_err(&pdev->dev, "Unable to allocate memory for vectors\n");
    		goto err_alloc_q_vectors;
    	}
    
    	err = igb_alloc_queues(adapter);
    	if (err) {
    		dev_err(&pdev->dev, "Unable to allocate memory for queues\n");
    		goto err_alloc_queues;
    	}
    
    	err = igb_map_ring_to_vector(adapter);
    	if (err) {
    		dev_err(&pdev->dev, "Invalid q_vector to ring mapping\n");
    		goto err_map_queues;
    	}
    
    
    	return 0;
    err_map_queues:
    	igb_free_queues(adapter);
    err_alloc_queues:
    	igb_free_q_vectors(adapter);
    err_alloc_q_vectors:
    	igb_reset_interrupt_capability(adapter);
    	return err;
    }
    
    
    /**
     * igb_request_irq - initialize interrupts
     *
     * Attempts to configure interrupts using the best available
     * capabilities of the hardware and kernel.
     **/
    static int igb_request_irq(struct igb_adapter *adapter)
    {
    	struct net_device *netdev = adapter->netdev;
    
    	struct pci_dev *pdev = adapter->pdev;
    
    	int err = 0;
    
    	if (adapter->msix_entries) {
    		err = igb_request_msix(adapter);
    
    		if (!err)
    
    			goto request_done;
    		/* fall back to MSI */
    
    		igb_clear_interrupt_scheme(adapter);
    
    		if (!pci_enable_msi(adapter->pdev))
    
    			adapter->flags |= IGB_FLAG_HAS_MSI;
    
    		igb_free_all_tx_resources(adapter);
    		igb_free_all_rx_resources(adapter);
    
    		adapter->num_rx_queues = 1;
    
    		adapter->num_q_vectors = 1;
    		err = igb_alloc_q_vectors(adapter);
    		if (err) {
    			dev_err(&pdev->dev,
    			        "Unable to allocate memory for vectors\n");
    			goto request_done;
    		}
    		err = igb_alloc_queues(adapter);
    		if (err) {
    			dev_err(&pdev->dev,
    			        "Unable to allocate memory for queues\n");
    			igb_free_q_vectors(adapter);
    			goto request_done;
    		}
    		igb_setup_all_tx_resources(adapter);
    		igb_setup_all_rx_resources(adapter);
    
    	} else {
    
    		igb_assign_vector(adapter->q_vector[0], 0);
    
    	if (adapter->flags & IGB_FLAG_HAS_MSI) {
    
    		err = request_irq(adapter->pdev->irq, igb_intr_msi, 0,
    
    		if (!err)
    			goto request_done;
    
    		/* fall back to legacy interrupts */
    		igb_reset_interrupt_capability(adapter);
    
    		adapter->flags &= ~IGB_FLAG_HAS_MSI;
    
    	err = request_irq(adapter->pdev->irq, igb_intr, IRQF_SHARED,
    
    	if (err)
    
    		dev_err(&adapter->pdev->dev, "Error %d getting interrupt\n",
    			err);
    
    request_done:
    	return err;
    }
    
    static void igb_free_irq(struct igb_adapter *adapter)
    {
    	if (adapter->msix_entries) {
    		int vector = 0, i;
    
    
    		free_irq(adapter->msix_entries[vector++].vector, adapter);
    
    		for (i = 0; i < adapter->num_q_vectors; i++) {
    			struct igb_q_vector *q_vector = adapter->q_vector[i];
    			free_irq(adapter->msix_entries[vector++].vector,
    			         q_vector);
    		}
    	} else {
    		free_irq(adapter->pdev->irq, adapter);
    
    	}
    }
    
    /**
     * igb_irq_disable - Mask off interrupt generation on the NIC
     * @adapter: board private structure
     **/
    static void igb_irq_disable(struct igb_adapter *adapter)
    {
    	struct e1000_hw *hw = &adapter->hw;
    
    
    	/*
    	 * we need to be careful when disabling interrupts.  The VFs are also
    	 * mapped into these registers and so clearing the bits can cause
    	 * issues on the VF drivers so we only need to clear what we set
    	 */
    
    	if (adapter->msix_entries) {
    
    		u32 regval = rd32(E1000_EIAM);
    		wr32(E1000_EIAM, regval & ~adapter->eims_enable_mask);
    		wr32(E1000_EIMC, adapter->eims_enable_mask);
    		regval = rd32(E1000_EIAC);
    		wr32(E1000_EIAC, regval & ~adapter->eims_enable_mask);
    
    
    	wr32(E1000_IAM, 0);
    
    	wr32(E1000_IMC, ~0);
    	wrfl();
    	synchronize_irq(adapter->pdev->irq);
    }
    
    /**
     * igb_irq_enable - Enable default interrupt generation settings
     * @adapter: board private structure
     **/
    static void igb_irq_enable(struct igb_adapter *adapter)
    {
    	struct e1000_hw *hw = &adapter->hw;
    
    	if (adapter->msix_entries) {
    
    		u32 ims = E1000_IMS_LSC | E1000_IMS_DOUTSYNC;
    
    		u32 regval = rd32(E1000_EIAC);
    		wr32(E1000_EIAC, regval | adapter->eims_enable_mask);