|
@@ -60,8 +60,8 @@
|
|
|
#include "igb.h"
|
|
|
|
|
|
#define MAJ 4
|
|
|
-#define MIN 0
|
|
|
-#define BUILD 17
|
|
|
+#define MIN 1
|
|
|
+#define BUILD 2
|
|
|
#define DRV_VERSION __stringify(MAJ) "." __stringify(MIN) "." \
|
|
|
__stringify(BUILD) "-k"
|
|
|
char igb_driver_name[] = "igb";
|
|
@@ -122,6 +122,7 @@ static void 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(struct igb_adapter *);
|
|
|
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 *);
|
|
@@ -831,17 +832,18 @@ 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;
|
|
|
+ int i, err = 0, vector = 0, free_vector = 0;
|
|
|
|
|
|
err = request_irq(adapter->msix_entries[vector].vector,
|
|
|
igb_msix_other, 0, netdev->name, adapter);
|
|
|
if (err)
|
|
|
- goto out;
|
|
|
- vector++;
|
|
|
+ goto err_out;
|
|
|
|
|
|
for (i = 0; i < adapter->num_q_vectors; i++) {
|
|
|
struct igb_q_vector *q_vector = adapter->q_vector[i];
|
|
|
|
|
|
+ vector++;
|
|
|
+
|
|
|
q_vector->itr_register = hw->hw_addr + E1000_EITR(vector);
|
|
|
|
|
|
if (q_vector->rx.ring && q_vector->tx.ring)
|
|
@@ -860,13 +862,22 @@ static int igb_request_msix(struct igb_adapter *adapter)
|
|
|
igb_msix_ring, 0, q_vector->name,
|
|
|
q_vector);
|
|
|
if (err)
|
|
|
- goto out;
|
|
|
- vector++;
|
|
|
+ goto err_free;
|
|
|
}
|
|
|
|
|
|
igb_configure_msix(adapter);
|
|
|
return 0;
|
|
|
-out:
|
|
|
+
|
|
|
+err_free:
|
|
|
+ /* free already assigned IRQs */
|
|
|
+ free_irq(adapter->msix_entries[free_vector++].vector, adapter);
|
|
|
+
|
|
|
+ vector--;
|
|
|
+ for (i = 0; i < vector; i++) {
|
|
|
+ free_irq(adapter->msix_entries[free_vector++].vector,
|
|
|
+ adapter->q_vector[i]);
|
|
|
+ }
|
|
|
+err_out:
|
|
|
return err;
|
|
|
}
|
|
|
|
|
@@ -948,11 +959,14 @@ static void igb_clear_interrupt_scheme(struct igb_adapter *adapter)
|
|
|
* Attempt to configure interrupts using the best available
|
|
|
* capabilities of the hardware and kernel.
|
|
|
**/
|
|
|
-static void igb_set_interrupt_capability(struct igb_adapter *adapter)
|
|
|
+static void igb_set_interrupt_capability(struct igb_adapter *adapter, bool msix)
|
|
|
{
|
|
|
int err;
|
|
|
int numvecs, i;
|
|
|
|
|
|
+ if (!msix)
|
|
|
+ goto msi_only;
|
|
|
+
|
|
|
/* Number of supported queues. */
|
|
|
adapter->num_rx_queues = adapter->rss_queues;
|
|
|
if (adapter->vfs_allocated_count)
|
|
@@ -1199,12 +1213,12 @@ err_out:
|
|
|
*
|
|
|
* This function initializes the interrupts and allocates all of the queues.
|
|
|
**/
|
|
|
-static int igb_init_interrupt_scheme(struct igb_adapter *adapter)
|
|
|
+static int igb_init_interrupt_scheme(struct igb_adapter *adapter, bool msix)
|
|
|
{
|
|
|
struct pci_dev *pdev = adapter->pdev;
|
|
|
int err;
|
|
|
|
|
|
- igb_set_interrupt_capability(adapter);
|
|
|
+ igb_set_interrupt_capability(adapter, msix);
|
|
|
|
|
|
err = igb_alloc_q_vectors(adapter);
|
|
|
if (err) {
|
|
@@ -1240,20 +1254,15 @@ static int igb_request_irq(struct igb_adapter *adapter)
|
|
|
/* fall back to MSI */
|
|
|
igb_free_all_tx_resources(adapter);
|
|
|
igb_free_all_rx_resources(adapter);
|
|
|
+
|
|
|
igb_clear_interrupt_scheme(adapter);
|
|
|
- if (!pci_enable_msi(pdev))
|
|
|
- adapter->flags |= IGB_FLAG_HAS_MSI;
|
|
|
- adapter->num_tx_queues = 1;
|
|
|
- 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");
|
|
|
+ err = igb_init_interrupt_scheme(adapter, false);
|
|
|
+ if (err)
|
|
|
goto request_done;
|
|
|
- }
|
|
|
+
|
|
|
igb_setup_all_tx_resources(adapter);
|
|
|
igb_setup_all_rx_resources(adapter);
|
|
|
+ igb_configure(adapter);
|
|
|
}
|
|
|
|
|
|
igb_assign_vector(adapter->q_vector[0], 0);
|
|
@@ -2444,7 +2453,7 @@ static int igb_sw_init(struct igb_adapter *adapter)
|
|
|
GFP_ATOMIC);
|
|
|
|
|
|
/* This call may decrease the number of queues */
|
|
|
- if (igb_init_interrupt_scheme(adapter)) {
|
|
|
+ if (igb_init_interrupt_scheme(adapter, true)) {
|
|
|
dev_err(&pdev->dev, "Unable to allocate memory for queues\n");
|
|
|
return -ENOMEM;
|
|
|
}
|
|
@@ -6818,7 +6827,7 @@ static int igb_resume(struct device *dev)
|
|
|
pci_enable_wake(pdev, PCI_D3hot, 0);
|
|
|
pci_enable_wake(pdev, PCI_D3cold, 0);
|
|
|
|
|
|
- if (igb_init_interrupt_scheme(adapter)) {
|
|
|
+ if (igb_init_interrupt_scheme(adapter, true)) {
|
|
|
dev_err(&pdev->dev, "Unable to allocate memory for queues\n");
|
|
|
return -ENOMEM;
|
|
|
}
|