[PATCH] forcedeth config: diagnostics
This patch adds support for diagnostic tests through ethtool support. Signed-Off-By: Ayaz Abdulla <aabdulla@nvidia.com> Signed-off-by: Jeff Garzik <jeff@garzik.org>
This commit is contained in:
parent
7a1854b797
commit
9589c77a0d
|
@ -166,6 +166,7 @@
|
|||
#define DEV_HAS_POWER_CNTRL 0x0100 /* device supports power savings */
|
||||
#define DEV_HAS_PAUSEFRAME_TX 0x0200 /* device supports tx pause frames */
|
||||
#define DEV_HAS_STATISTICS 0x0400 /* device supports hw statistics */
|
||||
#define DEV_HAS_TEST_EXTENDED 0x0800 /* device supports extended diagnostic test */
|
||||
|
||||
enum {
|
||||
NvRegIrqStatus = 0x000,
|
||||
|
@ -222,6 +223,7 @@ enum {
|
|||
#define NVREG_PFF_ALWAYS 0x7F0000
|
||||
#define NVREG_PFF_PROMISC 0x80
|
||||
#define NVREG_PFF_MYADDR 0x20
|
||||
#define NVREG_PFF_LOOPBACK 0x10
|
||||
|
||||
NvRegOffloadConfig = 0x90,
|
||||
#define NVREG_OFFLOAD_HOMEPHY 0x601
|
||||
|
@ -634,6 +636,32 @@ struct nv_ethtool_stats {
|
|||
u64 rx_errors_total;
|
||||
};
|
||||
|
||||
/* diagnostics */
|
||||
#define NV_TEST_COUNT_BASE 3
|
||||
#define NV_TEST_COUNT_EXTENDED 4
|
||||
|
||||
static const struct nv_ethtool_str nv_etests_str[] = {
|
||||
{ "link (online/offline)" },
|
||||
{ "register (offline) " },
|
||||
{ "interrupt (offline) " },
|
||||
{ "loopback (offline) " }
|
||||
};
|
||||
|
||||
struct register_test {
|
||||
u32 reg;
|
||||
u32 mask;
|
||||
};
|
||||
|
||||
static const struct register_test nv_registers_test[] = {
|
||||
{ NvRegUnknownSetupReg6, 0x01 },
|
||||
{ NvRegMisc1, 0x03c },
|
||||
{ NvRegOffloadConfig, 0x03ff },
|
||||
{ NvRegMulticastAddrA, 0xffffffff },
|
||||
{ NvRegUnknownSetupReg3, 0x0ff },
|
||||
{ NvRegWakeUpFlags, 0x07777 },
|
||||
{ 0,0 }
|
||||
};
|
||||
|
||||
/*
|
||||
* SMP locking:
|
||||
* All hardware access under dev->priv->lock, except the performance
|
||||
|
@ -662,6 +690,7 @@ struct fe_priv {
|
|||
int wolenabled;
|
||||
unsigned int phy_oui;
|
||||
u16 gigabit;
|
||||
int intr_test;
|
||||
|
||||
/* General data: RO fields */
|
||||
dma_addr_t ring_addr;
|
||||
|
@ -2502,6 +2531,36 @@ static irqreturn_t nv_nic_irq_other(int foo, void *data, struct pt_regs *regs)
|
|||
return IRQ_RETVAL(i);
|
||||
}
|
||||
|
||||
static irqreturn_t nv_nic_irq_test(int foo, void *data, struct pt_regs *regs)
|
||||
{
|
||||
struct net_device *dev = (struct net_device *) data;
|
||||
struct fe_priv *np = netdev_priv(dev);
|
||||
u8 __iomem *base = get_hwbase(dev);
|
||||
u32 events;
|
||||
|
||||
dprintk(KERN_DEBUG "%s: nv_nic_irq_test\n", dev->name);
|
||||
|
||||
if (!(np->msi_flags & NV_MSI_X_ENABLED)) {
|
||||
events = readl(base + NvRegIrqStatus) & NVREG_IRQSTAT_MASK;
|
||||
writel(NVREG_IRQ_TIMER, base + NvRegIrqStatus);
|
||||
} else {
|
||||
events = readl(base + NvRegMSIXIrqStatus) & NVREG_IRQSTAT_MASK;
|
||||
writel(NVREG_IRQ_TIMER, base + NvRegMSIXIrqStatus);
|
||||
}
|
||||
pci_push(base);
|
||||
dprintk(KERN_DEBUG "%s: irq: %08x\n", dev->name, events);
|
||||
if (!(events & NVREG_IRQ_TIMER))
|
||||
return IRQ_RETVAL(0);
|
||||
|
||||
spin_lock(&np->lock);
|
||||
np->intr_test = 1;
|
||||
spin_unlock(&np->lock);
|
||||
|
||||
dprintk(KERN_DEBUG "%s: nv_nic_irq_test completed\n", dev->name);
|
||||
|
||||
return IRQ_RETVAL(1);
|
||||
}
|
||||
|
||||
static void set_msix_vector_map(struct net_device *dev, u32 vector, u32 irqmask)
|
||||
{
|
||||
u8 __iomem *base = get_hwbase(dev);
|
||||
|
@ -2528,7 +2587,7 @@ static void set_msix_vector_map(struct net_device *dev, u32 vector, u32 irqmask)
|
|||
writel(readl(base + NvRegMSIXMap1) | msixmap, base + NvRegMSIXMap1);
|
||||
}
|
||||
|
||||
static int nv_request_irq(struct net_device *dev)
|
||||
static int nv_request_irq(struct net_device *dev, int intr_test)
|
||||
{
|
||||
struct fe_priv *np = get_nvpriv(dev);
|
||||
u8 __iomem *base = get_hwbase(dev);
|
||||
|
@ -2541,7 +2600,7 @@ static int nv_request_irq(struct net_device *dev)
|
|||
}
|
||||
if ((ret = pci_enable_msix(np->pci_dev, np->msi_x_entry, (np->msi_flags & NV_MSI_X_VECTORS_MASK))) == 0) {
|
||||
np->msi_flags |= NV_MSI_X_ENABLED;
|
||||
if (optimization_mode == NV_OPTIMIZATION_MODE_THROUGHPUT) {
|
||||
if (optimization_mode == NV_OPTIMIZATION_MODE_THROUGHPUT && !intr_test) {
|
||||
/* Request irq for rx handling */
|
||||
if (request_irq(np->msi_x_entry[NV_MSI_X_VECTOR_RX].vector, &nv_nic_irq_rx, SA_SHIRQ, dev->name, dev) != 0) {
|
||||
printk(KERN_INFO "forcedeth: request_irq failed for rx %d\n", ret);
|
||||
|
@ -2571,7 +2630,10 @@ static int nv_request_irq(struct net_device *dev)
|
|||
set_msix_vector_map(dev, NV_MSI_X_VECTOR_OTHER, NVREG_IRQ_OTHER);
|
||||
} else {
|
||||
/* Request irq for all interrupts */
|
||||
if (request_irq(np->msi_x_entry[NV_MSI_X_VECTOR_ALL].vector, &nv_nic_irq, SA_SHIRQ, dev->name, dev) != 0) {
|
||||
if ((!intr_test &&
|
||||
request_irq(np->msi_x_entry[NV_MSI_X_VECTOR_ALL].vector, &nv_nic_irq, SA_SHIRQ, dev->name, dev) != 0) ||
|
||||
(intr_test &&
|
||||
request_irq(np->msi_x_entry[NV_MSI_X_VECTOR_ALL].vector, &nv_nic_irq_test, SA_SHIRQ, dev->name, dev) != 0)) {
|
||||
printk(KERN_INFO "forcedeth: request_irq failed %d\n", ret);
|
||||
pci_disable_msix(np->pci_dev);
|
||||
np->msi_flags &= ~NV_MSI_X_ENABLED;
|
||||
|
@ -2587,7 +2649,8 @@ static int nv_request_irq(struct net_device *dev)
|
|||
if (ret != 0 && np->msi_flags & NV_MSI_CAPABLE) {
|
||||
if ((ret = pci_enable_msi(np->pci_dev)) == 0) {
|
||||
np->msi_flags |= NV_MSI_ENABLED;
|
||||
if (request_irq(np->pci_dev->irq, &nv_nic_irq, SA_SHIRQ, dev->name, dev) != 0) {
|
||||
if ((!intr_test && request_irq(np->pci_dev->irq, &nv_nic_irq, SA_SHIRQ, dev->name, dev) != 0) ||
|
||||
(intr_test && request_irq(np->pci_dev->irq, &nv_nic_irq_test, SA_SHIRQ, dev->name, dev) != 0)) {
|
||||
printk(KERN_INFO "forcedeth: request_irq failed %d\n", ret);
|
||||
pci_disable_msi(np->pci_dev);
|
||||
np->msi_flags &= ~NV_MSI_ENABLED;
|
||||
|
@ -2602,8 +2665,10 @@ static int nv_request_irq(struct net_device *dev)
|
|||
}
|
||||
}
|
||||
if (ret != 0) {
|
||||
if (request_irq(np->pci_dev->irq, &nv_nic_irq, SA_SHIRQ, dev->name, dev) != 0)
|
||||
if ((!intr_test && request_irq(np->pci_dev->irq, &nv_nic_irq, SA_SHIRQ, dev->name, dev) != 0) ||
|
||||
(intr_test && request_irq(np->pci_dev->irq, &nv_nic_irq_test, SA_SHIRQ, dev->name, dev) != 0))
|
||||
goto out_err;
|
||||
|
||||
}
|
||||
|
||||
return 0;
|
||||
|
@ -3387,12 +3452,335 @@ static void nv_get_ethtool_stats(struct net_device *dev, struct ethtool_stats *e
|
|||
memcpy(buffer, &np->estats, nv_get_stats_count(dev)*sizeof(u64));
|
||||
}
|
||||
|
||||
static int nv_self_test_count(struct net_device *dev)
|
||||
{
|
||||
struct fe_priv *np = netdev_priv(dev);
|
||||
|
||||
if (np->driver_data & DEV_HAS_TEST_EXTENDED)
|
||||
return NV_TEST_COUNT_EXTENDED;
|
||||
else
|
||||
return NV_TEST_COUNT_BASE;
|
||||
}
|
||||
|
||||
static int nv_link_test(struct net_device *dev)
|
||||
{
|
||||
struct fe_priv *np = netdev_priv(dev);
|
||||
int mii_status;
|
||||
|
||||
mii_rw(dev, np->phyaddr, MII_BMSR, MII_READ);
|
||||
mii_status = mii_rw(dev, np->phyaddr, MII_BMSR, MII_READ);
|
||||
|
||||
/* check phy link status */
|
||||
if (!(mii_status & BMSR_LSTATUS))
|
||||
return 0;
|
||||
else
|
||||
return 1;
|
||||
}
|
||||
|
||||
static int nv_register_test(struct net_device *dev)
|
||||
{
|
||||
u8 __iomem *base = get_hwbase(dev);
|
||||
int i = 0;
|
||||
u32 orig_read, new_read;
|
||||
|
||||
do {
|
||||
orig_read = readl(base + nv_registers_test[i].reg);
|
||||
|
||||
/* xor with mask to toggle bits */
|
||||
orig_read ^= nv_registers_test[i].mask;
|
||||
|
||||
writel(orig_read, base + nv_registers_test[i].reg);
|
||||
|
||||
new_read = readl(base + nv_registers_test[i].reg);
|
||||
|
||||
if ((new_read & nv_registers_test[i].mask) != (orig_read & nv_registers_test[i].mask))
|
||||
return 0;
|
||||
|
||||
/* restore original value */
|
||||
orig_read ^= nv_registers_test[i].mask;
|
||||
writel(orig_read, base + nv_registers_test[i].reg);
|
||||
|
||||
} while (nv_registers_test[++i].reg != 0);
|
||||
|
||||
return 1;
|
||||
}
|
||||
|
||||
static int nv_interrupt_test(struct net_device *dev)
|
||||
{
|
||||
struct fe_priv *np = netdev_priv(dev);
|
||||
u8 __iomem *base = get_hwbase(dev);
|
||||
int ret = 1;
|
||||
int testcnt;
|
||||
u32 save_msi_flags, save_poll_interval = 0;
|
||||
|
||||
if (netif_running(dev)) {
|
||||
/* free current irq */
|
||||
nv_free_irq(dev);
|
||||
save_poll_interval = readl(base+NvRegPollingInterval);
|
||||
}
|
||||
|
||||
/* flag to test interrupt handler */
|
||||
np->intr_test = 0;
|
||||
|
||||
/* setup test irq */
|
||||
save_msi_flags = np->msi_flags;
|
||||
np->msi_flags &= ~NV_MSI_X_VECTORS_MASK;
|
||||
np->msi_flags |= 0x001; /* setup 1 vector */
|
||||
if (nv_request_irq(dev, 1))
|
||||
return 0;
|
||||
|
||||
/* setup timer interrupt */
|
||||
writel(NVREG_POLL_DEFAULT_CPU, base + NvRegPollingInterval);
|
||||
writel(NVREG_UNKSETUP6_VAL, base + NvRegUnknownSetupReg6);
|
||||
|
||||
nv_enable_hw_interrupts(dev, NVREG_IRQ_TIMER);
|
||||
|
||||
/* wait for at least one interrupt */
|
||||
msleep(100);
|
||||
|
||||
spin_lock_irq(&np->lock);
|
||||
|
||||
/* flag should be set within ISR */
|
||||
testcnt = np->intr_test;
|
||||
if (!testcnt)
|
||||
ret = 2;
|
||||
|
||||
nv_disable_hw_interrupts(dev, NVREG_IRQ_TIMER);
|
||||
if (!(np->msi_flags & NV_MSI_X_ENABLED))
|
||||
writel(NVREG_IRQSTAT_MASK, base + NvRegIrqStatus);
|
||||
else
|
||||
writel(NVREG_IRQSTAT_MASK, base + NvRegMSIXIrqStatus);
|
||||
|
||||
spin_unlock_irq(&np->lock);
|
||||
|
||||
nv_free_irq(dev);
|
||||
|
||||
np->msi_flags = save_msi_flags;
|
||||
|
||||
if (netif_running(dev)) {
|
||||
writel(save_poll_interval, base + NvRegPollingInterval);
|
||||
writel(NVREG_UNKSETUP6_VAL, base + NvRegUnknownSetupReg6);
|
||||
/* restore original irq */
|
||||
if (nv_request_irq(dev, 0))
|
||||
return 0;
|
||||
}
|
||||
|
||||
return ret;
|
||||
}
|
||||
|
||||
static int nv_loopback_test(struct net_device *dev)
|
||||
{
|
||||
struct fe_priv *np = netdev_priv(dev);
|
||||
u8 __iomem *base = get_hwbase(dev);
|
||||
struct sk_buff *tx_skb, *rx_skb;
|
||||
dma_addr_t test_dma_addr;
|
||||
u32 tx_flags_extra = (np->desc_ver == DESC_VER_1 ? NV_TX_LASTPACKET : NV_TX2_LASTPACKET);
|
||||
u32 Flags;
|
||||
int len, i, pkt_len;
|
||||
u8 *pkt_data;
|
||||
u32 filter_flags = 0;
|
||||
u32 misc1_flags = 0;
|
||||
int ret = 1;
|
||||
|
||||
if (netif_running(dev)) {
|
||||
nv_disable_irq(dev);
|
||||
filter_flags = readl(base + NvRegPacketFilterFlags);
|
||||
misc1_flags = readl(base + NvRegMisc1);
|
||||
} else {
|
||||
nv_txrx_reset(dev);
|
||||
}
|
||||
|
||||
/* reinit driver view of the rx queue */
|
||||
set_bufsize(dev);
|
||||
nv_init_ring(dev);
|
||||
|
||||
/* setup hardware for loopback */
|
||||
writel(NVREG_MISC1_FORCE, base + NvRegMisc1);
|
||||
writel(NVREG_PFF_ALWAYS | NVREG_PFF_LOOPBACK, base + NvRegPacketFilterFlags);
|
||||
|
||||
/* reinit nic view of the rx queue */
|
||||
writel(np->rx_buf_sz, base + NvRegOffloadConfig);
|
||||
setup_hw_rings(dev, NV_SETUP_RX_RING | NV_SETUP_TX_RING);
|
||||
writel( ((np->rx_ring_size-1) << NVREG_RINGSZ_RXSHIFT) + ((np->tx_ring_size-1) << NVREG_RINGSZ_TXSHIFT),
|
||||
base + NvRegRingSizes);
|
||||
pci_push(base);
|
||||
|
||||
/* restart rx engine */
|
||||
nv_start_rx(dev);
|
||||
nv_start_tx(dev);
|
||||
|
||||
/* setup packet for tx */
|
||||
pkt_len = ETH_DATA_LEN;
|
||||
tx_skb = dev_alloc_skb(pkt_len);
|
||||
pkt_data = skb_put(tx_skb, pkt_len);
|
||||
for (i = 0; i < pkt_len; i++)
|
||||
pkt_data[i] = (u8)(i & 0xff);
|
||||
test_dma_addr = pci_map_single(np->pci_dev, tx_skb->data,
|
||||
tx_skb->end-tx_skb->data, PCI_DMA_FROMDEVICE);
|
||||
|
||||
if (np->desc_ver == DESC_VER_1 || np->desc_ver == DESC_VER_2) {
|
||||
np->tx_ring.orig[0].PacketBuffer = cpu_to_le32(test_dma_addr);
|
||||
np->tx_ring.orig[0].FlagLen = cpu_to_le32((pkt_len-1) | np->tx_flags | tx_flags_extra);
|
||||
} else {
|
||||
np->tx_ring.ex[0].PacketBufferHigh = cpu_to_le64(test_dma_addr) >> 32;
|
||||
np->tx_ring.ex[0].PacketBufferLow = cpu_to_le64(test_dma_addr) & 0x0FFFFFFFF;
|
||||
np->tx_ring.ex[0].FlagLen = cpu_to_le32((pkt_len-1) | np->tx_flags | tx_flags_extra);
|
||||
}
|
||||
writel(NVREG_TXRXCTL_KICK|np->txrxctl_bits, get_hwbase(dev) + NvRegTxRxControl);
|
||||
pci_push(get_hwbase(dev));
|
||||
|
||||
msleep(500);
|
||||
|
||||
/* check for rx of the packet */
|
||||
if (np->desc_ver == DESC_VER_1 || np->desc_ver == DESC_VER_2) {
|
||||
Flags = le32_to_cpu(np->rx_ring.orig[0].FlagLen);
|
||||
len = nv_descr_getlength(&np->rx_ring.orig[0], np->desc_ver);
|
||||
|
||||
} else {
|
||||
Flags = le32_to_cpu(np->rx_ring.ex[0].FlagLen);
|
||||
len = nv_descr_getlength_ex(&np->rx_ring.ex[0], np->desc_ver);
|
||||
}
|
||||
|
||||
if (Flags & NV_RX_AVAIL) {
|
||||
ret = 0;
|
||||
} else if (np->desc_ver == DESC_VER_1) {
|
||||
if (Flags & NV_RX_ERROR)
|
||||
ret = 0;
|
||||
} else {
|
||||
if (Flags & NV_RX2_ERROR) {
|
||||
ret = 0;
|
||||
}
|
||||
}
|
||||
|
||||
if (ret) {
|
||||
if (len != pkt_len) {
|
||||
ret = 0;
|
||||
dprintk(KERN_DEBUG "%s: loopback len mismatch %d vs %d\n",
|
||||
dev->name, len, pkt_len);
|
||||
} else {
|
||||
rx_skb = np->rx_skbuff[0];
|
||||
for (i = 0; i < pkt_len; i++) {
|
||||
if (rx_skb->data[i] != (u8)(i & 0xff)) {
|
||||
ret = 0;
|
||||
dprintk(KERN_DEBUG "%s: loopback pattern check failed on byte %d\n",
|
||||
dev->name, i);
|
||||
break;
|
||||
}
|
||||
}
|
||||
}
|
||||
} else {
|
||||
dprintk(KERN_DEBUG "%s: loopback - did not receive test packet\n", dev->name);
|
||||
}
|
||||
|
||||
pci_unmap_page(np->pci_dev, test_dma_addr,
|
||||
tx_skb->end-tx_skb->data,
|
||||
PCI_DMA_TODEVICE);
|
||||
dev_kfree_skb_any(tx_skb);
|
||||
|
||||
/* stop engines */
|
||||
nv_stop_rx(dev);
|
||||
nv_stop_tx(dev);
|
||||
nv_txrx_reset(dev);
|
||||
/* drain rx queue */
|
||||
nv_drain_rx(dev);
|
||||
nv_drain_tx(dev);
|
||||
|
||||
if (netif_running(dev)) {
|
||||
writel(misc1_flags, base + NvRegMisc1);
|
||||
writel(filter_flags, base + NvRegPacketFilterFlags);
|
||||
nv_enable_irq(dev);
|
||||
}
|
||||
|
||||
return ret;
|
||||
}
|
||||
|
||||
static void nv_self_test(struct net_device *dev, struct ethtool_test *test, u64 *buffer)
|
||||
{
|
||||
struct fe_priv *np = netdev_priv(dev);
|
||||
u8 __iomem *base = get_hwbase(dev);
|
||||
int result;
|
||||
memset(buffer, 0, nv_self_test_count(dev)*sizeof(u64));
|
||||
|
||||
if (!nv_link_test(dev)) {
|
||||
test->flags |= ETH_TEST_FL_FAILED;
|
||||
buffer[0] = 1;
|
||||
}
|
||||
|
||||
if (test->flags & ETH_TEST_FL_OFFLINE) {
|
||||
if (netif_running(dev)) {
|
||||
netif_stop_queue(dev);
|
||||
spin_lock_bh(&dev->xmit_lock);
|
||||
spin_lock_irq(&np->lock);
|
||||
nv_disable_hw_interrupts(dev, np->irqmask);
|
||||
if (!(np->msi_flags & NV_MSI_X_ENABLED)) {
|
||||
writel(NVREG_IRQSTAT_MASK, base + NvRegIrqStatus);
|
||||
} else {
|
||||
writel(NVREG_IRQSTAT_MASK, base + NvRegMSIXIrqStatus);
|
||||
}
|
||||
/* stop engines */
|
||||
nv_stop_rx(dev);
|
||||
nv_stop_tx(dev);
|
||||
nv_txrx_reset(dev);
|
||||
/* drain rx queue */
|
||||
nv_drain_rx(dev);
|
||||
nv_drain_tx(dev);
|
||||
spin_unlock_irq(&np->lock);
|
||||
spin_unlock_bh(&dev->xmit_lock);
|
||||
}
|
||||
|
||||
if (!nv_register_test(dev)) {
|
||||
test->flags |= ETH_TEST_FL_FAILED;
|
||||
buffer[1] = 1;
|
||||
}
|
||||
|
||||
result = nv_interrupt_test(dev);
|
||||
if (result != 1) {
|
||||
test->flags |= ETH_TEST_FL_FAILED;
|
||||
buffer[2] = 1;
|
||||
}
|
||||
if (result == 0) {
|
||||
/* bail out */
|
||||
return;
|
||||
}
|
||||
|
||||
if (!nv_loopback_test(dev)) {
|
||||
test->flags |= ETH_TEST_FL_FAILED;
|
||||
buffer[3] = 1;
|
||||
}
|
||||
|
||||
if (netif_running(dev)) {
|
||||
/* reinit driver view of the rx queue */
|
||||
set_bufsize(dev);
|
||||
if (nv_init_ring(dev)) {
|
||||
if (!np->in_shutdown)
|
||||
mod_timer(&np->oom_kick, jiffies + OOM_REFILL);
|
||||
}
|
||||
/* reinit nic view of the rx queue */
|
||||
writel(np->rx_buf_sz, base + NvRegOffloadConfig);
|
||||
setup_hw_rings(dev, NV_SETUP_RX_RING | NV_SETUP_TX_RING);
|
||||
writel( ((np->rx_ring_size-1) << NVREG_RINGSZ_RXSHIFT) + ((np->tx_ring_size-1) << NVREG_RINGSZ_TXSHIFT),
|
||||
base + NvRegRingSizes);
|
||||
pci_push(base);
|
||||
writel(NVREG_TXRXCTL_KICK|np->txrxctl_bits, get_hwbase(dev) + NvRegTxRxControl);
|
||||
pci_push(base);
|
||||
/* restart rx engine */
|
||||
nv_start_rx(dev);
|
||||
nv_start_tx(dev);
|
||||
netif_start_queue(dev);
|
||||
nv_enable_hw_interrupts(dev, np->irqmask);
|
||||
}
|
||||
}
|
||||
}
|
||||
|
||||
static void nv_get_strings(struct net_device *dev, u32 stringset, u8 *buffer)
|
||||
{
|
||||
switch (stringset) {
|
||||
case ETH_SS_STATS:
|
||||
memcpy(buffer, &nv_estats_str, nv_get_stats_count(dev)*sizeof(struct nv_ethtool_str));
|
||||
break;
|
||||
case ETH_SS_TEST:
|
||||
memcpy(buffer, &nv_etests_str, nv_self_test_count(dev)*sizeof(struct nv_ethtool_str));
|
||||
break;
|
||||
}
|
||||
}
|
||||
|
||||
|
@ -3422,6 +3810,8 @@ static struct ethtool_ops ops = {
|
|||
.get_strings = nv_get_strings,
|
||||
.get_stats_count = nv_get_stats_count,
|
||||
.get_ethtool_stats = nv_get_ethtool_stats,
|
||||
.self_test_count = nv_self_test_count,
|
||||
.self_test = nv_self_test,
|
||||
};
|
||||
|
||||
static void nv_vlan_rx_register(struct net_device *dev, struct vlan_group *grp)
|
||||
|
@ -3554,7 +3944,7 @@ static int nv_open(struct net_device *dev)
|
|||
writel(NVREG_IRQSTAT_MASK, base + NvRegIrqStatus);
|
||||
pci_push(base);
|
||||
|
||||
if (nv_request_irq(dev)) {
|
||||
if (nv_request_irq(dev, 0)) {
|
||||
goto out_drain;
|
||||
}
|
||||
|
||||
|
@ -4049,11 +4439,11 @@ static struct pci_device_id pci_tbl[] = {
|
|||
},
|
||||
{ /* MCP55 Ethernet Controller */
|
||||
PCI_DEVICE(PCI_VENDOR_ID_NVIDIA, PCI_DEVICE_ID_NVIDIA_NVENET_14),
|
||||
.driver_data = DEV_NEED_TIMERIRQ|DEV_NEED_LINKTIMER|DEV_HAS_LARGEDESC|DEV_HAS_CHECKSUM|DEV_HAS_HIGH_DMA|DEV_HAS_VLAN|DEV_HAS_MSI|DEV_HAS_MSI_X|DEV_HAS_POWER_CNTRL|DEV_HAS_PAUSEFRAME_TX|DEV_HAS_STATISTICS,
|
||||
.driver_data = DEV_NEED_TIMERIRQ|DEV_NEED_LINKTIMER|DEV_HAS_LARGEDESC|DEV_HAS_CHECKSUM|DEV_HAS_HIGH_DMA|DEV_HAS_VLAN|DEV_HAS_MSI|DEV_HAS_MSI_X|DEV_HAS_POWER_CNTRL|DEV_HAS_PAUSEFRAME_TX|DEV_HAS_STATISTICS|DEV_HAS_TEST_EXTENDED,
|
||||
},
|
||||
{ /* MCP55 Ethernet Controller */
|
||||
PCI_DEVICE(PCI_VENDOR_ID_NVIDIA, PCI_DEVICE_ID_NVIDIA_NVENET_15),
|
||||
.driver_data = DEV_NEED_TIMERIRQ|DEV_NEED_LINKTIMER|DEV_HAS_LARGEDESC|DEV_HAS_CHECKSUM|DEV_HAS_HIGH_DMA|DEV_HAS_VLAN|DEV_HAS_MSI|DEV_HAS_MSI_X|DEV_HAS_POWER_CNTRL|DEV_HAS_PAUSEFRAME_TX|DEV_HAS_STATISTICS,
|
||||
.driver_data = DEV_NEED_TIMERIRQ|DEV_NEED_LINKTIMER|DEV_HAS_LARGEDESC|DEV_HAS_CHECKSUM|DEV_HAS_HIGH_DMA|DEV_HAS_VLAN|DEV_HAS_MSI|DEV_HAS_MSI_X|DEV_HAS_POWER_CNTRL|DEV_HAS_PAUSEFRAME_TX|DEV_HAS_STATISTICS|DEV_HAS_TEST_EXTENDED,
|
||||
},
|
||||
{0,},
|
||||
};
|
||||
|
|
Loading…
Reference in New Issue