2010-01-13 08:37:25 +08:00
|
|
|
/*
|
2010-12-03 04:41:56 +08:00
|
|
|
* QLogic qlcnic NIC Driver
|
|
|
|
* Copyright (c) 2009-2010 QLogic Corporation
|
2010-01-13 08:37:25 +08:00
|
|
|
*
|
2010-12-03 04:41:56 +08:00
|
|
|
* See LICENSE.qlcnic for copyright and licensing details.
|
2010-01-13 08:37:25 +08:00
|
|
|
*/
|
|
|
|
|
include cleanup: Update gfp.h and slab.h includes to prepare for breaking implicit slab.h inclusion from percpu.h
percpu.h is included by sched.h and module.h and thus ends up being
included when building most .c files. percpu.h includes slab.h which
in turn includes gfp.h making everything defined by the two files
universally available and complicating inclusion dependencies.
percpu.h -> slab.h dependency is about to be removed. Prepare for
this change by updating users of gfp and slab facilities include those
headers directly instead of assuming availability. As this conversion
needs to touch large number of source files, the following script is
used as the basis of conversion.
http://userweb.kernel.org/~tj/misc/slabh-sweep.py
The script does the followings.
* Scan files for gfp and slab usages and update includes such that
only the necessary includes are there. ie. if only gfp is used,
gfp.h, if slab is used, slab.h.
* When the script inserts a new include, it looks at the include
blocks and try to put the new include such that its order conforms
to its surrounding. It's put in the include block which contains
core kernel includes, in the same order that the rest are ordered -
alphabetical, Christmas tree, rev-Xmas-tree or at the end if there
doesn't seem to be any matching order.
* If the script can't find a place to put a new include (mostly
because the file doesn't have fitting include block), it prints out
an error message indicating which .h file needs to be added to the
file.
The conversion was done in the following steps.
1. The initial automatic conversion of all .c files updated slightly
over 4000 files, deleting around 700 includes and adding ~480 gfp.h
and ~3000 slab.h inclusions. The script emitted errors for ~400
files.
2. Each error was manually checked. Some didn't need the inclusion,
some needed manual addition while adding it to implementation .h or
embedding .c file was more appropriate for others. This step added
inclusions to around 150 files.
3. The script was run again and the output was compared to the edits
from #2 to make sure no file was left behind.
4. Several build tests were done and a couple of problems were fixed.
e.g. lib/decompress_*.c used malloc/free() wrappers around slab
APIs requiring slab.h to be added manually.
5. The script was run on all .h files but without automatically
editing them as sprinkling gfp.h and slab.h inclusions around .h
files could easily lead to inclusion dependency hell. Most gfp.h
inclusion directives were ignored as stuff from gfp.h was usually
wildly available and often used in preprocessor macros. Each
slab.h inclusion directive was examined and added manually as
necessary.
6. percpu.h was updated not to include slab.h.
7. Build test were done on the following configurations and failures
were fixed. CONFIG_GCOV_KERNEL was turned off for all tests (as my
distributed build env didn't work with gcov compiles) and a few
more options had to be turned off depending on archs to make things
build (like ipr on powerpc/64 which failed due to missing writeq).
* x86 and x86_64 UP and SMP allmodconfig and a custom test config.
* powerpc and powerpc64 SMP allmodconfig
* sparc and sparc64 SMP allmodconfig
* ia64 SMP allmodconfig
* s390 SMP allmodconfig
* alpha SMP allmodconfig
* um on x86_64 SMP allmodconfig
8. percpu.h modifications were reverted so that it could be applied as
a separate patch and serve as bisection point.
Given the fact that I had only a couple of failures from tests on step
6, I'm fairly confident about the coverage of this conversion patch.
If there is a breakage, it's likely to be something in one of the arch
headers which should be easily discoverable easily on most builds of
the specific arch.
Signed-off-by: Tejun Heo <tj@kernel.org>
Guess-its-ok-by: Christoph Lameter <cl@linux-foundation.org>
Cc: Ingo Molnar <mingo@redhat.com>
Cc: Lee Schermerhorn <Lee.Schermerhorn@hp.com>
2010-03-24 16:04:11 +08:00
|
|
|
#include <linux/slab.h>
|
2010-01-13 08:37:25 +08:00
|
|
|
#include <linux/vmalloc.h>
|
|
|
|
#include <linux/interrupt.h>
|
|
|
|
|
|
|
|
#include "qlcnic.h"
|
|
|
|
|
2010-10-04 12:20:13 +08:00
|
|
|
#include <linux/swab.h>
|
2010-01-13 08:37:25 +08:00
|
|
|
#include <linux/dma-mapping.h>
|
|
|
|
#include <net/ip.h>
|
|
|
|
#include <linux/ipv6.h>
|
|
|
|
#include <linux/inetdevice.h>
|
|
|
|
#include <linux/sysfs.h>
|
2010-07-14 04:33:34 +08:00
|
|
|
#include <linux/aer.h>
|
2011-04-28 19:48:18 +08:00
|
|
|
#include <linux/log2.h>
|
2010-01-13 08:37:25 +08:00
|
|
|
|
2010-06-17 10:56:39 +08:00
|
|
|
MODULE_DESCRIPTION("QLogic 1/10 GbE Converged/Intelligent Ethernet Driver");
|
2010-01-13 08:37:25 +08:00
|
|
|
MODULE_LICENSE("GPL");
|
|
|
|
MODULE_VERSION(QLCNIC_LINUX_VERSIONID);
|
|
|
|
MODULE_FIRMWARE(QLCNIC_UNIFIED_ROMIMAGE_NAME);
|
|
|
|
|
|
|
|
char qlcnic_driver_name[] = "qlcnic";
|
2010-06-17 10:56:39 +08:00
|
|
|
static const char qlcnic_driver_string[] = "QLogic 1/10 GbE "
|
|
|
|
"Converged/Intelligent Ethernet Driver v" QLCNIC_LINUX_VERSIONID;
|
2010-01-13 08:37:25 +08:00
|
|
|
|
2010-10-08 07:46:05 +08:00
|
|
|
static struct workqueue_struct *qlcnic_wq;
|
2010-09-01 01:17:51 +08:00
|
|
|
static int qlcnic_mac_learn;
|
2011-01-10 08:15:23 +08:00
|
|
|
module_param(qlcnic_mac_learn, int, 0444);
|
2010-09-01 01:17:51 +08:00
|
|
|
MODULE_PARM_DESC(qlcnic_mac_learn, "Mac Filter (0=disabled, 1=enabled)");
|
|
|
|
|
2010-01-13 08:37:25 +08:00
|
|
|
static int use_msi = 1;
|
2011-01-10 08:15:23 +08:00
|
|
|
module_param(use_msi, int, 0444);
|
2010-01-13 08:37:25 +08:00
|
|
|
MODULE_PARM_DESC(use_msi, "MSI interrupt (0=disabled, 1=enabled");
|
|
|
|
|
|
|
|
static int use_msi_x = 1;
|
2011-01-10 08:15:23 +08:00
|
|
|
module_param(use_msi_x, int, 0444);
|
2010-01-13 08:37:25 +08:00
|
|
|
MODULE_PARM_DESC(use_msi_x, "MSI-X interrupt (0=disabled, 1=enabled");
|
|
|
|
|
2011-02-23 11:21:24 +08:00
|
|
|
static int auto_fw_reset = 1;
|
2010-01-13 08:37:25 +08:00
|
|
|
module_param(auto_fw_reset, int, 0644);
|
|
|
|
MODULE_PARM_DESC(auto_fw_reset, "Auto firmware reset (0=disabled, 1=enabled");
|
|
|
|
|
2010-05-17 09:22:11 +08:00
|
|
|
static int load_fw_file;
|
2011-01-10 08:15:23 +08:00
|
|
|
module_param(load_fw_file, int, 0444);
|
2010-05-17 09:22:11 +08:00
|
|
|
MODULE_PARM_DESC(load_fw_file, "Load firmware from (0=flash, 1=file");
|
|
|
|
|
2010-06-01 19:28:51 +08:00
|
|
|
static int qlcnic_config_npars;
|
2011-01-10 08:15:23 +08:00
|
|
|
module_param(qlcnic_config_npars, int, 0444);
|
2010-06-01 19:28:51 +08:00
|
|
|
MODULE_PARM_DESC(qlcnic_config_npars, "Configure NPARs (0=disabled, 1=enabled");
|
|
|
|
|
2010-01-13 08:37:25 +08:00
|
|
|
static int __devinit qlcnic_probe(struct pci_dev *pdev,
|
|
|
|
const struct pci_device_id *ent);
|
|
|
|
static void __devexit qlcnic_remove(struct pci_dev *pdev);
|
|
|
|
static int qlcnic_open(struct net_device *netdev);
|
|
|
|
static int qlcnic_close(struct net_device *netdev);
|
|
|
|
static void qlcnic_tx_timeout(struct net_device *netdev);
|
|
|
|
static void qlcnic_attach_work(struct work_struct *work);
|
|
|
|
static void qlcnic_fwinit_work(struct work_struct *work);
|
|
|
|
static void qlcnic_fw_poll_work(struct work_struct *work);
|
|
|
|
static void qlcnic_schedule_work(struct qlcnic_adapter *adapter,
|
|
|
|
work_func_t func, int delay);
|
|
|
|
static void qlcnic_cancel_fw_work(struct qlcnic_adapter *adapter);
|
|
|
|
#ifdef CONFIG_NET_POLL_CONTROLLER
|
|
|
|
static void qlcnic_poll_controller(struct net_device *netdev);
|
|
|
|
#endif
|
|
|
|
|
2010-05-13 11:07:50 +08:00
|
|
|
static void qlcnic_idc_debug_info(struct qlcnic_adapter *adapter, u8 encoding);
|
2010-08-19 13:08:29 +08:00
|
|
|
static void qlcnic_clr_all_drv_state(struct qlcnic_adapter *adapter, u8);
|
2010-01-13 08:37:25 +08:00
|
|
|
static int qlcnic_can_start_firmware(struct qlcnic_adapter *adapter);
|
|
|
|
|
2010-02-01 13:24:59 +08:00
|
|
|
static irqreturn_t qlcnic_tmp_intr(int irq, void *data);
|
2010-01-13 08:37:25 +08:00
|
|
|
static irqreturn_t qlcnic_intr(int irq, void *data);
|
|
|
|
static irqreturn_t qlcnic_msi_intr(int irq, void *data);
|
|
|
|
static irqreturn_t qlcnic_msix_intr(int irq, void *data);
|
|
|
|
|
|
|
|
static struct net_device_stats *qlcnic_get_stats(struct net_device *netdev);
|
2010-09-17 03:14:41 +08:00
|
|
|
static void qlcnic_restore_indev_addr(struct net_device *dev, unsigned long);
|
2010-06-01 19:33:09 +08:00
|
|
|
static int qlcnic_start_firmware(struct qlcnic_adapter *);
|
|
|
|
|
2010-09-01 01:17:51 +08:00
|
|
|
static void qlcnic_free_lb_filters_mem(struct qlcnic_adapter *adapter);
|
2010-06-01 19:33:09 +08:00
|
|
|
static void qlcnic_dev_set_npar_ready(struct qlcnic_adapter *);
|
|
|
|
static int qlcnicvf_start_firmware(struct qlcnic_adapter *);
|
2010-08-19 13:08:26 +08:00
|
|
|
static void qlcnic_set_netdev_features(struct qlcnic_adapter *,
|
|
|
|
struct qlcnic_esw_func_cfg *);
|
2011-12-09 08:52:37 +08:00
|
|
|
static int qlcnic_vlan_rx_add(struct net_device *, u16);
|
|
|
|
static int qlcnic_vlan_rx_del(struct net_device *, u16);
|
2011-04-01 22:28:15 +08:00
|
|
|
|
2010-01-13 08:37:25 +08:00
|
|
|
/* PCI Device ID Table */
|
|
|
|
#define ENTRY(device) \
|
|
|
|
{PCI_DEVICE(PCI_VENDOR_ID_QLOGIC, (device)), \
|
|
|
|
.class = PCI_CLASS_NETWORK_ETHERNET << 8, .class_mask = ~0}
|
|
|
|
|
|
|
|
#define PCI_DEVICE_ID_QLOGIC_QLE824X 0x8020
|
|
|
|
|
2010-02-01 13:24:54 +08:00
|
|
|
static DEFINE_PCI_DEVICE_TABLE(qlcnic_pci_tbl) = {
|
2010-01-13 08:37:25 +08:00
|
|
|
ENTRY(PCI_DEVICE_ID_QLOGIC_QLE824X),
|
|
|
|
{0,}
|
|
|
|
};
|
|
|
|
|
|
|
|
MODULE_DEVICE_TABLE(pci, qlcnic_pci_tbl);
|
|
|
|
|
|
|
|
|
2012-11-18 05:04:38 +08:00
|
|
|
inline void qlcnic_update_cmd_producer(struct qlcnic_host_tx_ring *tx_ring)
|
2010-01-13 08:37:25 +08:00
|
|
|
{
|
|
|
|
writel(tx_ring->producer, tx_ring->crb_cmd_producer);
|
|
|
|
}
|
|
|
|
|
|
|
|
static const u32 msi_tgt_status[8] = {
|
|
|
|
ISR_INT_TARGET_STATUS, ISR_INT_TARGET_STATUS_F1,
|
|
|
|
ISR_INT_TARGET_STATUS_F2, ISR_INT_TARGET_STATUS_F3,
|
|
|
|
ISR_INT_TARGET_STATUS_F4, ISR_INT_TARGET_STATUS_F5,
|
|
|
|
ISR_INT_TARGET_STATUS_F6, ISR_INT_TARGET_STATUS_F7
|
|
|
|
};
|
|
|
|
|
|
|
|
static const
|
|
|
|
struct qlcnic_legacy_intr_set legacy_intr[] = QLCNIC_LEGACY_INTR_CONFIG;
|
|
|
|
|
2012-11-28 12:34:26 +08:00
|
|
|
int qlcnic_alloc_sds_rings(struct qlcnic_recv_context *recv_ctx, int count)
|
2010-01-13 08:37:25 +08:00
|
|
|
{
|
|
|
|
int size = sizeof(struct qlcnic_host_sds_ring) * count;
|
|
|
|
|
|
|
|
recv_ctx->sds_rings = kzalloc(size, GFP_KERNEL);
|
|
|
|
|
2010-09-23 13:40:09 +08:00
|
|
|
return recv_ctx->sds_rings == NULL;
|
2010-01-13 08:37:25 +08:00
|
|
|
}
|
|
|
|
|
2012-11-28 12:34:26 +08:00
|
|
|
void qlcnic_free_sds_rings(struct qlcnic_recv_context *recv_ctx)
|
2010-01-13 08:37:25 +08:00
|
|
|
{
|
|
|
|
if (recv_ctx->sds_rings != NULL)
|
|
|
|
kfree(recv_ctx->sds_rings);
|
|
|
|
|
|
|
|
recv_ctx->sds_rings = NULL;
|
|
|
|
}
|
|
|
|
|
|
|
|
static void qlcnic_clear_stats(struct qlcnic_adapter *adapter)
|
|
|
|
{
|
|
|
|
memset(&adapter->stats, 0, sizeof(adapter->stats));
|
|
|
|
}
|
|
|
|
|
|
|
|
static void qlcnic_set_msix_bit(struct pci_dev *pdev, int enable)
|
|
|
|
{
|
|
|
|
u32 control;
|
|
|
|
int pos;
|
|
|
|
|
|
|
|
pos = pci_find_capability(pdev, PCI_CAP_ID_MSIX);
|
|
|
|
if (pos) {
|
|
|
|
pci_read_config_dword(pdev, pos, &control);
|
|
|
|
if (enable)
|
|
|
|
control |= PCI_MSIX_FLAGS_ENABLE;
|
|
|
|
else
|
|
|
|
control = 0;
|
|
|
|
pci_write_config_dword(pdev, pos, control);
|
|
|
|
}
|
|
|
|
}
|
|
|
|
|
|
|
|
static void qlcnic_init_msix_entries(struct qlcnic_adapter *adapter, int count)
|
|
|
|
{
|
|
|
|
int i;
|
|
|
|
|
|
|
|
for (i = 0; i < count; i++)
|
|
|
|
adapter->msix_entries[i].entry = i;
|
|
|
|
}
|
|
|
|
|
|
|
|
static int
|
|
|
|
qlcnic_read_mac_addr(struct qlcnic_adapter *adapter)
|
|
|
|
{
|
2010-06-01 19:28:51 +08:00
|
|
|
u8 mac_addr[ETH_ALEN];
|
2010-01-13 08:37:25 +08:00
|
|
|
struct net_device *netdev = adapter->netdev;
|
|
|
|
struct pci_dev *pdev = adapter->pdev;
|
|
|
|
|
2010-09-01 01:17:46 +08:00
|
|
|
if (qlcnic_get_mac_address(adapter, mac_addr) != 0)
|
2010-01-13 08:37:25 +08:00
|
|
|
return -EIO;
|
|
|
|
|
2010-06-01 19:28:51 +08:00
|
|
|
memcpy(netdev->dev_addr, mac_addr, ETH_ALEN);
|
2010-01-13 08:37:25 +08:00
|
|
|
memcpy(netdev->perm_addr, netdev->dev_addr, netdev->addr_len);
|
|
|
|
memcpy(adapter->mac_addr, netdev->dev_addr, netdev->addr_len);
|
|
|
|
|
|
|
|
/* set station address */
|
|
|
|
|
|
|
|
if (!is_valid_ether_addr(netdev->perm_addr))
|
|
|
|
dev_warn(&pdev->dev, "Bad MAC address %pM.\n",
|
|
|
|
netdev->dev_addr);
|
|
|
|
|
|
|
|
return 0;
|
|
|
|
}
|
|
|
|
|
|
|
|
static int qlcnic_set_mac(struct net_device *netdev, void *p)
|
|
|
|
{
|
|
|
|
struct qlcnic_adapter *adapter = netdev_priv(netdev);
|
|
|
|
struct sockaddr *addr = p;
|
|
|
|
|
2010-09-01 01:17:50 +08:00
|
|
|
if ((adapter->flags & QLCNIC_MAC_OVERRIDE_DISABLED))
|
|
|
|
return -EOPNOTSUPP;
|
|
|
|
|
2010-01-13 08:37:25 +08:00
|
|
|
if (!is_valid_ether_addr(addr->sa_data))
|
2012-02-21 10:07:49 +08:00
|
|
|
return -EADDRNOTAVAIL;
|
2010-01-13 08:37:25 +08:00
|
|
|
|
2010-06-22 11:19:01 +08:00
|
|
|
if (test_bit(__QLCNIC_DEV_UP, &adapter->state)) {
|
2010-01-13 08:37:25 +08:00
|
|
|
netif_device_detach(netdev);
|
|
|
|
qlcnic_napi_disable(adapter);
|
|
|
|
}
|
|
|
|
|
|
|
|
memcpy(adapter->mac_addr, addr->sa_data, netdev->addr_len);
|
|
|
|
memcpy(netdev->dev_addr, addr->sa_data, netdev->addr_len);
|
|
|
|
qlcnic_set_multi(adapter->netdev);
|
|
|
|
|
2010-06-22 11:19:01 +08:00
|
|
|
if (test_bit(__QLCNIC_DEV_UP, &adapter->state)) {
|
2010-01-13 08:37:25 +08:00
|
|
|
netif_device_attach(netdev);
|
|
|
|
qlcnic_napi_enable(adapter);
|
|
|
|
}
|
|
|
|
return 0;
|
|
|
|
}
|
|
|
|
|
|
|
|
static const struct net_device_ops qlcnic_netdev_ops = {
|
|
|
|
.ndo_open = qlcnic_open,
|
|
|
|
.ndo_stop = qlcnic_close,
|
|
|
|
.ndo_start_xmit = qlcnic_xmit_frame,
|
|
|
|
.ndo_get_stats = qlcnic_get_stats,
|
|
|
|
.ndo_validate_addr = eth_validate_addr,
|
2011-08-16 14:29:01 +08:00
|
|
|
.ndo_set_rx_mode = qlcnic_set_multi,
|
2010-01-13 08:37:25 +08:00
|
|
|
.ndo_set_mac_address = qlcnic_set_mac,
|
|
|
|
.ndo_change_mtu = qlcnic_change_mtu,
|
2011-04-19 11:03:57 +08:00
|
|
|
.ndo_fix_features = qlcnic_fix_features,
|
|
|
|
.ndo_set_features = qlcnic_set_features,
|
2010-01-13 08:37:25 +08:00
|
|
|
.ndo_tx_timeout = qlcnic_tx_timeout,
|
2011-04-01 22:28:15 +08:00
|
|
|
.ndo_vlan_rx_add_vid = qlcnic_vlan_rx_add,
|
|
|
|
.ndo_vlan_rx_kill_vid = qlcnic_vlan_rx_del,
|
2010-01-13 08:37:25 +08:00
|
|
|
#ifdef CONFIG_NET_POLL_CONTROLLER
|
|
|
|
.ndo_poll_controller = qlcnic_poll_controller,
|
|
|
|
#endif
|
|
|
|
};
|
|
|
|
|
2012-04-26 18:31:29 +08:00
|
|
|
static const struct net_device_ops qlcnic_netdev_failed_ops = {
|
|
|
|
.ndo_open = qlcnic_open,
|
|
|
|
};
|
|
|
|
|
2010-06-01 19:28:51 +08:00
|
|
|
static struct qlcnic_nic_template qlcnic_ops = {
|
|
|
|
.config_bridged_mode = qlcnic_config_bridged_mode,
|
|
|
|
.config_led = qlcnic_config_led,
|
2010-06-01 19:33:09 +08:00
|
|
|
.start_firmware = qlcnic_start_firmware
|
|
|
|
};
|
|
|
|
|
|
|
|
static struct qlcnic_nic_template qlcnic_vf_ops = {
|
|
|
|
.config_bridged_mode = qlcnicvf_config_bridged_mode,
|
|
|
|
.config_led = qlcnicvf_config_led,
|
|
|
|
.start_firmware = qlcnicvf_start_firmware
|
2010-06-01 19:28:51 +08:00
|
|
|
};
|
|
|
|
|
2011-04-28 19:48:18 +08:00
|
|
|
static int qlcnic_enable_msix(struct qlcnic_adapter *adapter, u32 num_msix)
|
2010-01-13 08:37:25 +08:00
|
|
|
{
|
|
|
|
struct pci_dev *pdev = adapter->pdev;
|
2011-04-28 19:48:18 +08:00
|
|
|
int err = -1;
|
2010-01-13 08:37:25 +08:00
|
|
|
|
|
|
|
adapter->max_sds_rings = 1;
|
|
|
|
adapter->flags &= ~(QLCNIC_MSI_ENABLED | QLCNIC_MSIX_ENABLED);
|
|
|
|
qlcnic_set_msix_bit(pdev, 0);
|
|
|
|
|
|
|
|
if (adapter->msix_supported) {
|
2011-04-28 19:48:18 +08:00
|
|
|
enable_msix:
|
2010-01-13 08:37:25 +08:00
|
|
|
qlcnic_init_msix_entries(adapter, num_msix);
|
|
|
|
err = pci_enable_msix(pdev, adapter->msix_entries, num_msix);
|
|
|
|
if (err == 0) {
|
|
|
|
adapter->flags |= QLCNIC_MSIX_ENABLED;
|
|
|
|
qlcnic_set_msix_bit(pdev, 1);
|
|
|
|
|
2011-04-01 22:28:05 +08:00
|
|
|
adapter->max_sds_rings = num_msix;
|
2010-01-13 08:37:25 +08:00
|
|
|
|
|
|
|
dev_info(&pdev->dev, "using msi-x interrupts\n");
|
2011-04-28 19:48:18 +08:00
|
|
|
return err;
|
2010-01-13 08:37:25 +08:00
|
|
|
}
|
2011-04-28 19:48:18 +08:00
|
|
|
if (err > 0) {
|
|
|
|
num_msix = rounddown_pow_of_two(err);
|
|
|
|
if (num_msix)
|
|
|
|
goto enable_msix;
|
|
|
|
}
|
|
|
|
}
|
|
|
|
return err;
|
|
|
|
}
|
2010-01-13 08:37:25 +08:00
|
|
|
|
|
|
|
|
2011-04-28 19:48:18 +08:00
|
|
|
static void qlcnic_enable_msi_legacy(struct qlcnic_adapter *adapter)
|
|
|
|
{
|
|
|
|
const struct qlcnic_legacy_intr_set *legacy_intrp;
|
|
|
|
struct pci_dev *pdev = adapter->pdev;
|
2010-01-13 08:37:25 +08:00
|
|
|
|
|
|
|
if (use_msi && !pci_enable_msi(pdev)) {
|
|
|
|
adapter->flags |= QLCNIC_MSI_ENABLED;
|
|
|
|
adapter->tgt_status_reg = qlcnic_get_ioaddr(adapter,
|
2011-04-01 22:28:05 +08:00
|
|
|
msi_tgt_status[adapter->ahw->pci_func]);
|
2010-01-13 08:37:25 +08:00
|
|
|
dev_info(&pdev->dev, "using msi interrupts\n");
|
|
|
|
adapter->msix_entries[0].vector = pdev->irq;
|
|
|
|
return;
|
|
|
|
}
|
|
|
|
|
2011-04-28 19:48:18 +08:00
|
|
|
legacy_intrp = &legacy_intr[adapter->ahw->pci_func];
|
|
|
|
|
|
|
|
adapter->int_vec_bit = legacy_intrp->int_vec_bit;
|
|
|
|
adapter->tgt_status_reg = qlcnic_get_ioaddr(adapter,
|
|
|
|
legacy_intrp->tgt_status_reg);
|
|
|
|
adapter->tgt_mask_reg = qlcnic_get_ioaddr(adapter,
|
|
|
|
legacy_intrp->tgt_mask_reg);
|
|
|
|
adapter->isr_int_vec = qlcnic_get_ioaddr(adapter, ISR_INT_VECTOR);
|
|
|
|
|
|
|
|
adapter->crb_int_state_reg = qlcnic_get_ioaddr(adapter,
|
|
|
|
ISR_INT_STATE_REG);
|
2010-01-13 08:37:25 +08:00
|
|
|
dev_info(&pdev->dev, "using legacy interrupts\n");
|
|
|
|
adapter->msix_entries[0].vector = pdev->irq;
|
|
|
|
}
|
|
|
|
|
2011-04-28 19:48:18 +08:00
|
|
|
static void
|
|
|
|
qlcnic_setup_intr(struct qlcnic_adapter *adapter)
|
|
|
|
{
|
|
|
|
int num_msix;
|
|
|
|
|
|
|
|
if (adapter->msix_supported) {
|
2011-06-22 10:52:20 +08:00
|
|
|
num_msix = rounddown_pow_of_two(min_t(int, num_online_cpus(),
|
|
|
|
QLCNIC_DEF_NUM_STS_DESC_RINGS));
|
2011-04-28 19:48:18 +08:00
|
|
|
} else
|
|
|
|
num_msix = 1;
|
|
|
|
|
|
|
|
if (!qlcnic_enable_msix(adapter, num_msix))
|
|
|
|
return;
|
|
|
|
|
|
|
|
qlcnic_enable_msi_legacy(adapter);
|
|
|
|
}
|
|
|
|
|
2010-01-13 08:37:25 +08:00
|
|
|
static void
|
|
|
|
qlcnic_teardown_intr(struct qlcnic_adapter *adapter)
|
|
|
|
{
|
|
|
|
if (adapter->flags & QLCNIC_MSIX_ENABLED)
|
|
|
|
pci_disable_msix(adapter->pdev);
|
|
|
|
if (adapter->flags & QLCNIC_MSI_ENABLED)
|
|
|
|
pci_disable_msi(adapter->pdev);
|
|
|
|
}
|
|
|
|
|
|
|
|
static void
|
|
|
|
qlcnic_cleanup_pci_map(struct qlcnic_adapter *adapter)
|
|
|
|
{
|
2011-04-01 22:28:05 +08:00
|
|
|
if (adapter->ahw->pci_base0 != NULL)
|
|
|
|
iounmap(adapter->ahw->pci_base0);
|
2010-01-13 08:37:25 +08:00
|
|
|
}
|
|
|
|
|
2010-06-29 16:01:20 +08:00
|
|
|
static int
|
|
|
|
qlcnic_init_pci_info(struct qlcnic_adapter *adapter)
|
|
|
|
{
|
2010-08-10 05:49:36 +08:00
|
|
|
struct qlcnic_pci_info *pci_info;
|
2010-08-16 18:23:51 +08:00
|
|
|
int i, ret = 0;
|
2010-06-29 16:01:20 +08:00
|
|
|
u8 pfn;
|
|
|
|
|
2010-08-10 05:49:36 +08:00
|
|
|
pci_info = kcalloc(QLCNIC_MAX_PCI_FUNC, sizeof(*pci_info), GFP_KERNEL);
|
|
|
|
if (!pci_info)
|
|
|
|
return -ENOMEM;
|
|
|
|
|
2010-08-10 05:47:56 +08:00
|
|
|
adapter->npars = kzalloc(sizeof(struct qlcnic_npar_info) *
|
2010-06-29 16:01:20 +08:00
|
|
|
QLCNIC_MAX_PCI_FUNC, GFP_KERNEL);
|
2010-08-10 05:49:36 +08:00
|
|
|
if (!adapter->npars) {
|
2010-08-16 18:23:51 +08:00
|
|
|
ret = -ENOMEM;
|
2010-08-10 05:49:36 +08:00
|
|
|
goto err_pci_info;
|
|
|
|
}
|
2010-06-29 16:01:20 +08:00
|
|
|
|
2010-08-10 05:47:56 +08:00
|
|
|
adapter->eswitch = kzalloc(sizeof(struct qlcnic_eswitch) *
|
2010-06-29 16:01:20 +08:00
|
|
|
QLCNIC_NIU_MAX_XG_PORTS, GFP_KERNEL);
|
|
|
|
if (!adapter->eswitch) {
|
2010-08-16 18:23:51 +08:00
|
|
|
ret = -ENOMEM;
|
2010-08-10 05:47:56 +08:00
|
|
|
goto err_npars;
|
2010-06-29 16:01:20 +08:00
|
|
|
}
|
|
|
|
|
|
|
|
ret = qlcnic_get_pci_info(adapter, pci_info);
|
2010-08-10 05:47:56 +08:00
|
|
|
if (ret)
|
|
|
|
goto err_eswitch;
|
2010-06-29 16:01:20 +08:00
|
|
|
|
2010-08-10 05:47:56 +08:00
|
|
|
for (i = 0; i < QLCNIC_MAX_PCI_FUNC; i++) {
|
|
|
|
pfn = pci_info[i].id;
|
2012-06-14 16:34:24 +08:00
|
|
|
if (pfn >= QLCNIC_MAX_PCI_FUNC) {
|
2011-04-01 22:27:59 +08:00
|
|
|
ret = QL_STATUS_INVALID_PARAM;
|
|
|
|
goto err_eswitch;
|
|
|
|
}
|
2010-10-04 12:20:14 +08:00
|
|
|
adapter->npars[pfn].active = (u8)pci_info[i].active;
|
|
|
|
adapter->npars[pfn].type = (u8)pci_info[i].type;
|
|
|
|
adapter->npars[pfn].phy_port = (u8)pci_info[i].default_port;
|
2010-08-10 05:47:56 +08:00
|
|
|
adapter->npars[pfn].min_bw = pci_info[i].tx_min_bw;
|
|
|
|
adapter->npars[pfn].max_bw = pci_info[i].tx_max_bw;
|
2010-06-29 16:01:20 +08:00
|
|
|
}
|
|
|
|
|
2010-08-10 05:47:56 +08:00
|
|
|
for (i = 0; i < QLCNIC_NIU_MAX_XG_PORTS; i++)
|
|
|
|
adapter->eswitch[i].flags |= QLCNIC_SWITCH_ENABLE;
|
|
|
|
|
2010-08-10 05:49:36 +08:00
|
|
|
kfree(pci_info);
|
2010-08-10 05:47:56 +08:00
|
|
|
return 0;
|
|
|
|
|
|
|
|
err_eswitch:
|
2010-06-29 16:01:20 +08:00
|
|
|
kfree(adapter->eswitch);
|
|
|
|
adapter->eswitch = NULL;
|
2010-08-10 05:47:56 +08:00
|
|
|
err_npars:
|
2010-06-29 16:01:20 +08:00
|
|
|
kfree(adapter->npars);
|
2010-08-10 05:47:56 +08:00
|
|
|
adapter->npars = NULL;
|
2010-08-10 05:49:36 +08:00
|
|
|
err_pci_info:
|
|
|
|
kfree(pci_info);
|
2010-06-29 16:01:20 +08:00
|
|
|
|
|
|
|
return ret;
|
|
|
|
}
|
|
|
|
|
2010-06-01 19:28:51 +08:00
|
|
|
static int
|
|
|
|
qlcnic_set_function_modes(struct qlcnic_adapter *adapter)
|
|
|
|
{
|
|
|
|
u8 id;
|
|
|
|
u32 ref_count;
|
|
|
|
int i, ret = 1;
|
|
|
|
u32 data = QLCNIC_MGMT_FUNC;
|
2011-04-01 22:28:05 +08:00
|
|
|
void __iomem *priv_op = adapter->ahw->pci_base0 + QLCNIC_DRV_OP_MODE;
|
2010-06-01 19:28:51 +08:00
|
|
|
|
|
|
|
/* If other drivers are not in use set their privilege level */
|
2010-08-25 12:03:05 +08:00
|
|
|
ref_count = QLCRD32(adapter, QLCNIC_CRB_DRV_ACTIVE);
|
2010-06-01 19:28:51 +08:00
|
|
|
ret = qlcnic_api_lock(adapter);
|
|
|
|
if (ret)
|
|
|
|
goto err_lock;
|
|
|
|
|
2010-06-16 17:07:27 +08:00
|
|
|
if (qlcnic_config_npars) {
|
|
|
|
for (i = 0; i < QLCNIC_MAX_PCI_FUNC; i++) {
|
2010-06-29 16:01:20 +08:00
|
|
|
id = i;
|
2010-06-16 17:07:27 +08:00
|
|
|
if (adapter->npars[i].type != QLCNIC_TYPE_NIC ||
|
2011-04-01 22:28:05 +08:00
|
|
|
id == adapter->ahw->pci_func)
|
2010-06-16 17:07:27 +08:00
|
|
|
continue;
|
|
|
|
data |= (qlcnic_config_npars &
|
|
|
|
QLC_DEV_SET_DRV(0xf, id));
|
|
|
|
}
|
|
|
|
} else {
|
|
|
|
data = readl(priv_op);
|
2011-04-01 22:28:05 +08:00
|
|
|
data = (data & ~QLC_DEV_SET_DRV(0xf, adapter->ahw->pci_func)) |
|
2010-06-16 17:07:27 +08:00
|
|
|
(QLC_DEV_SET_DRV(QLCNIC_MGMT_FUNC,
|
2011-04-01 22:28:05 +08:00
|
|
|
adapter->ahw->pci_func));
|
2010-06-01 19:28:51 +08:00
|
|
|
}
|
|
|
|
writel(data, priv_op);
|
|
|
|
qlcnic_api_unlock(adapter);
|
|
|
|
err_lock:
|
|
|
|
return ret;
|
|
|
|
}
|
|
|
|
|
2010-08-26 22:02:52 +08:00
|
|
|
static void
|
|
|
|
qlcnic_check_vf(struct qlcnic_adapter *adapter)
|
2010-06-01 19:28:51 +08:00
|
|
|
{
|
|
|
|
void __iomem *msix_base_addr;
|
|
|
|
void __iomem *priv_op;
|
|
|
|
u32 func;
|
|
|
|
u32 msix_base;
|
|
|
|
u32 op_mode, priv_level;
|
|
|
|
|
|
|
|
/* Determine FW API version */
|
2011-04-01 22:28:05 +08:00
|
|
|
adapter->fw_hal_version = readl(adapter->ahw->pci_base0 +
|
|
|
|
QLCNIC_FW_API);
|
2010-06-01 19:28:51 +08:00
|
|
|
|
|
|
|
/* Find PCI function number */
|
|
|
|
pci_read_config_dword(adapter->pdev, QLCNIC_MSIX_TABLE_OFFSET, &func);
|
2011-04-01 22:28:05 +08:00
|
|
|
msix_base_addr = adapter->ahw->pci_base0 + QLCNIC_MSIX_BASE;
|
2010-06-01 19:28:51 +08:00
|
|
|
msix_base = readl(msix_base_addr);
|
|
|
|
func = (func - msix_base)/QLCNIC_MSIX_TBL_PGSIZE;
|
2011-04-01 22:28:05 +08:00
|
|
|
adapter->ahw->pci_func = func;
|
2010-06-01 19:28:51 +08:00
|
|
|
|
|
|
|
/* Determine function privilege level */
|
2011-04-01 22:28:05 +08:00
|
|
|
priv_op = adapter->ahw->pci_base0 + QLCNIC_DRV_OP_MODE;
|
2010-06-01 19:28:51 +08:00
|
|
|
op_mode = readl(priv_op);
|
2010-06-16 17:07:27 +08:00
|
|
|
if (op_mode == QLC_DEV_DRV_DEFAULT)
|
2010-06-01 19:28:51 +08:00
|
|
|
priv_level = QLCNIC_MGMT_FUNC;
|
2010-06-16 17:07:27 +08:00
|
|
|
else
|
2011-04-01 22:28:05 +08:00
|
|
|
priv_level = QLC_DEV_GET_DRV(op_mode, adapter->ahw->pci_func);
|
2010-06-01 19:28:51 +08:00
|
|
|
|
2010-08-26 22:02:52 +08:00
|
|
|
if (priv_level == QLCNIC_NON_PRIV_FUNC) {
|
2010-06-01 19:33:09 +08:00
|
|
|
adapter->op_mode = QLCNIC_NON_PRIV_FUNC;
|
|
|
|
dev_info(&adapter->pdev->dev,
|
|
|
|
"HAL Version: %d Non Privileged function\n",
|
|
|
|
adapter->fw_hal_version);
|
|
|
|
adapter->nic_ops = &qlcnic_vf_ops;
|
2010-08-26 22:02:52 +08:00
|
|
|
} else
|
|
|
|
adapter->nic_ops = &qlcnic_ops;
|
2010-06-01 19:28:51 +08:00
|
|
|
}
|
|
|
|
|
2010-01-13 08:37:25 +08:00
|
|
|
static int
|
|
|
|
qlcnic_setup_pci_map(struct qlcnic_adapter *adapter)
|
|
|
|
{
|
|
|
|
void __iomem *mem_ptr0 = NULL;
|
|
|
|
resource_size_t mem_base;
|
|
|
|
unsigned long mem_len, pci_len0 = 0;
|
|
|
|
|
|
|
|
struct pci_dev *pdev = adapter->pdev;
|
|
|
|
|
|
|
|
/* remap phys address */
|
|
|
|
mem_base = pci_resource_start(pdev, 0); /* 0 is for BAR 0 */
|
|
|
|
mem_len = pci_resource_len(pdev, 0);
|
|
|
|
|
|
|
|
if (mem_len == QLCNIC_PCI_2MB_SIZE) {
|
|
|
|
|
|
|
|
mem_ptr0 = pci_ioremap_bar(pdev, 0);
|
|
|
|
if (mem_ptr0 == NULL) {
|
|
|
|
dev_err(&pdev->dev, "failed to map PCI bar 0\n");
|
|
|
|
return -EIO;
|
|
|
|
}
|
|
|
|
pci_len0 = mem_len;
|
|
|
|
} else {
|
|
|
|
return -EIO;
|
|
|
|
}
|
|
|
|
|
|
|
|
dev_info(&pdev->dev, "%dMB memory map\n", (int)(mem_len>>20));
|
|
|
|
|
2011-04-01 22:28:05 +08:00
|
|
|
adapter->ahw->pci_base0 = mem_ptr0;
|
|
|
|
adapter->ahw->pci_len0 = pci_len0;
|
2010-01-13 08:37:25 +08:00
|
|
|
|
2010-08-26 22:02:52 +08:00
|
|
|
qlcnic_check_vf(adapter);
|
2010-06-01 19:28:51 +08:00
|
|
|
|
2011-04-01 22:28:05 +08:00
|
|
|
adapter->ahw->ocm_win_crb = qlcnic_get_ioaddr(adapter,
|
|
|
|
QLCNIC_PCIX_PS_REG(PCIX_OCM_WINDOW_REG(
|
|
|
|
adapter->ahw->pci_func)));
|
2010-01-13 08:37:25 +08:00
|
|
|
|
|
|
|
return 0;
|
|
|
|
}
|
|
|
|
|
|
|
|
static void get_brd_name(struct qlcnic_adapter *adapter, char *name)
|
|
|
|
{
|
|
|
|
struct pci_dev *pdev = adapter->pdev;
|
|
|
|
int i, found = 0;
|
|
|
|
|
|
|
|
for (i = 0; i < NUM_SUPPORTED_BOARDS; ++i) {
|
|
|
|
if (qlcnic_boards[i].vendor == pdev->vendor &&
|
|
|
|
qlcnic_boards[i].device == pdev->device &&
|
|
|
|
qlcnic_boards[i].sub_vendor == pdev->subsystem_vendor &&
|
|
|
|
qlcnic_boards[i].sub_device == pdev->subsystem_device) {
|
2010-05-17 09:22:09 +08:00
|
|
|
sprintf(name, "%pM: %s" ,
|
|
|
|
adapter->mac_addr,
|
|
|
|
qlcnic_boards[i].short_name);
|
2010-01-13 08:37:25 +08:00
|
|
|
found = 1;
|
|
|
|
break;
|
|
|
|
}
|
|
|
|
|
|
|
|
}
|
|
|
|
|
|
|
|
if (!found)
|
2010-06-17 10:56:39 +08:00
|
|
|
sprintf(name, "%pM Gigabit Ethernet", adapter->mac_addr);
|
2010-01-13 08:37:25 +08:00
|
|
|
}
|
|
|
|
|
|
|
|
static void
|
|
|
|
qlcnic_check_options(struct qlcnic_adapter *adapter)
|
|
|
|
{
|
2011-07-29 21:30:28 +08:00
|
|
|
u32 fw_major, fw_minor, fw_build, prev_fw_version;
|
2010-01-13 08:37:25 +08:00
|
|
|
struct pci_dev *pdev = adapter->pdev;
|
2011-07-29 21:30:28 +08:00
|
|
|
struct qlcnic_fw_dump *fw_dump = &adapter->ahw->fw_dump;
|
|
|
|
|
|
|
|
prev_fw_version = adapter->fw_version;
|
2010-01-13 08:37:25 +08:00
|
|
|
|
|
|
|
fw_major = QLCRD32(adapter, QLCNIC_FW_VERSION_MAJOR);
|
|
|
|
fw_minor = QLCRD32(adapter, QLCNIC_FW_VERSION_MINOR);
|
|
|
|
fw_build = QLCRD32(adapter, QLCNIC_FW_VERSION_SUB);
|
|
|
|
|
|
|
|
adapter->fw_version = QLCNIC_VERSION_CODE(fw_major, fw_minor, fw_build);
|
|
|
|
|
2011-07-29 21:30:28 +08:00
|
|
|
if (adapter->op_mode != QLCNIC_NON_PRIV_FUNC) {
|
|
|
|
if (fw_dump->tmpl_hdr == NULL ||
|
|
|
|
adapter->fw_version > prev_fw_version) {
|
|
|
|
if (fw_dump->tmpl_hdr)
|
|
|
|
vfree(fw_dump->tmpl_hdr);
|
|
|
|
if (!qlcnic_fw_cmd_get_minidump_temp(adapter))
|
|
|
|
dev_info(&pdev->dev,
|
|
|
|
"Supports FW dump capability\n");
|
|
|
|
}
|
|
|
|
}
|
|
|
|
|
2010-05-13 11:07:46 +08:00
|
|
|
dev_info(&pdev->dev, "firmware v%d.%d.%d\n",
|
|
|
|
fw_major, fw_minor, fw_build);
|
2011-04-01 22:28:05 +08:00
|
|
|
if (adapter->ahw->port_type == QLCNIC_XGBE) {
|
2010-10-27 01:53:08 +08:00
|
|
|
if (adapter->flags & QLCNIC_ESWITCH_ENABLED) {
|
|
|
|
adapter->num_rxd = DEFAULT_RCV_DESCRIPTORS_VF;
|
|
|
|
adapter->max_rxd = MAX_RCV_DESCRIPTORS_VF;
|
|
|
|
} else {
|
|
|
|
adapter->num_rxd = DEFAULT_RCV_DESCRIPTORS_10G;
|
|
|
|
adapter->max_rxd = MAX_RCV_DESCRIPTORS_10G;
|
|
|
|
}
|
|
|
|
|
2010-01-13 08:37:25 +08:00
|
|
|
adapter->num_jumbo_rxd = MAX_JUMBO_RCV_DESCRIPTORS_10G;
|
2010-10-27 01:53:08 +08:00
|
|
|
adapter->max_jumbo_rxd = MAX_JUMBO_RCV_DESCRIPTORS_10G;
|
|
|
|
|
2011-04-01 22:28:05 +08:00
|
|
|
} else if (adapter->ahw->port_type == QLCNIC_GBE) {
|
2010-01-13 08:37:25 +08:00
|
|
|
adapter->num_rxd = DEFAULT_RCV_DESCRIPTORS_1G;
|
|
|
|
adapter->num_jumbo_rxd = MAX_JUMBO_RCV_DESCRIPTORS_1G;
|
2010-10-27 01:53:08 +08:00
|
|
|
adapter->max_jumbo_rxd = MAX_JUMBO_RCV_DESCRIPTORS_1G;
|
|
|
|
adapter->max_rxd = MAX_RCV_DESCRIPTORS_1G;
|
2010-01-13 08:37:25 +08:00
|
|
|
}
|
|
|
|
|
|
|
|
adapter->msix_supported = !!use_msi_x;
|
|
|
|
|
|
|
|
adapter->num_txd = MAX_CMD_DESCRIPTORS;
|
|
|
|
|
2010-08-19 13:08:24 +08:00
|
|
|
adapter->max_rds_rings = MAX_RDS_RINGS;
|
2010-01-13 08:37:25 +08:00
|
|
|
}
|
|
|
|
|
2010-09-01 01:17:47 +08:00
|
|
|
static int
|
|
|
|
qlcnic_initialize_nic(struct qlcnic_adapter *adapter)
|
|
|
|
{
|
|
|
|
int err;
|
|
|
|
struct qlcnic_info nic_info;
|
|
|
|
|
2011-04-01 22:28:05 +08:00
|
|
|
err = qlcnic_get_nic_info(adapter, &nic_info, adapter->ahw->pci_func);
|
2010-09-01 01:17:47 +08:00
|
|
|
if (err)
|
|
|
|
return err;
|
|
|
|
|
2010-10-04 12:20:14 +08:00
|
|
|
adapter->physical_port = (u8)nic_info.phys_port;
|
2010-09-01 01:17:47 +08:00
|
|
|
adapter->switch_mode = nic_info.switch_mode;
|
|
|
|
adapter->max_tx_ques = nic_info.max_tx_ques;
|
|
|
|
adapter->max_rx_ques = nic_info.max_rx_ques;
|
|
|
|
adapter->capabilities = nic_info.capabilities;
|
|
|
|
adapter->max_mac_filters = nic_info.max_mac_filters;
|
|
|
|
adapter->max_mtu = nic_info.max_mtu;
|
|
|
|
|
|
|
|
if (adapter->capabilities & BIT_6)
|
|
|
|
adapter->flags |= QLCNIC_ESWITCH_ENABLED;
|
|
|
|
else
|
|
|
|
adapter->flags &= ~QLCNIC_ESWITCH_ENABLED;
|
|
|
|
|
|
|
|
return err;
|
|
|
|
}
|
|
|
|
|
2012-11-28 12:34:28 +08:00
|
|
|
void qlcnic_set_vlan_config(struct qlcnic_adapter *adapter,
|
|
|
|
struct qlcnic_esw_func_cfg *esw_cfg)
|
2010-08-25 12:03:03 +08:00
|
|
|
{
|
|
|
|
if (esw_cfg->discard_tagged)
|
|
|
|
adapter->flags &= ~QLCNIC_TAGGING_ENABLED;
|
|
|
|
else
|
|
|
|
adapter->flags |= QLCNIC_TAGGING_ENABLED;
|
|
|
|
|
|
|
|
if (esw_cfg->vlan_id)
|
|
|
|
adapter->pvid = esw_cfg->vlan_id;
|
|
|
|
else
|
|
|
|
adapter->pvid = 0;
|
|
|
|
}
|
|
|
|
|
2011-12-09 08:52:37 +08:00
|
|
|
static int
|
2011-04-01 22:28:15 +08:00
|
|
|
qlcnic_vlan_rx_add(struct net_device *netdev, u16 vid)
|
|
|
|
{
|
|
|
|
struct qlcnic_adapter *adapter = netdev_priv(netdev);
|
|
|
|
set_bit(vid, adapter->vlans);
|
2011-12-09 08:52:37 +08:00
|
|
|
return 0;
|
2011-04-01 22:28:15 +08:00
|
|
|
}
|
|
|
|
|
2011-12-09 08:52:37 +08:00
|
|
|
static int
|
2011-04-01 22:28:15 +08:00
|
|
|
qlcnic_vlan_rx_del(struct net_device *netdev, u16 vid)
|
|
|
|
{
|
|
|
|
struct qlcnic_adapter *adapter = netdev_priv(netdev);
|
|
|
|
|
|
|
|
qlcnic_restore_indev_addr(netdev, NETDEV_DOWN);
|
|
|
|
clear_bit(vid, adapter->vlans);
|
2011-12-09 08:52:37 +08:00
|
|
|
return 0;
|
2011-04-01 22:28:15 +08:00
|
|
|
}
|
|
|
|
|
2012-11-28 12:34:28 +08:00
|
|
|
void qlcnic_set_eswitch_port_features(struct qlcnic_adapter *adapter,
|
|
|
|
struct qlcnic_esw_func_cfg *esw_cfg)
|
2010-08-19 13:08:26 +08:00
|
|
|
{
|
2010-10-08 07:46:09 +08:00
|
|
|
adapter->flags &= ~(QLCNIC_MACSPOOF | QLCNIC_MAC_OVERRIDE_DISABLED |
|
|
|
|
QLCNIC_PROMISC_DISABLED);
|
2010-09-01 01:17:48 +08:00
|
|
|
|
|
|
|
if (esw_cfg->mac_anti_spoof)
|
|
|
|
adapter->flags |= QLCNIC_MACSPOOF;
|
2010-08-19 13:08:27 +08:00
|
|
|
|
2010-09-01 01:17:50 +08:00
|
|
|
if (!esw_cfg->mac_override)
|
|
|
|
adapter->flags |= QLCNIC_MAC_OVERRIDE_DISABLED;
|
|
|
|
|
2010-10-08 07:46:09 +08:00
|
|
|
if (!esw_cfg->promisc_mode)
|
|
|
|
adapter->flags |= QLCNIC_PROMISC_DISABLED;
|
|
|
|
|
2010-08-19 13:08:26 +08:00
|
|
|
qlcnic_set_netdev_features(adapter, esw_cfg);
|
|
|
|
}
|
|
|
|
|
2012-11-28 12:34:28 +08:00
|
|
|
static int qlcnic_set_eswitch_port_config(struct qlcnic_adapter *adapter)
|
2010-08-19 13:08:26 +08:00
|
|
|
{
|
|
|
|
struct qlcnic_esw_func_cfg esw_cfg;
|
|
|
|
|
|
|
|
if (!(adapter->flags & QLCNIC_ESWITCH_ENABLED))
|
|
|
|
return 0;
|
|
|
|
|
2011-04-01 22:28:05 +08:00
|
|
|
esw_cfg.pci_func = adapter->ahw->pci_func;
|
2010-08-19 13:08:26 +08:00
|
|
|
if (qlcnic_get_eswitch_port_config(adapter, &esw_cfg))
|
|
|
|
return -EIO;
|
2010-08-25 12:03:03 +08:00
|
|
|
qlcnic_set_vlan_config(adapter, &esw_cfg);
|
2010-08-19 13:08:26 +08:00
|
|
|
qlcnic_set_eswitch_port_features(adapter, &esw_cfg);
|
|
|
|
|
|
|
|
return 0;
|
|
|
|
}
|
|
|
|
|
|
|
|
static void
|
|
|
|
qlcnic_set_netdev_features(struct qlcnic_adapter *adapter,
|
|
|
|
struct qlcnic_esw_func_cfg *esw_cfg)
|
|
|
|
{
|
|
|
|
struct net_device *netdev = adapter->netdev;
|
2011-11-15 23:29:55 +08:00
|
|
|
netdev_features_t features, vlan_features;
|
2010-08-19 13:08:26 +08:00
|
|
|
|
2011-04-19 11:03:57 +08:00
|
|
|
features = (NETIF_F_SG | NETIF_F_IP_CSUM | NETIF_F_RXCSUM |
|
2010-08-19 13:08:26 +08:00
|
|
|
NETIF_F_IPV6_CSUM | NETIF_F_GRO);
|
|
|
|
vlan_features = (NETIF_F_SG | NETIF_F_IP_CSUM |
|
2011-04-01 22:28:15 +08:00
|
|
|
NETIF_F_IPV6_CSUM | NETIF_F_HW_VLAN_FILTER);
|
2010-08-19 13:08:26 +08:00
|
|
|
|
|
|
|
if (adapter->capabilities & QLCNIC_FW_CAPABILITY_TSO) {
|
|
|
|
features |= (NETIF_F_TSO | NETIF_F_TSO6);
|
|
|
|
vlan_features |= (NETIF_F_TSO | NETIF_F_TSO6);
|
|
|
|
}
|
2011-04-01 22:28:31 +08:00
|
|
|
|
|
|
|
if (netdev->features & NETIF_F_LRO)
|
2010-08-19 13:08:26 +08:00
|
|
|
features |= NETIF_F_LRO;
|
|
|
|
|
|
|
|
if (esw_cfg->offload_flags & BIT_0) {
|
|
|
|
netdev->features |= features;
|
|
|
|
if (!(esw_cfg->offload_flags & BIT_1))
|
|
|
|
netdev->features &= ~NETIF_F_TSO;
|
|
|
|
if (!(esw_cfg->offload_flags & BIT_2))
|
|
|
|
netdev->features &= ~NETIF_F_TSO6;
|
|
|
|
} else {
|
|
|
|
netdev->features &= ~features;
|
|
|
|
}
|
|
|
|
|
|
|
|
netdev->vlan_features = (features & vlan_features);
|
|
|
|
}
|
|
|
|
|
2010-08-26 22:02:52 +08:00
|
|
|
static int
|
|
|
|
qlcnic_check_eswitch_mode(struct qlcnic_adapter *adapter)
|
|
|
|
{
|
|
|
|
void __iomem *priv_op;
|
|
|
|
u32 op_mode, priv_level;
|
|
|
|
int err = 0;
|
|
|
|
|
2010-09-01 01:17:47 +08:00
|
|
|
err = qlcnic_initialize_nic(adapter);
|
|
|
|
if (err)
|
|
|
|
return err;
|
|
|
|
|
2010-08-26 22:02:52 +08:00
|
|
|
if (adapter->flags & QLCNIC_ADAPTER_INITIALIZED)
|
|
|
|
return 0;
|
|
|
|
|
2011-04-01 22:28:05 +08:00
|
|
|
priv_op = adapter->ahw->pci_base0 + QLCNIC_DRV_OP_MODE;
|
2010-08-26 22:02:52 +08:00
|
|
|
op_mode = readl(priv_op);
|
2011-04-01 22:28:05 +08:00
|
|
|
priv_level = QLC_DEV_GET_DRV(op_mode, adapter->ahw->pci_func);
|
2010-08-26 22:02:52 +08:00
|
|
|
|
|
|
|
if (op_mode == QLC_DEV_DRV_DEFAULT)
|
|
|
|
priv_level = QLCNIC_MGMT_FUNC;
|
|
|
|
else
|
2011-04-01 22:28:05 +08:00
|
|
|
priv_level = QLC_DEV_GET_DRV(op_mode, adapter->ahw->pci_func);
|
2010-08-26 22:02:52 +08:00
|
|
|
|
2010-09-01 01:17:47 +08:00
|
|
|
if (adapter->flags & QLCNIC_ESWITCH_ENABLED) {
|
2010-08-26 22:02:52 +08:00
|
|
|
if (priv_level == QLCNIC_MGMT_FUNC) {
|
|
|
|
adapter->op_mode = QLCNIC_MGMT_FUNC;
|
|
|
|
err = qlcnic_init_pci_info(adapter);
|
|
|
|
if (err)
|
|
|
|
return err;
|
|
|
|
/* Set privilege level for other functions */
|
|
|
|
qlcnic_set_function_modes(adapter);
|
|
|
|
dev_info(&adapter->pdev->dev,
|
|
|
|
"HAL Version: %d, Management function\n",
|
|
|
|
adapter->fw_hal_version);
|
|
|
|
} else if (priv_level == QLCNIC_PRIV_FUNC) {
|
|
|
|
adapter->op_mode = QLCNIC_PRIV_FUNC;
|
|
|
|
dev_info(&adapter->pdev->dev,
|
|
|
|
"HAL Version: %d, Privileged function\n",
|
|
|
|
adapter->fw_hal_version);
|
|
|
|
}
|
2010-09-01 01:17:47 +08:00
|
|
|
}
|
2010-08-26 22:02:52 +08:00
|
|
|
|
|
|
|
adapter->flags |= QLCNIC_ADAPTER_INITIALIZED;
|
|
|
|
|
|
|
|
return err;
|
|
|
|
}
|
|
|
|
|
2010-08-19 13:08:26 +08:00
|
|
|
static int
|
|
|
|
qlcnic_set_default_offload_settings(struct qlcnic_adapter *adapter)
|
|
|
|
{
|
|
|
|
struct qlcnic_esw_func_cfg esw_cfg;
|
|
|
|
struct qlcnic_npar_info *npar;
|
|
|
|
u8 i;
|
|
|
|
|
2010-09-01 01:17:47 +08:00
|
|
|
if (adapter->need_fw_reset)
|
2010-08-19 13:08:26 +08:00
|
|
|
return 0;
|
|
|
|
|
|
|
|
for (i = 0; i < QLCNIC_MAX_PCI_FUNC; i++) {
|
|
|
|
if (adapter->npars[i].type != QLCNIC_TYPE_NIC)
|
|
|
|
continue;
|
|
|
|
memset(&esw_cfg, 0, sizeof(struct qlcnic_esw_func_cfg));
|
|
|
|
esw_cfg.pci_func = i;
|
|
|
|
esw_cfg.offload_flags = BIT_0;
|
2010-09-01 01:17:50 +08:00
|
|
|
esw_cfg.mac_override = BIT_0;
|
2010-10-08 07:46:09 +08:00
|
|
|
esw_cfg.promisc_mode = BIT_0;
|
2010-08-19 13:08:26 +08:00
|
|
|
if (adapter->capabilities & QLCNIC_FW_CAPABILITY_TSO)
|
|
|
|
esw_cfg.offload_flags |= (BIT_1 | BIT_2);
|
|
|
|
if (qlcnic_config_switch_port(adapter, &esw_cfg))
|
|
|
|
return -EIO;
|
|
|
|
npar = &adapter->npars[i];
|
|
|
|
npar->pvid = esw_cfg.vlan_id;
|
2010-09-01 01:17:50 +08:00
|
|
|
npar->mac_override = esw_cfg.mac_override;
|
2010-08-19 13:08:26 +08:00
|
|
|
npar->mac_anti_spoof = esw_cfg.mac_anti_spoof;
|
|
|
|
npar->discard_tagged = esw_cfg.discard_tagged;
|
|
|
|
npar->promisc_mode = esw_cfg.promisc_mode;
|
|
|
|
npar->offload_flags = esw_cfg.offload_flags;
|
|
|
|
}
|
|
|
|
|
|
|
|
return 0;
|
|
|
|
}
|
|
|
|
|
2010-08-19 13:08:25 +08:00
|
|
|
static int
|
|
|
|
qlcnic_reset_eswitch_config(struct qlcnic_adapter *adapter,
|
|
|
|
struct qlcnic_npar_info *npar, int pci_func)
|
|
|
|
{
|
|
|
|
struct qlcnic_esw_func_cfg esw_cfg;
|
|
|
|
esw_cfg.op_mode = QLCNIC_PORT_DEFAULTS;
|
|
|
|
esw_cfg.pci_func = pci_func;
|
|
|
|
esw_cfg.vlan_id = npar->pvid;
|
2010-09-01 01:17:50 +08:00
|
|
|
esw_cfg.mac_override = npar->mac_override;
|
2010-08-19 13:08:25 +08:00
|
|
|
esw_cfg.discard_tagged = npar->discard_tagged;
|
|
|
|
esw_cfg.mac_anti_spoof = npar->mac_anti_spoof;
|
|
|
|
esw_cfg.offload_flags = npar->offload_flags;
|
|
|
|
esw_cfg.promisc_mode = npar->promisc_mode;
|
|
|
|
if (qlcnic_config_switch_port(adapter, &esw_cfg))
|
|
|
|
return -EIO;
|
|
|
|
|
|
|
|
esw_cfg.op_mode = QLCNIC_ADD_VLAN;
|
|
|
|
if (qlcnic_config_switch_port(adapter, &esw_cfg))
|
|
|
|
return -EIO;
|
|
|
|
|
|
|
|
return 0;
|
|
|
|
}
|
|
|
|
|
2010-07-14 04:33:35 +08:00
|
|
|
static int
|
|
|
|
qlcnic_reset_npar_config(struct qlcnic_adapter *adapter)
|
|
|
|
{
|
2010-08-19 13:08:25 +08:00
|
|
|
int i, err;
|
2010-07-14 04:33:35 +08:00
|
|
|
struct qlcnic_npar_info *npar;
|
|
|
|
struct qlcnic_info nic_info;
|
|
|
|
|
2010-09-01 01:17:47 +08:00
|
|
|
if (!adapter->need_fw_reset)
|
2010-07-14 04:33:35 +08:00
|
|
|
return 0;
|
|
|
|
|
2010-08-19 13:08:25 +08:00
|
|
|
/* Set the NPAR config data after FW reset */
|
|
|
|
for (i = 0; i < QLCNIC_MAX_PCI_FUNC; i++) {
|
|
|
|
npar = &adapter->npars[i];
|
|
|
|
if (npar->type != QLCNIC_TYPE_NIC)
|
|
|
|
continue;
|
|
|
|
err = qlcnic_get_nic_info(adapter, &nic_info, i);
|
|
|
|
if (err)
|
|
|
|
return err;
|
|
|
|
nic_info.min_tx_bw = npar->min_bw;
|
|
|
|
nic_info.max_tx_bw = npar->max_bw;
|
|
|
|
err = qlcnic_set_nic_info(adapter, &nic_info);
|
|
|
|
if (err)
|
|
|
|
return err;
|
2010-07-14 04:33:35 +08:00
|
|
|
|
2010-08-19 13:08:25 +08:00
|
|
|
if (npar->enable_pm) {
|
|
|
|
err = qlcnic_config_port_mirroring(adapter,
|
|
|
|
npar->dest_npar, 1, i);
|
|
|
|
if (err)
|
|
|
|
return err;
|
2010-07-14 04:33:35 +08:00
|
|
|
}
|
2010-08-19 13:08:25 +08:00
|
|
|
err = qlcnic_reset_eswitch_config(adapter, npar, i);
|
|
|
|
if (err)
|
|
|
|
return err;
|
2010-07-14 04:33:35 +08:00
|
|
|
}
|
2010-08-19 13:08:25 +08:00
|
|
|
return 0;
|
2010-07-14 04:33:35 +08:00
|
|
|
}
|
|
|
|
|
2010-08-19 13:08:28 +08:00
|
|
|
static int qlcnic_check_npar_opertional(struct qlcnic_adapter *adapter)
|
|
|
|
{
|
|
|
|
u8 npar_opt_timeo = QLCNIC_DEV_NPAR_OPER_TIMEO;
|
|
|
|
u32 npar_state;
|
|
|
|
|
|
|
|
if (adapter->op_mode == QLCNIC_MGMT_FUNC)
|
|
|
|
return 0;
|
|
|
|
|
|
|
|
npar_state = QLCRD32(adapter, QLCNIC_CRB_DEV_NPAR_STATE);
|
|
|
|
while (npar_state != QLCNIC_DEV_NPAR_OPER && --npar_opt_timeo) {
|
|
|
|
msleep(1000);
|
|
|
|
npar_state = QLCRD32(adapter, QLCNIC_CRB_DEV_NPAR_STATE);
|
|
|
|
}
|
|
|
|
if (!npar_opt_timeo) {
|
|
|
|
dev_err(&adapter->pdev->dev,
|
|
|
|
"Waiting for NPAR state to opertional timeout\n");
|
|
|
|
return -EIO;
|
|
|
|
}
|
|
|
|
return 0;
|
|
|
|
}
|
|
|
|
|
2010-09-01 01:17:47 +08:00
|
|
|
static int
|
|
|
|
qlcnic_set_mgmt_operations(struct qlcnic_adapter *adapter)
|
|
|
|
{
|
|
|
|
int err;
|
|
|
|
|
|
|
|
if (!(adapter->flags & QLCNIC_ESWITCH_ENABLED) ||
|
|
|
|
adapter->op_mode != QLCNIC_MGMT_FUNC)
|
|
|
|
return 0;
|
|
|
|
|
|
|
|
err = qlcnic_set_default_offload_settings(adapter);
|
|
|
|
if (err)
|
|
|
|
return err;
|
|
|
|
|
|
|
|
err = qlcnic_reset_npar_config(adapter);
|
|
|
|
if (err)
|
|
|
|
return err;
|
|
|
|
|
|
|
|
qlcnic_dev_set_npar_ready(adapter);
|
|
|
|
|
|
|
|
return err;
|
|
|
|
}
|
|
|
|
|
2010-01-13 08:37:25 +08:00
|
|
|
static int
|
|
|
|
qlcnic_start_firmware(struct qlcnic_adapter *adapter)
|
|
|
|
{
|
2010-08-19 13:08:31 +08:00
|
|
|
int err;
|
2010-01-13 08:37:25 +08:00
|
|
|
|
2010-04-02 03:01:32 +08:00
|
|
|
err = qlcnic_can_start_firmware(adapter);
|
|
|
|
if (err < 0)
|
|
|
|
return err;
|
|
|
|
else if (!err)
|
2010-08-19 13:08:31 +08:00
|
|
|
goto check_fw_status;
|
2010-01-13 08:37:25 +08:00
|
|
|
|
2010-05-17 09:22:11 +08:00
|
|
|
if (load_fw_file)
|
|
|
|
qlcnic_request_firmware(adapter);
|
2010-06-17 10:56:40 +08:00
|
|
|
else {
|
2010-08-26 22:02:41 +08:00
|
|
|
err = qlcnic_check_flash_fw_ver(adapter);
|
|
|
|
if (err)
|
2010-06-17 10:56:40 +08:00
|
|
|
goto err_out;
|
|
|
|
|
2010-05-17 09:22:11 +08:00
|
|
|
adapter->fw_type = QLCNIC_FLASH_ROMIMAGE;
|
2010-06-17 10:56:40 +08:00
|
|
|
}
|
2010-01-13 08:37:25 +08:00
|
|
|
|
|
|
|
err = qlcnic_need_fw_reset(adapter);
|
|
|
|
if (err == 0)
|
2010-09-01 01:17:44 +08:00
|
|
|
goto check_fw_status;
|
2010-01-13 08:37:25 +08:00
|
|
|
|
2010-08-19 13:08:31 +08:00
|
|
|
err = qlcnic_pinit_from_rom(adapter);
|
|
|
|
if (err)
|
|
|
|
goto err_out;
|
2010-01-13 08:37:25 +08:00
|
|
|
|
|
|
|
err = qlcnic_load_firmware(adapter);
|
|
|
|
if (err)
|
|
|
|
goto err_out;
|
|
|
|
|
|
|
|
qlcnic_release_firmware(adapter);
|
2010-08-19 13:08:31 +08:00
|
|
|
QLCWR32(adapter, CRB_DRIVER_VERSION, QLCNIC_DRIVER_VERSION);
|
2010-01-13 08:37:25 +08:00
|
|
|
|
2010-08-19 13:08:31 +08:00
|
|
|
check_fw_status:
|
|
|
|
err = qlcnic_check_fw_status(adapter);
|
2010-01-13 08:37:25 +08:00
|
|
|
if (err)
|
|
|
|
goto err_out;
|
|
|
|
|
|
|
|
QLCWR32(adapter, QLCNIC_CRB_DEV_STATE, QLCNIC_DEV_READY);
|
2010-05-13 11:07:50 +08:00
|
|
|
qlcnic_idc_debug_info(adapter, 1);
|
2010-08-25 12:03:04 +08:00
|
|
|
|
2010-08-26 22:02:52 +08:00
|
|
|
err = qlcnic_check_eswitch_mode(adapter);
|
|
|
|
if (err) {
|
|
|
|
dev_err(&adapter->pdev->dev,
|
|
|
|
"Memory allocation failed for eswitch\n");
|
|
|
|
goto err_out;
|
|
|
|
}
|
2010-09-01 01:17:47 +08:00
|
|
|
err = qlcnic_set_mgmt_operations(adapter);
|
|
|
|
if (err)
|
|
|
|
goto err_out;
|
|
|
|
|
|
|
|
qlcnic_check_options(adapter);
|
2010-01-13 08:37:25 +08:00
|
|
|
adapter->need_fw_reset = 0;
|
|
|
|
|
2010-05-17 09:22:14 +08:00
|
|
|
qlcnic_release_firmware(adapter);
|
|
|
|
return 0;
|
2010-01-13 08:37:25 +08:00
|
|
|
|
|
|
|
err_out:
|
2010-05-17 09:22:14 +08:00
|
|
|
QLCWR32(adapter, QLCNIC_CRB_DEV_STATE, QLCNIC_DEV_FAILED);
|
|
|
|
dev_err(&adapter->pdev->dev, "Device state set to failed\n");
|
2010-08-26 22:02:52 +08:00
|
|
|
|
2010-01-13 08:37:25 +08:00
|
|
|
qlcnic_release_firmware(adapter);
|
|
|
|
return err;
|
|
|
|
}
|
|
|
|
|
|
|
|
static int
|
|
|
|
qlcnic_request_irq(struct qlcnic_adapter *adapter)
|
|
|
|
{
|
|
|
|
irq_handler_t handler;
|
|
|
|
struct qlcnic_host_sds_ring *sds_ring;
|
|
|
|
int err, ring;
|
|
|
|
|
|
|
|
unsigned long flags = 0;
|
|
|
|
struct net_device *netdev = adapter->netdev;
|
2011-04-01 22:28:05 +08:00
|
|
|
struct qlcnic_recv_context *recv_ctx = adapter->recv_ctx;
|
2010-01-13 08:37:25 +08:00
|
|
|
|
2010-02-01 13:24:59 +08:00
|
|
|
if (adapter->diag_test == QLCNIC_INTERRUPT_TEST) {
|
|
|
|
handler = qlcnic_tmp_intr;
|
|
|
|
if (!QLCNIC_IS_MSI_FAMILY(adapter))
|
|
|
|
flags |= IRQF_SHARED;
|
|
|
|
|
|
|
|
} else {
|
|
|
|
if (adapter->flags & QLCNIC_MSIX_ENABLED)
|
|
|
|
handler = qlcnic_msix_intr;
|
|
|
|
else if (adapter->flags & QLCNIC_MSI_ENABLED)
|
|
|
|
handler = qlcnic_msi_intr;
|
|
|
|
else {
|
|
|
|
flags |= IRQF_SHARED;
|
|
|
|
handler = qlcnic_intr;
|
|
|
|
}
|
2010-01-13 08:37:25 +08:00
|
|
|
}
|
|
|
|
adapter->irq = netdev->irq;
|
|
|
|
|
|
|
|
for (ring = 0; ring < adapter->max_sds_rings; ring++) {
|
|
|
|
sds_ring = &recv_ctx->sds_rings[ring];
|
|
|
|
sprintf(sds_ring->name, "%s[%d]", netdev->name, ring);
|
|
|
|
err = request_irq(sds_ring->irq, handler,
|
|
|
|
flags, sds_ring->name, sds_ring);
|
|
|
|
if (err)
|
|
|
|
return err;
|
|
|
|
}
|
|
|
|
|
|
|
|
return 0;
|
|
|
|
}
|
|
|
|
|
|
|
|
static void
|
|
|
|
qlcnic_free_irq(struct qlcnic_adapter *adapter)
|
|
|
|
{
|
|
|
|
int ring;
|
|
|
|
struct qlcnic_host_sds_ring *sds_ring;
|
|
|
|
|
2011-04-01 22:28:05 +08:00
|
|
|
struct qlcnic_recv_context *recv_ctx = adapter->recv_ctx;
|
2010-01-13 08:37:25 +08:00
|
|
|
|
|
|
|
for (ring = 0; ring < adapter->max_sds_rings; ring++) {
|
|
|
|
sds_ring = &recv_ctx->sds_rings[ring];
|
|
|
|
free_irq(sds_ring->irq, sds_ring);
|
|
|
|
}
|
|
|
|
}
|
|
|
|
|
|
|
|
static int
|
|
|
|
__qlcnic_up(struct qlcnic_adapter *adapter, struct net_device *netdev)
|
|
|
|
{
|
2010-06-22 11:19:01 +08:00
|
|
|
int ring;
|
2012-06-06 15:35:06 +08:00
|
|
|
u32 capab2;
|
|
|
|
|
2010-06-22 11:19:01 +08:00
|
|
|
struct qlcnic_host_rds_ring *rds_ring;
|
|
|
|
|
2010-01-13 08:37:25 +08:00
|
|
|
if (adapter->is_up != QLCNIC_ADAPTER_UP_MAGIC)
|
|
|
|
return -EIO;
|
|
|
|
|
2010-06-22 11:19:01 +08:00
|
|
|
if (test_bit(__QLCNIC_DEV_UP, &adapter->state))
|
|
|
|
return 0;
|
2010-08-19 13:08:26 +08:00
|
|
|
if (qlcnic_set_eswitch_port_config(adapter))
|
|
|
|
return -EIO;
|
2010-06-22 11:19:01 +08:00
|
|
|
|
2012-06-06 15:35:06 +08:00
|
|
|
if (adapter->capabilities & QLCNIC_FW_CAPABILITY_MORE_CAPS) {
|
|
|
|
capab2 = QLCRD32(adapter, CRB_FW_CAPABILITIES_2);
|
|
|
|
if (capab2 & QLCNIC_FW_CAPABILITY_2_LRO_MAX_TCP_SEG)
|
|
|
|
adapter->flags |= QLCNIC_FW_LRO_MSS_CAP;
|
|
|
|
}
|
|
|
|
|
2010-06-22 11:19:01 +08:00
|
|
|
if (qlcnic_fw_create_ctx(adapter))
|
|
|
|
return -EIO;
|
|
|
|
|
|
|
|
for (ring = 0; ring < adapter->max_rds_rings; ring++) {
|
2011-04-01 22:28:05 +08:00
|
|
|
rds_ring = &adapter->recv_ctx->rds_rings[ring];
|
|
|
|
qlcnic_post_rx_buffers(adapter, rds_ring);
|
2010-06-22 11:19:01 +08:00
|
|
|
}
|
|
|
|
|
2010-01-13 08:37:25 +08:00
|
|
|
qlcnic_set_multi(netdev);
|
|
|
|
qlcnic_fw_cmd_set_mtu(adapter, netdev->mtu);
|
|
|
|
|
2011-04-01 22:28:05 +08:00
|
|
|
adapter->ahw->linkup = 0;
|
2010-01-13 08:37:25 +08:00
|
|
|
|
|
|
|
if (adapter->max_sds_rings > 1)
|
|
|
|
qlcnic_config_rss(adapter, 1);
|
|
|
|
|
|
|
|
qlcnic_config_intr_coalesce(adapter);
|
|
|
|
|
2010-08-17 08:34:25 +08:00
|
|
|
if (netdev->features & NETIF_F_LRO)
|
2010-01-13 08:37:25 +08:00
|
|
|
qlcnic_config_hw_lro(adapter, QLCNIC_LRO_ENABLED);
|
|
|
|
|
|
|
|
qlcnic_napi_enable(adapter);
|
|
|
|
|
|
|
|
qlcnic_linkevent_request(adapter, 1);
|
|
|
|
|
2010-06-22 11:19:03 +08:00
|
|
|
adapter->reset_context = 0;
|
2010-01-13 08:37:25 +08:00
|
|
|
set_bit(__QLCNIC_DEV_UP, &adapter->state);
|
|
|
|
return 0;
|
|
|
|
}
|
|
|
|
|
|
|
|
/* Usage: During resume and firmware recovery module.*/
|
|
|
|
|
|
|
|
static int
|
|
|
|
qlcnic_up(struct qlcnic_adapter *adapter, struct net_device *netdev)
|
|
|
|
{
|
|
|
|
int err = 0;
|
|
|
|
|
|
|
|
rtnl_lock();
|
|
|
|
if (netif_running(netdev))
|
|
|
|
err = __qlcnic_up(adapter, netdev);
|
|
|
|
rtnl_unlock();
|
|
|
|
|
|
|
|
return err;
|
|
|
|
}
|
|
|
|
|
|
|
|
static void
|
|
|
|
__qlcnic_down(struct qlcnic_adapter *adapter, struct net_device *netdev)
|
|
|
|
{
|
|
|
|
if (adapter->is_up != QLCNIC_ADAPTER_UP_MAGIC)
|
|
|
|
return;
|
|
|
|
|
|
|
|
if (!test_and_clear_bit(__QLCNIC_DEV_UP, &adapter->state))
|
|
|
|
return;
|
|
|
|
|
|
|
|
smp_mb();
|
|
|
|
spin_lock(&adapter->tx_clean_lock);
|
|
|
|
netif_carrier_off(netdev);
|
|
|
|
netif_tx_disable(netdev);
|
|
|
|
|
|
|
|
qlcnic_free_mac_list(adapter);
|
|
|
|
|
2010-09-01 01:17:51 +08:00
|
|
|
if (adapter->fhash.fnum)
|
|
|
|
qlcnic_delete_lb_filters(adapter);
|
|
|
|
|
2010-01-13 08:37:25 +08:00
|
|
|
qlcnic_nic_set_promisc(adapter, QLCNIC_NIU_NON_PROMISC_MODE);
|
|
|
|
|
|
|
|
qlcnic_napi_disable(adapter);
|
|
|
|
|
2010-06-22 11:19:01 +08:00
|
|
|
qlcnic_fw_destroy_ctx(adapter);
|
2012-06-06 15:35:06 +08:00
|
|
|
adapter->flags &= ~QLCNIC_FW_LRO_MSS_CAP;
|
2010-06-22 11:19:01 +08:00
|
|
|
|
|
|
|
qlcnic_reset_rx_buffers_list(adapter);
|
2010-01-13 08:37:25 +08:00
|
|
|
qlcnic_release_tx_buffers(adapter);
|
|
|
|
spin_unlock(&adapter->tx_clean_lock);
|
|
|
|
}
|
|
|
|
|
|
|
|
/* Usage: During suspend and firmware recovery module */
|
|
|
|
|
|
|
|
static void
|
|
|
|
qlcnic_down(struct qlcnic_adapter *adapter, struct net_device *netdev)
|
|
|
|
{
|
|
|
|
rtnl_lock();
|
|
|
|
if (netif_running(netdev))
|
|
|
|
__qlcnic_down(adapter, netdev);
|
|
|
|
rtnl_unlock();
|
|
|
|
|
|
|
|
}
|
|
|
|
|
|
|
|
static int
|
|
|
|
qlcnic_attach(struct qlcnic_adapter *adapter)
|
|
|
|
{
|
|
|
|
struct net_device *netdev = adapter->netdev;
|
|
|
|
struct pci_dev *pdev = adapter->pdev;
|
2010-06-22 11:19:01 +08:00
|
|
|
int err;
|
2010-01-13 08:37:25 +08:00
|
|
|
|
|
|
|
if (adapter->is_up == QLCNIC_ADAPTER_UP_MAGIC)
|
|
|
|
return 0;
|
|
|
|
|
|
|
|
err = qlcnic_napi_add(adapter, netdev);
|
|
|
|
if (err)
|
|
|
|
return err;
|
|
|
|
|
|
|
|
err = qlcnic_alloc_sw_resources(adapter);
|
|
|
|
if (err) {
|
|
|
|
dev_err(&pdev->dev, "Error in setting sw resources\n");
|
2010-06-22 11:19:01 +08:00
|
|
|
goto err_out_napi_del;
|
2010-01-13 08:37:25 +08:00
|
|
|
}
|
|
|
|
|
|
|
|
err = qlcnic_alloc_hw_resources(adapter);
|
|
|
|
if (err) {
|
|
|
|
dev_err(&pdev->dev, "Error in setting hw resources\n");
|
|
|
|
goto err_out_free_sw;
|
|
|
|
}
|
|
|
|
|
|
|
|
err = qlcnic_request_irq(adapter);
|
|
|
|
if (err) {
|
|
|
|
dev_err(&pdev->dev, "failed to setup interrupt\n");
|
2010-06-22 11:19:01 +08:00
|
|
|
goto err_out_free_hw;
|
2010-01-13 08:37:25 +08:00
|
|
|
}
|
|
|
|
|
|
|
|
qlcnic_create_sysfs_entries(adapter);
|
|
|
|
|
|
|
|
adapter->is_up = QLCNIC_ADAPTER_UP_MAGIC;
|
|
|
|
return 0;
|
|
|
|
|
2010-06-22 11:19:01 +08:00
|
|
|
err_out_free_hw:
|
2010-01-13 08:37:25 +08:00
|
|
|
qlcnic_free_hw_resources(adapter);
|
|
|
|
err_out_free_sw:
|
|
|
|
qlcnic_free_sw_resources(adapter);
|
2010-06-22 11:19:01 +08:00
|
|
|
err_out_napi_del:
|
|
|
|
qlcnic_napi_del(adapter);
|
2010-01-13 08:37:25 +08:00
|
|
|
return err;
|
|
|
|
}
|
|
|
|
|
|
|
|
static void
|
|
|
|
qlcnic_detach(struct qlcnic_adapter *adapter)
|
|
|
|
{
|
|
|
|
if (adapter->is_up != QLCNIC_ADAPTER_UP_MAGIC)
|
|
|
|
return;
|
|
|
|
|
|
|
|
qlcnic_remove_sysfs_entries(adapter);
|
|
|
|
|
|
|
|
qlcnic_free_hw_resources(adapter);
|
|
|
|
qlcnic_release_rx_buffers(adapter);
|
|
|
|
qlcnic_free_irq(adapter);
|
|
|
|
qlcnic_napi_del(adapter);
|
|
|
|
qlcnic_free_sw_resources(adapter);
|
|
|
|
|
|
|
|
adapter->is_up = 0;
|
|
|
|
}
|
|
|
|
|
2010-02-01 13:24:59 +08:00
|
|
|
void qlcnic_diag_free_res(struct net_device *netdev, int max_sds_rings)
|
|
|
|
{
|
|
|
|
struct qlcnic_adapter *adapter = netdev_priv(netdev);
|
|
|
|
struct qlcnic_host_sds_ring *sds_ring;
|
|
|
|
int ring;
|
|
|
|
|
2010-05-17 09:22:12 +08:00
|
|
|
clear_bit(__QLCNIC_DEV_UP, &adapter->state);
|
2010-02-01 13:25:00 +08:00
|
|
|
if (adapter->diag_test == QLCNIC_INTERRUPT_TEST) {
|
|
|
|
for (ring = 0; ring < adapter->max_sds_rings; ring++) {
|
2011-04-01 22:28:05 +08:00
|
|
|
sds_ring = &adapter->recv_ctx->sds_rings[ring];
|
2010-02-01 13:25:00 +08:00
|
|
|
qlcnic_disable_int(sds_ring);
|
|
|
|
}
|
2010-02-01 13:24:59 +08:00
|
|
|
}
|
|
|
|
|
2010-06-22 11:19:01 +08:00
|
|
|
qlcnic_fw_destroy_ctx(adapter);
|
|
|
|
|
2010-02-01 13:24:59 +08:00
|
|
|
qlcnic_detach(adapter);
|
|
|
|
|
|
|
|
adapter->diag_test = 0;
|
|
|
|
adapter->max_sds_rings = max_sds_rings;
|
|
|
|
|
|
|
|
if (qlcnic_attach(adapter))
|
2010-04-02 03:01:34 +08:00
|
|
|
goto out;
|
2010-02-01 13:24:59 +08:00
|
|
|
|
|
|
|
if (netif_running(netdev))
|
|
|
|
__qlcnic_up(adapter, netdev);
|
2010-04-02 03:01:34 +08:00
|
|
|
out:
|
2010-02-01 13:24:59 +08:00
|
|
|
netif_device_attach(netdev);
|
|
|
|
}
|
|
|
|
|
2011-04-01 22:28:05 +08:00
|
|
|
static int qlcnic_alloc_adapter_resources(struct qlcnic_adapter *adapter)
|
|
|
|
{
|
|
|
|
int err = 0;
|
|
|
|
adapter->ahw = kzalloc(sizeof(struct qlcnic_hardware_context),
|
|
|
|
GFP_KERNEL);
|
|
|
|
if (!adapter->ahw) {
|
|
|
|
dev_err(&adapter->pdev->dev,
|
|
|
|
"Failed to allocate recv ctx resources for adapter\n");
|
|
|
|
err = -ENOMEM;
|
|
|
|
goto err_out;
|
|
|
|
}
|
|
|
|
adapter->recv_ctx = kzalloc(sizeof(struct qlcnic_recv_context),
|
|
|
|
GFP_KERNEL);
|
|
|
|
if (!adapter->recv_ctx) {
|
|
|
|
dev_err(&adapter->pdev->dev,
|
|
|
|
"Failed to allocate recv ctx resources for adapter\n");
|
|
|
|
kfree(adapter->ahw);
|
|
|
|
adapter->ahw = NULL;
|
|
|
|
err = -ENOMEM;
|
2011-04-01 22:28:21 +08:00
|
|
|
goto err_out;
|
2011-04-01 22:28:05 +08:00
|
|
|
}
|
2011-04-01 22:28:21 +08:00
|
|
|
/* Initialize interrupt coalesce parameters */
|
|
|
|
adapter->ahw->coal.flag = QLCNIC_INTR_DEFAULT;
|
|
|
|
adapter->ahw->coal.rx_time_us = QLCNIC_DEFAULT_INTR_COALESCE_RX_TIME_US;
|
|
|
|
adapter->ahw->coal.rx_packets = QLCNIC_DEFAULT_INTR_COALESCE_RX_PACKETS;
|
2011-04-01 22:28:05 +08:00
|
|
|
err_out:
|
|
|
|
return err;
|
|
|
|
}
|
|
|
|
|
|
|
|
static void qlcnic_free_adapter_resources(struct qlcnic_adapter *adapter)
|
|
|
|
{
|
|
|
|
kfree(adapter->recv_ctx);
|
|
|
|
adapter->recv_ctx = NULL;
|
|
|
|
|
2011-05-12 20:48:33 +08:00
|
|
|
if (adapter->ahw->fw_dump.tmpl_hdr) {
|
|
|
|
vfree(adapter->ahw->fw_dump.tmpl_hdr);
|
|
|
|
adapter->ahw->fw_dump.tmpl_hdr = NULL;
|
|
|
|
}
|
2011-04-01 22:28:05 +08:00
|
|
|
kfree(adapter->ahw);
|
|
|
|
adapter->ahw = NULL;
|
|
|
|
}
|
|
|
|
|
2010-02-01 13:24:59 +08:00
|
|
|
int qlcnic_diag_alloc_res(struct net_device *netdev, int test)
|
|
|
|
{
|
|
|
|
struct qlcnic_adapter *adapter = netdev_priv(netdev);
|
|
|
|
struct qlcnic_host_sds_ring *sds_ring;
|
2010-06-22 11:19:01 +08:00
|
|
|
struct qlcnic_host_rds_ring *rds_ring;
|
2010-02-01 13:24:59 +08:00
|
|
|
int ring;
|
|
|
|
int ret;
|
|
|
|
|
|
|
|
netif_device_detach(netdev);
|
|
|
|
|
|
|
|
if (netif_running(netdev))
|
|
|
|
__qlcnic_down(adapter, netdev);
|
|
|
|
|
|
|
|
qlcnic_detach(adapter);
|
|
|
|
|
|
|
|
adapter->max_sds_rings = 1;
|
|
|
|
adapter->diag_test = test;
|
|
|
|
|
|
|
|
ret = qlcnic_attach(adapter);
|
2010-04-02 03:01:34 +08:00
|
|
|
if (ret) {
|
|
|
|
netif_device_attach(netdev);
|
2010-02-01 13:24:59 +08:00
|
|
|
return ret;
|
2010-04-02 03:01:34 +08:00
|
|
|
}
|
2010-02-01 13:24:59 +08:00
|
|
|
|
2010-06-22 11:19:01 +08:00
|
|
|
ret = qlcnic_fw_create_ctx(adapter);
|
|
|
|
if (ret) {
|
|
|
|
qlcnic_detach(adapter);
|
2010-07-25 02:32:18 +08:00
|
|
|
netif_device_attach(netdev);
|
2010-06-22 11:19:01 +08:00
|
|
|
return ret;
|
|
|
|
}
|
|
|
|
|
|
|
|
for (ring = 0; ring < adapter->max_rds_rings; ring++) {
|
2011-04-01 22:28:05 +08:00
|
|
|
rds_ring = &adapter->recv_ctx->rds_rings[ring];
|
|
|
|
qlcnic_post_rx_buffers(adapter, rds_ring);
|
2010-06-22 11:19:01 +08:00
|
|
|
}
|
|
|
|
|
2010-02-01 13:25:00 +08:00
|
|
|
if (adapter->diag_test == QLCNIC_INTERRUPT_TEST) {
|
|
|
|
for (ring = 0; ring < adapter->max_sds_rings; ring++) {
|
2011-04-01 22:28:05 +08:00
|
|
|
sds_ring = &adapter->recv_ctx->sds_rings[ring];
|
2010-02-01 13:25:00 +08:00
|
|
|
qlcnic_enable_int(sds_ring);
|
|
|
|
}
|
2010-02-01 13:24:59 +08:00
|
|
|
}
|
2011-06-22 10:52:23 +08:00
|
|
|
|
|
|
|
if (adapter->diag_test == QLCNIC_LOOPBACK_TEST) {
|
|
|
|
adapter->ahw->loopback_state = 0;
|
|
|
|
qlcnic_linkevent_request(adapter, 1);
|
|
|
|
}
|
|
|
|
|
2010-05-17 09:22:12 +08:00
|
|
|
set_bit(__QLCNIC_DEV_UP, &adapter->state);
|
2010-02-01 13:24:59 +08:00
|
|
|
|
|
|
|
return 0;
|
|
|
|
}
|
|
|
|
|
2010-06-22 11:19:03 +08:00
|
|
|
/* Reset context in hardware only */
|
|
|
|
static int
|
|
|
|
qlcnic_reset_hw_context(struct qlcnic_adapter *adapter)
|
|
|
|
{
|
|
|
|
struct net_device *netdev = adapter->netdev;
|
|
|
|
|
|
|
|
if (test_and_set_bit(__QLCNIC_RESETTING, &adapter->state))
|
|
|
|
return -EBUSY;
|
|
|
|
|
|
|
|
netif_device_detach(netdev);
|
|
|
|
|
|
|
|
qlcnic_down(adapter, netdev);
|
|
|
|
|
|
|
|
qlcnic_up(adapter, netdev);
|
|
|
|
|
|
|
|
netif_device_attach(netdev);
|
|
|
|
|
|
|
|
clear_bit(__QLCNIC_RESETTING, &adapter->state);
|
|
|
|
return 0;
|
|
|
|
}
|
|
|
|
|
2010-01-13 08:37:25 +08:00
|
|
|
int
|
|
|
|
qlcnic_reset_context(struct qlcnic_adapter *adapter)
|
|
|
|
{
|
|
|
|
int err = 0;
|
|
|
|
struct net_device *netdev = adapter->netdev;
|
|
|
|
|
|
|
|
if (test_and_set_bit(__QLCNIC_RESETTING, &adapter->state))
|
|
|
|
return -EBUSY;
|
|
|
|
|
|
|
|
if (adapter->is_up == QLCNIC_ADAPTER_UP_MAGIC) {
|
|
|
|
|
|
|
|
netif_device_detach(netdev);
|
|
|
|
|
|
|
|
if (netif_running(netdev))
|
|
|
|
__qlcnic_down(adapter, netdev);
|
|
|
|
|
|
|
|
qlcnic_detach(adapter);
|
|
|
|
|
|
|
|
if (netif_running(netdev)) {
|
|
|
|
err = qlcnic_attach(adapter);
|
2012-03-23 14:32:34 +08:00
|
|
|
if (!err) {
|
2010-04-02 03:01:34 +08:00
|
|
|
__qlcnic_up(adapter, netdev);
|
2012-03-23 14:32:34 +08:00
|
|
|
qlcnic_restore_indev_addr(netdev, NETDEV_UP);
|
|
|
|
}
|
2010-01-13 08:37:25 +08:00
|
|
|
}
|
|
|
|
|
|
|
|
netif_device_attach(netdev);
|
|
|
|
}
|
|
|
|
|
|
|
|
clear_bit(__QLCNIC_RESETTING, &adapter->state);
|
|
|
|
return err;
|
|
|
|
}
|
|
|
|
|
|
|
|
static int
|
2012-11-18 05:04:38 +08:00
|
|
|
qlcnic_setup_netdev(struct qlcnic_adapter *adapter, struct net_device *netdev,
|
|
|
|
int pci_using_dac)
|
2010-01-13 08:37:25 +08:00
|
|
|
{
|
|
|
|
int err;
|
|
|
|
struct pci_dev *pdev = adapter->pdev;
|
|
|
|
|
|
|
|
adapter->mc_enabled = 0;
|
|
|
|
adapter->max_mc_count = 38;
|
|
|
|
|
|
|
|
netdev->netdev_ops = &qlcnic_netdev_ops;
|
2010-06-17 10:56:41 +08:00
|
|
|
netdev->watchdog_timeo = 5*HZ;
|
2010-01-13 08:37:25 +08:00
|
|
|
|
|
|
|
qlcnic_change_mtu(netdev, netdev->mtu);
|
|
|
|
|
|
|
|
SET_ETHTOOL_OPS(netdev, &qlcnic_ethtool_ops);
|
|
|
|
|
2011-04-19 11:03:57 +08:00
|
|
|
netdev->hw_features = NETIF_F_SG | NETIF_F_IP_CSUM |
|
|
|
|
NETIF_F_IPV6_CSUM | NETIF_F_RXCSUM;
|
2010-07-09 21:14:58 +08:00
|
|
|
|
2011-04-19 11:03:57 +08:00
|
|
|
if (adapter->capabilities & QLCNIC_FW_CAPABILITY_TSO)
|
|
|
|
netdev->hw_features |= NETIF_F_TSO | NETIF_F_TSO6;
|
2012-11-18 05:04:38 +08:00
|
|
|
if (pci_using_dac == 1)
|
2011-04-19 11:03:57 +08:00
|
|
|
netdev->hw_features |= NETIF_F_HIGHDMA;
|
2010-01-13 08:37:25 +08:00
|
|
|
|
2011-04-19 11:03:57 +08:00
|
|
|
netdev->vlan_features = netdev->hw_features;
|
2010-01-13 08:37:25 +08:00
|
|
|
|
|
|
|
if (adapter->capabilities & QLCNIC_FW_CAPABILITY_FVLANTX)
|
2011-04-19 11:03:57 +08:00
|
|
|
netdev->hw_features |= NETIF_F_HW_VLAN_TX;
|
2010-01-13 08:37:25 +08:00
|
|
|
if (adapter->capabilities & QLCNIC_FW_CAPABILITY_HW_LRO)
|
2011-04-19 11:03:57 +08:00
|
|
|
netdev->hw_features |= NETIF_F_LRO;
|
|
|
|
|
|
|
|
netdev->features |= netdev->hw_features |
|
|
|
|
NETIF_F_HW_VLAN_RX | NETIF_F_HW_VLAN_FILTER;
|
|
|
|
|
2010-01-13 08:37:25 +08:00
|
|
|
netdev->irq = adapter->msix_entries[0].vector;
|
|
|
|
|
|
|
|
err = register_netdev(netdev);
|
|
|
|
if (err) {
|
|
|
|
dev_err(&pdev->dev, "failed to register net device\n");
|
|
|
|
return err;
|
|
|
|
}
|
|
|
|
|
|
|
|
return 0;
|
|
|
|
}
|
|
|
|
|
2012-11-18 05:04:38 +08:00
|
|
|
static int qlcnic_set_dma_mask(struct pci_dev *pdev, int *pci_using_dac)
|
2010-05-14 18:07:46 +08:00
|
|
|
{
|
|
|
|
if (!pci_set_dma_mask(pdev, DMA_BIT_MASK(64)) &&
|
|
|
|
!pci_set_consistent_dma_mask(pdev, DMA_BIT_MASK(64)))
|
|
|
|
*pci_using_dac = 1;
|
|
|
|
else if (!pci_set_dma_mask(pdev, DMA_BIT_MASK(32)) &&
|
|
|
|
!pci_set_consistent_dma_mask(pdev, DMA_BIT_MASK(32)))
|
|
|
|
*pci_using_dac = 0;
|
|
|
|
else {
|
|
|
|
dev_err(&pdev->dev, "Unable to set DMA mask, aborting\n");
|
|
|
|
return -EIO;
|
|
|
|
}
|
|
|
|
|
|
|
|
return 0;
|
|
|
|
}
|
|
|
|
|
2011-04-28 19:48:18 +08:00
|
|
|
static int
|
|
|
|
qlcnic_alloc_msix_entries(struct qlcnic_adapter *adapter, u16 count)
|
|
|
|
{
|
|
|
|
adapter->msix_entries = kcalloc(count, sizeof(struct msix_entry),
|
|
|
|
GFP_KERNEL);
|
|
|
|
|
|
|
|
if (adapter->msix_entries)
|
|
|
|
return 0;
|
|
|
|
|
|
|
|
dev_err(&adapter->pdev->dev, "failed allocating msix_entries\n");
|
|
|
|
return -ENOMEM;
|
|
|
|
}
|
|
|
|
|
2010-01-13 08:37:25 +08:00
|
|
|
static int __devinit
|
|
|
|
qlcnic_probe(struct pci_dev *pdev, const struct pci_device_id *ent)
|
|
|
|
{
|
|
|
|
struct net_device *netdev = NULL;
|
|
|
|
struct qlcnic_adapter *adapter = NULL;
|
2012-11-18 05:04:38 +08:00
|
|
|
int err, pci_using_dac = -1;
|
2010-01-13 08:37:25 +08:00
|
|
|
uint8_t revision_id;
|
2010-09-01 01:17:46 +08:00
|
|
|
char brd_name[QLCNIC_MAX_BOARD_NAME_LEN];
|
2010-01-13 08:37:25 +08:00
|
|
|
|
|
|
|
err = pci_enable_device(pdev);
|
|
|
|
if (err)
|
|
|
|
return err;
|
|
|
|
|
|
|
|
if (!(pci_resource_flags(pdev, 0) & IORESOURCE_MEM)) {
|
|
|
|
err = -ENODEV;
|
|
|
|
goto err_out_disable_pdev;
|
|
|
|
}
|
|
|
|
|
2010-05-14 18:07:46 +08:00
|
|
|
err = qlcnic_set_dma_mask(pdev, &pci_using_dac);
|
|
|
|
if (err)
|
|
|
|
goto err_out_disable_pdev;
|
|
|
|
|
2010-01-13 08:37:25 +08:00
|
|
|
err = pci_request_regions(pdev, qlcnic_driver_name);
|
|
|
|
if (err)
|
|
|
|
goto err_out_disable_pdev;
|
|
|
|
|
|
|
|
pci_set_master(pdev);
|
2010-07-14 04:33:34 +08:00
|
|
|
pci_enable_pcie_error_reporting(pdev);
|
2010-01-13 08:37:25 +08:00
|
|
|
|
|
|
|
netdev = alloc_etherdev(sizeof(struct qlcnic_adapter));
|
|
|
|
if (!netdev) {
|
|
|
|
err = -ENOMEM;
|
|
|
|
goto err_out_free_res;
|
|
|
|
}
|
|
|
|
|
|
|
|
SET_NETDEV_DEV(netdev, &pdev->dev);
|
|
|
|
|
|
|
|
adapter = netdev_priv(netdev);
|
|
|
|
adapter->netdev = netdev;
|
|
|
|
adapter->pdev = pdev;
|
|
|
|
|
2012-10-05 20:10:50 +08:00
|
|
|
err = qlcnic_alloc_adapter_resources(adapter);
|
|
|
|
if (err)
|
2011-04-01 22:28:05 +08:00
|
|
|
goto err_out_free_netdev;
|
|
|
|
|
|
|
|
adapter->dev_rst_time = jiffies;
|
2010-01-13 08:37:25 +08:00
|
|
|
revision_id = pdev->revision;
|
2011-04-01 22:28:05 +08:00
|
|
|
adapter->ahw->revision_id = revision_id;
|
2011-07-14 11:16:52 +08:00
|
|
|
adapter->mac_learn = qlcnic_mac_learn;
|
2010-01-13 08:37:25 +08:00
|
|
|
|
2011-04-01 22:28:05 +08:00
|
|
|
rwlock_init(&adapter->ahw->crb_lock);
|
|
|
|
mutex_init(&adapter->ahw->mem_lock);
|
2010-01-13 08:37:25 +08:00
|
|
|
|
|
|
|
spin_lock_init(&adapter->tx_clean_lock);
|
|
|
|
INIT_LIST_HEAD(&adapter->mac_list);
|
|
|
|
|
|
|
|
err = qlcnic_setup_pci_map(adapter);
|
|
|
|
if (err)
|
2011-04-01 22:28:05 +08:00
|
|
|
goto err_out_free_hw;
|
2010-01-13 08:37:25 +08:00
|
|
|
|
|
|
|
/* This will be reset for mezz cards */
|
2011-04-01 22:28:05 +08:00
|
|
|
adapter->portnum = adapter->ahw->pci_func;
|
2010-01-13 08:37:25 +08:00
|
|
|
|
|
|
|
err = qlcnic_get_board_info(adapter);
|
|
|
|
if (err) {
|
|
|
|
dev_err(&pdev->dev, "Error getting board config info.\n");
|
|
|
|
goto err_out_iounmap;
|
|
|
|
}
|
|
|
|
|
2010-08-26 22:02:41 +08:00
|
|
|
err = qlcnic_setup_idc_param(adapter);
|
|
|
|
if (err)
|
2010-05-13 11:07:48 +08:00
|
|
|
goto err_out_iounmap;
|
2010-01-13 08:37:25 +08:00
|
|
|
|
2010-12-17 06:59:02 +08:00
|
|
|
adapter->flags |= QLCNIC_NEED_FLR;
|
2010-11-23 09:25:21 +08:00
|
|
|
|
2010-06-01 19:33:09 +08:00
|
|
|
err = adapter->nic_ops->start_firmware(adapter);
|
2010-05-17 09:22:14 +08:00
|
|
|
if (err) {
|
2012-04-26 18:31:29 +08:00
|
|
|
dev_err(&pdev->dev, "Loading fw failed. Please Reboot\n"
|
|
|
|
"\t\tIf reboot doesn't help, try flashing the card\n");
|
|
|
|
goto err_out_maintenance_mode;
|
2010-05-17 09:22:14 +08:00
|
|
|
}
|
2010-01-13 08:37:25 +08:00
|
|
|
|
2010-09-01 01:17:46 +08:00
|
|
|
if (qlcnic_read_mac_addr(adapter))
|
|
|
|
dev_warn(&pdev->dev, "failed to read mac addr\n");
|
|
|
|
|
|
|
|
if (adapter->portnum == 0) {
|
|
|
|
get_brd_name(adapter, brd_name);
|
|
|
|
|
|
|
|
pr_info("%s: %s Board Chip rev 0x%x\n",
|
|
|
|
module_name(THIS_MODULE),
|
2011-04-01 22:28:05 +08:00
|
|
|
brd_name, adapter->ahw->revision_id);
|
2010-09-01 01:17:46 +08:00
|
|
|
}
|
|
|
|
|
2010-01-13 08:37:25 +08:00
|
|
|
qlcnic_clear_stats(adapter);
|
|
|
|
|
2011-04-28 19:48:18 +08:00
|
|
|
err = qlcnic_alloc_msix_entries(adapter, adapter->max_rx_ques);
|
|
|
|
if (err)
|
|
|
|
goto err_out_decr_ref;
|
|
|
|
|
2010-01-13 08:37:25 +08:00
|
|
|
qlcnic_setup_intr(adapter);
|
|
|
|
|
2010-05-14 18:07:46 +08:00
|
|
|
err = qlcnic_setup_netdev(adapter, netdev, pci_using_dac);
|
2010-01-13 08:37:25 +08:00
|
|
|
if (err)
|
|
|
|
goto err_out_disable_msi;
|
|
|
|
|
|
|
|
pci_set_drvdata(pdev, adapter);
|
|
|
|
|
|
|
|
qlcnic_schedule_work(adapter, qlcnic_fw_poll_work, FW_POLL_DELAY);
|
|
|
|
|
2011-04-01 22:28:05 +08:00
|
|
|
switch (adapter->ahw->port_type) {
|
2010-01-13 08:37:25 +08:00
|
|
|
case QLCNIC_GBE:
|
|
|
|
dev_info(&adapter->pdev->dev, "%s: GbE port initialized\n",
|
|
|
|
adapter->netdev->name);
|
|
|
|
break;
|
|
|
|
case QLCNIC_XGBE:
|
|
|
|
dev_info(&adapter->pdev->dev, "%s: XGbE port initialized\n",
|
|
|
|
adapter->netdev->name);
|
|
|
|
break;
|
|
|
|
}
|
|
|
|
|
2011-07-14 11:16:52 +08:00
|
|
|
if (adapter->mac_learn)
|
|
|
|
qlcnic_alloc_lb_filters_mem(adapter);
|
|
|
|
|
2010-01-13 08:37:25 +08:00
|
|
|
qlcnic_create_diag_entries(adapter);
|
|
|
|
|
|
|
|
return 0;
|
|
|
|
|
|
|
|
err_out_disable_msi:
|
|
|
|
qlcnic_teardown_intr(adapter);
|
2011-04-28 19:48:18 +08:00
|
|
|
kfree(adapter->msix_entries);
|
2010-01-13 08:37:25 +08:00
|
|
|
|
|
|
|
err_out_decr_ref:
|
2010-08-19 13:08:29 +08:00
|
|
|
qlcnic_clr_all_drv_state(adapter, 0);
|
2010-01-13 08:37:25 +08:00
|
|
|
|
|
|
|
err_out_iounmap:
|
|
|
|
qlcnic_cleanup_pci_map(adapter);
|
|
|
|
|
2011-04-01 22:28:05 +08:00
|
|
|
err_out_free_hw:
|
|
|
|
qlcnic_free_adapter_resources(adapter);
|
|
|
|
|
2010-01-13 08:37:25 +08:00
|
|
|
err_out_free_netdev:
|
|
|
|
free_netdev(netdev);
|
|
|
|
|
|
|
|
err_out_free_res:
|
|
|
|
pci_release_regions(pdev);
|
|
|
|
|
|
|
|
err_out_disable_pdev:
|
|
|
|
pci_set_drvdata(pdev, NULL);
|
|
|
|
pci_disable_device(pdev);
|
|
|
|
return err;
|
2012-04-26 18:31:29 +08:00
|
|
|
|
|
|
|
err_out_maintenance_mode:
|
|
|
|
netdev->netdev_ops = &qlcnic_netdev_failed_ops;
|
|
|
|
SET_ETHTOOL_OPS(netdev, &qlcnic_ethtool_failed_ops);
|
|
|
|
err = register_netdev(netdev);
|
|
|
|
if (err) {
|
|
|
|
dev_err(&pdev->dev, "failed to register net device\n");
|
|
|
|
goto err_out_decr_ref;
|
|
|
|
}
|
|
|
|
pci_set_drvdata(pdev, adapter);
|
|
|
|
qlcnic_create_diag_entries(adapter);
|
|
|
|
return 0;
|
2010-01-13 08:37:25 +08:00
|
|
|
}
|
|
|
|
|
|
|
|
static void __devexit qlcnic_remove(struct pci_dev *pdev)
|
|
|
|
{
|
|
|
|
struct qlcnic_adapter *adapter;
|
|
|
|
struct net_device *netdev;
|
|
|
|
|
|
|
|
adapter = pci_get_drvdata(pdev);
|
|
|
|
if (adapter == NULL)
|
|
|
|
return;
|
|
|
|
|
|
|
|
netdev = adapter->netdev;
|
|
|
|
|
|
|
|
qlcnic_cancel_fw_work(adapter);
|
|
|
|
|
|
|
|
unregister_netdev(netdev);
|
|
|
|
|
|
|
|
qlcnic_detach(adapter);
|
|
|
|
|
2010-06-01 19:28:51 +08:00
|
|
|
if (adapter->npars != NULL)
|
|
|
|
kfree(adapter->npars);
|
|
|
|
if (adapter->eswitch != NULL)
|
|
|
|
kfree(adapter->eswitch);
|
|
|
|
|
2010-08-19 13:08:29 +08:00
|
|
|
qlcnic_clr_all_drv_state(adapter, 0);
|
2010-01-13 08:37:25 +08:00
|
|
|
|
|
|
|
clear_bit(__QLCNIC_RESETTING, &adapter->state);
|
|
|
|
|
2010-09-01 01:17:51 +08:00
|
|
|
qlcnic_free_lb_filters_mem(adapter);
|
|
|
|
|
2010-01-13 08:37:25 +08:00
|
|
|
qlcnic_teardown_intr(adapter);
|
2011-04-28 19:48:18 +08:00
|
|
|
kfree(adapter->msix_entries);
|
2010-01-13 08:37:25 +08:00
|
|
|
|
|
|
|
qlcnic_remove_diag_entries(adapter);
|
|
|
|
|
|
|
|
qlcnic_cleanup_pci_map(adapter);
|
|
|
|
|
|
|
|
qlcnic_release_firmware(adapter);
|
|
|
|
|
2010-07-14 04:33:34 +08:00
|
|
|
pci_disable_pcie_error_reporting(pdev);
|
2010-01-13 08:37:25 +08:00
|
|
|
pci_release_regions(pdev);
|
|
|
|
pci_disable_device(pdev);
|
|
|
|
pci_set_drvdata(pdev, NULL);
|
|
|
|
|
2011-04-01 22:28:05 +08:00
|
|
|
qlcnic_free_adapter_resources(adapter);
|
2010-01-13 08:37:25 +08:00
|
|
|
free_netdev(netdev);
|
|
|
|
}
|
|
|
|
static int __qlcnic_shutdown(struct pci_dev *pdev)
|
|
|
|
{
|
|
|
|
struct qlcnic_adapter *adapter = pci_get_drvdata(pdev);
|
|
|
|
struct net_device *netdev = adapter->netdev;
|
|
|
|
int retval;
|
|
|
|
|
|
|
|
netif_device_detach(netdev);
|
|
|
|
|
|
|
|
qlcnic_cancel_fw_work(adapter);
|
|
|
|
|
|
|
|
if (netif_running(netdev))
|
|
|
|
qlcnic_down(adapter, netdev);
|
|
|
|
|
2010-08-19 13:08:29 +08:00
|
|
|
qlcnic_clr_all_drv_state(adapter, 0);
|
2010-01-13 08:37:25 +08:00
|
|
|
|
|
|
|
clear_bit(__QLCNIC_RESETTING, &adapter->state);
|
|
|
|
|
|
|
|
retval = pci_save_state(pdev);
|
|
|
|
if (retval)
|
|
|
|
return retval;
|
|
|
|
|
|
|
|
if (qlcnic_wol_supported(adapter)) {
|
|
|
|
pci_enable_wake(pdev, PCI_D3cold, 1);
|
|
|
|
pci_enable_wake(pdev, PCI_D3hot, 1);
|
|
|
|
}
|
|
|
|
|
|
|
|
return 0;
|
|
|
|
}
|
|
|
|
|
|
|
|
static void qlcnic_shutdown(struct pci_dev *pdev)
|
|
|
|
{
|
|
|
|
if (__qlcnic_shutdown(pdev))
|
|
|
|
return;
|
|
|
|
|
|
|
|
pci_disable_device(pdev);
|
|
|
|
}
|
|
|
|
|
|
|
|
#ifdef CONFIG_PM
|
|
|
|
static int
|
|
|
|
qlcnic_suspend(struct pci_dev *pdev, pm_message_t state)
|
|
|
|
{
|
|
|
|
int retval;
|
|
|
|
|
|
|
|
retval = __qlcnic_shutdown(pdev);
|
|
|
|
if (retval)
|
|
|
|
return retval;
|
|
|
|
|
|
|
|
pci_set_power_state(pdev, pci_choose_state(pdev, state));
|
|
|
|
return 0;
|
|
|
|
}
|
|
|
|
|
|
|
|
static int
|
|
|
|
qlcnic_resume(struct pci_dev *pdev)
|
|
|
|
{
|
|
|
|
struct qlcnic_adapter *adapter = pci_get_drvdata(pdev);
|
|
|
|
struct net_device *netdev = adapter->netdev;
|
|
|
|
int err;
|
|
|
|
|
|
|
|
err = pci_enable_device(pdev);
|
|
|
|
if (err)
|
|
|
|
return err;
|
|
|
|
|
|
|
|
pci_set_power_state(pdev, PCI_D0);
|
|
|
|
pci_set_master(pdev);
|
|
|
|
pci_restore_state(pdev);
|
|
|
|
|
2010-06-01 19:33:09 +08:00
|
|
|
err = adapter->nic_ops->start_firmware(adapter);
|
2010-01-13 08:37:25 +08:00
|
|
|
if (err) {
|
|
|
|
dev_err(&pdev->dev, "failed to start firmware\n");
|
|
|
|
return err;
|
|
|
|
}
|
|
|
|
|
|
|
|
if (netif_running(netdev)) {
|
|
|
|
err = qlcnic_up(adapter, netdev);
|
|
|
|
if (err)
|
2010-06-22 11:19:02 +08:00
|
|
|
goto done;
|
2010-01-13 08:37:25 +08:00
|
|
|
|
2010-09-17 03:14:41 +08:00
|
|
|
qlcnic_restore_indev_addr(netdev, NETDEV_UP);
|
2010-01-13 08:37:25 +08:00
|
|
|
}
|
2010-06-22 11:19:02 +08:00
|
|
|
done:
|
2010-01-13 08:37:25 +08:00
|
|
|
netif_device_attach(netdev);
|
|
|
|
qlcnic_schedule_work(adapter, qlcnic_fw_poll_work, FW_POLL_DELAY);
|
|
|
|
return 0;
|
|
|
|
}
|
|
|
|
#endif
|
|
|
|
|
|
|
|
static int qlcnic_open(struct net_device *netdev)
|
|
|
|
{
|
|
|
|
struct qlcnic_adapter *adapter = netdev_priv(netdev);
|
2012-04-26 18:31:29 +08:00
|
|
|
u32 state = QLCRD32(adapter, QLCNIC_CRB_DEV_STATE);
|
2010-01-13 08:37:25 +08:00
|
|
|
int err;
|
|
|
|
|
2012-04-26 18:31:29 +08:00
|
|
|
if (state == QLCNIC_DEV_FAILED || (state == QLCNIC_DEV_BADBAD)) {
|
|
|
|
netdev_err(netdev, "Device in FAILED state\n");
|
|
|
|
return -EIO;
|
|
|
|
}
|
|
|
|
|
2011-06-22 10:52:21 +08:00
|
|
|
netif_carrier_off(netdev);
|
|
|
|
|
2010-01-13 08:37:25 +08:00
|
|
|
err = qlcnic_attach(adapter);
|
|
|
|
if (err)
|
|
|
|
return err;
|
|
|
|
|
|
|
|
err = __qlcnic_up(adapter, netdev);
|
|
|
|
if (err)
|
|
|
|
goto err_out;
|
|
|
|
|
|
|
|
netif_start_queue(netdev);
|
|
|
|
|
|
|
|
return 0;
|
|
|
|
|
|
|
|
err_out:
|
|
|
|
qlcnic_detach(adapter);
|
|
|
|
return err;
|
|
|
|
}
|
|
|
|
|
|
|
|
/*
|
|
|
|
* qlcnic_close - Disables a network interface entry point
|
|
|
|
*/
|
|
|
|
static int qlcnic_close(struct net_device *netdev)
|
|
|
|
{
|
|
|
|
struct qlcnic_adapter *adapter = netdev_priv(netdev);
|
|
|
|
|
|
|
|
__qlcnic_down(adapter, netdev);
|
|
|
|
return 0;
|
|
|
|
}
|
|
|
|
|
2011-07-14 11:16:52 +08:00
|
|
|
void qlcnic_alloc_lb_filters_mem(struct qlcnic_adapter *adapter)
|
2010-09-01 01:17:51 +08:00
|
|
|
{
|
|
|
|
void *head;
|
|
|
|
int i;
|
|
|
|
|
2011-07-14 11:16:52 +08:00
|
|
|
if (adapter->fhash.fmax && adapter->fhash.fhead)
|
2010-09-01 01:17:51 +08:00
|
|
|
return;
|
|
|
|
|
|
|
|
spin_lock_init(&adapter->mac_learn_lock);
|
|
|
|
|
|
|
|
head = kcalloc(QLCNIC_LB_MAX_FILTERS, sizeof(struct hlist_head),
|
|
|
|
GFP_KERNEL);
|
|
|
|
if (!head)
|
|
|
|
return;
|
|
|
|
|
|
|
|
adapter->fhash.fmax = QLCNIC_LB_MAX_FILTERS;
|
2011-06-17 03:08:06 +08:00
|
|
|
adapter->fhash.fhead = head;
|
2010-09-01 01:17:51 +08:00
|
|
|
|
|
|
|
for (i = 0; i < adapter->fhash.fmax; i++)
|
|
|
|
INIT_HLIST_HEAD(&adapter->fhash.fhead[i]);
|
|
|
|
}
|
|
|
|
|
|
|
|
static void qlcnic_free_lb_filters_mem(struct qlcnic_adapter *adapter)
|
|
|
|
{
|
|
|
|
if (adapter->fhash.fmax && adapter->fhash.fhead)
|
|
|
|
kfree(adapter->fhash.fhead);
|
|
|
|
|
|
|
|
adapter->fhash.fhead = NULL;
|
|
|
|
adapter->fhash.fmax = 0;
|
|
|
|
}
|
|
|
|
|
2010-01-13 08:37:25 +08:00
|
|
|
static int qlcnic_check_temp(struct qlcnic_adapter *adapter)
|
|
|
|
{
|
|
|
|
struct net_device *netdev = adapter->netdev;
|
|
|
|
u32 temp, temp_state, temp_val;
|
|
|
|
int rv = 0;
|
|
|
|
|
|
|
|
temp = QLCRD32(adapter, CRB_TEMP_STATE);
|
|
|
|
|
|
|
|
temp_state = qlcnic_get_temp_state(temp);
|
|
|
|
temp_val = qlcnic_get_temp_val(temp);
|
|
|
|
|
|
|
|
if (temp_state == QLCNIC_TEMP_PANIC) {
|
|
|
|
dev_err(&netdev->dev,
|
|
|
|
"Device temperature %d degrees C exceeds"
|
|
|
|
" maximum allowed. Hardware has been shut down.\n",
|
|
|
|
temp_val);
|
|
|
|
rv = 1;
|
|
|
|
} else if (temp_state == QLCNIC_TEMP_WARN) {
|
|
|
|
if (adapter->temp == QLCNIC_TEMP_NORMAL) {
|
|
|
|
dev_err(&netdev->dev,
|
|
|
|
"Device temperature %d degrees C "
|
|
|
|
"exceeds operating range."
|
|
|
|
" Immediate action needed.\n",
|
|
|
|
temp_val);
|
|
|
|
}
|
|
|
|
} else {
|
|
|
|
if (adapter->temp == QLCNIC_TEMP_WARN) {
|
|
|
|
dev_info(&netdev->dev,
|
|
|
|
"Device temperature is now %d degrees C"
|
|
|
|
" in normal range.\n", temp_val);
|
|
|
|
}
|
|
|
|
}
|
|
|
|
adapter->temp = temp_state;
|
|
|
|
return rv;
|
|
|
|
}
|
|
|
|
|
|
|
|
static void qlcnic_tx_timeout(struct net_device *netdev)
|
|
|
|
{
|
|
|
|
struct qlcnic_adapter *adapter = netdev_priv(netdev);
|
|
|
|
|
|
|
|
if (test_bit(__QLCNIC_RESETTING, &adapter->state))
|
|
|
|
return;
|
|
|
|
|
|
|
|
dev_err(&netdev->dev, "transmit timeout, resetting.\n");
|
|
|
|
|
|
|
|
if (++adapter->tx_timeo_cnt >= QLCNIC_MAX_TX_TIMEOUTS)
|
2010-06-22 11:19:03 +08:00
|
|
|
adapter->need_fw_reset = 1;
|
|
|
|
else
|
|
|
|
adapter->reset_context = 1;
|
2010-01-13 08:37:25 +08:00
|
|
|
}
|
|
|
|
|
|
|
|
static struct net_device_stats *qlcnic_get_stats(struct net_device *netdev)
|
|
|
|
{
|
|
|
|
struct qlcnic_adapter *adapter = netdev_priv(netdev);
|
|
|
|
struct net_device_stats *stats = &netdev->stats;
|
|
|
|
|
|
|
|
stats->rx_packets = adapter->stats.rx_pkts + adapter->stats.lro_pkts;
|
|
|
|
stats->tx_packets = adapter->stats.xmitfinished;
|
2010-05-17 09:22:10 +08:00
|
|
|
stats->rx_bytes = adapter->stats.rxbytes + adapter->stats.lrobytes;
|
2010-01-13 08:37:25 +08:00
|
|
|
stats->tx_bytes = adapter->stats.txbytes;
|
|
|
|
stats->rx_dropped = adapter->stats.rxdropped;
|
|
|
|
stats->tx_dropped = adapter->stats.txdropped;
|
|
|
|
|
|
|
|
return stats;
|
|
|
|
}
|
|
|
|
|
2010-02-01 13:24:59 +08:00
|
|
|
static irqreturn_t qlcnic_clear_legacy_intr(struct qlcnic_adapter *adapter)
|
2010-01-13 08:37:25 +08:00
|
|
|
{
|
|
|
|
u32 status;
|
|
|
|
|
|
|
|
status = readl(adapter->isr_int_vec);
|
|
|
|
|
|
|
|
if (!(status & adapter->int_vec_bit))
|
|
|
|
return IRQ_NONE;
|
|
|
|
|
|
|
|
/* check interrupt state machine, to be sure */
|
|
|
|
status = readl(adapter->crb_int_state_reg);
|
|
|
|
if (!ISR_LEGACY_INT_TRIGGERED(status))
|
|
|
|
return IRQ_NONE;
|
|
|
|
|
|
|
|
writel(0xffffffff, adapter->tgt_status_reg);
|
|
|
|
/* read twice to ensure write is flushed */
|
|
|
|
readl(adapter->isr_int_vec);
|
|
|
|
readl(adapter->isr_int_vec);
|
|
|
|
|
2010-02-01 13:24:59 +08:00
|
|
|
return IRQ_HANDLED;
|
|
|
|
}
|
|
|
|
|
|
|
|
static irqreturn_t qlcnic_tmp_intr(int irq, void *data)
|
|
|
|
{
|
|
|
|
struct qlcnic_host_sds_ring *sds_ring = data;
|
|
|
|
struct qlcnic_adapter *adapter = sds_ring->adapter;
|
|
|
|
|
|
|
|
if (adapter->flags & QLCNIC_MSIX_ENABLED)
|
|
|
|
goto done;
|
|
|
|
else if (adapter->flags & QLCNIC_MSI_ENABLED) {
|
|
|
|
writel(0xffffffff, adapter->tgt_status_reg);
|
|
|
|
goto done;
|
|
|
|
}
|
|
|
|
|
|
|
|
if (qlcnic_clear_legacy_intr(adapter) == IRQ_NONE)
|
|
|
|
return IRQ_NONE;
|
|
|
|
|
|
|
|
done:
|
|
|
|
adapter->diag_cnt++;
|
|
|
|
qlcnic_enable_int(sds_ring);
|
|
|
|
return IRQ_HANDLED;
|
|
|
|
}
|
|
|
|
|
|
|
|
static irqreturn_t qlcnic_intr(int irq, void *data)
|
|
|
|
{
|
|
|
|
struct qlcnic_host_sds_ring *sds_ring = data;
|
|
|
|
struct qlcnic_adapter *adapter = sds_ring->adapter;
|
|
|
|
|
|
|
|
if (qlcnic_clear_legacy_intr(adapter) == IRQ_NONE)
|
|
|
|
return IRQ_NONE;
|
|
|
|
|
2010-01-13 08:37:25 +08:00
|
|
|
napi_schedule(&sds_ring->napi);
|
|
|
|
|
|
|
|
return IRQ_HANDLED;
|
|
|
|
}
|
|
|
|
|
|
|
|
static irqreturn_t qlcnic_msi_intr(int irq, void *data)
|
|
|
|
{
|
|
|
|
struct qlcnic_host_sds_ring *sds_ring = data;
|
|
|
|
struct qlcnic_adapter *adapter = sds_ring->adapter;
|
|
|
|
|
|
|
|
/* clear interrupt */
|
|
|
|
writel(0xffffffff, adapter->tgt_status_reg);
|
|
|
|
|
|
|
|
napi_schedule(&sds_ring->napi);
|
|
|
|
return IRQ_HANDLED;
|
|
|
|
}
|
|
|
|
|
|
|
|
static irqreturn_t qlcnic_msix_intr(int irq, void *data)
|
|
|
|
{
|
|
|
|
struct qlcnic_host_sds_ring *sds_ring = data;
|
|
|
|
|
|
|
|
napi_schedule(&sds_ring->napi);
|
|
|
|
return IRQ_HANDLED;
|
|
|
|
}
|
|
|
|
|
|
|
|
#ifdef CONFIG_NET_POLL_CONTROLLER
|
|
|
|
static void qlcnic_poll_controller(struct net_device *netdev)
|
|
|
|
{
|
2010-08-23 05:57:56 +08:00
|
|
|
int ring;
|
|
|
|
struct qlcnic_host_sds_ring *sds_ring;
|
2010-01-13 08:37:25 +08:00
|
|
|
struct qlcnic_adapter *adapter = netdev_priv(netdev);
|
2011-04-01 22:28:05 +08:00
|
|
|
struct qlcnic_recv_context *recv_ctx = adapter->recv_ctx;
|
2010-08-23 05:57:56 +08:00
|
|
|
|
2010-01-13 08:37:25 +08:00
|
|
|
disable_irq(adapter->irq);
|
2010-08-23 05:57:56 +08:00
|
|
|
for (ring = 0; ring < adapter->max_sds_rings; ring++) {
|
|
|
|
sds_ring = &recv_ctx->sds_rings[ring];
|
|
|
|
qlcnic_intr(adapter->irq, sds_ring);
|
|
|
|
}
|
2010-01-13 08:37:25 +08:00
|
|
|
enable_irq(adapter->irq);
|
|
|
|
}
|
|
|
|
#endif
|
|
|
|
|
2010-05-13 11:07:50 +08:00
|
|
|
static void
|
|
|
|
qlcnic_idc_debug_info(struct qlcnic_adapter *adapter, u8 encoding)
|
|
|
|
{
|
|
|
|
u32 val;
|
|
|
|
|
|
|
|
val = adapter->portnum & 0xf;
|
|
|
|
val |= encoding << 7;
|
|
|
|
val |= (jiffies - adapter->dev_rst_time) << 8;
|
|
|
|
|
|
|
|
QLCWR32(adapter, QLCNIC_CRB_DRV_SCRATCH, val);
|
|
|
|
adapter->dev_rst_time = jiffies;
|
|
|
|
}
|
|
|
|
|
2010-04-22 10:51:39 +08:00
|
|
|
static int
|
|
|
|
qlcnic_set_drv_state(struct qlcnic_adapter *adapter, u8 state)
|
2010-01-13 08:37:25 +08:00
|
|
|
{
|
|
|
|
u32 val;
|
|
|
|
|
|
|
|
WARN_ON(state != QLCNIC_DEV_NEED_RESET &&
|
|
|
|
state != QLCNIC_DEV_NEED_QUISCENT);
|
|
|
|
|
|
|
|
if (qlcnic_api_lock(adapter))
|
2010-04-22 10:51:39 +08:00
|
|
|
return -EIO;
|
2010-01-13 08:37:25 +08:00
|
|
|
|
|
|
|
val = QLCRD32(adapter, QLCNIC_CRB_DRV_STATE);
|
|
|
|
|
|
|
|
if (state == QLCNIC_DEV_NEED_RESET)
|
2010-04-22 10:51:38 +08:00
|
|
|
QLC_DEV_SET_RST_RDY(val, adapter->portnum);
|
2010-01-13 08:37:25 +08:00
|
|
|
else if (state == QLCNIC_DEV_NEED_QUISCENT)
|
2010-04-22 10:51:38 +08:00
|
|
|
QLC_DEV_SET_QSCNT_RDY(val, adapter->portnum);
|
2010-01-13 08:37:25 +08:00
|
|
|
|
|
|
|
QLCWR32(adapter, QLCNIC_CRB_DRV_STATE, val);
|
|
|
|
|
|
|
|
qlcnic_api_unlock(adapter);
|
2010-04-22 10:51:39 +08:00
|
|
|
|
|
|
|
return 0;
|
2010-01-13 08:37:25 +08:00
|
|
|
}
|
|
|
|
|
2010-02-01 13:24:56 +08:00
|
|
|
static int
|
|
|
|
qlcnic_clr_drv_state(struct qlcnic_adapter *adapter)
|
|
|
|
{
|
|
|
|
u32 val;
|
|
|
|
|
|
|
|
if (qlcnic_api_lock(adapter))
|
|
|
|
return -EBUSY;
|
|
|
|
|
|
|
|
val = QLCRD32(adapter, QLCNIC_CRB_DRV_STATE);
|
2010-04-22 10:51:38 +08:00
|
|
|
QLC_DEV_CLR_RST_QSCNT(val, adapter->portnum);
|
2010-02-01 13:24:56 +08:00
|
|
|
QLCWR32(adapter, QLCNIC_CRB_DRV_STATE, val);
|
|
|
|
|
|
|
|
qlcnic_api_unlock(adapter);
|
|
|
|
|
|
|
|
return 0;
|
|
|
|
}
|
|
|
|
|
2010-01-13 08:37:25 +08:00
|
|
|
static void
|
2010-08-19 13:08:29 +08:00
|
|
|
qlcnic_clr_all_drv_state(struct qlcnic_adapter *adapter, u8 failed)
|
2010-01-13 08:37:25 +08:00
|
|
|
{
|
|
|
|
u32 val;
|
|
|
|
|
|
|
|
if (qlcnic_api_lock(adapter))
|
|
|
|
goto err;
|
|
|
|
|
2010-08-25 12:03:05 +08:00
|
|
|
val = QLCRD32(adapter, QLCNIC_CRB_DRV_ACTIVE);
|
2010-04-22 10:51:38 +08:00
|
|
|
QLC_DEV_CLR_REF_CNT(val, adapter->portnum);
|
2010-08-25 12:03:05 +08:00
|
|
|
QLCWR32(adapter, QLCNIC_CRB_DRV_ACTIVE, val);
|
2010-01-13 08:37:25 +08:00
|
|
|
|
2010-08-19 13:08:29 +08:00
|
|
|
if (failed) {
|
|
|
|
QLCWR32(adapter, QLCNIC_CRB_DEV_STATE, QLCNIC_DEV_FAILED);
|
|
|
|
dev_info(&adapter->pdev->dev,
|
|
|
|
"Device state set to Failed. Please Reboot\n");
|
|
|
|
} else if (!(val & 0x11111111))
|
2010-01-13 08:37:25 +08:00
|
|
|
QLCWR32(adapter, QLCNIC_CRB_DEV_STATE, QLCNIC_DEV_COLD);
|
|
|
|
|
|
|
|
val = QLCRD32(adapter, QLCNIC_CRB_DRV_STATE);
|
2010-04-22 10:51:38 +08:00
|
|
|
QLC_DEV_CLR_RST_QSCNT(val, adapter->portnum);
|
2010-01-13 08:37:25 +08:00
|
|
|
QLCWR32(adapter, QLCNIC_CRB_DRV_STATE, val);
|
|
|
|
|
|
|
|
qlcnic_api_unlock(adapter);
|
|
|
|
err:
|
|
|
|
adapter->fw_fail_cnt = 0;
|
2011-07-29 21:30:27 +08:00
|
|
|
adapter->flags &= ~QLCNIC_FW_HANG;
|
2010-01-13 08:37:25 +08:00
|
|
|
clear_bit(__QLCNIC_START_FW, &adapter->state);
|
|
|
|
clear_bit(__QLCNIC_RESETTING, &adapter->state);
|
|
|
|
}
|
|
|
|
|
2010-04-22 10:51:37 +08:00
|
|
|
/* Grab api lock, before checking state */
|
2010-01-13 08:37:25 +08:00
|
|
|
static int
|
|
|
|
qlcnic_check_drv_state(struct qlcnic_adapter *adapter)
|
|
|
|
{
|
2011-06-22 10:52:17 +08:00
|
|
|
int act, state, active_mask;
|
2010-01-13 08:37:25 +08:00
|
|
|
|
|
|
|
state = QLCRD32(adapter, QLCNIC_CRB_DRV_STATE);
|
2010-08-25 12:03:05 +08:00
|
|
|
act = QLCRD32(adapter, QLCNIC_CRB_DRV_ACTIVE);
|
2010-01-13 08:37:25 +08:00
|
|
|
|
2011-06-22 10:52:17 +08:00
|
|
|
if (adapter->flags & QLCNIC_FW_RESET_OWNER) {
|
|
|
|
active_mask = (~(1 << (adapter->ahw->pci_func * 4)));
|
|
|
|
act = act & active_mask;
|
|
|
|
}
|
|
|
|
|
2010-01-13 08:37:25 +08:00
|
|
|
if (((state & 0x11111111) == (act & 0x11111111)) ||
|
|
|
|
((act & 0x11111111) == ((state >> 1) & 0x11111111)))
|
|
|
|
return 0;
|
|
|
|
else
|
|
|
|
return 1;
|
|
|
|
}
|
|
|
|
|
2010-05-13 11:07:47 +08:00
|
|
|
static int qlcnic_check_idc_ver(struct qlcnic_adapter *adapter)
|
|
|
|
{
|
|
|
|
u32 val = QLCRD32(adapter, QLCNIC_CRB_DRV_IDC_VER);
|
|
|
|
|
|
|
|
if (val != QLCNIC_DRV_IDC_VER) {
|
|
|
|
dev_warn(&adapter->pdev->dev, "IDC Version mismatch, driver's"
|
|
|
|
" idc ver = %x; reqd = %x\n", QLCNIC_DRV_IDC_VER, val);
|
|
|
|
}
|
|
|
|
|
|
|
|
return 0;
|
|
|
|
}
|
|
|
|
|
2010-01-13 08:37:25 +08:00
|
|
|
static int
|
|
|
|
qlcnic_can_start_firmware(struct qlcnic_adapter *adapter)
|
|
|
|
{
|
|
|
|
u32 val, prev_state;
|
2010-04-02 03:01:32 +08:00
|
|
|
u8 dev_init_timeo = adapter->dev_init_timeo;
|
2010-04-22 10:51:38 +08:00
|
|
|
u8 portnum = adapter->portnum;
|
2010-05-13 11:07:47 +08:00
|
|
|
u8 ret;
|
2010-01-13 08:37:25 +08:00
|
|
|
|
2010-04-22 10:51:37 +08:00
|
|
|
if (test_and_clear_bit(__QLCNIC_START_FW, &adapter->state))
|
|
|
|
return 1;
|
|
|
|
|
2010-01-13 08:37:25 +08:00
|
|
|
if (qlcnic_api_lock(adapter))
|
|
|
|
return -1;
|
|
|
|
|
2010-08-25 12:03:05 +08:00
|
|
|
val = QLCRD32(adapter, QLCNIC_CRB_DRV_ACTIVE);
|
2010-04-22 10:51:38 +08:00
|
|
|
if (!(val & (1 << (portnum * 4)))) {
|
|
|
|
QLC_DEV_SET_REF_CNT(val, portnum);
|
2010-08-25 12:03:05 +08:00
|
|
|
QLCWR32(adapter, QLCNIC_CRB_DRV_ACTIVE, val);
|
2010-01-13 08:37:25 +08:00
|
|
|
}
|
|
|
|
|
|
|
|
prev_state = QLCRD32(adapter, QLCNIC_CRB_DEV_STATE);
|
2010-04-02 03:01:33 +08:00
|
|
|
QLCDB(adapter, HW, "Device state = %u\n", prev_state);
|
2010-01-13 08:37:25 +08:00
|
|
|
|
|
|
|
switch (prev_state) {
|
|
|
|
case QLCNIC_DEV_COLD:
|
2010-04-22 10:51:36 +08:00
|
|
|
QLCWR32(adapter, QLCNIC_CRB_DEV_STATE, QLCNIC_DEV_INITIALIZING);
|
2010-05-13 11:07:47 +08:00
|
|
|
QLCWR32(adapter, QLCNIC_CRB_DRV_IDC_VER, QLCNIC_DRV_IDC_VER);
|
2010-05-13 11:07:50 +08:00
|
|
|
qlcnic_idc_debug_info(adapter, 0);
|
2010-01-13 08:37:25 +08:00
|
|
|
qlcnic_api_unlock(adapter);
|
|
|
|
return 1;
|
|
|
|
|
|
|
|
case QLCNIC_DEV_READY:
|
2010-05-13 11:07:47 +08:00
|
|
|
ret = qlcnic_check_idc_ver(adapter);
|
2010-01-13 08:37:25 +08:00
|
|
|
qlcnic_api_unlock(adapter);
|
2010-05-13 11:07:47 +08:00
|
|
|
return ret;
|
2010-01-13 08:37:25 +08:00
|
|
|
|
|
|
|
case QLCNIC_DEV_NEED_RESET:
|
|
|
|
val = QLCRD32(adapter, QLCNIC_CRB_DRV_STATE);
|
2010-04-22 10:51:38 +08:00
|
|
|
QLC_DEV_SET_RST_RDY(val, portnum);
|
2010-01-13 08:37:25 +08:00
|
|
|
QLCWR32(adapter, QLCNIC_CRB_DRV_STATE, val);
|
|
|
|
break;
|
|
|
|
|
|
|
|
case QLCNIC_DEV_NEED_QUISCENT:
|
|
|
|
val = QLCRD32(adapter, QLCNIC_CRB_DRV_STATE);
|
2010-04-22 10:51:38 +08:00
|
|
|
QLC_DEV_SET_QSCNT_RDY(val, portnum);
|
2010-01-13 08:37:25 +08:00
|
|
|
QLCWR32(adapter, QLCNIC_CRB_DRV_STATE, val);
|
|
|
|
break;
|
|
|
|
|
|
|
|
case QLCNIC_DEV_FAILED:
|
2010-05-17 09:22:14 +08:00
|
|
|
dev_err(&adapter->pdev->dev, "Device in failed state.\n");
|
2010-01-13 08:37:25 +08:00
|
|
|
qlcnic_api_unlock(adapter);
|
|
|
|
return -1;
|
2010-04-22 10:51:36 +08:00
|
|
|
|
|
|
|
case QLCNIC_DEV_INITIALIZING:
|
|
|
|
case QLCNIC_DEV_QUISCENT:
|
|
|
|
break;
|
2010-01-13 08:37:25 +08:00
|
|
|
}
|
|
|
|
|
|
|
|
qlcnic_api_unlock(adapter);
|
2010-04-02 03:01:32 +08:00
|
|
|
|
|
|
|
do {
|
2010-01-13 08:37:25 +08:00
|
|
|
msleep(1000);
|
2010-05-13 11:07:49 +08:00
|
|
|
prev_state = QLCRD32(adapter, QLCNIC_CRB_DEV_STATE);
|
|
|
|
|
|
|
|
if (prev_state == QLCNIC_DEV_QUISCENT)
|
|
|
|
continue;
|
|
|
|
} while ((prev_state != QLCNIC_DEV_READY) && --dev_init_timeo);
|
2010-01-13 08:37:25 +08:00
|
|
|
|
2010-04-02 03:01:33 +08:00
|
|
|
if (!dev_init_timeo) {
|
|
|
|
dev_err(&adapter->pdev->dev,
|
|
|
|
"Waiting for device to initialize timeout\n");
|
2010-01-13 08:37:25 +08:00
|
|
|
return -1;
|
2010-04-02 03:01:33 +08:00
|
|
|
}
|
2010-01-13 08:37:25 +08:00
|
|
|
|
|
|
|
if (qlcnic_api_lock(adapter))
|
|
|
|
return -1;
|
|
|
|
|
|
|
|
val = QLCRD32(adapter, QLCNIC_CRB_DRV_STATE);
|
2010-04-22 10:51:38 +08:00
|
|
|
QLC_DEV_CLR_RST_QSCNT(val, portnum);
|
2010-01-13 08:37:25 +08:00
|
|
|
QLCWR32(adapter, QLCNIC_CRB_DRV_STATE, val);
|
|
|
|
|
2010-05-13 11:07:47 +08:00
|
|
|
ret = qlcnic_check_idc_ver(adapter);
|
2010-01-13 08:37:25 +08:00
|
|
|
qlcnic_api_unlock(adapter);
|
|
|
|
|
2010-05-13 11:07:47 +08:00
|
|
|
return ret;
|
2010-01-13 08:37:25 +08:00
|
|
|
}
|
|
|
|
|
|
|
|
static void
|
|
|
|
qlcnic_fwinit_work(struct work_struct *work)
|
|
|
|
{
|
|
|
|
struct qlcnic_adapter *adapter = container_of(work,
|
|
|
|
struct qlcnic_adapter, fw_work.work);
|
2010-08-17 08:34:20 +08:00
|
|
|
u32 dev_state = 0xf;
|
2011-07-14 11:16:50 +08:00
|
|
|
u32 val;
|
2010-01-13 08:37:25 +08:00
|
|
|
|
2010-04-22 10:51:37 +08:00
|
|
|
if (qlcnic_api_lock(adapter))
|
|
|
|
goto err_ret;
|
2010-01-13 08:37:25 +08:00
|
|
|
|
2010-05-13 11:07:49 +08:00
|
|
|
dev_state = QLCRD32(adapter, QLCNIC_CRB_DEV_STATE);
|
2010-10-08 07:46:06 +08:00
|
|
|
if (dev_state == QLCNIC_DEV_QUISCENT ||
|
|
|
|
dev_state == QLCNIC_DEV_NEED_QUISCENT) {
|
2010-05-13 11:07:49 +08:00
|
|
|
qlcnic_api_unlock(adapter);
|
|
|
|
qlcnic_schedule_work(adapter, qlcnic_fwinit_work,
|
|
|
|
FW_POLL_DELAY * 2);
|
|
|
|
return;
|
|
|
|
}
|
|
|
|
|
2010-06-01 19:33:09 +08:00
|
|
|
if (adapter->op_mode == QLCNIC_NON_PRIV_FUNC) {
|
2010-08-17 08:34:20 +08:00
|
|
|
qlcnic_api_unlock(adapter);
|
|
|
|
goto wait_npar;
|
2010-06-01 19:33:09 +08:00
|
|
|
}
|
|
|
|
|
2011-10-28 20:57:11 +08:00
|
|
|
if (dev_state == QLCNIC_DEV_INITIALIZING ||
|
|
|
|
dev_state == QLCNIC_DEV_READY) {
|
|
|
|
dev_info(&adapter->pdev->dev, "Detected state change from "
|
|
|
|
"DEV_NEED_RESET, skipping ack check\n");
|
|
|
|
goto skip_ack_check;
|
|
|
|
}
|
|
|
|
|
2010-04-22 10:51:37 +08:00
|
|
|
if (adapter->fw_wait_cnt++ > adapter->reset_ack_timeo) {
|
2011-10-28 20:57:11 +08:00
|
|
|
dev_info(&adapter->pdev->dev, "Reset:Failed to get ack %d sec\n",
|
2010-04-22 10:51:37 +08:00
|
|
|
adapter->reset_ack_timeo);
|
|
|
|
goto skip_ack_check;
|
|
|
|
}
|
|
|
|
|
|
|
|
if (!qlcnic_check_drv_state(adapter)) {
|
|
|
|
skip_ack_check:
|
|
|
|
dev_state = QLCRD32(adapter, QLCNIC_CRB_DEV_STATE);
|
2010-05-13 11:07:49 +08:00
|
|
|
|
2010-04-22 10:51:37 +08:00
|
|
|
if (dev_state == QLCNIC_DEV_NEED_RESET) {
|
|
|
|
QLCWR32(adapter, QLCNIC_CRB_DEV_STATE,
|
|
|
|
QLCNIC_DEV_INITIALIZING);
|
|
|
|
set_bit(__QLCNIC_START_FW, &adapter->state);
|
|
|
|
QLCDB(adapter, DRV, "Restarting fw\n");
|
2010-05-13 11:07:50 +08:00
|
|
|
qlcnic_idc_debug_info(adapter, 0);
|
2011-07-14 11:16:50 +08:00
|
|
|
val = QLCRD32(adapter, QLCNIC_CRB_DRV_STATE);
|
|
|
|
QLC_DEV_SET_RST_RDY(val, adapter->portnum);
|
|
|
|
QLCWR32(adapter, QLCNIC_CRB_DRV_STATE, val);
|
2010-01-13 08:37:25 +08:00
|
|
|
}
|
|
|
|
|
2010-04-22 10:51:37 +08:00
|
|
|
qlcnic_api_unlock(adapter);
|
|
|
|
|
2011-06-22 10:52:18 +08:00
|
|
|
rtnl_lock();
|
2011-07-14 11:16:50 +08:00
|
|
|
if (adapter->ahw->fw_dump.enable &&
|
|
|
|
(adapter->flags & QLCNIC_FW_RESET_OWNER)) {
|
2011-06-22 10:52:22 +08:00
|
|
|
QLCDB(adapter, DRV, "Take FW dump\n");
|
|
|
|
qlcnic_dump_fw(adapter);
|
2011-07-29 21:30:27 +08:00
|
|
|
adapter->flags |= QLCNIC_FW_HANG;
|
2011-06-22 10:52:22 +08:00
|
|
|
}
|
2011-06-22 10:52:18 +08:00
|
|
|
rtnl_unlock();
|
2011-07-14 11:16:50 +08:00
|
|
|
|
|
|
|
adapter->flags &= ~QLCNIC_FW_RESET_OWNER;
|
2010-06-01 19:33:09 +08:00
|
|
|
if (!adapter->nic_ops->start_firmware(adapter)) {
|
2010-01-13 08:37:25 +08:00
|
|
|
qlcnic_schedule_work(adapter, qlcnic_attach_work, 0);
|
2010-08-25 12:03:04 +08:00
|
|
|
adapter->fw_wait_cnt = 0;
|
2010-01-13 08:37:25 +08:00
|
|
|
return;
|
|
|
|
}
|
|
|
|
goto err_ret;
|
|
|
|
}
|
|
|
|
|
2010-04-22 10:51:37 +08:00
|
|
|
qlcnic_api_unlock(adapter);
|
2010-04-02 03:01:32 +08:00
|
|
|
|
2010-06-01 19:33:09 +08:00
|
|
|
wait_npar:
|
2010-01-13 08:37:25 +08:00
|
|
|
dev_state = QLCRD32(adapter, QLCNIC_CRB_DEV_STATE);
|
2010-04-22 10:51:37 +08:00
|
|
|
QLCDB(adapter, HW, "Func waiting: Device state=%u\n", dev_state);
|
2010-04-02 03:01:33 +08:00
|
|
|
|
2010-01-13 08:37:25 +08:00
|
|
|
switch (dev_state) {
|
2010-08-17 08:34:20 +08:00
|
|
|
case QLCNIC_DEV_READY:
|
2010-06-01 19:33:09 +08:00
|
|
|
if (!adapter->nic_ops->start_firmware(adapter)) {
|
2010-04-22 10:51:37 +08:00
|
|
|
qlcnic_schedule_work(adapter, qlcnic_attach_work, 0);
|
2010-08-25 12:03:04 +08:00
|
|
|
adapter->fw_wait_cnt = 0;
|
2010-04-22 10:51:37 +08:00
|
|
|
return;
|
|
|
|
}
|
2010-08-17 08:34:20 +08:00
|
|
|
case QLCNIC_DEV_FAILED:
|
|
|
|
break;
|
|
|
|
default:
|
|
|
|
qlcnic_schedule_work(adapter,
|
|
|
|
qlcnic_fwinit_work, FW_POLL_DELAY);
|
|
|
|
return;
|
2010-01-13 08:37:25 +08:00
|
|
|
}
|
|
|
|
|
|
|
|
err_ret:
|
2010-04-22 10:51:37 +08:00
|
|
|
dev_err(&adapter->pdev->dev, "Fwinit work failed state=%u "
|
|
|
|
"fw_wait_cnt=%u\n", dev_state, adapter->fw_wait_cnt);
|
2010-04-02 03:01:34 +08:00
|
|
|
netif_device_attach(adapter->netdev);
|
2010-08-19 13:08:29 +08:00
|
|
|
qlcnic_clr_all_drv_state(adapter, 0);
|
2010-01-13 08:37:25 +08:00
|
|
|
}
|
|
|
|
|
|
|
|
static void
|
|
|
|
qlcnic_detach_work(struct work_struct *work)
|
|
|
|
{
|
|
|
|
struct qlcnic_adapter *adapter = container_of(work,
|
|
|
|
struct qlcnic_adapter, fw_work.work);
|
|
|
|
struct net_device *netdev = adapter->netdev;
|
|
|
|
u32 status;
|
|
|
|
|
|
|
|
netif_device_detach(netdev);
|
|
|
|
|
2010-10-08 07:46:06 +08:00
|
|
|
/* Dont grab rtnl lock during Quiscent mode */
|
|
|
|
if (adapter->dev_state == QLCNIC_DEV_NEED_QUISCENT) {
|
|
|
|
if (netif_running(netdev))
|
|
|
|
__qlcnic_down(adapter, netdev);
|
|
|
|
} else
|
|
|
|
qlcnic_down(adapter, netdev);
|
2010-01-13 08:37:25 +08:00
|
|
|
|
|
|
|
status = QLCRD32(adapter, QLCNIC_PEG_HALT_STATUS1);
|
|
|
|
|
2011-08-29 20:50:26 +08:00
|
|
|
if (status & QLCNIC_RCODE_FATAL_ERROR) {
|
|
|
|
dev_err(&adapter->pdev->dev,
|
|
|
|
"Detaching the device: peg halt status1=0x%x\n",
|
|
|
|
status);
|
|
|
|
|
|
|
|
if (QLCNIC_FWERROR_CODE(status) == QLCNIC_FWERROR_FAN_FAILURE) {
|
|
|
|
dev_err(&adapter->pdev->dev,
|
|
|
|
"On board active cooling fan failed. "
|
|
|
|
"Device has been halted.\n");
|
|
|
|
dev_err(&adapter->pdev->dev,
|
|
|
|
"Replace the adapter.\n");
|
|
|
|
}
|
|
|
|
|
2010-01-13 08:37:25 +08:00
|
|
|
goto err_ret;
|
2011-08-29 20:50:26 +08:00
|
|
|
}
|
2010-01-13 08:37:25 +08:00
|
|
|
|
2011-08-29 20:50:26 +08:00
|
|
|
if (adapter->temp == QLCNIC_TEMP_PANIC) {
|
|
|
|
dev_err(&adapter->pdev->dev, "Detaching the device: temp=%d\n",
|
|
|
|
adapter->temp);
|
2010-01-13 08:37:25 +08:00
|
|
|
goto err_ret;
|
2011-08-29 20:50:26 +08:00
|
|
|
}
|
|
|
|
|
2011-06-22 10:52:17 +08:00
|
|
|
/* Dont ack if this instance is the reset owner */
|
|
|
|
if (!(adapter->flags & QLCNIC_FW_RESET_OWNER)) {
|
2011-08-29 20:50:26 +08:00
|
|
|
if (qlcnic_set_drv_state(adapter, adapter->dev_state)) {
|
|
|
|
dev_err(&adapter->pdev->dev,
|
|
|
|
"Failed to set driver state,"
|
|
|
|
"detaching the device.\n");
|
2011-06-22 10:52:17 +08:00
|
|
|
goto err_ret;
|
2011-08-29 20:50:26 +08:00
|
|
|
}
|
2011-06-22 10:52:17 +08:00
|
|
|
}
|
2010-01-13 08:37:25 +08:00
|
|
|
|
|
|
|
adapter->fw_wait_cnt = 0;
|
|
|
|
|
|
|
|
qlcnic_schedule_work(adapter, qlcnic_fwinit_work, FW_POLL_DELAY);
|
|
|
|
|
|
|
|
return;
|
|
|
|
|
|
|
|
err_ret:
|
2010-04-02 03:01:34 +08:00
|
|
|
netif_device_attach(netdev);
|
2010-08-19 13:08:29 +08:00
|
|
|
qlcnic_clr_all_drv_state(adapter, 1);
|
2010-01-13 08:37:25 +08:00
|
|
|
}
|
|
|
|
|
2010-08-17 08:34:20 +08:00
|
|
|
/*Transit NPAR state to NON Operational */
|
|
|
|
static void
|
|
|
|
qlcnic_set_npar_non_operational(struct qlcnic_adapter *adapter)
|
|
|
|
{
|
|
|
|
u32 state;
|
|
|
|
|
|
|
|
state = QLCRD32(adapter, QLCNIC_CRB_DEV_NPAR_STATE);
|
|
|
|
if (state == QLCNIC_DEV_NPAR_NON_OPER)
|
|
|
|
return;
|
|
|
|
|
|
|
|
if (qlcnic_api_lock(adapter))
|
|
|
|
return;
|
|
|
|
QLCWR32(adapter, QLCNIC_CRB_DEV_NPAR_STATE, QLCNIC_DEV_NPAR_NON_OPER);
|
|
|
|
qlcnic_api_unlock(adapter);
|
|
|
|
}
|
|
|
|
|
2010-04-22 10:51:37 +08:00
|
|
|
/*Transit to RESET state from READY state only */
|
2011-05-12 20:48:33 +08:00
|
|
|
void
|
2010-01-13 08:37:25 +08:00
|
|
|
qlcnic_dev_request_reset(struct qlcnic_adapter *adapter)
|
|
|
|
{
|
2012-02-03 21:45:41 +08:00
|
|
|
u32 state, xg_val = 0, gb_val = 0;
|
|
|
|
|
|
|
|
qlcnic_xg_set_xg0_mask(xg_val);
|
|
|
|
qlcnic_xg_set_xg1_mask(xg_val);
|
|
|
|
QLCWR32(adapter, QLCNIC_NIU_XG_PAUSE_CTL, xg_val);
|
|
|
|
qlcnic_gb_set_gb0_mask(gb_val);
|
|
|
|
qlcnic_gb_set_gb1_mask(gb_val);
|
|
|
|
qlcnic_gb_set_gb2_mask(gb_val);
|
|
|
|
qlcnic_gb_set_gb3_mask(gb_val);
|
|
|
|
QLCWR32(adapter, QLCNIC_NIU_GB_PAUSE_CTL, gb_val);
|
|
|
|
dev_info(&adapter->pdev->dev, "Pause control frames disabled"
|
|
|
|
" on all ports\n");
|
2010-07-14 04:33:35 +08:00
|
|
|
adapter->need_fw_reset = 1;
|
2010-01-13 08:37:25 +08:00
|
|
|
if (qlcnic_api_lock(adapter))
|
|
|
|
return;
|
|
|
|
|
|
|
|
state = QLCRD32(adapter, QLCNIC_CRB_DEV_STATE);
|
2012-04-26 18:31:29 +08:00
|
|
|
if (state == QLCNIC_DEV_FAILED || (state == QLCNIC_DEV_BADBAD)) {
|
|
|
|
netdev_err(adapter->netdev,
|
|
|
|
"Device is in FAILED state, Please Reboot\n");
|
|
|
|
qlcnic_api_unlock(adapter);
|
|
|
|
return;
|
|
|
|
}
|
2010-01-13 08:37:25 +08:00
|
|
|
|
2010-04-22 10:51:37 +08:00
|
|
|
if (state == QLCNIC_DEV_READY) {
|
2010-01-13 08:37:25 +08:00
|
|
|
QLCWR32(adapter, QLCNIC_CRB_DEV_STATE, QLCNIC_DEV_NEED_RESET);
|
2011-06-22 10:52:17 +08:00
|
|
|
adapter->flags |= QLCNIC_FW_RESET_OWNER;
|
2010-04-02 03:01:33 +08:00
|
|
|
QLCDB(adapter, DRV, "NEED_RESET state set\n");
|
2010-05-13 11:07:50 +08:00
|
|
|
qlcnic_idc_debug_info(adapter, 0);
|
2010-01-13 08:37:25 +08:00
|
|
|
}
|
|
|
|
|
2010-08-17 08:34:20 +08:00
|
|
|
QLCWR32(adapter, QLCNIC_CRB_DEV_NPAR_STATE, QLCNIC_DEV_NPAR_NON_OPER);
|
2010-01-13 08:37:25 +08:00
|
|
|
qlcnic_api_unlock(adapter);
|
|
|
|
}
|
|
|
|
|
2010-06-01 19:33:09 +08:00
|
|
|
/* Transit to NPAR READY state from NPAR NOT READY state */
|
|
|
|
static void
|
|
|
|
qlcnic_dev_set_npar_ready(struct qlcnic_adapter *adapter)
|
|
|
|
{
|
|
|
|
if (qlcnic_api_lock(adapter))
|
|
|
|
return;
|
|
|
|
|
2010-08-17 08:34:20 +08:00
|
|
|
QLCWR32(adapter, QLCNIC_CRB_DEV_NPAR_STATE, QLCNIC_DEV_NPAR_OPER);
|
|
|
|
QLCDB(adapter, DRV, "NPAR operational state set\n");
|
2010-06-01 19:33:09 +08:00
|
|
|
|
|
|
|
qlcnic_api_unlock(adapter);
|
|
|
|
}
|
|
|
|
|
2010-01-13 08:37:25 +08:00
|
|
|
static void
|
|
|
|
qlcnic_schedule_work(struct qlcnic_adapter *adapter,
|
|
|
|
work_func_t func, int delay)
|
|
|
|
{
|
2010-07-14 04:33:34 +08:00
|
|
|
if (test_bit(__QLCNIC_AER, &adapter->state))
|
|
|
|
return;
|
|
|
|
|
2010-01-13 08:37:25 +08:00
|
|
|
INIT_DELAYED_WORK(&adapter->fw_work, func);
|
2010-10-08 07:46:05 +08:00
|
|
|
queue_delayed_work(qlcnic_wq, &adapter->fw_work,
|
|
|
|
round_jiffies_relative(delay));
|
2010-01-13 08:37:25 +08:00
|
|
|
}
|
|
|
|
|
|
|
|
static void
|
|
|
|
qlcnic_cancel_fw_work(struct qlcnic_adapter *adapter)
|
|
|
|
{
|
|
|
|
while (test_and_set_bit(__QLCNIC_RESETTING, &adapter->state))
|
|
|
|
msleep(10);
|
|
|
|
|
2012-04-26 18:31:29 +08:00
|
|
|
if (!adapter->fw_work.work.func)
|
|
|
|
return;
|
|
|
|
|
2010-01-13 08:37:25 +08:00
|
|
|
cancel_delayed_work_sync(&adapter->fw_work);
|
|
|
|
}
|
|
|
|
|
|
|
|
static void
|
|
|
|
qlcnic_attach_work(struct work_struct *work)
|
|
|
|
{
|
|
|
|
struct qlcnic_adapter *adapter = container_of(work,
|
|
|
|
struct qlcnic_adapter, fw_work.work);
|
|
|
|
struct net_device *netdev = adapter->netdev;
|
2010-08-25 12:03:04 +08:00
|
|
|
u32 npar_state;
|
2010-01-13 08:37:25 +08:00
|
|
|
|
2010-08-25 12:03:04 +08:00
|
|
|
if (adapter->op_mode != QLCNIC_MGMT_FUNC) {
|
|
|
|
npar_state = QLCRD32(adapter, QLCNIC_CRB_DEV_NPAR_STATE);
|
|
|
|
if (adapter->fw_wait_cnt++ > QLCNIC_DEV_NPAR_OPER_TIMEO)
|
|
|
|
qlcnic_clr_all_drv_state(adapter, 0);
|
|
|
|
else if (npar_state != QLCNIC_DEV_NPAR_OPER)
|
|
|
|
qlcnic_schedule_work(adapter, qlcnic_attach_work,
|
|
|
|
FW_POLL_DELAY);
|
|
|
|
else
|
|
|
|
goto attach;
|
|
|
|
QLCDB(adapter, DRV, "Waiting for NPAR state to operational\n");
|
|
|
|
return;
|
|
|
|
}
|
|
|
|
attach:
|
2010-01-13 08:37:25 +08:00
|
|
|
if (netif_running(netdev)) {
|
2010-06-22 11:19:02 +08:00
|
|
|
if (qlcnic_up(adapter, netdev))
|
2010-01-13 08:37:25 +08:00
|
|
|
goto done;
|
|
|
|
|
2010-09-17 03:14:41 +08:00
|
|
|
qlcnic_restore_indev_addr(netdev, NETDEV_UP);
|
2010-01-13 08:37:25 +08:00
|
|
|
}
|
|
|
|
|
|
|
|
done:
|
2010-04-02 03:01:34 +08:00
|
|
|
netif_device_attach(netdev);
|
2010-01-13 08:37:25 +08:00
|
|
|
adapter->fw_fail_cnt = 0;
|
2011-07-29 21:30:27 +08:00
|
|
|
adapter->flags &= ~QLCNIC_FW_HANG;
|
2010-01-13 08:37:25 +08:00
|
|
|
clear_bit(__QLCNIC_RESETTING, &adapter->state);
|
2010-02-01 13:24:56 +08:00
|
|
|
|
|
|
|
if (!qlcnic_clr_drv_state(adapter))
|
|
|
|
qlcnic_schedule_work(adapter, qlcnic_fw_poll_work,
|
|
|
|
FW_POLL_DELAY);
|
2010-01-13 08:37:25 +08:00
|
|
|
}
|
|
|
|
|
|
|
|
static int
|
|
|
|
qlcnic_check_health(struct qlcnic_adapter *adapter)
|
|
|
|
{
|
2010-09-01 01:17:44 +08:00
|
|
|
u32 state = 0, heartbeat;
|
2011-09-13 16:06:17 +08:00
|
|
|
u32 peg_status;
|
2010-01-13 08:37:25 +08:00
|
|
|
|
|
|
|
if (qlcnic_check_temp(adapter))
|
|
|
|
goto detach;
|
|
|
|
|
2010-05-13 11:07:42 +08:00
|
|
|
if (adapter->need_fw_reset)
|
2010-01-13 08:37:25 +08:00
|
|
|
qlcnic_dev_request_reset(adapter);
|
|
|
|
|
|
|
|
state = QLCRD32(adapter, QLCNIC_CRB_DEV_STATE);
|
2010-10-08 07:46:06 +08:00
|
|
|
if (state == QLCNIC_DEV_NEED_RESET) {
|
2010-08-17 08:34:20 +08:00
|
|
|
qlcnic_set_npar_non_operational(adapter);
|
2010-01-13 08:37:25 +08:00
|
|
|
adapter->need_fw_reset = 1;
|
2010-10-08 07:46:06 +08:00
|
|
|
} else if (state == QLCNIC_DEV_NEED_QUISCENT)
|
|
|
|
goto detach;
|
2010-01-13 08:37:25 +08:00
|
|
|
|
2010-09-01 01:17:44 +08:00
|
|
|
heartbeat = QLCRD32(adapter, QLCNIC_PEG_ALIVE_COUNTER);
|
|
|
|
if (heartbeat != adapter->heartbeat) {
|
|
|
|
adapter->heartbeat = heartbeat;
|
2010-01-13 08:37:25 +08:00
|
|
|
adapter->fw_fail_cnt = 0;
|
|
|
|
if (adapter->need_fw_reset)
|
|
|
|
goto detach;
|
2010-06-22 11:19:03 +08:00
|
|
|
|
2011-02-23 11:21:24 +08:00
|
|
|
if (adapter->reset_context && auto_fw_reset) {
|
2010-06-22 11:19:03 +08:00
|
|
|
qlcnic_reset_hw_context(adapter);
|
|
|
|
adapter->netdev->trans_start = jiffies;
|
|
|
|
}
|
|
|
|
|
2010-01-13 08:37:25 +08:00
|
|
|
return 0;
|
|
|
|
}
|
|
|
|
|
|
|
|
if (++adapter->fw_fail_cnt < FW_FAIL_THRESH)
|
|
|
|
return 0;
|
|
|
|
|
2011-07-29 21:30:27 +08:00
|
|
|
adapter->flags |= QLCNIC_FW_HANG;
|
|
|
|
|
2010-01-13 08:37:25 +08:00
|
|
|
qlcnic_dev_request_reset(adapter);
|
|
|
|
|
2011-02-23 11:21:24 +08:00
|
|
|
if (auto_fw_reset)
|
2010-07-14 04:33:32 +08:00
|
|
|
clear_bit(__QLCNIC_FW_ATTACHED, &adapter->state);
|
2010-01-13 08:37:25 +08:00
|
|
|
|
2011-09-13 16:06:17 +08:00
|
|
|
dev_err(&adapter->pdev->dev, "firmware hang detected\n");
|
|
|
|
dev_err(&adapter->pdev->dev, "Dumping hw/fw registers\n"
|
2011-07-29 21:30:29 +08:00
|
|
|
"PEG_HALT_STATUS1: 0x%x, PEG_HALT_STATUS2: 0x%x,\n"
|
|
|
|
"PEG_NET_0_PC: 0x%x, PEG_NET_1_PC: 0x%x,\n"
|
|
|
|
"PEG_NET_2_PC: 0x%x, PEG_NET_3_PC: 0x%x,\n"
|
|
|
|
"PEG_NET_4_PC: 0x%x\n",
|
|
|
|
QLCRD32(adapter, QLCNIC_PEG_HALT_STATUS1),
|
|
|
|
QLCRD32(adapter, QLCNIC_PEG_HALT_STATUS2),
|
|
|
|
QLCRD32(adapter, QLCNIC_CRB_PEG_NET_0 + 0x3c),
|
|
|
|
QLCRD32(adapter, QLCNIC_CRB_PEG_NET_1 + 0x3c),
|
|
|
|
QLCRD32(adapter, QLCNIC_CRB_PEG_NET_2 + 0x3c),
|
|
|
|
QLCRD32(adapter, QLCNIC_CRB_PEG_NET_3 + 0x3c),
|
|
|
|
QLCRD32(adapter, QLCNIC_CRB_PEG_NET_4 + 0x3c));
|
2011-09-13 16:06:17 +08:00
|
|
|
peg_status = QLCRD32(adapter, QLCNIC_PEG_HALT_STATUS1);
|
2012-02-03 21:45:43 +08:00
|
|
|
if (QLCNIC_FWERROR_CODE(peg_status) == 0x67)
|
2011-09-13 16:06:17 +08:00
|
|
|
dev_err(&adapter->pdev->dev,
|
|
|
|
"Firmware aborted with error code 0x00006700. "
|
|
|
|
"Device is being reset.\n");
|
2010-01-13 08:37:25 +08:00
|
|
|
detach:
|
|
|
|
adapter->dev_state = (state == QLCNIC_DEV_NEED_QUISCENT) ? state :
|
|
|
|
QLCNIC_DEV_NEED_RESET;
|
|
|
|
|
2011-02-23 11:21:24 +08:00
|
|
|
if (auto_fw_reset &&
|
2010-04-02 03:01:33 +08:00
|
|
|
!test_and_set_bit(__QLCNIC_RESETTING, &adapter->state)) {
|
|
|
|
|
2010-01-13 08:37:25 +08:00
|
|
|
qlcnic_schedule_work(adapter, qlcnic_detach_work, 0);
|
2010-04-02 03:01:33 +08:00
|
|
|
QLCDB(adapter, DRV, "fw recovery scheduled.\n");
|
|
|
|
}
|
2010-01-13 08:37:25 +08:00
|
|
|
|
|
|
|
return 1;
|
|
|
|
}
|
|
|
|
|
|
|
|
static void
|
|
|
|
qlcnic_fw_poll_work(struct work_struct *work)
|
|
|
|
{
|
|
|
|
struct qlcnic_adapter *adapter = container_of(work,
|
|
|
|
struct qlcnic_adapter, fw_work.work);
|
|
|
|
|
|
|
|
if (test_bit(__QLCNIC_RESETTING, &adapter->state))
|
|
|
|
goto reschedule;
|
|
|
|
|
|
|
|
|
|
|
|
if (qlcnic_check_health(adapter))
|
|
|
|
return;
|
|
|
|
|
2010-09-01 01:17:51 +08:00
|
|
|
if (adapter->fhash.fnum)
|
|
|
|
qlcnic_prune_lb_filters(adapter);
|
|
|
|
|
2010-01-13 08:37:25 +08:00
|
|
|
reschedule:
|
|
|
|
qlcnic_schedule_work(adapter, qlcnic_fw_poll_work, FW_POLL_DELAY);
|
|
|
|
}
|
|
|
|
|
2010-07-14 04:33:34 +08:00
|
|
|
static int qlcnic_is_first_func(struct pci_dev *pdev)
|
|
|
|
{
|
|
|
|
struct pci_dev *oth_pdev;
|
|
|
|
int val = pdev->devfn;
|
|
|
|
|
|
|
|
while (val-- > 0) {
|
|
|
|
oth_pdev = pci_get_domain_bus_and_slot(pci_domain_nr
|
|
|
|
(pdev->bus), pdev->bus->number,
|
|
|
|
PCI_DEVFN(PCI_SLOT(pdev->devfn), val));
|
2010-07-19 05:51:59 +08:00
|
|
|
if (!oth_pdev)
|
|
|
|
continue;
|
2010-07-14 04:33:34 +08:00
|
|
|
|
2010-07-19 05:51:59 +08:00
|
|
|
if (oth_pdev->current_state != PCI_D3cold) {
|
|
|
|
pci_dev_put(oth_pdev);
|
2010-07-14 04:33:34 +08:00
|
|
|
return 0;
|
2010-07-19 05:51:59 +08:00
|
|
|
}
|
|
|
|
pci_dev_put(oth_pdev);
|
2010-07-14 04:33:34 +08:00
|
|
|
}
|
|
|
|
return 1;
|
|
|
|
}
|
|
|
|
|
|
|
|
static int qlcnic_attach_func(struct pci_dev *pdev)
|
|
|
|
{
|
|
|
|
int err, first_func;
|
|
|
|
struct qlcnic_adapter *adapter = pci_get_drvdata(pdev);
|
|
|
|
struct net_device *netdev = adapter->netdev;
|
|
|
|
|
|
|
|
pdev->error_state = pci_channel_io_normal;
|
|
|
|
|
|
|
|
err = pci_enable_device(pdev);
|
|
|
|
if (err)
|
|
|
|
return err;
|
|
|
|
|
|
|
|
pci_set_power_state(pdev, PCI_D0);
|
|
|
|
pci_set_master(pdev);
|
|
|
|
pci_restore_state(pdev);
|
|
|
|
|
|
|
|
first_func = qlcnic_is_first_func(pdev);
|
|
|
|
|
|
|
|
if (qlcnic_api_lock(adapter))
|
|
|
|
return -EINVAL;
|
|
|
|
|
2010-08-17 08:34:19 +08:00
|
|
|
if (adapter->op_mode != QLCNIC_NON_PRIV_FUNC && first_func) {
|
2010-07-14 04:33:34 +08:00
|
|
|
adapter->need_fw_reset = 1;
|
|
|
|
set_bit(__QLCNIC_START_FW, &adapter->state);
|
|
|
|
QLCWR32(adapter, QLCNIC_CRB_DEV_STATE, QLCNIC_DEV_INITIALIZING);
|
|
|
|
QLCDB(adapter, DRV, "Restarting fw\n");
|
|
|
|
}
|
|
|
|
qlcnic_api_unlock(adapter);
|
|
|
|
|
|
|
|
err = adapter->nic_ops->start_firmware(adapter);
|
|
|
|
if (err)
|
|
|
|
return err;
|
|
|
|
|
|
|
|
qlcnic_clr_drv_state(adapter);
|
|
|
|
qlcnic_setup_intr(adapter);
|
|
|
|
|
|
|
|
if (netif_running(netdev)) {
|
|
|
|
err = qlcnic_attach(adapter);
|
|
|
|
if (err) {
|
2010-08-19 13:08:29 +08:00
|
|
|
qlcnic_clr_all_drv_state(adapter, 1);
|
2010-07-14 04:33:34 +08:00
|
|
|
clear_bit(__QLCNIC_AER, &adapter->state);
|
|
|
|
netif_device_attach(netdev);
|
|
|
|
return err;
|
|
|
|
}
|
|
|
|
|
|
|
|
err = qlcnic_up(adapter, netdev);
|
|
|
|
if (err)
|
|
|
|
goto done;
|
|
|
|
|
2010-09-17 03:14:41 +08:00
|
|
|
qlcnic_restore_indev_addr(netdev, NETDEV_UP);
|
2010-07-14 04:33:34 +08:00
|
|
|
}
|
|
|
|
done:
|
|
|
|
netif_device_attach(netdev);
|
|
|
|
return err;
|
|
|
|
}
|
|
|
|
|
|
|
|
static pci_ers_result_t qlcnic_io_error_detected(struct pci_dev *pdev,
|
|
|
|
pci_channel_state_t state)
|
|
|
|
{
|
|
|
|
struct qlcnic_adapter *adapter = pci_get_drvdata(pdev);
|
|
|
|
struct net_device *netdev = adapter->netdev;
|
|
|
|
|
|
|
|
if (state == pci_channel_io_perm_failure)
|
|
|
|
return PCI_ERS_RESULT_DISCONNECT;
|
|
|
|
|
|
|
|
if (state == pci_channel_io_normal)
|
|
|
|
return PCI_ERS_RESULT_RECOVERED;
|
|
|
|
|
|
|
|
set_bit(__QLCNIC_AER, &adapter->state);
|
|
|
|
netif_device_detach(netdev);
|
|
|
|
|
|
|
|
cancel_delayed_work_sync(&adapter->fw_work);
|
|
|
|
|
|
|
|
if (netif_running(netdev))
|
|
|
|
qlcnic_down(adapter, netdev);
|
|
|
|
|
|
|
|
qlcnic_detach(adapter);
|
|
|
|
qlcnic_teardown_intr(adapter);
|
|
|
|
|
|
|
|
clear_bit(__QLCNIC_RESETTING, &adapter->state);
|
|
|
|
|
|
|
|
pci_save_state(pdev);
|
|
|
|
pci_disable_device(pdev);
|
|
|
|
|
|
|
|
return PCI_ERS_RESULT_NEED_RESET;
|
|
|
|
}
|
|
|
|
|
|
|
|
static pci_ers_result_t qlcnic_io_slot_reset(struct pci_dev *pdev)
|
|
|
|
{
|
|
|
|
return qlcnic_attach_func(pdev) ? PCI_ERS_RESULT_DISCONNECT :
|
|
|
|
PCI_ERS_RESULT_RECOVERED;
|
|
|
|
}
|
|
|
|
|
|
|
|
static void qlcnic_io_resume(struct pci_dev *pdev)
|
|
|
|
{
|
|
|
|
struct qlcnic_adapter *adapter = pci_get_drvdata(pdev);
|
|
|
|
|
|
|
|
pci_cleanup_aer_uncorrect_error_status(pdev);
|
|
|
|
|
|
|
|
if (QLCRD32(adapter, QLCNIC_CRB_DEV_STATE) == QLCNIC_DEV_READY &&
|
|
|
|
test_and_clear_bit(__QLCNIC_AER, &adapter->state))
|
|
|
|
qlcnic_schedule_work(adapter, qlcnic_fw_poll_work,
|
|
|
|
FW_POLL_DELAY);
|
|
|
|
}
|
|
|
|
|
2010-06-03 15:50:56 +08:00
|
|
|
static int
|
|
|
|
qlcnicvf_start_firmware(struct qlcnic_adapter *adapter)
|
|
|
|
{
|
|
|
|
int err;
|
|
|
|
|
|
|
|
err = qlcnic_can_start_firmware(adapter);
|
|
|
|
if (err)
|
|
|
|
return err;
|
|
|
|
|
2010-08-19 13:08:28 +08:00
|
|
|
err = qlcnic_check_npar_opertional(adapter);
|
|
|
|
if (err)
|
|
|
|
return err;
|
2010-08-17 08:34:20 +08:00
|
|
|
|
2010-09-01 01:17:47 +08:00
|
|
|
err = qlcnic_initialize_nic(adapter);
|
|
|
|
if (err)
|
|
|
|
return err;
|
|
|
|
|
2010-06-03 15:50:56 +08:00
|
|
|
qlcnic_check_options(adapter);
|
|
|
|
|
2010-09-01 01:17:50 +08:00
|
|
|
err = qlcnic_set_eswitch_port_config(adapter);
|
|
|
|
if (err)
|
|
|
|
return err;
|
|
|
|
|
2010-06-03 15:50:56 +08:00
|
|
|
adapter->need_fw_reset = 0;
|
|
|
|
|
|
|
|
return err;
|
|
|
|
}
|
|
|
|
|
2011-04-28 19:48:18 +08:00
|
|
|
int qlcnic_validate_max_rss(struct net_device *netdev, u8 max_hw, u8 val)
|
|
|
|
{
|
|
|
|
if (!use_msi_x && !use_msi) {
|
|
|
|
netdev_info(netdev, "no msix or msi support, hence no rss\n");
|
|
|
|
return -EINVAL;
|
|
|
|
}
|
|
|
|
|
|
|
|
if ((val > max_hw) || (val < 2) || !is_power_of_2(val)) {
|
|
|
|
netdev_info(netdev, "rss_ring valid range [2 - %x] in "
|
|
|
|
" powers of 2\n", max_hw);
|
|
|
|
return -EINVAL;
|
|
|
|
}
|
|
|
|
return 0;
|
|
|
|
|
|
|
|
}
|
|
|
|
|
|
|
|
int qlcnic_set_max_rss(struct qlcnic_adapter *adapter, u8 data)
|
|
|
|
{
|
|
|
|
struct net_device *netdev = adapter->netdev;
|
|
|
|
int err = 0;
|
|
|
|
|
|
|
|
if (test_and_set_bit(__QLCNIC_RESETTING, &adapter->state))
|
|
|
|
return -EBUSY;
|
|
|
|
|
|
|
|
netif_device_detach(netdev);
|
|
|
|
if (netif_running(netdev))
|
|
|
|
__qlcnic_down(adapter, netdev);
|
|
|
|
qlcnic_detach(adapter);
|
|
|
|
qlcnic_teardown_intr(adapter);
|
|
|
|
|
|
|
|
if (qlcnic_enable_msix(adapter, data)) {
|
|
|
|
netdev_info(netdev, "failed setting max_rss; rss disabled\n");
|
|
|
|
qlcnic_enable_msi_legacy(adapter);
|
|
|
|
}
|
|
|
|
|
|
|
|
if (netif_running(netdev)) {
|
|
|
|
err = qlcnic_attach(adapter);
|
|
|
|
if (err)
|
|
|
|
goto done;
|
|
|
|
err = __qlcnic_up(adapter, netdev);
|
|
|
|
if (err)
|
|
|
|
goto done;
|
|
|
|
qlcnic_restore_indev_addr(netdev, NETDEV_UP);
|
|
|
|
}
|
|
|
|
done:
|
|
|
|
netif_device_attach(netdev);
|
|
|
|
clear_bit(__QLCNIC_RESETTING, &adapter->state);
|
|
|
|
return err;
|
|
|
|
}
|
|
|
|
|
2010-01-13 08:37:25 +08:00
|
|
|
#ifdef CONFIG_INET
|
|
|
|
|
|
|
|
#define is_qlcnic_netdev(dev) (dev->netdev_ops == &qlcnic_netdev_ops)
|
|
|
|
|
|
|
|
static void
|
2010-09-17 03:14:41 +08:00
|
|
|
qlcnic_config_indev_addr(struct qlcnic_adapter *adapter,
|
|
|
|
struct net_device *dev, unsigned long event)
|
2010-01-13 08:37:25 +08:00
|
|
|
{
|
|
|
|
struct in_device *indev;
|
|
|
|
|
|
|
|
indev = in_dev_get(dev);
|
|
|
|
if (!indev)
|
|
|
|
return;
|
|
|
|
|
|
|
|
for_ifa(indev) {
|
|
|
|
switch (event) {
|
|
|
|
case NETDEV_UP:
|
|
|
|
qlcnic_config_ipaddr(adapter,
|
|
|
|
ifa->ifa_address, QLCNIC_IP_UP);
|
|
|
|
break;
|
|
|
|
case NETDEV_DOWN:
|
|
|
|
qlcnic_config_ipaddr(adapter,
|
|
|
|
ifa->ifa_address, QLCNIC_IP_DOWN);
|
|
|
|
break;
|
|
|
|
default:
|
|
|
|
break;
|
|
|
|
}
|
|
|
|
} endfor_ifa(indev);
|
|
|
|
|
|
|
|
in_dev_put(indev);
|
|
|
|
}
|
|
|
|
|
2010-09-17 03:14:41 +08:00
|
|
|
static void
|
|
|
|
qlcnic_restore_indev_addr(struct net_device *netdev, unsigned long event)
|
|
|
|
{
|
|
|
|
struct qlcnic_adapter *adapter = netdev_priv(netdev);
|
|
|
|
struct net_device *dev;
|
|
|
|
u16 vid;
|
|
|
|
|
|
|
|
qlcnic_config_indev_addr(adapter, netdev, event);
|
|
|
|
|
2011-04-01 22:28:15 +08:00
|
|
|
for_each_set_bit(vid, adapter->vlans, VLAN_N_VID) {
|
2011-07-20 12:54:44 +08:00
|
|
|
dev = __vlan_find_dev_deep(netdev, vid);
|
2010-09-17 03:14:41 +08:00
|
|
|
if (!dev)
|
|
|
|
continue;
|
|
|
|
qlcnic_config_indev_addr(adapter, dev, event);
|
|
|
|
}
|
|
|
|
}
|
|
|
|
|
2010-01-13 08:37:25 +08:00
|
|
|
static int qlcnic_netdev_event(struct notifier_block *this,
|
|
|
|
unsigned long event, void *ptr)
|
|
|
|
{
|
|
|
|
struct qlcnic_adapter *adapter;
|
|
|
|
struct net_device *dev = (struct net_device *)ptr;
|
|
|
|
|
|
|
|
recheck:
|
|
|
|
if (dev == NULL)
|
|
|
|
goto done;
|
|
|
|
|
|
|
|
if (dev->priv_flags & IFF_802_1Q_VLAN) {
|
|
|
|
dev = vlan_dev_real_dev(dev);
|
|
|
|
goto recheck;
|
|
|
|
}
|
|
|
|
|
|
|
|
if (!is_qlcnic_netdev(dev))
|
|
|
|
goto done;
|
|
|
|
|
|
|
|
adapter = netdev_priv(dev);
|
|
|
|
|
|
|
|
if (!adapter)
|
|
|
|
goto done;
|
|
|
|
|
2010-06-22 11:19:01 +08:00
|
|
|
if (!test_bit(__QLCNIC_DEV_UP, &adapter->state))
|
2010-01-13 08:37:25 +08:00
|
|
|
goto done;
|
|
|
|
|
2010-09-17 03:14:41 +08:00
|
|
|
qlcnic_config_indev_addr(adapter, dev, event);
|
2010-01-13 08:37:25 +08:00
|
|
|
done:
|
|
|
|
return NOTIFY_DONE;
|
|
|
|
}
|
|
|
|
|
|
|
|
static int
|
|
|
|
qlcnic_inetaddr_event(struct notifier_block *this,
|
|
|
|
unsigned long event, void *ptr)
|
|
|
|
{
|
|
|
|
struct qlcnic_adapter *adapter;
|
|
|
|
struct net_device *dev;
|
|
|
|
|
|
|
|
struct in_ifaddr *ifa = (struct in_ifaddr *)ptr;
|
|
|
|
|
|
|
|
dev = ifa->ifa_dev ? ifa->ifa_dev->dev : NULL;
|
|
|
|
|
|
|
|
recheck:
|
2010-09-17 03:14:41 +08:00
|
|
|
if (dev == NULL)
|
2010-01-13 08:37:25 +08:00
|
|
|
goto done;
|
|
|
|
|
|
|
|
if (dev->priv_flags & IFF_802_1Q_VLAN) {
|
|
|
|
dev = vlan_dev_real_dev(dev);
|
|
|
|
goto recheck;
|
|
|
|
}
|
|
|
|
|
|
|
|
if (!is_qlcnic_netdev(dev))
|
|
|
|
goto done;
|
|
|
|
|
|
|
|
adapter = netdev_priv(dev);
|
|
|
|
|
2010-05-13 11:07:46 +08:00
|
|
|
if (!adapter)
|
2010-01-13 08:37:25 +08:00
|
|
|
goto done;
|
|
|
|
|
2010-06-22 11:19:01 +08:00
|
|
|
if (!test_bit(__QLCNIC_DEV_UP, &adapter->state))
|
2010-01-13 08:37:25 +08:00
|
|
|
goto done;
|
|
|
|
|
|
|
|
switch (event) {
|
|
|
|
case NETDEV_UP:
|
|
|
|
qlcnic_config_ipaddr(adapter, ifa->ifa_address, QLCNIC_IP_UP);
|
|
|
|
break;
|
|
|
|
case NETDEV_DOWN:
|
|
|
|
qlcnic_config_ipaddr(adapter, ifa->ifa_address, QLCNIC_IP_DOWN);
|
|
|
|
break;
|
|
|
|
default:
|
|
|
|
break;
|
|
|
|
}
|
|
|
|
|
|
|
|
done:
|
|
|
|
return NOTIFY_DONE;
|
|
|
|
}
|
|
|
|
|
|
|
|
static struct notifier_block qlcnic_netdev_cb = {
|
|
|
|
.notifier_call = qlcnic_netdev_event,
|
|
|
|
};
|
|
|
|
|
|
|
|
static struct notifier_block qlcnic_inetaddr_cb = {
|
|
|
|
.notifier_call = qlcnic_inetaddr_event,
|
|
|
|
};
|
|
|
|
#else
|
|
|
|
static void
|
2010-09-17 03:14:41 +08:00
|
|
|
qlcnic_restore_indev_addr(struct net_device *dev, unsigned long event)
|
2010-01-13 08:37:25 +08:00
|
|
|
{ }
|
|
|
|
#endif
|
2012-11-24 07:56:52 +08:00
|
|
|
static struct pci_error_handlers qlcnic_err_handler = {
|
2010-07-14 04:33:34 +08:00
|
|
|
.error_detected = qlcnic_io_error_detected,
|
|
|
|
.slot_reset = qlcnic_io_slot_reset,
|
|
|
|
.resume = qlcnic_io_resume,
|
|
|
|
};
|
2010-01-13 08:37:25 +08:00
|
|
|
|
|
|
|
static struct pci_driver qlcnic_driver = {
|
|
|
|
.name = qlcnic_driver_name,
|
|
|
|
.id_table = qlcnic_pci_tbl,
|
|
|
|
.probe = qlcnic_probe,
|
|
|
|
.remove = __devexit_p(qlcnic_remove),
|
|
|
|
#ifdef CONFIG_PM
|
|
|
|
.suspend = qlcnic_suspend,
|
|
|
|
.resume = qlcnic_resume,
|
|
|
|
#endif
|
2010-07-14 04:33:34 +08:00
|
|
|
.shutdown = qlcnic_shutdown,
|
|
|
|
.err_handler = &qlcnic_err_handler
|
|
|
|
|
2010-01-13 08:37:25 +08:00
|
|
|
};
|
|
|
|
|
|
|
|
static int __init qlcnic_init_module(void)
|
|
|
|
{
|
2010-07-14 04:33:33 +08:00
|
|
|
int ret;
|
2010-01-13 08:37:25 +08:00
|
|
|
|
|
|
|
printk(KERN_INFO "%s\n", qlcnic_driver_string);
|
|
|
|
|
2010-10-08 07:46:05 +08:00
|
|
|
qlcnic_wq = create_singlethread_workqueue("qlcnic");
|
|
|
|
if (qlcnic_wq == NULL) {
|
|
|
|
printk(KERN_ERR "qlcnic: cannot create workqueue\n");
|
|
|
|
return -ENOMEM;
|
|
|
|
}
|
|
|
|
|
2010-01-13 08:37:25 +08:00
|
|
|
#ifdef CONFIG_INET
|
|
|
|
register_netdevice_notifier(&qlcnic_netdev_cb);
|
|
|
|
register_inetaddr_notifier(&qlcnic_inetaddr_cb);
|
|
|
|
#endif
|
|
|
|
|
2010-07-14 04:33:33 +08:00
|
|
|
ret = pci_register_driver(&qlcnic_driver);
|
|
|
|
if (ret) {
|
|
|
|
#ifdef CONFIG_INET
|
|
|
|
unregister_inetaddr_notifier(&qlcnic_inetaddr_cb);
|
|
|
|
unregister_netdevice_notifier(&qlcnic_netdev_cb);
|
|
|
|
#endif
|
2010-10-08 07:46:05 +08:00
|
|
|
destroy_workqueue(qlcnic_wq);
|
2010-07-14 04:33:33 +08:00
|
|
|
}
|
2010-01-13 08:37:25 +08:00
|
|
|
|
2010-07-14 04:33:33 +08:00
|
|
|
return ret;
|
2010-01-13 08:37:25 +08:00
|
|
|
}
|
|
|
|
|
|
|
|
module_init(qlcnic_init_module);
|
|
|
|
|
|
|
|
static void __exit qlcnic_exit_module(void)
|
|
|
|
{
|
|
|
|
|
|
|
|
pci_unregister_driver(&qlcnic_driver);
|
|
|
|
|
|
|
|
#ifdef CONFIG_INET
|
|
|
|
unregister_inetaddr_notifier(&qlcnic_inetaddr_cb);
|
|
|
|
unregister_netdevice_notifier(&qlcnic_netdev_cb);
|
|
|
|
#endif
|
2010-10-08 07:46:05 +08:00
|
|
|
destroy_workqueue(qlcnic_wq);
|
2010-01-13 08:37:25 +08:00
|
|
|
}
|
|
|
|
|
|
|
|
module_exit(qlcnic_exit_module);
|