mirror of
https://github.com/torvalds/linux.git
synced 2024-12-04 01:51:34 +00:00
ixgbe: Add SR-IOV features to main module
Adds SR-IOV features supported by the 82599 controller to the main driver module. If the CONFIG_PCI_IOV kernel option is selected then the SR-IOV features are enabled. Use the max_vfs module option to allocate up to 63 virtual functions per physical port. Signed-off-by: Greg Rose <gregory.v.rose@intel.com> Signed-off-by: Jeff Kirsher <jeffrey.t.kirsher@intel.com> Signed-off-by: David S. Miller <davem@davemloft.net>
This commit is contained in:
parent
096a58fdec
commit
1cdd1ec878
@ -45,6 +45,7 @@
|
|||||||
#include "ixgbe.h"
|
#include "ixgbe.h"
|
||||||
#include "ixgbe_common.h"
|
#include "ixgbe_common.h"
|
||||||
#include "ixgbe_dcb_82599.h"
|
#include "ixgbe_dcb_82599.h"
|
||||||
|
#include "ixgbe_sriov.h"
|
||||||
|
|
||||||
char ixgbe_driver_name[] = "ixgbe";
|
char ixgbe_driver_name[] = "ixgbe";
|
||||||
static const char ixgbe_driver_string[] =
|
static const char ixgbe_driver_string[] =
|
||||||
@ -124,6 +125,13 @@ static struct notifier_block dca_notifier = {
|
|||||||
};
|
};
|
||||||
#endif
|
#endif
|
||||||
|
|
||||||
|
#ifdef CONFIG_PCI_IOV
|
||||||
|
static unsigned int max_vfs;
|
||||||
|
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 */
|
||||||
|
|
||||||
MODULE_AUTHOR("Intel Corporation, <linux.nics@intel.com>");
|
MODULE_AUTHOR("Intel Corporation, <linux.nics@intel.com>");
|
||||||
MODULE_DESCRIPTION("Intel(R) 10 Gigabit PCI Express Network Driver");
|
MODULE_DESCRIPTION("Intel(R) 10 Gigabit PCI Express Network Driver");
|
||||||
MODULE_LICENSE("GPL");
|
MODULE_LICENSE("GPL");
|
||||||
@ -131,6 +139,41 @@ MODULE_VERSION(DRV_VERSION);
|
|||||||
|
|
||||||
#define DEFAULT_DEBUG_LEVEL_SHIFT 3
|
#define DEFAULT_DEBUG_LEVEL_SHIFT 3
|
||||||
|
|
||||||
|
static inline void ixgbe_disable_sriov(struct ixgbe_adapter *adapter)
|
||||||
|
{
|
||||||
|
struct ixgbe_hw *hw = &adapter->hw;
|
||||||
|
u32 gcr;
|
||||||
|
u32 gpie;
|
||||||
|
u32 vmdctl;
|
||||||
|
|
||||||
|
#ifdef CONFIG_PCI_IOV
|
||||||
|
/* disable iov and allow time for transactions to clear */
|
||||||
|
pci_disable_sriov(adapter->pdev);
|
||||||
|
#endif
|
||||||
|
|
||||||
|
/* turn off device IOV mode */
|
||||||
|
gcr = IXGBE_READ_REG(hw, IXGBE_GCR_EXT);
|
||||||
|
gcr &= ~(IXGBE_GCR_EXT_SRIOV);
|
||||||
|
IXGBE_WRITE_REG(hw, IXGBE_GCR_EXT, gcr);
|
||||||
|
gpie = IXGBE_READ_REG(hw, IXGBE_GPIE);
|
||||||
|
gpie &= ~IXGBE_GPIE_VTMODE_MASK;
|
||||||
|
IXGBE_WRITE_REG(hw, IXGBE_GPIE, gpie);
|
||||||
|
|
||||||
|
/* set default pool back to 0 */
|
||||||
|
vmdctl = IXGBE_READ_REG(hw, IXGBE_VT_CTL);
|
||||||
|
vmdctl &= ~IXGBE_VT_CTL_POOL_MASK;
|
||||||
|
IXGBE_WRITE_REG(hw, IXGBE_VT_CTL, vmdctl);
|
||||||
|
|
||||||
|
/* take a breather then clean up driver data */
|
||||||
|
msleep(100);
|
||||||
|
if (adapter->vfinfo)
|
||||||
|
kfree(adapter->vfinfo);
|
||||||
|
adapter->vfinfo = NULL;
|
||||||
|
|
||||||
|
adapter->num_vfs = 0;
|
||||||
|
adapter->flags &= ~IXGBE_FLAG_SRIOV_ENABLED;
|
||||||
|
}
|
||||||
|
|
||||||
static void ixgbe_release_hw_control(struct ixgbe_adapter *adapter)
|
static void ixgbe_release_hw_control(struct ixgbe_adapter *adapter)
|
||||||
{
|
{
|
||||||
u32 ctrl_ext;
|
u32 ctrl_ext;
|
||||||
@ -1020,7 +1063,12 @@ static void ixgbe_configure_msix(struct ixgbe_adapter *adapter)
|
|||||||
|
|
||||||
/* set up to autoclear timer, and the vectors */
|
/* set up to autoclear timer, and the vectors */
|
||||||
mask = IXGBE_EIMS_ENABLE_MASK;
|
mask = IXGBE_EIMS_ENABLE_MASK;
|
||||||
mask &= ~(IXGBE_EIMS_OTHER | IXGBE_EIMS_LSC);
|
if (adapter->num_vfs)
|
||||||
|
mask &= ~(IXGBE_EIMS_OTHER |
|
||||||
|
IXGBE_EIMS_MAILBOX |
|
||||||
|
IXGBE_EIMS_LSC);
|
||||||
|
else
|
||||||
|
mask &= ~(IXGBE_EIMS_OTHER | IXGBE_EIMS_LSC);
|
||||||
IXGBE_WRITE_REG(&adapter->hw, IXGBE_EIAC, mask);
|
IXGBE_WRITE_REG(&adapter->hw, IXGBE_EIAC, mask);
|
||||||
}
|
}
|
||||||
|
|
||||||
@ -1249,6 +1297,9 @@ static irqreturn_t ixgbe_msix_lsc(int irq, void *data)
|
|||||||
if (eicr & IXGBE_EICR_LSC)
|
if (eicr & IXGBE_EICR_LSC)
|
||||||
ixgbe_check_lsc(adapter);
|
ixgbe_check_lsc(adapter);
|
||||||
|
|
||||||
|
if (eicr & IXGBE_EICR_MAILBOX)
|
||||||
|
ixgbe_msg_task(adapter);
|
||||||
|
|
||||||
if (hw->mac.type == ixgbe_mac_82598EB)
|
if (hw->mac.type == ixgbe_mac_82598EB)
|
||||||
ixgbe_check_fan_failure(adapter, eicr);
|
ixgbe_check_fan_failure(adapter, eicr);
|
||||||
|
|
||||||
@ -1763,6 +1814,8 @@ static inline void ixgbe_irq_enable(struct ixgbe_adapter *adapter)
|
|||||||
mask |= IXGBE_EIMS_ECC;
|
mask |= IXGBE_EIMS_ECC;
|
||||||
mask |= IXGBE_EIMS_GPI_SDP1;
|
mask |= IXGBE_EIMS_GPI_SDP1;
|
||||||
mask |= IXGBE_EIMS_GPI_SDP2;
|
mask |= IXGBE_EIMS_GPI_SDP2;
|
||||||
|
if (adapter->num_vfs)
|
||||||
|
mask |= IXGBE_EIMS_MAILBOX;
|
||||||
}
|
}
|
||||||
if (adapter->flags & IXGBE_FLAG_FDIR_HASH_CAPABLE ||
|
if (adapter->flags & IXGBE_FLAG_FDIR_HASH_CAPABLE ||
|
||||||
adapter->flags & IXGBE_FLAG_FDIR_PERFECT_CAPABLE)
|
adapter->flags & IXGBE_FLAG_FDIR_PERFECT_CAPABLE)
|
||||||
@ -1771,6 +1824,11 @@ static inline void ixgbe_irq_enable(struct ixgbe_adapter *adapter)
|
|||||||
IXGBE_WRITE_REG(&adapter->hw, IXGBE_EIMS, mask);
|
IXGBE_WRITE_REG(&adapter->hw, IXGBE_EIMS, mask);
|
||||||
ixgbe_irq_enable_queues(adapter, ~0);
|
ixgbe_irq_enable_queues(adapter, ~0);
|
||||||
IXGBE_WRITE_FLUSH(&adapter->hw);
|
IXGBE_WRITE_FLUSH(&adapter->hw);
|
||||||
|
|
||||||
|
if (adapter->num_vfs > 32) {
|
||||||
|
u32 eitrsel = (1 << (adapter->num_vfs - 32)) - 1;
|
||||||
|
IXGBE_WRITE_REG(&adapter->hw, IXGBE_EITRSEL, eitrsel);
|
||||||
|
}
|
||||||
}
|
}
|
||||||
|
|
||||||
/**
|
/**
|
||||||
@ -1900,6 +1958,8 @@ static inline void ixgbe_irq_disable(struct ixgbe_adapter *adapter)
|
|||||||
IXGBE_WRITE_REG(&adapter->hw, IXGBE_EIMC, 0xFFFF0000);
|
IXGBE_WRITE_REG(&adapter->hw, IXGBE_EIMC, 0xFFFF0000);
|
||||||
IXGBE_WRITE_REG(&adapter->hw, IXGBE_EIMC_EX(0), ~0);
|
IXGBE_WRITE_REG(&adapter->hw, IXGBE_EIMC_EX(0), ~0);
|
||||||
IXGBE_WRITE_REG(&adapter->hw, IXGBE_EIMC_EX(1), ~0);
|
IXGBE_WRITE_REG(&adapter->hw, IXGBE_EIMC_EX(1), ~0);
|
||||||
|
if (adapter->num_vfs > 32)
|
||||||
|
IXGBE_WRITE_REG(&adapter->hw, IXGBE_EITRSEL, 0);
|
||||||
}
|
}
|
||||||
IXGBE_WRITE_FLUSH(&adapter->hw);
|
IXGBE_WRITE_FLUSH(&adapter->hw);
|
||||||
if (adapter->flags & IXGBE_FLAG_MSIX_ENABLED) {
|
if (adapter->flags & IXGBE_FLAG_MSIX_ENABLED) {
|
||||||
@ -1984,18 +2044,32 @@ static void ixgbe_configure_tx(struct ixgbe_adapter *adapter)
|
|||||||
|
|
||||||
if (hw->mac.type == ixgbe_mac_82599EB) {
|
if (hw->mac.type == ixgbe_mac_82599EB) {
|
||||||
u32 rttdcs;
|
u32 rttdcs;
|
||||||
|
u32 mask;
|
||||||
|
|
||||||
/* disable the arbiter while setting MTQC */
|
/* disable the arbiter while setting MTQC */
|
||||||
rttdcs = IXGBE_READ_REG(hw, IXGBE_RTTDCS);
|
rttdcs = IXGBE_READ_REG(hw, IXGBE_RTTDCS);
|
||||||
rttdcs |= IXGBE_RTTDCS_ARBDIS;
|
rttdcs |= IXGBE_RTTDCS_ARBDIS;
|
||||||
IXGBE_WRITE_REG(hw, IXGBE_RTTDCS, rttdcs);
|
IXGBE_WRITE_REG(hw, IXGBE_RTTDCS, rttdcs);
|
||||||
|
|
||||||
/* We enable 8 traffic classes, DCB only */
|
/* set transmit pool layout */
|
||||||
if (adapter->flags & IXGBE_FLAG_DCB_ENABLED)
|
mask = (IXGBE_FLAG_SRIOV_ENABLED | IXGBE_FLAG_DCB_ENABLED);
|
||||||
IXGBE_WRITE_REG(hw, IXGBE_MTQC, (IXGBE_MTQC_RT_ENA |
|
switch (adapter->flags & mask) {
|
||||||
IXGBE_MTQC_8TC_8TQ));
|
|
||||||
else
|
case (IXGBE_FLAG_SRIOV_ENABLED):
|
||||||
|
IXGBE_WRITE_REG(hw, IXGBE_MTQC,
|
||||||
|
(IXGBE_MTQC_VT_ENA | IXGBE_MTQC_64VF));
|
||||||
|
break;
|
||||||
|
|
||||||
|
case (IXGBE_FLAG_DCB_ENABLED):
|
||||||
|
/* We enable 8 traffic classes, DCB only */
|
||||||
|
IXGBE_WRITE_REG(hw, IXGBE_MTQC,
|
||||||
|
(IXGBE_MTQC_RT_ENA | IXGBE_MTQC_8TC_8TQ));
|
||||||
|
break;
|
||||||
|
|
||||||
|
default:
|
||||||
IXGBE_WRITE_REG(hw, IXGBE_MTQC, IXGBE_MTQC_64Q_1PB);
|
IXGBE_WRITE_REG(hw, IXGBE_MTQC, IXGBE_MTQC_64Q_1PB);
|
||||||
|
break;
|
||||||
|
}
|
||||||
|
|
||||||
/* re-eable the arbiter */
|
/* re-eable the arbiter */
|
||||||
rttdcs &= ~IXGBE_RTTDCS_ARBDIS;
|
rttdcs &= ~IXGBE_RTTDCS_ARBDIS;
|
||||||
@ -2054,12 +2128,16 @@ static u32 ixgbe_setup_mrqc(struct ixgbe_adapter *adapter)
|
|||||||
#ifdef CONFIG_IXGBE_DCB
|
#ifdef CONFIG_IXGBE_DCB
|
||||||
| IXGBE_FLAG_DCB_ENABLED
|
| IXGBE_FLAG_DCB_ENABLED
|
||||||
#endif
|
#endif
|
||||||
|
| IXGBE_FLAG_SRIOV_ENABLED
|
||||||
);
|
);
|
||||||
|
|
||||||
switch (mask) {
|
switch (mask) {
|
||||||
case (IXGBE_FLAG_RSS_ENABLED):
|
case (IXGBE_FLAG_RSS_ENABLED):
|
||||||
mrqc = IXGBE_MRQC_RSSEN;
|
mrqc = IXGBE_MRQC_RSSEN;
|
||||||
break;
|
break;
|
||||||
|
case (IXGBE_FLAG_SRIOV_ENABLED):
|
||||||
|
mrqc = IXGBE_MRQC_VMDQEN;
|
||||||
|
break;
|
||||||
#ifdef CONFIG_IXGBE_DCB
|
#ifdef CONFIG_IXGBE_DCB
|
||||||
case (IXGBE_FLAG_DCB_ENABLED):
|
case (IXGBE_FLAG_DCB_ENABLED):
|
||||||
mrqc = IXGBE_MRQC_RT8TCEN;
|
mrqc = IXGBE_MRQC_RT8TCEN;
|
||||||
@ -2140,7 +2218,9 @@ static void ixgbe_configure_rx(struct ixgbe_adapter *adapter)
|
|||||||
int rx_buf_len;
|
int rx_buf_len;
|
||||||
|
|
||||||
/* Decide whether to use packet split mode or not */
|
/* Decide whether to use packet split mode or not */
|
||||||
adapter->flags |= IXGBE_FLAG_RX_PS_ENABLED;
|
/* Do not use packet split if we're in SR-IOV Mode */
|
||||||
|
if (!adapter->num_vfs)
|
||||||
|
adapter->flags |= IXGBE_FLAG_RX_PS_ENABLED;
|
||||||
|
|
||||||
/* Set the RX buffer length according to the mode */
|
/* Set the RX buffer length according to the mode */
|
||||||
if (adapter->flags & IXGBE_FLAG_RX_PS_ENABLED) {
|
if (adapter->flags & IXGBE_FLAG_RX_PS_ENABLED) {
|
||||||
@ -2152,7 +2232,9 @@ static void ixgbe_configure_rx(struct ixgbe_adapter *adapter)
|
|||||||
IXGBE_PSRTYPE_IPV4HDR |
|
IXGBE_PSRTYPE_IPV4HDR |
|
||||||
IXGBE_PSRTYPE_IPV6HDR |
|
IXGBE_PSRTYPE_IPV6HDR |
|
||||||
IXGBE_PSRTYPE_L2HDR;
|
IXGBE_PSRTYPE_L2HDR;
|
||||||
IXGBE_WRITE_REG(hw, IXGBE_PSRTYPE(0), psrtype);
|
IXGBE_WRITE_REG(hw,
|
||||||
|
IXGBE_PSRTYPE(adapter->num_vfs),
|
||||||
|
psrtype);
|
||||||
}
|
}
|
||||||
} else {
|
} else {
|
||||||
if (!(adapter->flags2 & IXGBE_FLAG2_RSC_ENABLED) &&
|
if (!(adapter->flags2 & IXGBE_FLAG2_RSC_ENABLED) &&
|
||||||
@ -2238,6 +2320,30 @@ static void ixgbe_configure_rx(struct ixgbe_adapter *adapter)
|
|||||||
IXGBE_WRITE_REG(hw, IXGBE_RDRXCTL, rdrxctl);
|
IXGBE_WRITE_REG(hw, IXGBE_RDRXCTL, rdrxctl);
|
||||||
}
|
}
|
||||||
|
|
||||||
|
if (adapter->flags & IXGBE_FLAG_SRIOV_ENABLED) {
|
||||||
|
u32 vt_reg_bits;
|
||||||
|
u32 reg_offset, vf_shift;
|
||||||
|
u32 vmdctl = IXGBE_READ_REG(hw, IXGBE_VT_CTL);
|
||||||
|
vt_reg_bits = IXGBE_VMD_CTL_VMDQ_EN
|
||||||
|
| IXGBE_VT_CTL_REPLEN;
|
||||||
|
vt_reg_bits |= (adapter->num_vfs <<
|
||||||
|
IXGBE_VT_CTL_POOL_SHIFT);
|
||||||
|
IXGBE_WRITE_REG(hw, IXGBE_VT_CTL, vmdctl | vt_reg_bits);
|
||||||
|
IXGBE_WRITE_REG(hw, IXGBE_MRQC, 0);
|
||||||
|
|
||||||
|
vf_shift = adapter->num_vfs % 32;
|
||||||
|
reg_offset = adapter->num_vfs / 32;
|
||||||
|
IXGBE_WRITE_REG(hw, IXGBE_VFRE(0), 0);
|
||||||
|
IXGBE_WRITE_REG(hw, IXGBE_VFRE(1), 0);
|
||||||
|
IXGBE_WRITE_REG(hw, IXGBE_VFTE(0), 0);
|
||||||
|
IXGBE_WRITE_REG(hw, IXGBE_VFTE(1), 0);
|
||||||
|
/* Enable only the PF's pool for Tx/Rx */
|
||||||
|
IXGBE_WRITE_REG(hw, IXGBE_VFRE(reg_offset), (1 << vf_shift));
|
||||||
|
IXGBE_WRITE_REG(hw, IXGBE_VFTE(reg_offset), (1 << vf_shift));
|
||||||
|
IXGBE_WRITE_REG(hw, IXGBE_PFDTXGSWC, IXGBE_PFDTXGSWC_VT_LBEN);
|
||||||
|
ixgbe_set_vmolr(hw, adapter->num_vfs);
|
||||||
|
}
|
||||||
|
|
||||||
/* Program MRQC for the distribution of queues */
|
/* Program MRQC for the distribution of queues */
|
||||||
mrqc = ixgbe_setup_mrqc(adapter);
|
mrqc = ixgbe_setup_mrqc(adapter);
|
||||||
|
|
||||||
@ -2269,6 +2375,20 @@ static void ixgbe_configure_rx(struct ixgbe_adapter *adapter)
|
|||||||
}
|
}
|
||||||
IXGBE_WRITE_REG(hw, IXGBE_MRQC, mrqc);
|
IXGBE_WRITE_REG(hw, IXGBE_MRQC, mrqc);
|
||||||
|
|
||||||
|
if (adapter->num_vfs) {
|
||||||
|
u32 reg;
|
||||||
|
|
||||||
|
/* Map PF MAC address in RAR Entry 0 to first pool
|
||||||
|
* following VFs */
|
||||||
|
hw->mac.ops.set_vmdq(hw, 0, adapter->num_vfs);
|
||||||
|
|
||||||
|
/* Set up VF register offsets for selected VT Mode, i.e.
|
||||||
|
* 64 VFs for SR-IOV */
|
||||||
|
reg = IXGBE_READ_REG(hw, IXGBE_GCR_EXT);
|
||||||
|
reg |= IXGBE_GCR_EXT_SRIOV;
|
||||||
|
IXGBE_WRITE_REG(hw, IXGBE_GCR_EXT, reg);
|
||||||
|
}
|
||||||
|
|
||||||
rxcsum = IXGBE_READ_REG(hw, IXGBE_RXCSUM);
|
rxcsum = IXGBE_READ_REG(hw, IXGBE_RXCSUM);
|
||||||
|
|
||||||
if (adapter->flags & IXGBE_FLAG_RSS_ENABLED ||
|
if (adapter->flags & IXGBE_FLAG_RSS_ENABLED ||
|
||||||
@ -2449,6 +2569,8 @@ void ixgbe_set_rx_mode(struct net_device *netdev)
|
|||||||
addr_list = netdev->mc_list->dmi_addr;
|
addr_list = netdev->mc_list->dmi_addr;
|
||||||
hw->mac.ops.update_mc_addr_list(hw, addr_list, addr_count,
|
hw->mac.ops.update_mc_addr_list(hw, addr_list, addr_count,
|
||||||
ixgbe_addr_list_itr);
|
ixgbe_addr_list_itr);
|
||||||
|
if (adapter->num_vfs)
|
||||||
|
ixgbe_restore_vf_multicasts(adapter);
|
||||||
}
|
}
|
||||||
|
|
||||||
static void ixgbe_napi_enable_all(struct ixgbe_adapter *adapter)
|
static void ixgbe_napi_enable_all(struct ixgbe_adapter *adapter)
|
||||||
@ -2709,6 +2831,10 @@ static int ixgbe_up_complete(struct ixgbe_adapter *adapter)
|
|||||||
/* MSI only */
|
/* MSI only */
|
||||||
gpie = 0;
|
gpie = 0;
|
||||||
}
|
}
|
||||||
|
if (adapter->flags & IXGBE_FLAG_SRIOV_ENABLED) {
|
||||||
|
gpie &= ~IXGBE_GPIE_VTMODE_MASK;
|
||||||
|
gpie |= IXGBE_GPIE_VTMODE_64;
|
||||||
|
}
|
||||||
/* XXX: to interrupt immediately for EICS writes, enable this */
|
/* XXX: to interrupt immediately for EICS writes, enable this */
|
||||||
/* gpie |= IXGBE_GPIE_EIMEN; */
|
/* gpie |= IXGBE_GPIE_EIMEN; */
|
||||||
IXGBE_WRITE_REG(hw, IXGBE_GPIE, gpie);
|
IXGBE_WRITE_REG(hw, IXGBE_GPIE, gpie);
|
||||||
@ -2783,6 +2909,18 @@ static int ixgbe_up_complete(struct ixgbe_adapter *adapter)
|
|||||||
txdctl = IXGBE_READ_REG(hw, IXGBE_TXDCTL(j));
|
txdctl = IXGBE_READ_REG(hw, IXGBE_TXDCTL(j));
|
||||||
txdctl |= IXGBE_TXDCTL_ENABLE;
|
txdctl |= IXGBE_TXDCTL_ENABLE;
|
||||||
IXGBE_WRITE_REG(hw, IXGBE_TXDCTL(j), txdctl);
|
IXGBE_WRITE_REG(hw, IXGBE_TXDCTL(j), txdctl);
|
||||||
|
if (hw->mac.type == ixgbe_mac_82599EB) {
|
||||||
|
int wait_loop = 10;
|
||||||
|
/* poll for Tx Enable ready */
|
||||||
|
do {
|
||||||
|
msleep(1);
|
||||||
|
txdctl = IXGBE_READ_REG(hw, IXGBE_TXDCTL(j));
|
||||||
|
} while (--wait_loop &&
|
||||||
|
!(txdctl & IXGBE_TXDCTL_ENABLE));
|
||||||
|
if (!wait_loop)
|
||||||
|
DPRINTK(DRV, ERR, "Could not enable "
|
||||||
|
"Tx Queue %d\n", j);
|
||||||
|
}
|
||||||
}
|
}
|
||||||
|
|
||||||
for (i = 0; i < num_rx_rings; i++) {
|
for (i = 0; i < num_rx_rings; i++) {
|
||||||
@ -2918,7 +3056,8 @@ void ixgbe_reset(struct ixgbe_adapter *adapter)
|
|||||||
}
|
}
|
||||||
|
|
||||||
/* reprogram the RAR[0] in case user changed it. */
|
/* reprogram the RAR[0] in case user changed it. */
|
||||||
hw->mac.ops.set_rar(hw, 0, hw->mac.addr, 0, IXGBE_RAH_AV);
|
hw->mac.ops.set_rar(hw, 0, hw->mac.addr, adapter->num_vfs,
|
||||||
|
IXGBE_RAH_AV);
|
||||||
}
|
}
|
||||||
|
|
||||||
/**
|
/**
|
||||||
@ -3286,6 +3425,19 @@ static inline bool ixgbe_set_fcoe_queues(struct ixgbe_adapter *adapter)
|
|||||||
}
|
}
|
||||||
|
|
||||||
#endif /* IXGBE_FCOE */
|
#endif /* IXGBE_FCOE */
|
||||||
|
/**
|
||||||
|
* ixgbe_set_sriov_queues: Allocate queues for IOV use
|
||||||
|
* @adapter: board private structure to initialize
|
||||||
|
*
|
||||||
|
* IOV doesn't actually use anything, so just NAK the
|
||||||
|
* request for now and let the other queue routines
|
||||||
|
* figure out what to do.
|
||||||
|
*/
|
||||||
|
static inline bool ixgbe_set_sriov_queues(struct ixgbe_adapter *adapter)
|
||||||
|
{
|
||||||
|
return false;
|
||||||
|
}
|
||||||
|
|
||||||
/*
|
/*
|
||||||
* ixgbe_set_num_queues: Allocate queues for device, feature dependant
|
* ixgbe_set_num_queues: Allocate queues for device, feature dependant
|
||||||
* @adapter: board private structure to initialize
|
* @adapter: board private structure to initialize
|
||||||
@ -3299,6 +3451,15 @@ static inline bool ixgbe_set_fcoe_queues(struct ixgbe_adapter *adapter)
|
|||||||
**/
|
**/
|
||||||
static void ixgbe_set_num_queues(struct ixgbe_adapter *adapter)
|
static void ixgbe_set_num_queues(struct ixgbe_adapter *adapter)
|
||||||
{
|
{
|
||||||
|
/* Start with base case */
|
||||||
|
adapter->num_rx_queues = 1;
|
||||||
|
adapter->num_tx_queues = 1;
|
||||||
|
adapter->num_rx_pools = adapter->num_rx_queues;
|
||||||
|
adapter->num_rx_queues_per_pool = 1;
|
||||||
|
|
||||||
|
if (ixgbe_set_sriov_queues(adapter))
|
||||||
|
return;
|
||||||
|
|
||||||
#ifdef IXGBE_FCOE
|
#ifdef IXGBE_FCOE
|
||||||
if (ixgbe_set_fcoe_queues(adapter))
|
if (ixgbe_set_fcoe_queues(adapter))
|
||||||
goto done;
|
goto done;
|
||||||
@ -3569,6 +3730,24 @@ static inline bool ixgbe_cache_ring_fcoe(struct ixgbe_adapter *adapter)
|
|||||||
}
|
}
|
||||||
|
|
||||||
#endif /* IXGBE_FCOE */
|
#endif /* IXGBE_FCOE */
|
||||||
|
/**
|
||||||
|
* ixgbe_cache_ring_sriov - Descriptor ring to register mapping for sriov
|
||||||
|
* @adapter: board private structure to initialize
|
||||||
|
*
|
||||||
|
* SR-IOV doesn't use any descriptor rings but changes the default if
|
||||||
|
* no other mapping is used.
|
||||||
|
*
|
||||||
|
*/
|
||||||
|
static inline bool ixgbe_cache_ring_sriov(struct ixgbe_adapter *adapter)
|
||||||
|
{
|
||||||
|
adapter->rx_ring[0].reg_idx = adapter->num_vfs * 2;
|
||||||
|
adapter->tx_ring[0].reg_idx = adapter->num_vfs * 2;
|
||||||
|
if (adapter->num_vfs)
|
||||||
|
return true;
|
||||||
|
else
|
||||||
|
return false;
|
||||||
|
}
|
||||||
|
|
||||||
/**
|
/**
|
||||||
* ixgbe_cache_ring_register - Descriptor ring to register mapping
|
* ixgbe_cache_ring_register - Descriptor ring to register mapping
|
||||||
* @adapter: board private structure to initialize
|
* @adapter: board private structure to initialize
|
||||||
@ -3586,6 +3765,9 @@ static void ixgbe_cache_ring_register(struct ixgbe_adapter *adapter)
|
|||||||
adapter->rx_ring[0].reg_idx = 0;
|
adapter->rx_ring[0].reg_idx = 0;
|
||||||
adapter->tx_ring[0].reg_idx = 0;
|
adapter->tx_ring[0].reg_idx = 0;
|
||||||
|
|
||||||
|
if (ixgbe_cache_ring_sriov(adapter))
|
||||||
|
return;
|
||||||
|
|
||||||
#ifdef IXGBE_FCOE
|
#ifdef IXGBE_FCOE
|
||||||
if (ixgbe_cache_ring_fcoe(adapter))
|
if (ixgbe_cache_ring_fcoe(adapter))
|
||||||
return;
|
return;
|
||||||
@ -3695,6 +3877,9 @@ static int ixgbe_set_interrupt_capability(struct ixgbe_adapter *adapter)
|
|||||||
adapter->flags &= ~IXGBE_FLAG_FDIR_HASH_CAPABLE;
|
adapter->flags &= ~IXGBE_FLAG_FDIR_HASH_CAPABLE;
|
||||||
adapter->flags &= ~IXGBE_FLAG_FDIR_PERFECT_CAPABLE;
|
adapter->flags &= ~IXGBE_FLAG_FDIR_PERFECT_CAPABLE;
|
||||||
adapter->atr_sample_rate = 0;
|
adapter->atr_sample_rate = 0;
|
||||||
|
if (adapter->flags & IXGBE_FLAG_SRIOV_ENABLED)
|
||||||
|
ixgbe_disable_sriov(adapter);
|
||||||
|
|
||||||
ixgbe_set_num_queues(adapter);
|
ixgbe_set_num_queues(adapter);
|
||||||
|
|
||||||
err = pci_enable_msi(adapter->pdev);
|
err = pci_enable_msi(adapter->pdev);
|
||||||
@ -5474,7 +5659,8 @@ static int ixgbe_set_mac(struct net_device *netdev, void *p)
|
|||||||
memcpy(netdev->dev_addr, addr->sa_data, netdev->addr_len);
|
memcpy(netdev->dev_addr, addr->sa_data, netdev->addr_len);
|
||||||
memcpy(hw->mac.addr, addr->sa_data, netdev->addr_len);
|
memcpy(hw->mac.addr, addr->sa_data, netdev->addr_len);
|
||||||
|
|
||||||
hw->mac.ops.set_rar(hw, 0, hw->mac.addr, 0, IXGBE_RAH_AV);
|
hw->mac.ops.set_rar(hw, 0, hw->mac.addr, adapter->num_vfs,
|
||||||
|
IXGBE_RAH_AV);
|
||||||
|
|
||||||
return 0;
|
return 0;
|
||||||
}
|
}
|
||||||
@ -5607,6 +5793,61 @@ static const struct net_device_ops ixgbe_netdev_ops = {
|
|||||||
#endif /* IXGBE_FCOE */
|
#endif /* IXGBE_FCOE */
|
||||||
};
|
};
|
||||||
|
|
||||||
|
static void __devinit ixgbe_probe_vf(struct ixgbe_adapter *adapter,
|
||||||
|
const struct ixgbe_info *ii)
|
||||||
|
{
|
||||||
|
#ifdef CONFIG_PCI_IOV
|
||||||
|
struct ixgbe_hw *hw = &adapter->hw;
|
||||||
|
int err;
|
||||||
|
|
||||||
|
if (hw->mac.type != ixgbe_mac_82599EB || !max_vfs)
|
||||||
|
return;
|
||||||
|
|
||||||
|
/* The 82599 supports up to 64 VFs per physical function
|
||||||
|
* but this implementation limits allocation to 63 so that
|
||||||
|
* basic networking resources are still available to the
|
||||||
|
* physical function
|
||||||
|
*/
|
||||||
|
adapter->num_vfs = (max_vfs > 63) ? 63 : max_vfs;
|
||||||
|
adapter->flags |= IXGBE_FLAG_SRIOV_ENABLED;
|
||||||
|
err = pci_enable_sriov(adapter->pdev, adapter->num_vfs);
|
||||||
|
if (err) {
|
||||||
|
DPRINTK(PROBE, ERR,
|
||||||
|
"Failed to enable PCI sriov: %d\n", err);
|
||||||
|
goto err_novfs;
|
||||||
|
}
|
||||||
|
/* If call to enable VFs succeeded then allocate memory
|
||||||
|
* for per VF control structures.
|
||||||
|
*/
|
||||||
|
adapter->vfinfo =
|
||||||
|
kcalloc(adapter->num_vfs,
|
||||||
|
sizeof(struct vf_data_storage), GFP_KERNEL);
|
||||||
|
if (adapter->vfinfo) {
|
||||||
|
/* Now that we're sure SR-IOV is enabled
|
||||||
|
* and memory allocated set up the mailbox parameters
|
||||||
|
*/
|
||||||
|
ixgbe_init_mbx_params_pf(hw);
|
||||||
|
memcpy(&hw->mbx.ops, ii->mbx_ops,
|
||||||
|
sizeof(hw->mbx.ops));
|
||||||
|
|
||||||
|
/* Disable RSC when in SR-IOV mode */
|
||||||
|
adapter->flags2 &= ~(IXGBE_FLAG2_RSC_CAPABLE |
|
||||||
|
IXGBE_FLAG2_RSC_ENABLED);
|
||||||
|
return;
|
||||||
|
}
|
||||||
|
|
||||||
|
/* Oh oh */
|
||||||
|
DPRINTK(PROBE, ERR,
|
||||||
|
"Unable to allocate memory for VF "
|
||||||
|
"Data Storage - SRIOV disabled\n");
|
||||||
|
pci_disable_sriov(adapter->pdev);
|
||||||
|
|
||||||
|
err_novfs:
|
||||||
|
adapter->flags &= ~IXGBE_FLAG_SRIOV_ENABLED;
|
||||||
|
adapter->num_vfs = 0;
|
||||||
|
#endif /* CONFIG_PCI_IOV */
|
||||||
|
}
|
||||||
|
|
||||||
/**
|
/**
|
||||||
* ixgbe_probe - Device Initialization Routine
|
* ixgbe_probe - Device Initialization Routine
|
||||||
* @pdev: PCI device information struct
|
* @pdev: PCI device information struct
|
||||||
@ -5781,6 +6022,8 @@ static int __devinit ixgbe_probe(struct pci_dev *pdev,
|
|||||||
goto err_sw_init;
|
goto err_sw_init;
|
||||||
}
|
}
|
||||||
|
|
||||||
|
ixgbe_probe_vf(adapter, ii);
|
||||||
|
|
||||||
netdev->features = NETIF_F_SG |
|
netdev->features = NETIF_F_SG |
|
||||||
NETIF_F_IP_CSUM |
|
NETIF_F_IP_CSUM |
|
||||||
NETIF_F_HW_VLAN_TX |
|
NETIF_F_HW_VLAN_TX |
|
||||||
@ -5801,6 +6044,9 @@ static int __devinit ixgbe_probe(struct pci_dev *pdev,
|
|||||||
netdev->vlan_features |= NETIF_F_IPV6_CSUM;
|
netdev->vlan_features |= NETIF_F_IPV6_CSUM;
|
||||||
netdev->vlan_features |= NETIF_F_SG;
|
netdev->vlan_features |= NETIF_F_SG;
|
||||||
|
|
||||||
|
if (adapter->flags & IXGBE_FLAG_SRIOV_ENABLED)
|
||||||
|
adapter->flags &= ~(IXGBE_FLAG_RSS_ENABLED |
|
||||||
|
IXGBE_FLAG_DCB_ENABLED);
|
||||||
if (adapter->flags & IXGBE_FLAG_DCB_ENABLED)
|
if (adapter->flags & IXGBE_FLAG_DCB_ENABLED)
|
||||||
adapter->flags &= ~IXGBE_FLAG_RSS_ENABLED;
|
adapter->flags &= ~IXGBE_FLAG_RSS_ENABLED;
|
||||||
|
|
||||||
@ -5927,6 +6173,13 @@ static int __devinit ixgbe_probe(struct pci_dev *pdev,
|
|||||||
ixgbe_setup_dca(adapter);
|
ixgbe_setup_dca(adapter);
|
||||||
}
|
}
|
||||||
#endif
|
#endif
|
||||||
|
if (adapter->flags & IXGBE_FLAG_SRIOV_ENABLED) {
|
||||||
|
DPRINTK(PROBE, INFO, "IOV is enabled with %d VFs\n",
|
||||||
|
adapter->num_vfs);
|
||||||
|
for (i = 0; i < adapter->num_vfs; i++)
|
||||||
|
ixgbe_vf_configuration(pdev, (i | 0x10000000));
|
||||||
|
}
|
||||||
|
|
||||||
/* add san mac addr to netdev */
|
/* add san mac addr to netdev */
|
||||||
ixgbe_add_sanmac_netdev(netdev);
|
ixgbe_add_sanmac_netdev(netdev);
|
||||||
|
|
||||||
@ -5939,6 +6192,8 @@ err_register:
|
|||||||
ixgbe_clear_interrupt_scheme(adapter);
|
ixgbe_clear_interrupt_scheme(adapter);
|
||||||
err_sw_init:
|
err_sw_init:
|
||||||
err_eeprom:
|
err_eeprom:
|
||||||
|
if (adapter->flags & IXGBE_FLAG_SRIOV_ENABLED)
|
||||||
|
ixgbe_disable_sriov(adapter);
|
||||||
clear_bit(__IXGBE_SFP_MODULE_NOT_FOUND, &adapter->state);
|
clear_bit(__IXGBE_SFP_MODULE_NOT_FOUND, &adapter->state);
|
||||||
del_timer_sync(&adapter->sfp_timer);
|
del_timer_sync(&adapter->sfp_timer);
|
||||||
cancel_work_sync(&adapter->sfp_task);
|
cancel_work_sync(&adapter->sfp_task);
|
||||||
@ -6007,6 +6262,9 @@ static void __devexit ixgbe_remove(struct pci_dev *pdev)
|
|||||||
if (netdev->reg_state == NETREG_REGISTERED)
|
if (netdev->reg_state == NETREG_REGISTERED)
|
||||||
unregister_netdev(netdev);
|
unregister_netdev(netdev);
|
||||||
|
|
||||||
|
if (adapter->flags & IXGBE_FLAG_SRIOV_ENABLED)
|
||||||
|
ixgbe_disable_sriov(adapter);
|
||||||
|
|
||||||
ixgbe_clear_interrupt_scheme(adapter);
|
ixgbe_clear_interrupt_scheme(adapter);
|
||||||
|
|
||||||
ixgbe_release_hw_control(adapter);
|
ixgbe_release_hw_control(adapter);
|
||||||
|
Loading…
Reference in New Issue
Block a user