2006-05-23 18:10:15 +08:00
|
|
|
/*************************************************************************
|
|
|
|
* myri10ge.c: Myricom Myri-10G Ethernet driver.
|
|
|
|
*
|
2011-06-27 13:05:07 +08:00
|
|
|
* Copyright (C) 2005 - 2011 Myricom, Inc.
|
2006-05-23 18:10:15 +08:00
|
|
|
* All rights reserved.
|
|
|
|
*
|
|
|
|
* Redistribution and use in source and binary forms, with or without
|
|
|
|
* modification, are permitted provided that the following conditions
|
|
|
|
* are met:
|
|
|
|
* 1. Redistributions of source code must retain the above copyright
|
|
|
|
* notice, this list of conditions and the following disclaimer.
|
|
|
|
* 2. Redistributions in binary form must reproduce the above copyright
|
|
|
|
* notice, this list of conditions and the following disclaimer in the
|
|
|
|
* documentation and/or other materials provided with the distribution.
|
|
|
|
* 3. Neither the name of Myricom, Inc. nor the names of its contributors
|
|
|
|
* may be used to endorse or promote products derived from this software
|
|
|
|
* without specific prior written permission.
|
|
|
|
*
|
2007-02-28 00:18:40 +08:00
|
|
|
* THIS SOFTWARE IS PROVIDED BY THE COPYRIGHT HOLDERS AND CONTRIBUTORS "AS IS"
|
|
|
|
* AND ANY EXPRESS OR IMPLIED WARRANTIES, INCLUDING, BUT NOT LIMITED TO, THE
|
2006-05-23 18:10:15 +08:00
|
|
|
* IMPLIED WARRANTIES OF MERCHANTABILITY AND FITNESS FOR A PARTICULAR PURPOSE
|
2007-02-28 00:18:40 +08:00
|
|
|
* ARE DISCLAIMED. IN NO EVENT SHALL THE COPYRIGHT OWNER OR CONTRIBUTORS BE
|
|
|
|
* LIABLE FOR ANY DIRECT, INDIRECT, INCIDENTAL, SPECIAL, EXEMPLARY, OR
|
|
|
|
* CONSEQUENTIAL DAMAGES (INCLUDING, BUT NOT LIMITED TO, PROCUREMENT OF
|
|
|
|
* SUBSTITUTE GOODS OR SERVICES; LOSS OF USE, DATA, OR PROFITS; OR BUSINESS
|
|
|
|
* INTERRUPTION) HOWEVER CAUSED AND ON ANY THEORY OF LIABILITY, WHETHER IN
|
|
|
|
* CONTRACT, STRICT LIABILITY, OR TORT (INCLUDING NEGLIGENCE OR OTHERWISE)
|
|
|
|
* ARISING IN ANY WAY OUT OF THE USE OF THIS SOFTWARE, EVEN IF ADVISED OF THE
|
|
|
|
* POSSIBILITY OF SUCH DAMAGE.
|
2006-05-23 18:10:15 +08:00
|
|
|
*
|
|
|
|
*
|
|
|
|
* If the eeprom on your board is not recent enough, you will need to get a
|
|
|
|
* newer firmware image at:
|
|
|
|
* http://www.myri.com/scs/download-Myri10GE.html
|
|
|
|
*
|
|
|
|
* Contact Information:
|
|
|
|
* <help@myri.com>
|
|
|
|
* Myricom, Inc., 325N Santa Anita Avenue, Arcadia, CA 91006
|
|
|
|
*************************************************************************/
|
|
|
|
|
2010-02-23 00:56:58 +08:00
|
|
|
#define pr_fmt(fmt) KBUILD_MODNAME ": " fmt
|
|
|
|
|
2006-05-23 18:10:15 +08:00
|
|
|
#include <linux/tcp.h>
|
|
|
|
#include <linux/netdevice.h>
|
|
|
|
#include <linux/skbuff.h>
|
|
|
|
#include <linux/string.h>
|
|
|
|
#include <linux/module.h>
|
|
|
|
#include <linux/pci.h>
|
2006-06-08 22:25:00 +08:00
|
|
|
#include <linux/dma-mapping.h>
|
2006-05-23 18:10:15 +08:00
|
|
|
#include <linux/etherdevice.h>
|
|
|
|
#include <linux/if_ether.h>
|
|
|
|
#include <linux/if_vlan.h>
|
2008-05-09 08:22:16 +08:00
|
|
|
#include <linux/dca.h>
|
2006-05-23 18:10:15 +08:00
|
|
|
#include <linux/ip.h>
|
|
|
|
#include <linux/inet.h>
|
|
|
|
#include <linux/in.h>
|
|
|
|
#include <linux/ethtool.h>
|
|
|
|
#include <linux/firmware.h>
|
|
|
|
#include <linux/delay.h>
|
|
|
|
#include <linux/timer.h>
|
|
|
|
#include <linux/vmalloc.h>
|
|
|
|
#include <linux/crc32.h>
|
|
|
|
#include <linux/moduleparam.h>
|
|
|
|
#include <linux/io.h>
|
2007-07-10 02:50:22 +08:00
|
|
|
#include <linux/log2.h>
|
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>
|
2011-05-23 04:47:17 +08:00
|
|
|
#include <linux/prefetch.h>
|
2006-05-23 18:10:15 +08:00
|
|
|
#include <net/checksum.h>
|
2007-09-18 02:37:42 +08:00
|
|
|
#include <net/ip.h>
|
|
|
|
#include <net/tcp.h>
|
2006-05-23 18:10:15 +08:00
|
|
|
#include <asm/byteorder.h>
|
|
|
|
#include <asm/processor.h>
|
|
|
|
|
|
|
|
#include "myri10ge_mcp.h"
|
|
|
|
#include "myri10ge_mcp_gen_header.h"
|
|
|
|
|
2011-06-27 13:05:07 +08:00
|
|
|
#define MYRI10GE_VERSION_STR "1.5.3-1.534"
|
2006-05-23 18:10:15 +08:00
|
|
|
|
|
|
|
MODULE_DESCRIPTION("Myricom 10G driver (10GbE)");
|
|
|
|
MODULE_AUTHOR("Maintainer: help@myri.com");
|
|
|
|
MODULE_VERSION(MYRI10GE_VERSION_STR);
|
|
|
|
MODULE_LICENSE("Dual BSD/GPL");
|
|
|
|
|
|
|
|
#define MYRI10GE_MAX_ETHER_MTU 9014
|
|
|
|
|
|
|
|
#define MYRI10GE_ETH_STOPPED 0
|
|
|
|
#define MYRI10GE_ETH_STOPPING 1
|
|
|
|
#define MYRI10GE_ETH_STARTING 2
|
|
|
|
#define MYRI10GE_ETH_RUNNING 3
|
|
|
|
#define MYRI10GE_ETH_OPEN_FAILED 4
|
|
|
|
|
|
|
|
#define MYRI10GE_EEPROM_STRINGS_SIZE 256
|
|
|
|
#define MYRI10GE_MAX_SEND_DESC_TSO ((65536 / 2048) * 2)
|
|
|
|
|
2006-11-21 02:48:32 +08:00
|
|
|
#define MYRI10GE_NO_CONFIRM_DATA htonl(0xffffffff)
|
2006-05-23 18:10:15 +08:00
|
|
|
#define MYRI10GE_NO_RESPONSE_RESULT 0xffffffff
|
|
|
|
|
2006-12-11 18:25:09 +08:00
|
|
|
#define MYRI10GE_ALLOC_ORDER 0
|
|
|
|
#define MYRI10GE_ALLOC_SIZE ((1 << MYRI10GE_ALLOC_ORDER) * PAGE_SIZE)
|
|
|
|
#define MYRI10GE_MAX_FRAGS_PER_FRAME (MYRI10GE_MAX_ETHER_MTU/MYRI10GE_ALLOC_SIZE + 1)
|
|
|
|
|
2008-09-28 23:34:21 +08:00
|
|
|
#define MYRI10GE_MAX_SLICES 32
|
|
|
|
|
2006-05-23 18:10:15 +08:00
|
|
|
struct myri10ge_rx_buffer_state {
|
2006-12-11 18:25:09 +08:00
|
|
|
struct page *page;
|
|
|
|
int page_offset;
|
2010-04-12 22:32:10 +08:00
|
|
|
DEFINE_DMA_UNMAP_ADDR(bus);
|
|
|
|
DEFINE_DMA_UNMAP_LEN(len);
|
2006-05-23 18:10:15 +08:00
|
|
|
};
|
|
|
|
|
|
|
|
struct myri10ge_tx_buffer_state {
|
|
|
|
struct sk_buff *skb;
|
|
|
|
int last;
|
2010-04-12 22:32:10 +08:00
|
|
|
DEFINE_DMA_UNMAP_ADDR(bus);
|
|
|
|
DEFINE_DMA_UNMAP_LEN(len);
|
2006-05-23 18:10:15 +08:00
|
|
|
};
|
|
|
|
|
|
|
|
struct myri10ge_cmd {
|
|
|
|
u32 data0;
|
|
|
|
u32 data1;
|
|
|
|
u32 data2;
|
|
|
|
};
|
|
|
|
|
|
|
|
struct myri10ge_rx_buf {
|
|
|
|
struct mcp_kreq_ether_recv __iomem *lanai; /* lanai ptr for recv ring */
|
|
|
|
struct mcp_kreq_ether_recv *shadow; /* host shadow of recv ring */
|
|
|
|
struct myri10ge_rx_buffer_state *info;
|
2006-12-11 18:25:09 +08:00
|
|
|
struct page *page;
|
|
|
|
dma_addr_t bus;
|
|
|
|
int page_offset;
|
2006-05-23 18:10:15 +08:00
|
|
|
int cnt;
|
2006-12-11 18:25:09 +08:00
|
|
|
int fill_cnt;
|
2006-05-23 18:10:15 +08:00
|
|
|
int alloc_fail;
|
|
|
|
int mask; /* number of rx slots -1 */
|
2006-12-11 18:25:09 +08:00
|
|
|
int watchdog_needed;
|
2006-05-23 18:10:15 +08:00
|
|
|
};
|
|
|
|
|
|
|
|
struct myri10ge_tx_buf {
|
|
|
|
struct mcp_kreq_ether_send __iomem *lanai; /* lanai ptr for sendq */
|
2008-09-28 23:34:21 +08:00
|
|
|
__be32 __iomem *send_go; /* "go" doorbell ptr */
|
|
|
|
__be32 __iomem *send_stop; /* "stop" doorbell ptr */
|
2006-05-23 18:10:15 +08:00
|
|
|
struct mcp_kreq_ether_send *req_list; /* host shadow of sendq */
|
|
|
|
char *req_bytes;
|
|
|
|
struct myri10ge_tx_buffer_state *info;
|
|
|
|
int mask; /* number of transmit slots -1 */
|
|
|
|
int req ____cacheline_aligned; /* transmit slots submitted */
|
|
|
|
int pkt_start; /* packets started */
|
2008-05-09 08:20:03 +08:00
|
|
|
int stop_queue;
|
|
|
|
int linearized;
|
2006-05-23 18:10:15 +08:00
|
|
|
int done ____cacheline_aligned; /* transmit slots completed */
|
|
|
|
int pkt_done; /* packets completed */
|
2008-05-09 08:20:03 +08:00
|
|
|
int wake_queue;
|
2008-09-28 23:34:21 +08:00
|
|
|
int queue_active;
|
2006-05-23 18:10:15 +08:00
|
|
|
};
|
|
|
|
|
|
|
|
struct myri10ge_rx_done {
|
|
|
|
struct mcp_slot *entry;
|
|
|
|
dma_addr_t bus;
|
|
|
|
int cnt;
|
|
|
|
int idx;
|
|
|
|
};
|
|
|
|
|
2008-05-09 08:20:03 +08:00
|
|
|
struct myri10ge_slice_netstats {
|
|
|
|
unsigned long rx_packets;
|
|
|
|
unsigned long tx_packets;
|
|
|
|
unsigned long rx_bytes;
|
|
|
|
unsigned long tx_bytes;
|
|
|
|
unsigned long rx_dropped;
|
|
|
|
unsigned long tx_dropped;
|
|
|
|
};
|
|
|
|
|
|
|
|
struct myri10ge_slice_state {
|
2006-05-23 18:10:15 +08:00
|
|
|
struct myri10ge_tx_buf tx; /* transmit ring */
|
|
|
|
struct myri10ge_rx_buf rx_small;
|
|
|
|
struct myri10ge_rx_buf rx_big;
|
|
|
|
struct myri10ge_rx_done rx_done;
|
2008-05-09 08:20:03 +08:00
|
|
|
struct net_device *dev;
|
|
|
|
struct napi_struct napi;
|
|
|
|
struct myri10ge_priv *mgp;
|
|
|
|
struct myri10ge_slice_netstats stats;
|
|
|
|
__be32 __iomem *irq_claim;
|
|
|
|
struct mcp_irq_data *fw_stats;
|
|
|
|
dma_addr_t fw_stats_bus;
|
|
|
|
int watchdog_tx_done;
|
|
|
|
int watchdog_tx_req;
|
2009-08-07 18:44:22 +08:00
|
|
|
int watchdog_rx_done;
|
2011-06-28 01:57:28 +08:00
|
|
|
int stuck;
|
2008-10-16 17:09:31 +08:00
|
|
|
#ifdef CONFIG_MYRI10GE_DCA
|
2008-05-09 08:22:16 +08:00
|
|
|
int cached_dca_tag;
|
|
|
|
int cpu;
|
|
|
|
__be32 __iomem *dca_tag;
|
|
|
|
#endif
|
2008-05-09 08:21:49 +08:00
|
|
|
char irq_desc[32];
|
2008-05-09 08:20:03 +08:00
|
|
|
};
|
|
|
|
|
|
|
|
struct myri10ge_priv {
|
2008-05-09 08:21:49 +08:00
|
|
|
struct myri10ge_slice_state *ss;
|
2008-05-09 08:20:03 +08:00
|
|
|
int tx_boundary; /* boundary transmits cannot cross */
|
2008-05-09 08:21:49 +08:00
|
|
|
int num_slices;
|
2008-05-09 08:20:03 +08:00
|
|
|
int running; /* running? */
|
2006-05-23 18:10:15 +08:00
|
|
|
int small_bytes;
|
2006-12-11 18:25:09 +08:00
|
|
|
int big_bytes;
|
2008-05-09 08:20:25 +08:00
|
|
|
int max_intr_slots;
|
2006-05-23 18:10:15 +08:00
|
|
|
struct net_device *dev;
|
|
|
|
u8 __iomem *sram;
|
|
|
|
int sram_size;
|
|
|
|
unsigned long board_span;
|
|
|
|
unsigned long iomem_base;
|
2006-11-21 02:48:32 +08:00
|
|
|
__be32 __iomem *irq_deassert;
|
2006-05-23 18:10:15 +08:00
|
|
|
char *mac_addr_string;
|
|
|
|
struct mcp_cmd_response *cmd;
|
|
|
|
dma_addr_t cmd_bus;
|
|
|
|
struct pci_dev *pdev;
|
|
|
|
int msi_enabled;
|
2008-05-09 08:21:49 +08:00
|
|
|
int msix_enabled;
|
|
|
|
struct msix_entry *msix_vectors;
|
2008-10-16 17:09:31 +08:00
|
|
|
#ifdef CONFIG_MYRI10GE_DCA
|
2008-05-09 08:22:16 +08:00
|
|
|
int dca_enabled;
|
2010-09-28 16:13:12 +08:00
|
|
|
int relaxed_order;
|
2008-05-09 08:22:16 +08:00
|
|
|
#endif
|
2007-12-23 02:56:43 +08:00
|
|
|
u32 link_state;
|
2006-05-23 18:10:15 +08:00
|
|
|
unsigned int rdma_tags_available;
|
|
|
|
int intr_coal_delay;
|
2006-11-21 02:48:32 +08:00
|
|
|
__be32 __iomem *intr_coal_delay_ptr;
|
2015-04-22 04:09:45 +08:00
|
|
|
int wc_cookie;
|
2006-05-23 18:10:15 +08:00
|
|
|
int down_cnt;
|
|
|
|
wait_queue_head_t down_wq;
|
|
|
|
struct work_struct watchdog_work;
|
|
|
|
struct timer_list watchdog_timer;
|
|
|
|
int watchdog_resets;
|
2008-05-09 08:20:03 +08:00
|
|
|
int watchdog_pause;
|
2006-05-23 18:10:15 +08:00
|
|
|
int pause;
|
2010-08-12 13:04:31 +08:00
|
|
|
bool fw_name_allocated;
|
2006-05-23 18:10:15 +08:00
|
|
|
char *fw_name;
|
|
|
|
char eeprom_strings[MYRI10GE_EEPROM_STRINGS_SIZE];
|
2008-05-09 08:18:24 +08:00
|
|
|
char *product_code_string;
|
2006-05-23 18:10:15 +08:00
|
|
|
char fw_version[128];
|
2007-02-22 01:05:17 +08:00
|
|
|
int fw_ver_major;
|
|
|
|
int fw_ver_minor;
|
|
|
|
int fw_ver_tiny;
|
|
|
|
int adopted_rx_filter_bug;
|
2013-08-02 07:17:49 +08:00
|
|
|
u8 mac_addr[ETH_ALEN]; /* eeprom mac address */
|
2006-05-23 18:10:15 +08:00
|
|
|
unsigned long serial_number;
|
|
|
|
int vendor_specific_offset;
|
2006-08-22 05:36:56 +08:00
|
|
|
int fw_multicast_support;
|
2011-01-25 07:32:47 +08:00
|
|
|
u32 features;
|
2007-10-13 18:34:01 +08:00
|
|
|
u32 max_tso6;
|
2006-05-23 18:10:15 +08:00
|
|
|
u32 read_dma;
|
|
|
|
u32 write_dma;
|
|
|
|
u32 read_write_dma;
|
2006-08-22 05:36:49 +08:00
|
|
|
u32 link_changes;
|
|
|
|
u32 msg_enable;
|
2009-04-16 10:24:59 +08:00
|
|
|
unsigned int board_number;
|
2009-08-07 18:44:22 +08:00
|
|
|
int rebooted;
|
2006-05-23 18:10:15 +08:00
|
|
|
};
|
|
|
|
|
|
|
|
static char *myri10ge_fw_unaligned = "myri10ge_ethp_z8e.dat";
|
|
|
|
static char *myri10ge_fw_aligned = "myri10ge_eth_z8e.dat";
|
2008-05-09 08:21:49 +08:00
|
|
|
static char *myri10ge_fw_rss_unaligned = "myri10ge_rss_ethp_z8e.dat";
|
|
|
|
static char *myri10ge_fw_rss_aligned = "myri10ge_rss_eth_z8e.dat";
|
2009-11-07 19:54:44 +08:00
|
|
|
MODULE_FIRMWARE("myri10ge_ethp_z8e.dat");
|
|
|
|
MODULE_FIRMWARE("myri10ge_eth_z8e.dat");
|
|
|
|
MODULE_FIRMWARE("myri10ge_rss_ethp_z8e.dat");
|
|
|
|
MODULE_FIRMWARE("myri10ge_rss_eth_z8e.dat");
|
2006-05-23 18:10:15 +08:00
|
|
|
|
module: add per-module param_lock
Add a "param_lock" mutex to each module, and update params.c to use
the correct built-in or module mutex while locking kernel params.
Remove the kparam_block_sysfs_r/w() macros, replace them with direct
calls to kernel_param_[un]lock(module).
The kernel param code currently uses a single mutex to protect
modification of any and all kernel params. While this generally works,
there is one specific problem with it; a module callback function
cannot safely load another module, i.e. with request_module() or even
with indirect calls such as crypto_has_alg(). If the module to be
loaded has any of its params configured (e.g. with a /etc/modprobe.d/*
config file), then the attempt will result in a deadlock between the
first module param callback waiting for modprobe, and modprobe trying to
lock the single kernel param mutex to set the new module's param.
This fixes that by using per-module mutexes, so that each individual module
is protected against concurrent changes in its own kernel params, but is
not blocked by changes to other module params. All built-in modules
continue to use the built-in mutex, since they will always be loaded at
runtime and references (e.g. request_module(), crypto_has_alg()) to them
will never cause load-time param changing.
This also simplifies the interface used by modules to block sysfs access
to their params; while there are currently functions to block and unblock
sysfs param access which are split up by read and write and expect a single
kernel param to be passed, their actual operation is identical and applies
to all params, not just the one passed to them; they simply lock and unlock
the global param mutex. They are replaced with direct calls to
kernel_param_[un]lock(THIS_MODULE), which locks THIS_MODULE's param_lock, or
if the module is built-in, it locks the built-in mutex.
Suggested-by: Rusty Russell <rusty@rustcorp.com.au>
Signed-off-by: Dan Streetman <ddstreet@ieee.org>
Signed-off-by: Rusty Russell <rusty@rustcorp.com.au>
2015-06-17 04:48:52 +08:00
|
|
|
/* Careful: must be accessed under kernel_param_lock() */
|
2006-05-23 18:10:15 +08:00
|
|
|
static char *myri10ge_fw_name = NULL;
|
2018-03-24 07:34:44 +08:00
|
|
|
module_param(myri10ge_fw_name, charp, 0644);
|
2008-05-09 08:16:53 +08:00
|
|
|
MODULE_PARM_DESC(myri10ge_fw_name, "Firmware image name");
|
2006-05-23 18:10:15 +08:00
|
|
|
|
2009-04-16 10:24:59 +08:00
|
|
|
#define MYRI10GE_MAX_BOARDS 8
|
|
|
|
static char *myri10ge_fw_names[MYRI10GE_MAX_BOARDS] =
|
2009-04-18 06:45:15 +08:00
|
|
|
{[0 ... (MYRI10GE_MAX_BOARDS - 1)] = NULL };
|
2009-04-16 10:24:59 +08:00
|
|
|
module_param_array_named(myri10ge_fw_names, myri10ge_fw_names, charp, NULL,
|
|
|
|
0444);
|
2016-10-21 01:01:56 +08:00
|
|
|
MODULE_PARM_DESC(myri10ge_fw_names, "Firmware image names per board");
|
2009-04-16 10:24:59 +08:00
|
|
|
|
2006-05-23 18:10:15 +08:00
|
|
|
static int myri10ge_ecrc_enable = 1;
|
2018-03-24 07:34:44 +08:00
|
|
|
module_param(myri10ge_ecrc_enable, int, 0444);
|
2008-05-09 08:16:53 +08:00
|
|
|
MODULE_PARM_DESC(myri10ge_ecrc_enable, "Enable Extended CRC on PCI-E");
|
2006-05-23 18:10:15 +08:00
|
|
|
|
|
|
|
static int myri10ge_small_bytes = -1; /* -1 == auto */
|
2018-03-24 07:34:44 +08:00
|
|
|
module_param(myri10ge_small_bytes, int, 0644);
|
2008-05-09 08:16:53 +08:00
|
|
|
MODULE_PARM_DESC(myri10ge_small_bytes, "Threshold of small packets");
|
2006-05-23 18:10:15 +08:00
|
|
|
|
|
|
|
static int myri10ge_msi = 1; /* enable msi by default */
|
2018-03-24 07:34:44 +08:00
|
|
|
module_param(myri10ge_msi, int, 0644);
|
2008-05-09 08:16:53 +08:00
|
|
|
MODULE_PARM_DESC(myri10ge_msi, "Enable Message Signalled Interrupts");
|
2006-05-23 18:10:15 +08:00
|
|
|
|
2007-03-22 02:45:56 +08:00
|
|
|
static int myri10ge_intr_coal_delay = 75;
|
2018-03-24 07:34:44 +08:00
|
|
|
module_param(myri10ge_intr_coal_delay, int, 0444);
|
2008-05-09 08:16:53 +08:00
|
|
|
MODULE_PARM_DESC(myri10ge_intr_coal_delay, "Interrupt coalescing delay");
|
2006-05-23 18:10:15 +08:00
|
|
|
|
|
|
|
static int myri10ge_flow_control = 1;
|
2018-03-24 07:34:44 +08:00
|
|
|
module_param(myri10ge_flow_control, int, 0444);
|
2008-05-09 08:16:53 +08:00
|
|
|
MODULE_PARM_DESC(myri10ge_flow_control, "Pause parameter");
|
2006-05-23 18:10:15 +08:00
|
|
|
|
|
|
|
static int myri10ge_deassert_wait = 1;
|
2018-03-24 07:34:44 +08:00
|
|
|
module_param(myri10ge_deassert_wait, int, 0644);
|
2006-05-23 18:10:15 +08:00
|
|
|
MODULE_PARM_DESC(myri10ge_deassert_wait,
|
2008-05-09 08:16:53 +08:00
|
|
|
"Wait when deasserting legacy interrupts");
|
2006-05-23 18:10:15 +08:00
|
|
|
|
|
|
|
static int myri10ge_force_firmware = 0;
|
2018-03-24 07:34:44 +08:00
|
|
|
module_param(myri10ge_force_firmware, int, 0444);
|
2006-05-23 18:10:15 +08:00
|
|
|
MODULE_PARM_DESC(myri10ge_force_firmware,
|
2008-05-09 08:16:53 +08:00
|
|
|
"Force firmware to assume aligned completions");
|
2006-05-23 18:10:15 +08:00
|
|
|
|
|
|
|
static int myri10ge_initial_mtu = MYRI10GE_MAX_ETHER_MTU - ETH_HLEN;
|
2018-03-24 07:34:44 +08:00
|
|
|
module_param(myri10ge_initial_mtu, int, 0444);
|
2008-05-09 08:16:53 +08:00
|
|
|
MODULE_PARM_DESC(myri10ge_initial_mtu, "Initial MTU");
|
2006-05-23 18:10:15 +08:00
|
|
|
|
|
|
|
static int myri10ge_napi_weight = 64;
|
2018-03-24 07:34:44 +08:00
|
|
|
module_param(myri10ge_napi_weight, int, 0444);
|
2008-05-09 08:16:53 +08:00
|
|
|
MODULE_PARM_DESC(myri10ge_napi_weight, "Set NAPI weight");
|
2006-05-23 18:10:15 +08:00
|
|
|
|
|
|
|
static int myri10ge_watchdog_timeout = 1;
|
2018-03-24 07:34:44 +08:00
|
|
|
module_param(myri10ge_watchdog_timeout, int, 0444);
|
2008-05-09 08:16:53 +08:00
|
|
|
MODULE_PARM_DESC(myri10ge_watchdog_timeout, "Set watchdog timeout");
|
2006-05-23 18:10:15 +08:00
|
|
|
|
|
|
|
static int myri10ge_max_irq_loops = 1048576;
|
2018-03-24 07:34:44 +08:00
|
|
|
module_param(myri10ge_max_irq_loops, int, 0444);
|
2006-05-23 18:10:15 +08:00
|
|
|
MODULE_PARM_DESC(myri10ge_max_irq_loops,
|
2008-05-09 08:16:53 +08:00
|
|
|
"Set stuck legacy IRQ detection threshold");
|
2006-05-23 18:10:15 +08:00
|
|
|
|
2006-08-22 05:36:49 +08:00
|
|
|
#define MYRI10GE_MSG_DEFAULT NETIF_MSG_LINK
|
|
|
|
|
|
|
|
static int myri10ge_debug = -1; /* defaults above */
|
|
|
|
module_param(myri10ge_debug, int, 0);
|
|
|
|
MODULE_PARM_DESC(myri10ge_debug, "Debug level (0=none,...,16=all)");
|
|
|
|
|
2006-12-11 18:25:09 +08:00
|
|
|
static int myri10ge_fill_thresh = 256;
|
2018-03-24 07:34:44 +08:00
|
|
|
module_param(myri10ge_fill_thresh, int, 0644);
|
2008-05-09 08:16:53 +08:00
|
|
|
MODULE_PARM_DESC(myri10ge_fill_thresh, "Number of empty rx slots allowed");
|
2006-12-11 18:25:09 +08:00
|
|
|
|
2007-06-12 02:26:31 +08:00
|
|
|
static int myri10ge_reset_recover = 1;
|
|
|
|
|
2008-05-09 08:21:49 +08:00
|
|
|
static int myri10ge_max_slices = 1;
|
2018-03-24 07:34:44 +08:00
|
|
|
module_param(myri10ge_max_slices, int, 0444);
|
2008-05-09 08:21:49 +08:00
|
|
|
MODULE_PARM_DESC(myri10ge_max_slices, "Max tx/rx queues");
|
|
|
|
|
2009-12-09 12:24:35 +08:00
|
|
|
static int myri10ge_rss_hash = MXGEFW_RSS_HASH_TYPE_SRC_DST_PORT;
|
2018-03-24 07:34:44 +08:00
|
|
|
module_param(myri10ge_rss_hash, int, 0444);
|
2008-05-09 08:21:49 +08:00
|
|
|
MODULE_PARM_DESC(myri10ge_rss_hash, "Type of RSS hashing to do");
|
|
|
|
|
2008-05-09 08:22:16 +08:00
|
|
|
static int myri10ge_dca = 1;
|
2018-03-24 07:34:44 +08:00
|
|
|
module_param(myri10ge_dca, int, 0444);
|
2008-05-09 08:22:16 +08:00
|
|
|
MODULE_PARM_DESC(myri10ge_dca, "Enable DCA if possible");
|
|
|
|
|
2006-05-23 18:10:15 +08:00
|
|
|
#define MYRI10GE_FW_OFFSET 1024*1024
|
|
|
|
#define MYRI10GE_HIGHPART_TO_U32(X) \
|
|
|
|
(sizeof (X) == 8) ? ((u32)((u64)(X) >> 32)) : (0)
|
|
|
|
#define MYRI10GE_LOWPART_TO_U32(X) ((u32)(X))
|
|
|
|
|
|
|
|
#define myri10ge_pio_copy(to,from,size) __iowrite64_copy(to,from,size/8)
|
|
|
|
|
2007-05-08 05:50:37 +08:00
|
|
|
static void myri10ge_set_multicast_list(struct net_device *dev);
|
2009-09-01 03:50:58 +08:00
|
|
|
static netdev_tx_t myri10ge_sw_tso(struct sk_buff *skb,
|
|
|
|
struct net_device *dev);
|
2007-05-08 05:50:37 +08:00
|
|
|
|
2006-12-11 18:24:37 +08:00
|
|
|
static inline void put_be32(__be32 val, __be32 __iomem * p)
|
2006-11-21 02:48:32 +08:00
|
|
|
{
|
2006-12-11 18:24:37 +08:00
|
|
|
__raw_writel((__force __u32) val, (__force void __iomem *)p);
|
2006-11-21 02:48:32 +08:00
|
|
|
}
|
|
|
|
|
2017-01-07 11:12:52 +08:00
|
|
|
static void myri10ge_get_stats(struct net_device *dev,
|
|
|
|
struct rtnl_link_stats64 *stats);
|
2009-04-16 10:23:56 +08:00
|
|
|
|
2010-08-12 13:04:31 +08:00
|
|
|
static void set_fw_name(struct myri10ge_priv *mgp, char *name, bool allocated)
|
|
|
|
{
|
|
|
|
if (mgp->fw_name_allocated)
|
|
|
|
kfree(mgp->fw_name);
|
|
|
|
mgp->fw_name = name;
|
|
|
|
mgp->fw_name_allocated = allocated;
|
|
|
|
}
|
|
|
|
|
2006-05-23 18:10:15 +08:00
|
|
|
static int
|
|
|
|
myri10ge_send_cmd(struct myri10ge_priv *mgp, u32 cmd,
|
|
|
|
struct myri10ge_cmd *data, int atomic)
|
|
|
|
{
|
|
|
|
struct mcp_cmd *buf;
|
|
|
|
char buf_bytes[sizeof(*buf) + 8];
|
|
|
|
struct mcp_cmd_response *response = mgp->cmd;
|
2006-08-15 05:52:54 +08:00
|
|
|
char __iomem *cmd_addr = mgp->sram + MXGEFW_ETH_CMD;
|
2006-05-23 18:10:15 +08:00
|
|
|
u32 dma_low, dma_high, result, value;
|
|
|
|
int sleep_total = 0;
|
|
|
|
|
|
|
|
/* ensure buf is aligned to 8 bytes */
|
|
|
|
buf = (struct mcp_cmd *)ALIGN((unsigned long)buf_bytes, 8);
|
|
|
|
|
|
|
|
buf->data0 = htonl(data->data0);
|
|
|
|
buf->data1 = htonl(data->data1);
|
|
|
|
buf->data2 = htonl(data->data2);
|
|
|
|
buf->cmd = htonl(cmd);
|
|
|
|
dma_low = MYRI10GE_LOWPART_TO_U32(mgp->cmd_bus);
|
|
|
|
dma_high = MYRI10GE_HIGHPART_TO_U32(mgp->cmd_bus);
|
|
|
|
|
|
|
|
buf->response_addr.low = htonl(dma_low);
|
|
|
|
buf->response_addr.high = htonl(dma_high);
|
2006-11-21 02:48:32 +08:00
|
|
|
response->result = htonl(MYRI10GE_NO_RESPONSE_RESULT);
|
2006-05-23 18:10:15 +08:00
|
|
|
mb();
|
|
|
|
myri10ge_pio_copy(cmd_addr, buf, sizeof(*buf));
|
|
|
|
|
|
|
|
/* wait up to 15ms. Longest command is the DMA benchmark,
|
|
|
|
* which is capped at 5ms, but runs from a timeout handler
|
|
|
|
* that runs every 7.8ms. So a 15ms timeout leaves us with
|
|
|
|
* a 2.2ms margin
|
|
|
|
*/
|
|
|
|
if (atomic) {
|
|
|
|
/* if atomic is set, do not sleep,
|
|
|
|
* and try to get the completion quickly
|
|
|
|
* (1ms will be enough for those commands) */
|
|
|
|
for (sleep_total = 0;
|
2009-12-03 15:58:21 +08:00
|
|
|
sleep_total < 1000 &&
|
|
|
|
response->result == htonl(MYRI10GE_NO_RESPONSE_RESULT);
|
2008-05-09 08:18:45 +08:00
|
|
|
sleep_total += 10) {
|
2006-05-23 18:10:15 +08:00
|
|
|
udelay(10);
|
2008-05-09 08:18:45 +08:00
|
|
|
mb();
|
|
|
|
}
|
2006-05-23 18:10:15 +08:00
|
|
|
} else {
|
|
|
|
/* use msleep for most command */
|
|
|
|
for (sleep_total = 0;
|
2009-12-03 15:58:21 +08:00
|
|
|
sleep_total < 15 &&
|
|
|
|
response->result == htonl(MYRI10GE_NO_RESPONSE_RESULT);
|
2006-05-23 18:10:15 +08:00
|
|
|
sleep_total++)
|
|
|
|
msleep(1);
|
|
|
|
}
|
|
|
|
|
|
|
|
result = ntohl(response->result);
|
|
|
|
value = ntohl(response->data);
|
|
|
|
if (result != MYRI10GE_NO_RESPONSE_RESULT) {
|
|
|
|
if (result == 0) {
|
|
|
|
data->data0 = value;
|
|
|
|
return 0;
|
2006-08-22 05:36:56 +08:00
|
|
|
} else if (result == MXGEFW_CMD_UNKNOWN) {
|
|
|
|
return -ENOSYS;
|
2007-05-08 05:52:22 +08:00
|
|
|
} else if (result == MXGEFW_CMD_ERROR_UNALIGNED) {
|
|
|
|
return -E2BIG;
|
2008-09-28 23:34:21 +08:00
|
|
|
} else if (result == MXGEFW_CMD_ERROR_RANGE &&
|
|
|
|
cmd == MXGEFW_CMD_ENABLE_RSS_QUEUES &&
|
|
|
|
(data->
|
|
|
|
data1 & MXGEFW_SLICE_ENABLE_MULTIPLE_TX_QUEUES) !=
|
|
|
|
0) {
|
|
|
|
return -ERANGE;
|
2006-05-23 18:10:15 +08:00
|
|
|
} else {
|
|
|
|
dev_err(&mgp->pdev->dev,
|
|
|
|
"command %d failed, result = %d\n",
|
|
|
|
cmd, result);
|
|
|
|
return -ENXIO;
|
|
|
|
}
|
|
|
|
}
|
|
|
|
|
|
|
|
dev_err(&mgp->pdev->dev, "command %d timed out, result = %d\n",
|
|
|
|
cmd, result);
|
|
|
|
return -EAGAIN;
|
|
|
|
}
|
|
|
|
|
|
|
|
/*
|
|
|
|
* The eeprom strings on the lanaiX have the format
|
|
|
|
* SN=x\0
|
|
|
|
* MAC=x:x:x:x:x:x\0
|
|
|
|
* PT:ddd mmm xx xx:xx:xx xx\0
|
|
|
|
* PV:ddd mmm xx xx:xx:xx xx\0
|
|
|
|
*/
|
|
|
|
static int myri10ge_read_mac_addr(struct myri10ge_priv *mgp)
|
|
|
|
{
|
|
|
|
char *ptr, *limit;
|
|
|
|
int i;
|
|
|
|
|
|
|
|
ptr = mgp->eeprom_strings;
|
|
|
|
limit = mgp->eeprom_strings + MYRI10GE_EEPROM_STRINGS_SIZE;
|
|
|
|
|
|
|
|
while (*ptr != '\0' && ptr < limit) {
|
|
|
|
if (memcmp(ptr, "MAC=", 4) == 0) {
|
|
|
|
ptr += 4;
|
|
|
|
mgp->mac_addr_string = ptr;
|
|
|
|
for (i = 0; i < 6; i++) {
|
|
|
|
if ((ptr + 2) > limit)
|
|
|
|
goto abort;
|
|
|
|
mgp->mac_addr[i] =
|
|
|
|
simple_strtoul(ptr, &ptr, 16);
|
|
|
|
ptr += 1;
|
|
|
|
}
|
|
|
|
}
|
2008-05-09 08:18:24 +08:00
|
|
|
if (memcmp(ptr, "PC=", 3) == 0) {
|
|
|
|
ptr += 3;
|
|
|
|
mgp->product_code_string = ptr;
|
|
|
|
}
|
2006-05-23 18:10:15 +08:00
|
|
|
if (memcmp((const void *)ptr, "SN=", 3) == 0) {
|
|
|
|
ptr += 3;
|
|
|
|
mgp->serial_number = simple_strtoul(ptr, &ptr, 10);
|
|
|
|
}
|
|
|
|
while (ptr < limit && *ptr++) ;
|
|
|
|
}
|
|
|
|
|
|
|
|
return 0;
|
|
|
|
|
|
|
|
abort:
|
|
|
|
dev_err(&mgp->pdev->dev, "failed to parse eeprom_strings\n");
|
|
|
|
return -ENXIO;
|
|
|
|
}
|
|
|
|
|
|
|
|
/*
|
|
|
|
* Enable or disable periodic RDMAs from the host to make certain
|
|
|
|
* chipsets resend dropped PCIe messages
|
|
|
|
*/
|
|
|
|
|
|
|
|
static void myri10ge_dummy_rdma(struct myri10ge_priv *mgp, int enable)
|
|
|
|
{
|
|
|
|
char __iomem *submit;
|
2008-05-09 08:17:37 +08:00
|
|
|
__be32 buf[16] __attribute__ ((__aligned__(8)));
|
2006-05-23 18:10:15 +08:00
|
|
|
u32 dma_low, dma_high;
|
|
|
|
int i;
|
|
|
|
|
|
|
|
/* clear confirmation addr */
|
|
|
|
mgp->cmd->data = 0;
|
|
|
|
mb();
|
|
|
|
|
|
|
|
/* send a rdma command to the PCIe engine, and wait for the
|
|
|
|
* response in the confirmation address. The firmware should
|
|
|
|
* write a -1 there to indicate it is alive and well
|
|
|
|
*/
|
|
|
|
dma_low = MYRI10GE_LOWPART_TO_U32(mgp->cmd_bus);
|
|
|
|
dma_high = MYRI10GE_HIGHPART_TO_U32(mgp->cmd_bus);
|
|
|
|
|
|
|
|
buf[0] = htonl(dma_high); /* confirm addr MSW */
|
|
|
|
buf[1] = htonl(dma_low); /* confirm addr LSW */
|
2006-11-21 02:48:32 +08:00
|
|
|
buf[2] = MYRI10GE_NO_CONFIRM_DATA; /* confirm data */
|
2006-05-23 18:10:15 +08:00
|
|
|
buf[3] = htonl(dma_high); /* dummy addr MSW */
|
|
|
|
buf[4] = htonl(dma_low); /* dummy addr LSW */
|
|
|
|
buf[5] = htonl(enable); /* enable? */
|
|
|
|
|
2006-08-15 05:52:54 +08:00
|
|
|
submit = mgp->sram + MXGEFW_BOOT_DUMMY_RDMA;
|
2006-05-23 18:10:15 +08:00
|
|
|
|
|
|
|
myri10ge_pio_copy(submit, &buf, sizeof(buf));
|
|
|
|
for (i = 0; mgp->cmd->data != MYRI10GE_NO_CONFIRM_DATA && i < 20; i++)
|
|
|
|
msleep(1);
|
|
|
|
if (mgp->cmd->data != MYRI10GE_NO_CONFIRM_DATA)
|
|
|
|
dev_err(&mgp->pdev->dev, "dummy rdma %s failed\n",
|
|
|
|
(enable ? "enable" : "disable"));
|
|
|
|
}
|
|
|
|
|
|
|
|
static int
|
|
|
|
myri10ge_validate_firmware(struct myri10ge_priv *mgp,
|
|
|
|
struct mcp_gen_header *hdr)
|
|
|
|
{
|
|
|
|
struct device *dev = &mgp->pdev->dev;
|
|
|
|
|
|
|
|
/* check firmware type */
|
|
|
|
if (ntohl(hdr->mcp_type) != MCP_TYPE_ETH) {
|
|
|
|
dev_err(dev, "Bad firmware type: 0x%x\n", ntohl(hdr->mcp_type));
|
|
|
|
return -EINVAL;
|
|
|
|
}
|
|
|
|
|
|
|
|
/* save firmware version for ethtool */
|
|
|
|
strncpy(mgp->fw_version, hdr->version, sizeof(mgp->fw_version));
|
2014-08-12 03:18:16 +08:00
|
|
|
mgp->fw_version[sizeof(mgp->fw_version) - 1] = '\0';
|
2006-05-23 18:10:15 +08:00
|
|
|
|
2007-02-22 01:05:17 +08:00
|
|
|
sscanf(mgp->fw_version, "%d.%d.%d", &mgp->fw_ver_major,
|
|
|
|
&mgp->fw_ver_minor, &mgp->fw_ver_tiny);
|
2006-05-23 18:10:15 +08:00
|
|
|
|
2009-12-03 15:58:21 +08:00
|
|
|
if (!(mgp->fw_ver_major == MXGEFW_VERSION_MAJOR &&
|
|
|
|
mgp->fw_ver_minor == MXGEFW_VERSION_MINOR)) {
|
2006-05-23 18:10:15 +08:00
|
|
|
dev_err(dev, "Found firmware version %s\n", mgp->fw_version);
|
|
|
|
dev_err(dev, "Driver needs %d.%d\n", MXGEFW_VERSION_MAJOR,
|
|
|
|
MXGEFW_VERSION_MINOR);
|
|
|
|
return -EINVAL;
|
|
|
|
}
|
|
|
|
return 0;
|
|
|
|
}
|
|
|
|
|
|
|
|
static int myri10ge_load_hotplug_firmware(struct myri10ge_priv *mgp, u32 * size)
|
|
|
|
{
|
|
|
|
unsigned crc, reread_crc;
|
|
|
|
const struct firmware *fw;
|
|
|
|
struct device *dev = &mgp->pdev->dev;
|
2008-05-24 07:00:07 +08:00
|
|
|
unsigned char *fw_readback;
|
2006-05-23 18:10:15 +08:00
|
|
|
struct mcp_gen_header *hdr;
|
|
|
|
size_t hdr_offset;
|
|
|
|
int status;
|
2006-07-30 12:14:09 +08:00
|
|
|
unsigned i;
|
2006-05-23 18:10:15 +08:00
|
|
|
|
|
|
|
if ((status = request_firmware(&fw, mgp->fw_name, dev)) < 0) {
|
|
|
|
dev_err(dev, "Unable to load %s firmware image via hotplug\n",
|
|
|
|
mgp->fw_name);
|
|
|
|
status = -EINVAL;
|
|
|
|
goto abort_with_nothing;
|
|
|
|
}
|
|
|
|
|
|
|
|
/* check size */
|
|
|
|
|
|
|
|
if (fw->size >= mgp->sram_size - MYRI10GE_FW_OFFSET ||
|
|
|
|
fw->size < MCP_HEADER_PTR_OFFSET + 4) {
|
|
|
|
dev_err(dev, "Firmware size invalid:%d\n", (int)fw->size);
|
|
|
|
status = -EINVAL;
|
|
|
|
goto abort_with_fw;
|
|
|
|
}
|
|
|
|
|
|
|
|
/* check id */
|
2006-11-21 02:48:32 +08:00
|
|
|
hdr_offset = ntohl(*(__be32 *) (fw->data + MCP_HEADER_PTR_OFFSET));
|
2006-05-23 18:10:15 +08:00
|
|
|
if ((hdr_offset & 3) || hdr_offset + sizeof(*hdr) > fw->size) {
|
|
|
|
dev_err(dev, "Bad firmware file\n");
|
|
|
|
status = -EINVAL;
|
|
|
|
goto abort_with_fw;
|
|
|
|
}
|
|
|
|
hdr = (void *)(fw->data + hdr_offset);
|
|
|
|
|
|
|
|
status = myri10ge_validate_firmware(mgp, hdr);
|
|
|
|
if (status != 0)
|
|
|
|
goto abort_with_fw;
|
|
|
|
|
|
|
|
crc = crc32(~0, fw->data, fw->size);
|
2006-07-30 12:14:09 +08:00
|
|
|
for (i = 0; i < fw->size; i += 256) {
|
|
|
|
myri10ge_pio_copy(mgp->sram + MYRI10GE_FW_OFFSET + i,
|
|
|
|
fw->data + i,
|
|
|
|
min(256U, (unsigned)(fw->size - i)));
|
|
|
|
mb();
|
|
|
|
readb(mgp->sram);
|
2006-06-08 22:25:00 +08:00
|
|
|
}
|
2008-05-24 07:00:07 +08:00
|
|
|
fw_readback = vmalloc(fw->size);
|
|
|
|
if (!fw_readback) {
|
|
|
|
status = -ENOMEM;
|
|
|
|
goto abort_with_fw;
|
|
|
|
}
|
2006-05-23 18:10:15 +08:00
|
|
|
/* corruption checking is good for parity recovery and buggy chipset */
|
2008-05-24 07:00:07 +08:00
|
|
|
memcpy_fromio(fw_readback, mgp->sram + MYRI10GE_FW_OFFSET, fw->size);
|
|
|
|
reread_crc = crc32(~0, fw_readback, fw->size);
|
|
|
|
vfree(fw_readback);
|
2006-05-23 18:10:15 +08:00
|
|
|
if (crc != reread_crc) {
|
|
|
|
dev_err(dev, "CRC failed(fw-len=%u), got 0x%x (expect 0x%x)\n",
|
|
|
|
(unsigned)fw->size, reread_crc, crc);
|
|
|
|
status = -EIO;
|
|
|
|
goto abort_with_fw;
|
|
|
|
}
|
|
|
|
*size = (u32) fw->size;
|
|
|
|
|
|
|
|
abort_with_fw:
|
|
|
|
release_firmware(fw);
|
|
|
|
|
|
|
|
abort_with_nothing:
|
|
|
|
return status;
|
|
|
|
}
|
|
|
|
|
|
|
|
static int myri10ge_adopt_running_firmware(struct myri10ge_priv *mgp)
|
|
|
|
{
|
|
|
|
struct mcp_gen_header *hdr;
|
|
|
|
struct device *dev = &mgp->pdev->dev;
|
|
|
|
const size_t bytes = sizeof(struct mcp_gen_header);
|
|
|
|
size_t hdr_offset;
|
|
|
|
int status;
|
|
|
|
|
|
|
|
/* find running firmware header */
|
2007-12-23 02:56:43 +08:00
|
|
|
hdr_offset = swab32(readl(mgp->sram + MCP_HEADER_PTR_OFFSET));
|
2006-05-23 18:10:15 +08:00
|
|
|
|
|
|
|
if ((hdr_offset & 3) || hdr_offset + sizeof(*hdr) > mgp->sram_size) {
|
|
|
|
dev_err(dev, "Running firmware has bad header offset (%d)\n",
|
|
|
|
(int)hdr_offset);
|
|
|
|
return -EIO;
|
|
|
|
}
|
|
|
|
|
|
|
|
/* copy header of running firmware from SRAM to host memory to
|
|
|
|
* validate firmware */
|
|
|
|
hdr = kmalloc(bytes, GFP_KERNEL);
|
2013-02-04 01:43:58 +08:00
|
|
|
if (hdr == NULL)
|
2006-05-23 18:10:15 +08:00
|
|
|
return -ENOMEM;
|
2013-02-04 01:43:58 +08:00
|
|
|
|
2006-05-23 18:10:15 +08:00
|
|
|
memcpy_fromio(hdr, mgp->sram + hdr_offset, bytes);
|
|
|
|
status = myri10ge_validate_firmware(mgp, hdr);
|
|
|
|
kfree(hdr);
|
2007-02-22 01:05:17 +08:00
|
|
|
|
|
|
|
/* check to see if adopted firmware has bug where adopting
|
|
|
|
* it will cause broadcasts to be filtered unless the NIC
|
|
|
|
* is kept in ALLMULTI mode */
|
|
|
|
if (mgp->fw_ver_major == 1 && mgp->fw_ver_minor == 4 &&
|
|
|
|
mgp->fw_ver_tiny >= 4 && mgp->fw_ver_tiny <= 11) {
|
|
|
|
mgp->adopted_rx_filter_bug = 1;
|
|
|
|
dev_warn(dev, "Adopting fw %d.%d.%d: "
|
|
|
|
"working around rx filter bug\n",
|
|
|
|
mgp->fw_ver_major, mgp->fw_ver_minor,
|
|
|
|
mgp->fw_ver_tiny);
|
|
|
|
}
|
2006-05-23 18:10:15 +08:00
|
|
|
return status;
|
|
|
|
}
|
|
|
|
|
2008-05-20 05:53:00 +08:00
|
|
|
static int myri10ge_get_firmware_capabilities(struct myri10ge_priv *mgp)
|
2008-05-09 08:20:25 +08:00
|
|
|
{
|
|
|
|
struct myri10ge_cmd cmd;
|
|
|
|
int status;
|
|
|
|
|
|
|
|
/* probe for IPv6 TSO support */
|
|
|
|
mgp->features = NETIF_F_SG | NETIF_F_HW_CSUM | NETIF_F_TSO;
|
|
|
|
status = myri10ge_send_cmd(mgp, MXGEFW_CMD_GET_MAX_TSO6_HDR_SIZE,
|
|
|
|
&cmd, 0);
|
|
|
|
if (status == 0) {
|
|
|
|
mgp->max_tso6 = cmd.data0;
|
|
|
|
mgp->features |= NETIF_F_TSO6;
|
|
|
|
}
|
|
|
|
|
|
|
|
status = myri10ge_send_cmd(mgp, MXGEFW_CMD_GET_RX_RING_SIZE, &cmd, 0);
|
|
|
|
if (status != 0) {
|
|
|
|
dev_err(&mgp->pdev->dev,
|
|
|
|
"failed MXGEFW_CMD_GET_RX_RING_SIZE\n");
|
|
|
|
return -ENXIO;
|
|
|
|
}
|
|
|
|
|
|
|
|
mgp->max_intr_slots = 2 * (cmd.data0 / sizeof(struct mcp_dma_addr));
|
|
|
|
|
|
|
|
return 0;
|
|
|
|
}
|
|
|
|
|
2008-05-09 08:21:49 +08:00
|
|
|
static int myri10ge_load_firmware(struct myri10ge_priv *mgp, int adopt)
|
2006-05-23 18:10:15 +08:00
|
|
|
{
|
|
|
|
char __iomem *submit;
|
2008-05-09 08:17:37 +08:00
|
|
|
__be32 buf[16] __attribute__ ((__aligned__(8)));
|
2006-05-23 18:10:15 +08:00
|
|
|
u32 dma_low, dma_high, size;
|
|
|
|
int status, i;
|
|
|
|
|
2006-06-08 22:25:00 +08:00
|
|
|
size = 0;
|
2006-05-23 18:10:15 +08:00
|
|
|
status = myri10ge_load_hotplug_firmware(mgp, &size);
|
|
|
|
if (status) {
|
2008-05-09 08:21:49 +08:00
|
|
|
if (!adopt)
|
|
|
|
return status;
|
2006-05-23 18:10:15 +08:00
|
|
|
dev_warn(&mgp->pdev->dev, "hotplug firmware loading failed\n");
|
|
|
|
|
|
|
|
/* Do not attempt to adopt firmware if there
|
|
|
|
* was a bad crc */
|
|
|
|
if (status == -EIO)
|
|
|
|
return status;
|
|
|
|
|
|
|
|
status = myri10ge_adopt_running_firmware(mgp);
|
|
|
|
if (status != 0) {
|
|
|
|
dev_err(&mgp->pdev->dev,
|
|
|
|
"failed to adopt running firmware\n");
|
|
|
|
return status;
|
|
|
|
}
|
|
|
|
dev_info(&mgp->pdev->dev,
|
|
|
|
"Successfully adopted running firmware\n");
|
2008-05-09 08:20:03 +08:00
|
|
|
if (mgp->tx_boundary == 4096) {
|
2006-05-23 18:10:15 +08:00
|
|
|
dev_warn(&mgp->pdev->dev,
|
|
|
|
"Using firmware currently running on NIC"
|
|
|
|
". For optimal\n");
|
|
|
|
dev_warn(&mgp->pdev->dev,
|
|
|
|
"performance consider loading optimized "
|
|
|
|
"firmware\n");
|
|
|
|
dev_warn(&mgp->pdev->dev, "via hotplug\n");
|
|
|
|
}
|
|
|
|
|
2010-08-12 13:04:31 +08:00
|
|
|
set_fw_name(mgp, "adopted", false);
|
2008-05-09 08:20:03 +08:00
|
|
|
mgp->tx_boundary = 2048;
|
2008-05-09 08:20:25 +08:00
|
|
|
myri10ge_dummy_rdma(mgp, 1);
|
|
|
|
status = myri10ge_get_firmware_capabilities(mgp);
|
2006-05-23 18:10:15 +08:00
|
|
|
return status;
|
|
|
|
}
|
|
|
|
|
|
|
|
/* clear confirmation addr */
|
|
|
|
mgp->cmd->data = 0;
|
|
|
|
mb();
|
|
|
|
|
|
|
|
/* send a reload command to the bootstrap MCP, and wait for the
|
|
|
|
* response in the confirmation address. The firmware should
|
|
|
|
* write a -1 there to indicate it is alive and well
|
|
|
|
*/
|
|
|
|
dma_low = MYRI10GE_LOWPART_TO_U32(mgp->cmd_bus);
|
|
|
|
dma_high = MYRI10GE_HIGHPART_TO_U32(mgp->cmd_bus);
|
|
|
|
|
|
|
|
buf[0] = htonl(dma_high); /* confirm addr MSW */
|
|
|
|
buf[1] = htonl(dma_low); /* confirm addr LSW */
|
2006-11-21 02:48:32 +08:00
|
|
|
buf[2] = MYRI10GE_NO_CONFIRM_DATA; /* confirm data */
|
2006-05-23 18:10:15 +08:00
|
|
|
|
|
|
|
/* FIX: All newest firmware should un-protect the bottom of
|
|
|
|
* the sram before handoff. However, the very first interfaces
|
|
|
|
* do not. Therefore the handoff copy must skip the first 8 bytes
|
|
|
|
*/
|
|
|
|
buf[3] = htonl(MYRI10GE_FW_OFFSET + 8); /* where the code starts */
|
|
|
|
buf[4] = htonl(size - 8); /* length of code */
|
|
|
|
buf[5] = htonl(8); /* where to copy to */
|
|
|
|
buf[6] = htonl(0); /* where to jump to */
|
|
|
|
|
2006-08-15 05:52:54 +08:00
|
|
|
submit = mgp->sram + MXGEFW_BOOT_HANDOFF;
|
2006-05-23 18:10:15 +08:00
|
|
|
|
|
|
|
myri10ge_pio_copy(submit, &buf, sizeof(buf));
|
|
|
|
mb();
|
|
|
|
msleep(1);
|
|
|
|
mb();
|
|
|
|
i = 0;
|
2008-05-09 08:17:16 +08:00
|
|
|
while (mgp->cmd->data != MYRI10GE_NO_CONFIRM_DATA && i < 9) {
|
|
|
|
msleep(1 << i);
|
2006-05-23 18:10:15 +08:00
|
|
|
i++;
|
|
|
|
}
|
|
|
|
if (mgp->cmd->data != MYRI10GE_NO_CONFIRM_DATA) {
|
|
|
|
dev_err(&mgp->pdev->dev, "handoff failed\n");
|
|
|
|
return -ENXIO;
|
|
|
|
}
|
2006-07-22 03:49:32 +08:00
|
|
|
myri10ge_dummy_rdma(mgp, 1);
|
2008-05-09 08:20:25 +08:00
|
|
|
status = myri10ge_get_firmware_capabilities(mgp);
|
2006-05-23 18:10:15 +08:00
|
|
|
|
2008-05-09 08:20:25 +08:00
|
|
|
return status;
|
2006-05-23 18:10:15 +08:00
|
|
|
}
|
|
|
|
|
|
|
|
static int myri10ge_update_mac_address(struct myri10ge_priv *mgp, u8 * addr)
|
|
|
|
{
|
|
|
|
struct myri10ge_cmd cmd;
|
|
|
|
int status;
|
|
|
|
|
|
|
|
cmd.data0 = ((addr[0] << 24) | (addr[1] << 16)
|
|
|
|
| (addr[2] << 8) | addr[3]);
|
|
|
|
|
|
|
|
cmd.data1 = ((addr[4] << 8) | (addr[5]));
|
|
|
|
|
|
|
|
status = myri10ge_send_cmd(mgp, MXGEFW_SET_MAC_ADDRESS, &cmd, 0);
|
|
|
|
return status;
|
|
|
|
}
|
|
|
|
|
|
|
|
static int myri10ge_change_pause(struct myri10ge_priv *mgp, int pause)
|
|
|
|
{
|
|
|
|
struct myri10ge_cmd cmd;
|
|
|
|
int status, ctl;
|
|
|
|
|
|
|
|
ctl = pause ? MXGEFW_ENABLE_FLOW_CONTROL : MXGEFW_DISABLE_FLOW_CONTROL;
|
|
|
|
status = myri10ge_send_cmd(mgp, ctl, &cmd, 0);
|
|
|
|
|
|
|
|
if (status) {
|
2010-02-23 00:56:58 +08:00
|
|
|
netdev_err(mgp->dev, "Failed to set flow control mode\n");
|
2006-05-23 18:10:15 +08:00
|
|
|
return status;
|
|
|
|
}
|
|
|
|
mgp->pause = pause;
|
|
|
|
return 0;
|
|
|
|
}
|
|
|
|
|
|
|
|
static void
|
|
|
|
myri10ge_change_promisc(struct myri10ge_priv *mgp, int promisc, int atomic)
|
|
|
|
{
|
|
|
|
struct myri10ge_cmd cmd;
|
|
|
|
int status, ctl;
|
|
|
|
|
|
|
|
ctl = promisc ? MXGEFW_ENABLE_PROMISC : MXGEFW_DISABLE_PROMISC;
|
|
|
|
status = myri10ge_send_cmd(mgp, ctl, &cmd, atomic);
|
|
|
|
if (status)
|
2010-02-23 00:56:58 +08:00
|
|
|
netdev_err(mgp->dev, "Failed to set promisc mode\n");
|
2006-05-23 18:10:15 +08:00
|
|
|
}
|
|
|
|
|
2007-05-08 05:51:45 +08:00
|
|
|
static int myri10ge_dma_test(struct myri10ge_priv *mgp, int test_type)
|
2006-05-23 18:10:15 +08:00
|
|
|
{
|
|
|
|
struct myri10ge_cmd cmd;
|
|
|
|
int status;
|
|
|
|
u32 len;
|
2007-03-08 03:00:45 +08:00
|
|
|
struct page *dmatest_page;
|
|
|
|
dma_addr_t dmatest_bus;
|
2007-05-08 05:51:45 +08:00
|
|
|
char *test = " ";
|
|
|
|
|
|
|
|
dmatest_page = alloc_page(GFP_KERNEL);
|
|
|
|
if (!dmatest_page)
|
|
|
|
return -ENOMEM;
|
|
|
|
dmatest_bus = pci_map_page(mgp->pdev, dmatest_page, 0, PAGE_SIZE,
|
|
|
|
DMA_BIDIRECTIONAL);
|
2014-08-12 16:35:19 +08:00
|
|
|
if (unlikely(pci_dma_mapping_error(mgp->pdev, dmatest_bus))) {
|
|
|
|
__free_page(dmatest_page);
|
|
|
|
return -ENOMEM;
|
|
|
|
}
|
2007-05-08 05:51:45 +08:00
|
|
|
|
|
|
|
/* Run a small DMA test.
|
|
|
|
* The magic multipliers to the length tell the firmware
|
|
|
|
* to do DMA read, write, or read+write tests. The
|
|
|
|
* results are returned in cmd.data0. The upper 16
|
|
|
|
* bits or the return is the number of transfers completed.
|
|
|
|
* The lower 16 bits is the time in 0.5us ticks that the
|
|
|
|
* transfers took to complete.
|
|
|
|
*/
|
|
|
|
|
2008-05-09 08:20:03 +08:00
|
|
|
len = mgp->tx_boundary;
|
2007-05-08 05:51:45 +08:00
|
|
|
|
|
|
|
cmd.data0 = MYRI10GE_LOWPART_TO_U32(dmatest_bus);
|
|
|
|
cmd.data1 = MYRI10GE_HIGHPART_TO_U32(dmatest_bus);
|
|
|
|
cmd.data2 = len * 0x10000;
|
|
|
|
status = myri10ge_send_cmd(mgp, test_type, &cmd, 0);
|
|
|
|
if (status != 0) {
|
|
|
|
test = "read";
|
|
|
|
goto abort;
|
|
|
|
}
|
|
|
|
mgp->read_dma = ((cmd.data0 >> 16) * len * 2) / (cmd.data0 & 0xffff);
|
|
|
|
cmd.data0 = MYRI10GE_LOWPART_TO_U32(dmatest_bus);
|
|
|
|
cmd.data1 = MYRI10GE_HIGHPART_TO_U32(dmatest_bus);
|
|
|
|
cmd.data2 = len * 0x1;
|
|
|
|
status = myri10ge_send_cmd(mgp, test_type, &cmd, 0);
|
|
|
|
if (status != 0) {
|
|
|
|
test = "write";
|
|
|
|
goto abort;
|
|
|
|
}
|
|
|
|
mgp->write_dma = ((cmd.data0 >> 16) * len * 2) / (cmd.data0 & 0xffff);
|
|
|
|
|
|
|
|
cmd.data0 = MYRI10GE_LOWPART_TO_U32(dmatest_bus);
|
|
|
|
cmd.data1 = MYRI10GE_HIGHPART_TO_U32(dmatest_bus);
|
|
|
|
cmd.data2 = len * 0x10001;
|
|
|
|
status = myri10ge_send_cmd(mgp, test_type, &cmd, 0);
|
|
|
|
if (status != 0) {
|
|
|
|
test = "read/write";
|
|
|
|
goto abort;
|
|
|
|
}
|
|
|
|
mgp->read_write_dma = ((cmd.data0 >> 16) * len * 2 * 2) /
|
|
|
|
(cmd.data0 & 0xffff);
|
|
|
|
|
|
|
|
abort:
|
|
|
|
pci_unmap_page(mgp->pdev, dmatest_bus, PAGE_SIZE, DMA_BIDIRECTIONAL);
|
|
|
|
put_page(dmatest_page);
|
|
|
|
|
|
|
|
if (status != 0 && test_type != MXGEFW_CMD_UNALIGNED_TEST)
|
|
|
|
dev_warn(&mgp->pdev->dev, "DMA %s benchmark failed: %d\n",
|
|
|
|
test, status);
|
|
|
|
|
|
|
|
return status;
|
|
|
|
}
|
|
|
|
|
|
|
|
static int myri10ge_reset(struct myri10ge_priv *mgp)
|
|
|
|
{
|
|
|
|
struct myri10ge_cmd cmd;
|
2008-05-09 08:21:49 +08:00
|
|
|
struct myri10ge_slice_state *ss;
|
|
|
|
int i, status;
|
2007-05-08 05:51:45 +08:00
|
|
|
size_t bytes;
|
2008-10-16 17:09:31 +08:00
|
|
|
#ifdef CONFIG_MYRI10GE_DCA
|
2008-05-09 08:22:16 +08:00
|
|
|
unsigned long dca_tag_off;
|
|
|
|
#endif
|
2006-05-23 18:10:15 +08:00
|
|
|
|
|
|
|
/* try to send a reset command to the card to see if it
|
|
|
|
* is alive */
|
|
|
|
memset(&cmd, 0, sizeof(cmd));
|
|
|
|
status = myri10ge_send_cmd(mgp, MXGEFW_CMD_RESET, &cmd, 0);
|
|
|
|
if (status != 0) {
|
|
|
|
dev_err(&mgp->pdev->dev, "failed reset\n");
|
|
|
|
return -ENXIO;
|
|
|
|
}
|
2007-05-08 05:51:45 +08:00
|
|
|
|
|
|
|
(void)myri10ge_dma_test(mgp, MXGEFW_DMA_TEST);
|
2008-05-09 08:21:49 +08:00
|
|
|
/*
|
|
|
|
* Use non-ndis mcp_slot (eg, 4 bytes total,
|
|
|
|
* no toeplitz hash value returned. Older firmware will
|
|
|
|
* not understand this command, but will use the correct
|
|
|
|
* sized mcp_slot, so we ignore error returns
|
|
|
|
*/
|
|
|
|
cmd.data0 = MXGEFW_RSS_MCP_SLOT_TYPE_MIN;
|
|
|
|
(void)myri10ge_send_cmd(mgp, MXGEFW_CMD_SET_RSS_MCP_SLOT_TYPE, &cmd, 0);
|
2006-05-23 18:10:15 +08:00
|
|
|
|
|
|
|
/* Now exchange information about interrupts */
|
|
|
|
|
2008-05-09 08:21:49 +08:00
|
|
|
bytes = mgp->max_intr_slots * sizeof(*mgp->ss[0].rx_done.entry);
|
2006-05-23 18:10:15 +08:00
|
|
|
cmd.data0 = (u32) bytes;
|
|
|
|
status = myri10ge_send_cmd(mgp, MXGEFW_CMD_SET_INTRQ_SIZE, &cmd, 0);
|
2008-05-09 08:21:49 +08:00
|
|
|
|
|
|
|
/*
|
|
|
|
* Even though we already know how many slices are supported
|
|
|
|
* via myri10ge_probe_slices() MXGEFW_CMD_GET_MAX_RSS_QUEUES
|
|
|
|
* has magic side effects, and must be called after a reset.
|
|
|
|
* It must be called prior to calling any RSS related cmds,
|
|
|
|
* including assigning an interrupt queue for anything but
|
|
|
|
* slice 0. It must also be called *after*
|
|
|
|
* MXGEFW_CMD_SET_INTRQ_SIZE, since the intrq size is used by
|
|
|
|
* the firmware to compute offsets.
|
|
|
|
*/
|
|
|
|
|
|
|
|
if (mgp->num_slices > 1) {
|
|
|
|
|
|
|
|
/* ask the maximum number of slices it supports */
|
|
|
|
status = myri10ge_send_cmd(mgp, MXGEFW_CMD_GET_MAX_RSS_QUEUES,
|
|
|
|
&cmd, 0);
|
|
|
|
if (status != 0) {
|
|
|
|
dev_err(&mgp->pdev->dev,
|
|
|
|
"failed to get number of slices\n");
|
|
|
|
}
|
|
|
|
|
|
|
|
/*
|
|
|
|
* MXGEFW_CMD_ENABLE_RSS_QUEUES must be called prior
|
|
|
|
* to setting up the interrupt queue DMA
|
|
|
|
*/
|
|
|
|
|
|
|
|
cmd.data0 = mgp->num_slices;
|
2008-09-28 23:34:21 +08:00
|
|
|
cmd.data1 = MXGEFW_SLICE_INTR_MODE_ONE_PER_SLICE;
|
|
|
|
if (mgp->dev->real_num_tx_queues > 1)
|
|
|
|
cmd.data1 |= MXGEFW_SLICE_ENABLE_MULTIPLE_TX_QUEUES;
|
2008-05-09 08:21:49 +08:00
|
|
|
status = myri10ge_send_cmd(mgp, MXGEFW_CMD_ENABLE_RSS_QUEUES,
|
|
|
|
&cmd, 0);
|
2008-09-28 23:34:21 +08:00
|
|
|
|
|
|
|
/* Firmware older than 1.4.32 only supports multiple
|
|
|
|
* RX queues, so if we get an error, first retry using a
|
|
|
|
* single TX queue before giving up */
|
|
|
|
if (status != 0 && mgp->dev->real_num_tx_queues > 1) {
|
2010-09-27 16:30:34 +08:00
|
|
|
netif_set_real_num_tx_queues(mgp->dev, 1);
|
2008-09-28 23:34:21 +08:00
|
|
|
cmd.data0 = mgp->num_slices;
|
|
|
|
cmd.data1 = MXGEFW_SLICE_INTR_MODE_ONE_PER_SLICE;
|
|
|
|
status = myri10ge_send_cmd(mgp,
|
|
|
|
MXGEFW_CMD_ENABLE_RSS_QUEUES,
|
|
|
|
&cmd, 0);
|
|
|
|
}
|
|
|
|
|
2008-05-09 08:21:49 +08:00
|
|
|
if (status != 0) {
|
|
|
|
dev_err(&mgp->pdev->dev,
|
|
|
|
"failed to set number of slices\n");
|
|
|
|
|
|
|
|
return status;
|
|
|
|
}
|
|
|
|
}
|
|
|
|
for (i = 0; i < mgp->num_slices; i++) {
|
|
|
|
ss = &mgp->ss[i];
|
|
|
|
cmd.data0 = MYRI10GE_LOWPART_TO_U32(ss->rx_done.bus);
|
|
|
|
cmd.data1 = MYRI10GE_HIGHPART_TO_U32(ss->rx_done.bus);
|
|
|
|
cmd.data2 = i;
|
|
|
|
status |= myri10ge_send_cmd(mgp, MXGEFW_CMD_SET_INTRQ_DMA,
|
|
|
|
&cmd, 0);
|
2011-06-03 19:51:20 +08:00
|
|
|
}
|
2006-05-23 18:10:15 +08:00
|
|
|
|
|
|
|
status |=
|
|
|
|
myri10ge_send_cmd(mgp, MXGEFW_CMD_GET_IRQ_ACK_OFFSET, &cmd, 0);
|
2008-05-09 08:21:49 +08:00
|
|
|
for (i = 0; i < mgp->num_slices; i++) {
|
|
|
|
ss = &mgp->ss[i];
|
|
|
|
ss->irq_claim =
|
|
|
|
(__iomem __be32 *) (mgp->sram + cmd.data0 + 8 * i);
|
|
|
|
}
|
2006-12-18 18:50:40 +08:00
|
|
|
status |= myri10ge_send_cmd(mgp, MXGEFW_CMD_GET_IRQ_DEASSERT_OFFSET,
|
|
|
|
&cmd, 0);
|
|
|
|
mgp->irq_deassert = (__iomem __be32 *) (mgp->sram + cmd.data0);
|
2006-05-23 18:10:15 +08:00
|
|
|
|
|
|
|
status |= myri10ge_send_cmd
|
|
|
|
(mgp, MXGEFW_CMD_GET_INTR_COAL_DELAY_OFFSET, &cmd, 0);
|
2006-11-21 02:48:32 +08:00
|
|
|
mgp->intr_coal_delay_ptr = (__iomem __be32 *) (mgp->sram + cmd.data0);
|
2006-05-23 18:10:15 +08:00
|
|
|
if (status != 0) {
|
|
|
|
dev_err(&mgp->pdev->dev, "failed set interrupt parameters\n");
|
|
|
|
return status;
|
|
|
|
}
|
2006-11-21 02:48:32 +08:00
|
|
|
put_be32(htonl(mgp->intr_coal_delay), mgp->intr_coal_delay_ptr);
|
2006-05-23 18:10:15 +08:00
|
|
|
|
2008-10-16 17:09:31 +08:00
|
|
|
#ifdef CONFIG_MYRI10GE_DCA
|
2008-05-09 08:22:16 +08:00
|
|
|
status = myri10ge_send_cmd(mgp, MXGEFW_CMD_GET_DCA_OFFSET, &cmd, 0);
|
|
|
|
dca_tag_off = cmd.data0;
|
|
|
|
for (i = 0; i < mgp->num_slices; i++) {
|
|
|
|
ss = &mgp->ss[i];
|
|
|
|
if (status == 0) {
|
|
|
|
ss->dca_tag = (__iomem __be32 *)
|
|
|
|
(mgp->sram + dca_tag_off + 4 * i);
|
|
|
|
} else {
|
|
|
|
ss->dca_tag = NULL;
|
|
|
|
}
|
|
|
|
}
|
2008-11-24 07:49:28 +08:00
|
|
|
#endif /* CONFIG_MYRI10GE_DCA */
|
2008-05-09 08:22:16 +08:00
|
|
|
|
2006-05-23 18:10:15 +08:00
|
|
|
/* reset mcp/driver shared state back to 0 */
|
2008-05-09 08:21:49 +08:00
|
|
|
|
2006-08-22 05:36:49 +08:00
|
|
|
mgp->link_changes = 0;
|
2008-05-09 08:21:49 +08:00
|
|
|
for (i = 0; i < mgp->num_slices; i++) {
|
|
|
|
ss = &mgp->ss[i];
|
|
|
|
|
|
|
|
memset(ss->rx_done.entry, 0, bytes);
|
|
|
|
ss->tx.req = 0;
|
|
|
|
ss->tx.done = 0;
|
|
|
|
ss->tx.pkt_start = 0;
|
|
|
|
ss->tx.pkt_done = 0;
|
|
|
|
ss->rx_big.cnt = 0;
|
|
|
|
ss->rx_small.cnt = 0;
|
|
|
|
ss->rx_done.idx = 0;
|
|
|
|
ss->rx_done.cnt = 0;
|
|
|
|
ss->tx.wake_queue = 0;
|
|
|
|
ss->tx.stop_queue = 0;
|
|
|
|
}
|
|
|
|
|
2006-05-23 18:10:15 +08:00
|
|
|
status = myri10ge_update_mac_address(mgp, mgp->dev->dev_addr);
|
|
|
|
myri10ge_change_pause(mgp, mgp->pause);
|
2007-05-08 05:50:37 +08:00
|
|
|
myri10ge_set_multicast_list(mgp->dev);
|
2006-05-23 18:10:15 +08:00
|
|
|
return status;
|
|
|
|
}
|
|
|
|
|
2008-10-16 17:09:31 +08:00
|
|
|
#ifdef CONFIG_MYRI10GE_DCA
|
2010-09-28 16:13:12 +08:00
|
|
|
static int myri10ge_toggle_relaxed(struct pci_dev *pdev, int on)
|
|
|
|
{
|
2012-07-24 17:20:22 +08:00
|
|
|
int ret;
|
2010-09-28 16:13:12 +08:00
|
|
|
u16 ctl;
|
|
|
|
|
2012-07-24 17:20:22 +08:00
|
|
|
pcie_capability_read_word(pdev, PCI_EXP_DEVCTL, &ctl);
|
2011-06-27 18:56:41 +08:00
|
|
|
|
2010-09-28 16:13:12 +08:00
|
|
|
ret = (ctl & PCI_EXP_DEVCTL_RELAX_EN) >> 4;
|
|
|
|
if (ret != on) {
|
|
|
|
ctl &= ~PCI_EXP_DEVCTL_RELAX_EN;
|
|
|
|
ctl |= (on << 4);
|
2012-07-24 17:20:22 +08:00
|
|
|
pcie_capability_write_word(pdev, PCI_EXP_DEVCTL, ctl);
|
2010-09-28 16:13:12 +08:00
|
|
|
}
|
|
|
|
return ret;
|
|
|
|
}
|
|
|
|
|
2008-05-09 08:22:16 +08:00
|
|
|
static void
|
|
|
|
myri10ge_write_dca(struct myri10ge_slice_state *ss, int cpu, int tag)
|
|
|
|
{
|
|
|
|
ss->cached_dca_tag = tag;
|
|
|
|
put_be32(htonl(tag), ss->dca_tag);
|
|
|
|
}
|
|
|
|
|
|
|
|
static inline void myri10ge_update_dca(struct myri10ge_slice_state *ss)
|
|
|
|
{
|
|
|
|
int cpu = get_cpu();
|
|
|
|
int tag;
|
|
|
|
|
|
|
|
if (cpu != ss->cpu) {
|
2010-09-28 16:13:12 +08:00
|
|
|
tag = dca3_get_tag(&ss->mgp->pdev->dev, cpu);
|
2008-05-09 08:22:16 +08:00
|
|
|
if (ss->cached_dca_tag != tag)
|
|
|
|
myri10ge_write_dca(ss, cpu, tag);
|
2010-09-28 16:13:12 +08:00
|
|
|
ss->cpu = cpu;
|
2008-05-09 08:22:16 +08:00
|
|
|
}
|
|
|
|
put_cpu();
|
|
|
|
}
|
|
|
|
|
|
|
|
static void myri10ge_setup_dca(struct myri10ge_priv *mgp)
|
|
|
|
{
|
|
|
|
int err, i;
|
|
|
|
struct pci_dev *pdev = mgp->pdev;
|
|
|
|
|
|
|
|
if (mgp->ss[0].dca_tag == NULL || mgp->dca_enabled)
|
|
|
|
return;
|
|
|
|
if (!myri10ge_dca) {
|
|
|
|
dev_err(&pdev->dev, "dca disabled by administrator\n");
|
|
|
|
return;
|
|
|
|
}
|
|
|
|
err = dca_add_requester(&pdev->dev);
|
|
|
|
if (err) {
|
2008-09-13 01:47:26 +08:00
|
|
|
if (err != -ENODEV)
|
|
|
|
dev_err(&pdev->dev,
|
|
|
|
"dca_add_requester() failed, err=%d\n", err);
|
2008-05-09 08:22:16 +08:00
|
|
|
return;
|
|
|
|
}
|
2010-09-28 16:13:12 +08:00
|
|
|
mgp->relaxed_order = myri10ge_toggle_relaxed(pdev, 0);
|
2008-05-09 08:22:16 +08:00
|
|
|
mgp->dca_enabled = 1;
|
2010-09-28 16:13:12 +08:00
|
|
|
for (i = 0; i < mgp->num_slices; i++) {
|
|
|
|
mgp->ss[i].cpu = -1;
|
|
|
|
mgp->ss[i].cached_dca_tag = -1;
|
|
|
|
myri10ge_update_dca(&mgp->ss[i]);
|
2011-06-27 18:56:41 +08:00
|
|
|
}
|
2008-05-09 08:22:16 +08:00
|
|
|
}
|
|
|
|
|
|
|
|
static void myri10ge_teardown_dca(struct myri10ge_priv *mgp)
|
|
|
|
{
|
|
|
|
struct pci_dev *pdev = mgp->pdev;
|
|
|
|
|
|
|
|
if (!mgp->dca_enabled)
|
|
|
|
return;
|
|
|
|
mgp->dca_enabled = 0;
|
2010-09-28 16:13:12 +08:00
|
|
|
if (mgp->relaxed_order)
|
|
|
|
myri10ge_toggle_relaxed(pdev, 1);
|
2011-06-27 18:56:41 +08:00
|
|
|
dca_remove_requester(&pdev->dev);
|
2008-05-09 08:22:16 +08:00
|
|
|
}
|
|
|
|
|
|
|
|
static int myri10ge_notify_dca_device(struct device *dev, void *data)
|
|
|
|
{
|
|
|
|
struct myri10ge_priv *mgp;
|
|
|
|
unsigned long event;
|
|
|
|
|
|
|
|
mgp = dev_get_drvdata(dev);
|
|
|
|
event = *(unsigned long *)data;
|
|
|
|
|
|
|
|
if (event == DCA_PROVIDER_ADD)
|
|
|
|
myri10ge_setup_dca(mgp);
|
|
|
|
else if (event == DCA_PROVIDER_REMOVE)
|
|
|
|
myri10ge_teardown_dca(mgp);
|
|
|
|
return 0;
|
|
|
|
}
|
2008-11-24 07:49:28 +08:00
|
|
|
#endif /* CONFIG_MYRI10GE_DCA */
|
2008-05-09 08:22:16 +08:00
|
|
|
|
2006-05-23 18:10:15 +08:00
|
|
|
static inline void
|
|
|
|
myri10ge_submit_8rx(struct mcp_kreq_ether_recv __iomem * dst,
|
|
|
|
struct mcp_kreq_ether_recv *src)
|
|
|
|
{
|
2006-11-21 02:48:32 +08:00
|
|
|
__be32 low;
|
2006-05-23 18:10:15 +08:00
|
|
|
|
|
|
|
low = src->addr_low;
|
2009-04-07 10:01:15 +08:00
|
|
|
src->addr_low = htonl(DMA_BIT_MASK(32));
|
2006-12-06 00:26:27 +08:00
|
|
|
myri10ge_pio_copy(dst, src, 4 * sizeof(*src));
|
|
|
|
mb();
|
|
|
|
myri10ge_pio_copy(dst + 4, src + 4, 4 * sizeof(*src));
|
2006-05-23 18:10:15 +08:00
|
|
|
mb();
|
|
|
|
src->addr_low = low;
|
2006-11-21 02:48:32 +08:00
|
|
|
put_be32(low, &dst->addr_low);
|
2006-05-23 18:10:15 +08:00
|
|
|
mb();
|
|
|
|
}
|
|
|
|
|
2006-11-21 02:48:32 +08:00
|
|
|
static inline void myri10ge_vlan_ip_csum(struct sk_buff *skb, __wsum hw_csum)
|
2006-05-23 18:10:15 +08:00
|
|
|
{
|
|
|
|
struct vlan_hdr *vh = (struct vlan_hdr *)(skb->data);
|
|
|
|
|
2006-11-21 02:48:32 +08:00
|
|
|
if ((skb->protocol == htons(ETH_P_8021Q)) &&
|
2006-05-23 18:10:15 +08:00
|
|
|
(vh->h_vlan_encapsulated_proto == htons(ETH_P_IP) ||
|
|
|
|
vh->h_vlan_encapsulated_proto == htons(ETH_P_IPV6))) {
|
|
|
|
skb->csum = hw_csum;
|
2006-08-30 07:44:56 +08:00
|
|
|
skb->ip_summed = CHECKSUM_COMPLETE;
|
2006-05-23 18:10:15 +08:00
|
|
|
}
|
|
|
|
}
|
|
|
|
|
2006-12-11 18:25:09 +08:00
|
|
|
static void
|
|
|
|
myri10ge_alloc_rx_pages(struct myri10ge_priv *mgp, struct myri10ge_rx_buf *rx,
|
|
|
|
int bytes, int watchdog)
|
|
|
|
{
|
|
|
|
struct page *page;
|
2014-08-12 16:35:19 +08:00
|
|
|
dma_addr_t bus;
|
2006-12-11 18:25:09 +08:00
|
|
|
int idx;
|
2010-02-24 20:11:19 +08:00
|
|
|
#if MYRI10GE_ALLOC_SIZE > 4096
|
|
|
|
int end_offset;
|
|
|
|
#endif
|
2006-12-11 18:25:09 +08:00
|
|
|
|
|
|
|
if (unlikely(rx->watchdog_needed && !watchdog))
|
|
|
|
return;
|
|
|
|
|
|
|
|
/* try to refill entire ring */
|
|
|
|
while (rx->fill_cnt != (rx->cnt + rx->mask + 1)) {
|
|
|
|
idx = rx->fill_cnt & rx->mask;
|
2007-04-11 03:21:08 +08:00
|
|
|
if (rx->page_offset + bytes <= MYRI10GE_ALLOC_SIZE) {
|
2006-12-11 18:25:09 +08:00
|
|
|
/* we can use part of previous page */
|
|
|
|
get_page(rx->page);
|
|
|
|
} else {
|
|
|
|
/* we need a new page */
|
|
|
|
page =
|
|
|
|
alloc_pages(GFP_ATOMIC | __GFP_COMP,
|
|
|
|
MYRI10GE_ALLOC_ORDER);
|
|
|
|
if (unlikely(page == NULL)) {
|
|
|
|
if (rx->fill_cnt - rx->cnt < 16)
|
|
|
|
rx->watchdog_needed = 1;
|
|
|
|
return;
|
|
|
|
}
|
2014-08-12 16:35:19 +08:00
|
|
|
|
|
|
|
bus = pci_map_page(mgp->pdev, page, 0,
|
|
|
|
MYRI10GE_ALLOC_SIZE,
|
|
|
|
PCI_DMA_FROMDEVICE);
|
|
|
|
if (unlikely(pci_dma_mapping_error(mgp->pdev, bus))) {
|
|
|
|
__free_pages(page, MYRI10GE_ALLOC_ORDER);
|
|
|
|
if (rx->fill_cnt - rx->cnt < 16)
|
|
|
|
rx->watchdog_needed = 1;
|
|
|
|
return;
|
|
|
|
}
|
|
|
|
|
2006-12-11 18:25:09 +08:00
|
|
|
rx->page = page;
|
|
|
|
rx->page_offset = 0;
|
2014-08-12 16:35:19 +08:00
|
|
|
rx->bus = bus;
|
|
|
|
|
2006-12-11 18:25:09 +08:00
|
|
|
}
|
|
|
|
rx->info[idx].page = rx->page;
|
|
|
|
rx->info[idx].page_offset = rx->page_offset;
|
|
|
|
/* note that this is the address of the start of the
|
|
|
|
* page */
|
2010-04-12 22:32:10 +08:00
|
|
|
dma_unmap_addr_set(&rx->info[idx], bus, rx->bus);
|
2006-12-11 18:25:09 +08:00
|
|
|
rx->shadow[idx].addr_low =
|
|
|
|
htonl(MYRI10GE_LOWPART_TO_U32(rx->bus) + rx->page_offset);
|
|
|
|
rx->shadow[idx].addr_high =
|
|
|
|
htonl(MYRI10GE_HIGHPART_TO_U32(rx->bus));
|
|
|
|
|
|
|
|
/* start next packet on a cacheline boundary */
|
|
|
|
rx->page_offset += SKB_DATA_ALIGN(bytes);
|
2007-04-11 03:21:08 +08:00
|
|
|
|
|
|
|
#if MYRI10GE_ALLOC_SIZE > 4096
|
|
|
|
/* don't cross a 4KB boundary */
|
2010-02-24 20:11:19 +08:00
|
|
|
end_offset = rx->page_offset + bytes - 1;
|
|
|
|
if ((unsigned)(rx->page_offset ^ end_offset) > 4095)
|
|
|
|
rx->page_offset = end_offset & ~4095;
|
2007-04-11 03:21:08 +08:00
|
|
|
#endif
|
2006-12-11 18:25:09 +08:00
|
|
|
rx->fill_cnt++;
|
|
|
|
|
|
|
|
/* copy 8 descriptors to the firmware at a time */
|
|
|
|
if ((idx & 7) == 7) {
|
2008-07-21 16:25:50 +08:00
|
|
|
myri10ge_submit_8rx(&rx->lanai[idx - 7],
|
|
|
|
&rx->shadow[idx - 7]);
|
2006-12-11 18:25:09 +08:00
|
|
|
}
|
|
|
|
}
|
|
|
|
}
|
|
|
|
|
|
|
|
static inline void
|
|
|
|
myri10ge_unmap_rx_page(struct pci_dev *pdev,
|
|
|
|
struct myri10ge_rx_buffer_state *info, int bytes)
|
|
|
|
{
|
|
|
|
/* unmap the recvd page if we're the only or last user of it */
|
|
|
|
if (bytes >= MYRI10GE_ALLOC_SIZE / 2 ||
|
|
|
|
(info->page_offset + 2 * bytes) > MYRI10GE_ALLOC_SIZE) {
|
2010-04-12 22:32:10 +08:00
|
|
|
pci_unmap_page(pdev, (dma_unmap_addr(info, bus)
|
2006-12-11 18:25:09 +08:00
|
|
|
& ~(MYRI10GE_ALLOC_SIZE - 1)),
|
|
|
|
MYRI10GE_ALLOC_SIZE, PCI_DMA_FROMDEVICE);
|
|
|
|
}
|
|
|
|
}
|
|
|
|
|
2012-11-30 16:31:59 +08:00
|
|
|
/*
|
|
|
|
* GRO does not support acceleration of tagged vlan frames, and
|
|
|
|
* this NIC does not support vlan tag offload, so we must pop
|
|
|
|
* the tag ourselves to be able to achieve GRO performance that
|
|
|
|
* is comparable to LRO.
|
|
|
|
*/
|
|
|
|
|
|
|
|
static inline void
|
|
|
|
myri10ge_vlan_rx(struct net_device *dev, void *addr, struct sk_buff *skb)
|
|
|
|
{
|
|
|
|
u8 *va;
|
|
|
|
struct vlan_ethhdr *veh;
|
2019-07-23 11:08:25 +08:00
|
|
|
skb_frag_t *frag;
|
2012-11-30 16:31:59 +08:00
|
|
|
__wsum vsum;
|
|
|
|
|
|
|
|
va = addr;
|
|
|
|
va += MXGEFW_PAD;
|
|
|
|
veh = (struct vlan_ethhdr *)va;
|
2013-04-19 10:04:27 +08:00
|
|
|
if ((dev->features & NETIF_F_HW_VLAN_CTAG_RX) ==
|
|
|
|
NETIF_F_HW_VLAN_CTAG_RX &&
|
2012-11-30 20:31:26 +08:00
|
|
|
veh->h_vlan_proto == htons(ETH_P_8021Q)) {
|
2012-11-30 16:31:59 +08:00
|
|
|
/* fixup csum if needed */
|
|
|
|
if (skb->ip_summed == CHECKSUM_COMPLETE) {
|
|
|
|
vsum = csum_partial(va + ETH_HLEN, VLAN_HLEN, 0);
|
|
|
|
skb->csum = csum_sub(skb->csum, vsum);
|
|
|
|
}
|
|
|
|
/* pop tag */
|
2013-04-19 10:04:30 +08:00
|
|
|
__vlan_hwaccel_put_tag(skb, htons(ETH_P_8021Q), ntohs(veh->h_vlan_TCI));
|
2012-11-30 16:31:59 +08:00
|
|
|
memmove(va + VLAN_HLEN, va, 2 * ETH_ALEN);
|
|
|
|
skb->len -= VLAN_HLEN;
|
|
|
|
skb->data_len -= VLAN_HLEN;
|
|
|
|
frag = skb_shinfo(skb)->frags;
|
2019-07-30 22:40:33 +08:00
|
|
|
skb_frag_off_add(frag, VLAN_HLEN);
|
|
|
|
skb_frag_size_sub(frag, VLAN_HLEN);
|
2012-11-30 16:31:59 +08:00
|
|
|
}
|
|
|
|
}
|
|
|
|
|
2013-08-19 17:02:19 +08:00
|
|
|
#define MYRI10GE_HLEN 64 /* Bytes to copy from page to skb linear memory */
|
|
|
|
|
2006-12-11 18:25:09 +08:00
|
|
|
static inline int
|
2012-11-30 16:31:58 +08:00
|
|
|
myri10ge_rx_done(struct myri10ge_slice_state *ss, int len, __wsum csum)
|
2006-12-11 18:25:09 +08:00
|
|
|
{
|
2008-05-09 08:20:03 +08:00
|
|
|
struct myri10ge_priv *mgp = ss->mgp;
|
2006-12-11 18:25:09 +08:00
|
|
|
struct sk_buff *skb;
|
2019-07-23 11:08:25 +08:00
|
|
|
skb_frag_t *rx_frags;
|
2011-03-25 09:21:51 +08:00
|
|
|
struct myri10ge_rx_buf *rx;
|
2012-11-30 16:31:58 +08:00
|
|
|
int i, idx, remainder, bytes;
|
2006-12-11 18:25:09 +08:00
|
|
|
struct pci_dev *pdev = mgp->pdev;
|
|
|
|
struct net_device *dev = mgp->dev;
|
|
|
|
u8 *va;
|
|
|
|
|
2011-03-25 09:21:51 +08:00
|
|
|
if (len <= mgp->small_bytes) {
|
|
|
|
rx = &ss->rx_small;
|
|
|
|
bytes = mgp->small_bytes;
|
|
|
|
} else {
|
|
|
|
rx = &ss->rx_big;
|
|
|
|
bytes = mgp->big_bytes;
|
|
|
|
}
|
|
|
|
|
2006-12-11 18:25:09 +08:00
|
|
|
len += MXGEFW_PAD;
|
|
|
|
idx = rx->cnt & rx->mask;
|
|
|
|
va = page_address(rx->info[idx].page) + rx->info[idx].page_offset;
|
|
|
|
prefetch(va);
|
2012-11-30 16:31:58 +08:00
|
|
|
|
2017-02-03 02:50:48 +08:00
|
|
|
skb = napi_get_frags(&ss->napi);
|
2012-11-30 16:31:58 +08:00
|
|
|
if (unlikely(skb == NULL)) {
|
|
|
|
ss->stats.rx_dropped++;
|
|
|
|
for (i = 0, remainder = len; remainder > 0; i++) {
|
|
|
|
myri10ge_unmap_rx_page(pdev, &rx->info[idx], bytes);
|
|
|
|
put_page(rx->info[idx].page);
|
|
|
|
rx->cnt++;
|
|
|
|
idx = rx->cnt & rx->mask;
|
|
|
|
remainder -= MYRI10GE_ALLOC_SIZE;
|
|
|
|
}
|
|
|
|
return 0;
|
|
|
|
}
|
|
|
|
rx_frags = skb_shinfo(skb)->frags;
|
2019-07-23 11:08:25 +08:00
|
|
|
/* Fill skb_frag_t(s) with data from our receive */
|
2006-12-11 18:25:09 +08:00
|
|
|
for (i = 0, remainder = len; remainder > 0; i++) {
|
|
|
|
myri10ge_unmap_rx_page(pdev, &rx->info[idx], bytes);
|
2012-11-30 16:31:58 +08:00
|
|
|
skb_fill_page_desc(skb, i, rx->info[idx].page,
|
|
|
|
rx->info[idx].page_offset,
|
|
|
|
remainder < MYRI10GE_ALLOC_SIZE ?
|
|
|
|
remainder : MYRI10GE_ALLOC_SIZE);
|
2006-12-11 18:25:09 +08:00
|
|
|
rx->cnt++;
|
|
|
|
idx = rx->cnt & rx->mask;
|
|
|
|
remainder -= MYRI10GE_ALLOC_SIZE;
|
|
|
|
}
|
|
|
|
|
2012-11-30 16:31:58 +08:00
|
|
|
/* remove padding */
|
2019-07-30 22:40:33 +08:00
|
|
|
skb_frag_off_add(&rx_frags[0], MXGEFW_PAD);
|
2019-07-23 11:08:25 +08:00
|
|
|
skb_frag_size_sub(&rx_frags[0], MXGEFW_PAD);
|
2012-11-30 16:31:58 +08:00
|
|
|
len -= MXGEFW_PAD;
|
2006-12-11 18:25:09 +08:00
|
|
|
|
2012-11-30 16:31:58 +08:00
|
|
|
skb->len = len;
|
|
|
|
skb->data_len = len;
|
|
|
|
skb->truesize += len;
|
|
|
|
if (dev->features & NETIF_F_RXCSUM) {
|
|
|
|
skb->ip_summed = CHECKSUM_COMPLETE;
|
|
|
|
skb->csum = csum;
|
2006-12-11 18:25:09 +08:00
|
|
|
}
|
2012-11-30 16:31:59 +08:00
|
|
|
myri10ge_vlan_rx(mgp->dev, va, skb);
|
2009-01-28 08:22:32 +08:00
|
|
|
skb_record_rx_queue(skb, ss - &mgp->ss[0]);
|
2013-08-19 17:02:19 +08:00
|
|
|
|
2017-02-03 02:50:48 +08:00
|
|
|
napi_gro_frags(&ss->napi);
|
2006-12-11 18:25:09 +08:00
|
|
|
|
|
|
|
return 1;
|
|
|
|
}
|
|
|
|
|
2008-05-09 08:20:03 +08:00
|
|
|
static inline void
|
|
|
|
myri10ge_tx_done(struct myri10ge_slice_state *ss, int mcp_index)
|
2006-05-23 18:10:15 +08:00
|
|
|
{
|
2008-05-09 08:20:03 +08:00
|
|
|
struct pci_dev *pdev = ss->mgp->pdev;
|
|
|
|
struct myri10ge_tx_buf *tx = &ss->tx;
|
2008-09-28 23:34:21 +08:00
|
|
|
struct netdev_queue *dev_queue;
|
2006-05-23 18:10:15 +08:00
|
|
|
struct sk_buff *skb;
|
|
|
|
int idx, len;
|
|
|
|
|
|
|
|
while (tx->pkt_done != mcp_index) {
|
|
|
|
idx = tx->done & tx->mask;
|
|
|
|
skb = tx->info[idx].skb;
|
|
|
|
|
|
|
|
/* Mark as free */
|
|
|
|
tx->info[idx].skb = NULL;
|
|
|
|
if (tx->info[idx].last) {
|
|
|
|
tx->pkt_done++;
|
|
|
|
tx->info[idx].last = 0;
|
|
|
|
}
|
|
|
|
tx->done++;
|
2010-04-12 22:32:10 +08:00
|
|
|
len = dma_unmap_len(&tx->info[idx], len);
|
|
|
|
dma_unmap_len_set(&tx->info[idx], len, 0);
|
2006-05-23 18:10:15 +08:00
|
|
|
if (skb) {
|
2008-05-09 08:20:03 +08:00
|
|
|
ss->stats.tx_bytes += skb->len;
|
|
|
|
ss->stats.tx_packets++;
|
2019-02-13 23:15:43 +08:00
|
|
|
dev_consume_skb_irq(skb);
|
2006-05-23 18:10:15 +08:00
|
|
|
if (len)
|
|
|
|
pci_unmap_single(pdev,
|
2010-04-12 22:32:10 +08:00
|
|
|
dma_unmap_addr(&tx->info[idx],
|
2006-05-23 18:10:15 +08:00
|
|
|
bus), len,
|
|
|
|
PCI_DMA_TODEVICE);
|
|
|
|
} else {
|
|
|
|
if (len)
|
|
|
|
pci_unmap_page(pdev,
|
2010-04-12 22:32:10 +08:00
|
|
|
dma_unmap_addr(&tx->info[idx],
|
2006-05-23 18:10:15 +08:00
|
|
|
bus), len,
|
|
|
|
PCI_DMA_TODEVICE);
|
|
|
|
}
|
|
|
|
}
|
2008-09-28 23:34:21 +08:00
|
|
|
|
|
|
|
dev_queue = netdev_get_tx_queue(ss->dev, ss - ss->mgp->ss);
|
|
|
|
/*
|
|
|
|
* Make a minimal effort to prevent the NIC from polling an
|
|
|
|
* idle tx queue. If we can't get the lock we leave the queue
|
|
|
|
* active. In this case, either a thread was about to start
|
|
|
|
* using the queue anyway, or we lost a race and the NIC will
|
|
|
|
* waste some of its resources polling an inactive queue for a
|
|
|
|
* while.
|
|
|
|
*/
|
|
|
|
|
|
|
|
if ((ss->mgp->dev->real_num_tx_queues > 1) &&
|
|
|
|
__netif_tx_trylock(dev_queue)) {
|
|
|
|
if (tx->req == tx->done) {
|
|
|
|
tx->queue_active = 0;
|
|
|
|
put_be32(htonl(1), tx->send_stop);
|
2008-11-10 20:58:41 +08:00
|
|
|
mb();
|
2008-09-28 23:34:21 +08:00
|
|
|
}
|
|
|
|
__netif_tx_unlock(dev_queue);
|
|
|
|
}
|
|
|
|
|
2006-05-23 18:10:15 +08:00
|
|
|
/* start the queue if we've stopped it */
|
2009-12-03 15:58:21 +08:00
|
|
|
if (netif_tx_queue_stopped(dev_queue) &&
|
2011-06-27 13:05:00 +08:00
|
|
|
tx->req - tx->done < (tx->mask >> 1) &&
|
|
|
|
ss->mgp->running == MYRI10GE_ETH_RUNNING) {
|
2008-05-09 08:20:03 +08:00
|
|
|
tx->wake_queue++;
|
2008-09-28 23:34:21 +08:00
|
|
|
netif_tx_wake_queue(dev_queue);
|
2006-05-23 18:10:15 +08:00
|
|
|
}
|
|
|
|
}
|
|
|
|
|
2008-05-09 08:20:03 +08:00
|
|
|
static inline int
|
|
|
|
myri10ge_clean_rx_done(struct myri10ge_slice_state *ss, int budget)
|
2006-05-23 18:10:15 +08:00
|
|
|
{
|
2008-05-09 08:20:03 +08:00
|
|
|
struct myri10ge_rx_done *rx_done = &ss->rx_done;
|
|
|
|
struct myri10ge_priv *mgp = ss->mgp;
|
2006-05-23 18:10:15 +08:00
|
|
|
unsigned long rx_bytes = 0;
|
|
|
|
unsigned long rx_packets = 0;
|
|
|
|
unsigned long rx_ok;
|
|
|
|
int idx = rx_done->idx;
|
|
|
|
int cnt = rx_done->cnt;
|
[NET]: Make NAPI polling independent of struct net_device objects.
Several devices have multiple independant RX queues per net
device, and some have a single interrupt doorbell for several
queues.
In either case, it's easier to support layouts like that if the
structure representing the poll is independant from the net
device itself.
The signature of the ->poll() call back goes from:
int foo_poll(struct net_device *dev, int *budget)
to
int foo_poll(struct napi_struct *napi, int budget)
The caller is returned the number of RX packets processed (or
the number of "NAPI credits" consumed if you want to get
abstract). The callee no longer messes around bumping
dev->quota, *budget, etc. because that is all handled in the
caller upon return.
The napi_struct is to be embedded in the device driver private data
structures.
Furthermore, it is the driver's responsibility to disable all NAPI
instances in it's ->stop() device close handler. Since the
napi_struct is privatized into the driver's private data structures,
only the driver knows how to get at all of the napi_struct instances
it may have per-device.
With lots of help and suggestions from Rusty Russell, Roland Dreier,
Michael Chan, Jeff Garzik, and Jamal Hadi Salim.
Bug fixes from Thomas Graf, Roland Dreier, Peter Zijlstra,
Joseph Fannin, Scott Wood, Hans J. Koch, and Michael Chan.
[ Ported to current tree and all drivers converted. Integrated
Stephen's follow-on kerneldoc additions, and restored poll_list
handling to the old style to fix mutual exclusion issues. -DaveM ]
Signed-off-by: Stephen Hemminger <shemminger@linux-foundation.org>
Signed-off-by: David S. Miller <davem@davemloft.net>
2007-10-04 07:41:36 +08:00
|
|
|
int work_done = 0;
|
2006-05-23 18:10:15 +08:00
|
|
|
u16 length;
|
2006-11-21 02:48:32 +08:00
|
|
|
__wsum checksum;
|
2006-05-23 18:10:15 +08:00
|
|
|
|
2007-11-01 05:40:06 +08:00
|
|
|
while (rx_done->entry[idx].length != 0 && work_done < budget) {
|
2006-05-23 18:10:15 +08:00
|
|
|
length = ntohs(rx_done->entry[idx].length);
|
|
|
|
rx_done->entry[idx].length = 0;
|
2006-11-21 02:48:32 +08:00
|
|
|
checksum = csum_unfold(rx_done->entry[idx].checksum);
|
2012-11-30 16:31:58 +08:00
|
|
|
rx_ok = myri10ge_rx_done(ss, length, checksum);
|
2006-05-23 18:10:15 +08:00
|
|
|
rx_packets += rx_ok;
|
|
|
|
rx_bytes += rx_ok * (unsigned long)length;
|
|
|
|
cnt++;
|
2008-05-09 08:20:47 +08:00
|
|
|
idx = cnt & (mgp->max_intr_slots - 1);
|
2007-11-01 05:40:06 +08:00
|
|
|
work_done++;
|
2006-05-23 18:10:15 +08:00
|
|
|
}
|
|
|
|
rx_done->idx = idx;
|
|
|
|
rx_done->cnt = cnt;
|
2008-05-09 08:20:03 +08:00
|
|
|
ss->stats.rx_packets += rx_packets;
|
|
|
|
ss->stats.rx_bytes += rx_bytes;
|
2006-12-11 18:25:42 +08:00
|
|
|
|
|
|
|
/* restock receive rings if needed */
|
2008-05-09 08:20:03 +08:00
|
|
|
if (ss->rx_small.fill_cnt - ss->rx_small.cnt < myri10ge_fill_thresh)
|
|
|
|
myri10ge_alloc_rx_pages(mgp, &ss->rx_small,
|
2006-12-11 18:25:42 +08:00
|
|
|
mgp->small_bytes + MXGEFW_PAD, 0);
|
2008-05-09 08:20:03 +08:00
|
|
|
if (ss->rx_big.fill_cnt - ss->rx_big.cnt < myri10ge_fill_thresh)
|
|
|
|
myri10ge_alloc_rx_pages(mgp, &ss->rx_big, mgp->big_bytes, 0);
|
2006-12-11 18:25:42 +08:00
|
|
|
|
[NET]: Make NAPI polling independent of struct net_device objects.
Several devices have multiple independant RX queues per net
device, and some have a single interrupt doorbell for several
queues.
In either case, it's easier to support layouts like that if the
structure representing the poll is independant from the net
device itself.
The signature of the ->poll() call back goes from:
int foo_poll(struct net_device *dev, int *budget)
to
int foo_poll(struct napi_struct *napi, int budget)
The caller is returned the number of RX packets processed (or
the number of "NAPI credits" consumed if you want to get
abstract). The callee no longer messes around bumping
dev->quota, *budget, etc. because that is all handled in the
caller upon return.
The napi_struct is to be embedded in the device driver private data
structures.
Furthermore, it is the driver's responsibility to disable all NAPI
instances in it's ->stop() device close handler. Since the
napi_struct is privatized into the driver's private data structures,
only the driver knows how to get at all of the napi_struct instances
it may have per-device.
With lots of help and suggestions from Rusty Russell, Roland Dreier,
Michael Chan, Jeff Garzik, and Jamal Hadi Salim.
Bug fixes from Thomas Graf, Roland Dreier, Peter Zijlstra,
Joseph Fannin, Scott Wood, Hans J. Koch, and Michael Chan.
[ Ported to current tree and all drivers converted. Integrated
Stephen's follow-on kerneldoc additions, and restored poll_list
handling to the old style to fix mutual exclusion issues. -DaveM ]
Signed-off-by: Stephen Hemminger <shemminger@linux-foundation.org>
Signed-off-by: David S. Miller <davem@davemloft.net>
2007-10-04 07:41:36 +08:00
|
|
|
return work_done;
|
2006-05-23 18:10:15 +08:00
|
|
|
}
|
|
|
|
|
|
|
|
static inline void myri10ge_check_statblock(struct myri10ge_priv *mgp)
|
|
|
|
{
|
2008-05-09 08:21:49 +08:00
|
|
|
struct mcp_irq_data *stats = mgp->ss[0].fw_stats;
|
2006-05-23 18:10:15 +08:00
|
|
|
|
|
|
|
if (unlikely(stats->stats_updated)) {
|
2007-06-12 02:26:50 +08:00
|
|
|
unsigned link_up = ntohl(stats->link_up);
|
|
|
|
if (mgp->link_state != link_up) {
|
|
|
|
mgp->link_state = link_up;
|
|
|
|
|
|
|
|
if (mgp->link_state == MXGEFW_LINK_UP) {
|
2011-06-27 18:56:41 +08:00
|
|
|
netif_info(mgp, link, mgp->dev, "link up\n");
|
2006-05-23 18:10:15 +08:00
|
|
|
netif_carrier_on(mgp->dev);
|
2006-08-22 05:36:49 +08:00
|
|
|
mgp->link_changes++;
|
2006-05-23 18:10:15 +08:00
|
|
|
} else {
|
2011-06-27 18:56:41 +08:00
|
|
|
netif_info(mgp, link, mgp->dev, "link %s\n",
|
|
|
|
(link_up == MXGEFW_LINK_MYRINET ?
|
2010-02-23 00:56:58 +08:00
|
|
|
"mismatch (Myrinet detected)" :
|
2011-06-27 18:56:41 +08:00
|
|
|
"down"));
|
2006-05-23 18:10:15 +08:00
|
|
|
netif_carrier_off(mgp->dev);
|
2006-08-22 05:36:49 +08:00
|
|
|
mgp->link_changes++;
|
2006-05-23 18:10:15 +08:00
|
|
|
}
|
|
|
|
}
|
|
|
|
if (mgp->rdma_tags_available !=
|
2008-05-09 08:20:03 +08:00
|
|
|
ntohl(stats->rdma_tags_available)) {
|
2006-05-23 18:10:15 +08:00
|
|
|
mgp->rdma_tags_available =
|
2008-05-09 08:20:03 +08:00
|
|
|
ntohl(stats->rdma_tags_available);
|
2010-02-23 00:56:58 +08:00
|
|
|
netdev_warn(mgp->dev, "RDMA timed out! %d tags left\n",
|
|
|
|
mgp->rdma_tags_available);
|
2006-05-23 18:10:15 +08:00
|
|
|
}
|
|
|
|
mgp->down_cnt += stats->link_down;
|
|
|
|
if (stats->link_down)
|
|
|
|
wake_up(&mgp->down_wq);
|
|
|
|
}
|
|
|
|
}
|
|
|
|
|
[NET]: Make NAPI polling independent of struct net_device objects.
Several devices have multiple independant RX queues per net
device, and some have a single interrupt doorbell for several
queues.
In either case, it's easier to support layouts like that if the
structure representing the poll is independant from the net
device itself.
The signature of the ->poll() call back goes from:
int foo_poll(struct net_device *dev, int *budget)
to
int foo_poll(struct napi_struct *napi, int budget)
The caller is returned the number of RX packets processed (or
the number of "NAPI credits" consumed if you want to get
abstract). The callee no longer messes around bumping
dev->quota, *budget, etc. because that is all handled in the
caller upon return.
The napi_struct is to be embedded in the device driver private data
structures.
Furthermore, it is the driver's responsibility to disable all NAPI
instances in it's ->stop() device close handler. Since the
napi_struct is privatized into the driver's private data structures,
only the driver knows how to get at all of the napi_struct instances
it may have per-device.
With lots of help and suggestions from Rusty Russell, Roland Dreier,
Michael Chan, Jeff Garzik, and Jamal Hadi Salim.
Bug fixes from Thomas Graf, Roland Dreier, Peter Zijlstra,
Joseph Fannin, Scott Wood, Hans J. Koch, and Michael Chan.
[ Ported to current tree and all drivers converted. Integrated
Stephen's follow-on kerneldoc additions, and restored poll_list
handling to the old style to fix mutual exclusion issues. -DaveM ]
Signed-off-by: Stephen Hemminger <shemminger@linux-foundation.org>
Signed-off-by: David S. Miller <davem@davemloft.net>
2007-10-04 07:41:36 +08:00
|
|
|
static int myri10ge_poll(struct napi_struct *napi, int budget)
|
2006-05-23 18:10:15 +08:00
|
|
|
{
|
2008-05-09 08:20:03 +08:00
|
|
|
struct myri10ge_slice_state *ss =
|
|
|
|
container_of(napi, struct myri10ge_slice_state, napi);
|
[NET]: Make NAPI polling independent of struct net_device objects.
Several devices have multiple independant RX queues per net
device, and some have a single interrupt doorbell for several
queues.
In either case, it's easier to support layouts like that if the
structure representing the poll is independant from the net
device itself.
The signature of the ->poll() call back goes from:
int foo_poll(struct net_device *dev, int *budget)
to
int foo_poll(struct napi_struct *napi, int budget)
The caller is returned the number of RX packets processed (or
the number of "NAPI credits" consumed if you want to get
abstract). The callee no longer messes around bumping
dev->quota, *budget, etc. because that is all handled in the
caller upon return.
The napi_struct is to be embedded in the device driver private data
structures.
Furthermore, it is the driver's responsibility to disable all NAPI
instances in it's ->stop() device close handler. Since the
napi_struct is privatized into the driver's private data structures,
only the driver knows how to get at all of the napi_struct instances
it may have per-device.
With lots of help and suggestions from Rusty Russell, Roland Dreier,
Michael Chan, Jeff Garzik, and Jamal Hadi Salim.
Bug fixes from Thomas Graf, Roland Dreier, Peter Zijlstra,
Joseph Fannin, Scott Wood, Hans J. Koch, and Michael Chan.
[ Ported to current tree and all drivers converted. Integrated
Stephen's follow-on kerneldoc additions, and restored poll_list
handling to the old style to fix mutual exclusion issues. -DaveM ]
Signed-off-by: Stephen Hemminger <shemminger@linux-foundation.org>
Signed-off-by: David S. Miller <davem@davemloft.net>
2007-10-04 07:41:36 +08:00
|
|
|
int work_done;
|
2006-05-23 18:10:15 +08:00
|
|
|
|
2008-10-16 17:09:31 +08:00
|
|
|
#ifdef CONFIG_MYRI10GE_DCA
|
2008-05-09 08:22:16 +08:00
|
|
|
if (ss->mgp->dca_enabled)
|
|
|
|
myri10ge_update_dca(ss);
|
|
|
|
#endif
|
2006-05-23 18:10:15 +08:00
|
|
|
/* process as many rx events as NAPI will allow */
|
2008-05-09 08:20:03 +08:00
|
|
|
work_done = myri10ge_clean_rx_done(ss, budget);
|
2006-05-23 18:10:15 +08:00
|
|
|
|
2008-01-08 12:48:21 +08:00
|
|
|
if (work_done < budget) {
|
2017-01-31 00:22:01 +08:00
|
|
|
napi_complete_done(napi, work_done);
|
2008-05-09 08:20:03 +08:00
|
|
|
put_be32(htonl(3), ss->irq_claim);
|
2006-05-23 18:10:15 +08:00
|
|
|
}
|
[NET]: Make NAPI polling independent of struct net_device objects.
Several devices have multiple independant RX queues per net
device, and some have a single interrupt doorbell for several
queues.
In either case, it's easier to support layouts like that if the
structure representing the poll is independant from the net
device itself.
The signature of the ->poll() call back goes from:
int foo_poll(struct net_device *dev, int *budget)
to
int foo_poll(struct napi_struct *napi, int budget)
The caller is returned the number of RX packets processed (or
the number of "NAPI credits" consumed if you want to get
abstract). The callee no longer messes around bumping
dev->quota, *budget, etc. because that is all handled in the
caller upon return.
The napi_struct is to be embedded in the device driver private data
structures.
Furthermore, it is the driver's responsibility to disable all NAPI
instances in it's ->stop() device close handler. Since the
napi_struct is privatized into the driver's private data structures,
only the driver knows how to get at all of the napi_struct instances
it may have per-device.
With lots of help and suggestions from Rusty Russell, Roland Dreier,
Michael Chan, Jeff Garzik, and Jamal Hadi Salim.
Bug fixes from Thomas Graf, Roland Dreier, Peter Zijlstra,
Joseph Fannin, Scott Wood, Hans J. Koch, and Michael Chan.
[ Ported to current tree and all drivers converted. Integrated
Stephen's follow-on kerneldoc additions, and restored poll_list
handling to the old style to fix mutual exclusion issues. -DaveM ]
Signed-off-by: Stephen Hemminger <shemminger@linux-foundation.org>
Signed-off-by: David S. Miller <davem@davemloft.net>
2007-10-04 07:41:36 +08:00
|
|
|
return work_done;
|
2006-05-23 18:10:15 +08:00
|
|
|
}
|
|
|
|
|
IRQ: Maintain regs pointer globally rather than passing to IRQ handlers
Maintain a per-CPU global "struct pt_regs *" variable which can be used instead
of passing regs around manually through all ~1800 interrupt handlers in the
Linux kernel.
The regs pointer is used in few places, but it potentially costs both stack
space and code to pass it around. On the FRV arch, removing the regs parameter
from all the genirq function results in a 20% speed up of the IRQ exit path
(ie: from leaving timer_interrupt() to leaving do_IRQ()).
Where appropriate, an arch may override the generic storage facility and do
something different with the variable. On FRV, for instance, the address is
maintained in GR28 at all times inside the kernel as part of general exception
handling.
Having looked over the code, it appears that the parameter may be handed down
through up to twenty or so layers of functions. Consider a USB character
device attached to a USB hub, attached to a USB controller that posts its
interrupts through a cascaded auxiliary interrupt controller. A character
device driver may want to pass regs to the sysrq handler through the input
layer which adds another few layers of parameter passing.
I've build this code with allyesconfig for x86_64 and i386. I've runtested the
main part of the code on FRV and i386, though I can't test most of the drivers.
I've also done partial conversion for powerpc and MIPS - these at least compile
with minimal configurations.
This will affect all archs. Mostly the changes should be relatively easy.
Take do_IRQ(), store the regs pointer at the beginning, saving the old one:
struct pt_regs *old_regs = set_irq_regs(regs);
And put the old one back at the end:
set_irq_regs(old_regs);
Don't pass regs through to generic_handle_irq() or __do_IRQ().
In timer_interrupt(), this sort of change will be necessary:
- update_process_times(user_mode(regs));
- profile_tick(CPU_PROFILING, regs);
+ update_process_times(user_mode(get_irq_regs()));
+ profile_tick(CPU_PROFILING);
I'd like to move update_process_times()'s use of get_irq_regs() into itself,
except that i386, alone of the archs, uses something other than user_mode().
Some notes on the interrupt handling in the drivers:
(*) input_dev() is now gone entirely. The regs pointer is no longer stored in
the input_dev struct.
(*) finish_unlinks() in drivers/usb/host/ohci-q.c needs checking. It does
something different depending on whether it's been supplied with a regs
pointer or not.
(*) Various IRQ handler function pointers have been moved to type
irq_handler_t.
Signed-Off-By: David Howells <dhowells@redhat.com>
(cherry picked from 1b16e7ac850969f38b375e511e3fa2f474a33867 commit)
2006-10-05 21:55:46 +08:00
|
|
|
static irqreturn_t myri10ge_intr(int irq, void *arg)
|
2006-05-23 18:10:15 +08:00
|
|
|
{
|
2008-05-09 08:20:03 +08:00
|
|
|
struct myri10ge_slice_state *ss = arg;
|
|
|
|
struct myri10ge_priv *mgp = ss->mgp;
|
|
|
|
struct mcp_irq_data *stats = ss->fw_stats;
|
|
|
|
struct myri10ge_tx_buf *tx = &ss->tx;
|
2006-05-23 18:10:15 +08:00
|
|
|
u32 send_done_count;
|
|
|
|
int i;
|
|
|
|
|
2008-09-28 23:34:21 +08:00
|
|
|
/* an interrupt on a non-zero receive-only slice is implicitly
|
|
|
|
* valid since MSI-X irqs are not shared */
|
|
|
|
if ((mgp->dev->real_num_tx_queues == 1) && (ss != mgp->ss)) {
|
2009-01-20 08:43:59 +08:00
|
|
|
napi_schedule(&ss->napi);
|
2010-09-23 13:40:09 +08:00
|
|
|
return IRQ_HANDLED;
|
2008-05-09 08:21:49 +08:00
|
|
|
}
|
|
|
|
|
2006-05-23 18:10:15 +08:00
|
|
|
/* make sure it is our IRQ, and that the DMA has finished */
|
|
|
|
if (unlikely(!stats->valid))
|
2010-09-23 13:40:09 +08:00
|
|
|
return IRQ_NONE;
|
2006-05-23 18:10:15 +08:00
|
|
|
|
|
|
|
/* low bit indicates receives are present, so schedule
|
|
|
|
* napi poll handler */
|
|
|
|
if (stats->valid & 1)
|
2009-01-20 08:43:59 +08:00
|
|
|
napi_schedule(&ss->napi);
|
2006-05-23 18:10:15 +08:00
|
|
|
|
2008-05-09 08:21:49 +08:00
|
|
|
if (!mgp->msi_enabled && !mgp->msix_enabled) {
|
2006-11-21 02:48:32 +08:00
|
|
|
put_be32(0, mgp->irq_deassert);
|
2006-05-23 18:10:15 +08:00
|
|
|
if (!myri10ge_deassert_wait)
|
|
|
|
stats->valid = 0;
|
|
|
|
mb();
|
|
|
|
} else
|
|
|
|
stats->valid = 0;
|
|
|
|
|
|
|
|
/* Wait for IRQ line to go low, if using INTx */
|
|
|
|
i = 0;
|
|
|
|
while (1) {
|
|
|
|
i++;
|
|
|
|
/* check for transmit completes and receives */
|
|
|
|
send_done_count = ntohl(stats->send_done_count);
|
|
|
|
if (send_done_count != tx->pkt_done)
|
2008-05-09 08:20:03 +08:00
|
|
|
myri10ge_tx_done(ss, (int)send_done_count);
|
2006-05-23 18:10:15 +08:00
|
|
|
if (unlikely(i > myri10ge_max_irq_loops)) {
|
2011-06-27 18:56:41 +08:00
|
|
|
netdev_warn(mgp->dev, "irq stuck?\n");
|
2006-05-23 18:10:15 +08:00
|
|
|
stats->valid = 0;
|
|
|
|
schedule_work(&mgp->watchdog_work);
|
|
|
|
}
|
|
|
|
if (likely(stats->valid == 0))
|
|
|
|
break;
|
|
|
|
cpu_relax();
|
|
|
|
barrier();
|
|
|
|
}
|
|
|
|
|
2008-09-28 23:34:21 +08:00
|
|
|
/* Only slice 0 updates stats */
|
|
|
|
if (ss == mgp->ss)
|
|
|
|
myri10ge_check_statblock(mgp);
|
2006-05-23 18:10:15 +08:00
|
|
|
|
2008-05-09 08:20:03 +08:00
|
|
|
put_be32(htonl(3), ss->irq_claim + 1);
|
2010-09-23 13:40:09 +08:00
|
|
|
return IRQ_HANDLED;
|
2006-05-23 18:10:15 +08:00
|
|
|
}
|
|
|
|
|
|
|
|
static int
|
2017-02-10 06:17:23 +08:00
|
|
|
myri10ge_get_link_ksettings(struct net_device *netdev,
|
|
|
|
struct ethtool_link_ksettings *cmd)
|
2006-05-23 18:10:15 +08:00
|
|
|
{
|
2008-05-09 08:18:24 +08:00
|
|
|
struct myri10ge_priv *mgp = netdev_priv(netdev);
|
|
|
|
char *ptr;
|
|
|
|
int i;
|
|
|
|
|
2017-02-10 06:17:23 +08:00
|
|
|
cmd->base.autoneg = AUTONEG_DISABLE;
|
|
|
|
cmd->base.speed = SPEED_10000;
|
|
|
|
cmd->base.duplex = DUPLEX_FULL;
|
2008-05-09 08:18:24 +08:00
|
|
|
|
|
|
|
/*
|
|
|
|
* parse the product code to deterimine the interface type
|
|
|
|
* (CX4, XFP, Quad Ribbon Fiber) by looking at the character
|
|
|
|
* after the 3rd dash in the driver's cached copy of the
|
|
|
|
* EEPROM's product code string.
|
|
|
|
*/
|
|
|
|
ptr = mgp->product_code_string;
|
|
|
|
if (ptr == NULL) {
|
2010-02-23 00:56:58 +08:00
|
|
|
netdev_err(netdev, "Missing product code\n");
|
2008-05-09 08:18:24 +08:00
|
|
|
return 0;
|
|
|
|
}
|
|
|
|
for (i = 0; i < 3; i++, ptr++) {
|
|
|
|
ptr = strchr(ptr, '-');
|
|
|
|
if (ptr == NULL) {
|
2010-02-23 00:56:58 +08:00
|
|
|
netdev_err(netdev, "Invalid product code %s\n",
|
|
|
|
mgp->product_code_string);
|
2008-05-09 08:18:24 +08:00
|
|
|
return 0;
|
|
|
|
}
|
|
|
|
}
|
2009-10-23 12:43:43 +08:00
|
|
|
if (*ptr == '2')
|
|
|
|
ptr++;
|
|
|
|
if (*ptr == 'R' || *ptr == 'Q' || *ptr == 'S') {
|
|
|
|
/* We've found either an XFP, quad ribbon fiber, or SFP+ */
|
2017-02-10 06:17:23 +08:00
|
|
|
cmd->base.port = PORT_FIBRE;
|
|
|
|
ethtool_link_ksettings_add_link_mode(cmd, supported, FIBRE);
|
|
|
|
ethtool_link_ksettings_add_link_mode(cmd, advertising, FIBRE);
|
2009-10-23 12:43:43 +08:00
|
|
|
} else {
|
2017-02-10 06:17:23 +08:00
|
|
|
cmd->base.port = PORT_OTHER;
|
2008-05-09 08:18:24 +08:00
|
|
|
}
|
2009-10-23 12:43:43 +08:00
|
|
|
|
2006-05-23 18:10:15 +08:00
|
|
|
return 0;
|
|
|
|
}
|
|
|
|
|
|
|
|
static void
|
|
|
|
myri10ge_get_drvinfo(struct net_device *netdev, struct ethtool_drvinfo *info)
|
|
|
|
{
|
|
|
|
struct myri10ge_priv *mgp = netdev_priv(netdev);
|
|
|
|
|
|
|
|
strlcpy(info->driver, "myri10ge", sizeof(info->driver));
|
|
|
|
strlcpy(info->version, MYRI10GE_VERSION_STR, sizeof(info->version));
|
|
|
|
strlcpy(info->fw_version, mgp->fw_version, sizeof(info->fw_version));
|
|
|
|
strlcpy(info->bus_info, pci_name(mgp->pdev), sizeof(info->bus_info));
|
|
|
|
}
|
|
|
|
|
|
|
|
static int
|
|
|
|
myri10ge_get_coalesce(struct net_device *netdev, struct ethtool_coalesce *coal)
|
|
|
|
{
|
|
|
|
struct myri10ge_priv *mgp = netdev_priv(netdev);
|
2008-05-09 08:19:08 +08:00
|
|
|
|
2006-05-23 18:10:15 +08:00
|
|
|
coal->rx_coalesce_usecs = mgp->intr_coal_delay;
|
|
|
|
return 0;
|
|
|
|
}
|
|
|
|
|
|
|
|
static int
|
|
|
|
myri10ge_set_coalesce(struct net_device *netdev, struct ethtool_coalesce *coal)
|
|
|
|
{
|
|
|
|
struct myri10ge_priv *mgp = netdev_priv(netdev);
|
|
|
|
|
|
|
|
mgp->intr_coal_delay = coal->rx_coalesce_usecs;
|
2006-11-21 02:48:32 +08:00
|
|
|
put_be32(htonl(mgp->intr_coal_delay), mgp->intr_coal_delay_ptr);
|
2006-05-23 18:10:15 +08:00
|
|
|
return 0;
|
|
|
|
}
|
|
|
|
|
|
|
|
static void
|
|
|
|
myri10ge_get_pauseparam(struct net_device *netdev,
|
|
|
|
struct ethtool_pauseparam *pause)
|
|
|
|
{
|
|
|
|
struct myri10ge_priv *mgp = netdev_priv(netdev);
|
|
|
|
|
|
|
|
pause->autoneg = 0;
|
|
|
|
pause->rx_pause = mgp->pause;
|
|
|
|
pause->tx_pause = mgp->pause;
|
|
|
|
}
|
|
|
|
|
|
|
|
static int
|
|
|
|
myri10ge_set_pauseparam(struct net_device *netdev,
|
|
|
|
struct ethtool_pauseparam *pause)
|
|
|
|
{
|
|
|
|
struct myri10ge_priv *mgp = netdev_priv(netdev);
|
|
|
|
|
|
|
|
if (pause->tx_pause != mgp->pause)
|
|
|
|
return myri10ge_change_pause(mgp, pause->tx_pause);
|
|
|
|
if (pause->rx_pause != mgp->pause)
|
2010-04-08 13:23:45 +08:00
|
|
|
return myri10ge_change_pause(mgp, pause->rx_pause);
|
2006-05-23 18:10:15 +08:00
|
|
|
if (pause->autoneg != 0)
|
|
|
|
return -EINVAL;
|
|
|
|
return 0;
|
|
|
|
}
|
|
|
|
|
|
|
|
static void
|
|
|
|
myri10ge_get_ringparam(struct net_device *netdev,
|
|
|
|
struct ethtool_ringparam *ring)
|
|
|
|
{
|
|
|
|
struct myri10ge_priv *mgp = netdev_priv(netdev);
|
|
|
|
|
2008-05-09 08:21:49 +08:00
|
|
|
ring->rx_mini_max_pending = mgp->ss[0].rx_small.mask + 1;
|
|
|
|
ring->rx_max_pending = mgp->ss[0].rx_big.mask + 1;
|
2006-05-23 18:10:15 +08:00
|
|
|
ring->rx_jumbo_max_pending = 0;
|
2009-04-17 08:56:57 +08:00
|
|
|
ring->tx_max_pending = mgp->ss[0].tx.mask + 1;
|
2006-05-23 18:10:15 +08:00
|
|
|
ring->rx_mini_pending = ring->rx_mini_max_pending;
|
|
|
|
ring->rx_pending = ring->rx_max_pending;
|
|
|
|
ring->rx_jumbo_pending = ring->rx_jumbo_max_pending;
|
|
|
|
ring->tx_pending = ring->tx_max_pending;
|
|
|
|
}
|
|
|
|
|
2008-05-09 08:20:03 +08:00
|
|
|
static const char myri10ge_gstrings_main_stats[][ETH_GSTRING_LEN] = {
|
2006-05-23 18:10:15 +08:00
|
|
|
"rx_packets", "tx_packets", "rx_bytes", "tx_bytes", "rx_errors",
|
|
|
|
"tx_errors", "rx_dropped", "tx_dropped", "multicast", "collisions",
|
|
|
|
"rx_length_errors", "rx_over_errors", "rx_crc_errors",
|
|
|
|
"rx_frame_errors", "rx_fifo_errors", "rx_missed_errors",
|
|
|
|
"tx_aborted_errors", "tx_carrier_errors", "tx_fifo_errors",
|
|
|
|
"tx_heartbeat_errors", "tx_window_errors",
|
|
|
|
/* device-specific stats */
|
2015-04-22 04:09:45 +08:00
|
|
|
"tx_boundary", "irq", "MSI", "MSIX",
|
2006-05-23 18:10:15 +08:00
|
|
|
"read_dma_bw_MBs", "write_dma_bw_MBs", "read_write_dma_bw_MBs",
|
2008-05-09 08:20:03 +08:00
|
|
|
"serial_number", "watchdog_resets",
|
2008-10-16 17:09:31 +08:00
|
|
|
#ifdef CONFIG_MYRI10GE_DCA
|
2008-09-13 01:48:06 +08:00
|
|
|
"dca_capable_firmware", "dca_device_present",
|
2008-05-09 08:22:16 +08:00
|
|
|
#endif
|
2006-08-22 05:36:49 +08:00
|
|
|
"link_changes", "link_up", "dropped_link_overflow",
|
2007-05-08 05:49:25 +08:00
|
|
|
"dropped_link_error_or_filtered",
|
|
|
|
"dropped_pause", "dropped_bad_phy", "dropped_bad_crc32",
|
|
|
|
"dropped_unicast_filtered", "dropped_multicast_filtered",
|
2006-05-23 18:10:15 +08:00
|
|
|
"dropped_runt", "dropped_overrun", "dropped_no_small_buffer",
|
2008-05-09 08:20:03 +08:00
|
|
|
"dropped_no_big_buffer"
|
|
|
|
};
|
|
|
|
|
|
|
|
static const char myri10ge_gstrings_slice_stats[][ETH_GSTRING_LEN] = {
|
|
|
|
"----------- slice ---------",
|
|
|
|
"tx_pkt_start", "tx_pkt_done", "tx_req", "tx_done",
|
|
|
|
"rx_small_cnt", "rx_big_cnt",
|
2011-06-27 18:56:41 +08:00
|
|
|
"wake_queue", "stop_queue", "tx_linearized",
|
2006-05-23 18:10:15 +08:00
|
|
|
};
|
|
|
|
|
|
|
|
#define MYRI10GE_NET_STATS_LEN 21
|
2008-05-09 08:20:03 +08:00
|
|
|
#define MYRI10GE_MAIN_STATS_LEN ARRAY_SIZE(myri10ge_gstrings_main_stats)
|
|
|
|
#define MYRI10GE_SLICE_STATS_LEN ARRAY_SIZE(myri10ge_gstrings_slice_stats)
|
2006-05-23 18:10:15 +08:00
|
|
|
|
|
|
|
static void
|
|
|
|
myri10ge_get_strings(struct net_device *netdev, u32 stringset, u8 * data)
|
|
|
|
{
|
2008-05-09 08:21:49 +08:00
|
|
|
struct myri10ge_priv *mgp = netdev_priv(netdev);
|
|
|
|
int i;
|
|
|
|
|
2006-05-23 18:10:15 +08:00
|
|
|
switch (stringset) {
|
|
|
|
case ETH_SS_STATS:
|
2008-05-09 08:20:03 +08:00
|
|
|
memcpy(data, *myri10ge_gstrings_main_stats,
|
|
|
|
sizeof(myri10ge_gstrings_main_stats));
|
|
|
|
data += sizeof(myri10ge_gstrings_main_stats);
|
2008-05-09 08:21:49 +08:00
|
|
|
for (i = 0; i < mgp->num_slices; i++) {
|
|
|
|
memcpy(data, *myri10ge_gstrings_slice_stats,
|
|
|
|
sizeof(myri10ge_gstrings_slice_stats));
|
|
|
|
data += sizeof(myri10ge_gstrings_slice_stats);
|
|
|
|
}
|
2006-05-23 18:10:15 +08:00
|
|
|
break;
|
|
|
|
}
|
|
|
|
}
|
|
|
|
|
2007-10-04 09:07:32 +08:00
|
|
|
static int myri10ge_get_sset_count(struct net_device *netdev, int sset)
|
2006-05-23 18:10:15 +08:00
|
|
|
{
|
2008-05-09 08:21:49 +08:00
|
|
|
struct myri10ge_priv *mgp = netdev_priv(netdev);
|
|
|
|
|
2007-10-04 09:07:32 +08:00
|
|
|
switch (sset) {
|
|
|
|
case ETH_SS_STATS:
|
2008-05-09 08:21:49 +08:00
|
|
|
return MYRI10GE_MAIN_STATS_LEN +
|
|
|
|
mgp->num_slices * MYRI10GE_SLICE_STATS_LEN;
|
2007-10-04 09:07:32 +08:00
|
|
|
default:
|
|
|
|
return -EOPNOTSUPP;
|
|
|
|
}
|
2006-05-23 18:10:15 +08:00
|
|
|
}
|
|
|
|
|
|
|
|
static void
|
|
|
|
myri10ge_get_ethtool_stats(struct net_device *netdev,
|
|
|
|
struct ethtool_stats *stats, u64 * data)
|
|
|
|
{
|
|
|
|
struct myri10ge_priv *mgp = netdev_priv(netdev);
|
2008-05-09 08:20:03 +08:00
|
|
|
struct myri10ge_slice_state *ss;
|
2011-06-08 22:54:03 +08:00
|
|
|
struct rtnl_link_stats64 link_stats;
|
2008-05-09 08:21:49 +08:00
|
|
|
int slice;
|
2006-05-23 18:10:15 +08:00
|
|
|
int i;
|
|
|
|
|
2009-04-16 10:23:56 +08:00
|
|
|
/* force stats update */
|
2011-06-20 04:07:46 +08:00
|
|
|
memset(&link_stats, 0, sizeof(link_stats));
|
2011-06-08 22:54:03 +08:00
|
|
|
(void)myri10ge_get_stats(netdev, &link_stats);
|
2006-05-23 18:10:15 +08:00
|
|
|
for (i = 0; i < MYRI10GE_NET_STATS_LEN; i++)
|
2011-06-08 22:54:03 +08:00
|
|
|
data[i] = ((u64 *)&link_stats)[i];
|
2006-05-23 18:10:15 +08:00
|
|
|
|
2008-05-09 08:20:03 +08:00
|
|
|
data[i++] = (unsigned int)mgp->tx_boundary;
|
2006-07-04 06:16:46 +08:00
|
|
|
data[i++] = (unsigned int)mgp->pdev->irq;
|
|
|
|
data[i++] = (unsigned int)mgp->msi_enabled;
|
2008-05-09 08:21:49 +08:00
|
|
|
data[i++] = (unsigned int)mgp->msix_enabled;
|
2006-05-23 18:10:15 +08:00
|
|
|
data[i++] = (unsigned int)mgp->read_dma;
|
|
|
|
data[i++] = (unsigned int)mgp->write_dma;
|
|
|
|
data[i++] = (unsigned int)mgp->read_write_dma;
|
|
|
|
data[i++] = (unsigned int)mgp->serial_number;
|
|
|
|
data[i++] = (unsigned int)mgp->watchdog_resets;
|
2008-10-16 17:09:31 +08:00
|
|
|
#ifdef CONFIG_MYRI10GE_DCA
|
2008-05-09 08:22:16 +08:00
|
|
|
data[i++] = (unsigned int)(mgp->ss[0].dca_tag != NULL);
|
|
|
|
data[i++] = (unsigned int)(mgp->dca_enabled);
|
|
|
|
#endif
|
2006-08-22 05:36:49 +08:00
|
|
|
data[i++] = (unsigned int)mgp->link_changes;
|
2008-05-09 08:20:03 +08:00
|
|
|
|
|
|
|
/* firmware stats are useful only in the first slice */
|
2008-05-09 08:21:49 +08:00
|
|
|
ss = &mgp->ss[0];
|
2008-05-09 08:20:03 +08:00
|
|
|
data[i++] = (unsigned int)ntohl(ss->fw_stats->link_up);
|
|
|
|
data[i++] = (unsigned int)ntohl(ss->fw_stats->dropped_link_overflow);
|
2007-05-08 05:49:25 +08:00
|
|
|
data[i++] =
|
2008-05-09 08:20:03 +08:00
|
|
|
(unsigned int)ntohl(ss->fw_stats->dropped_link_error_or_filtered);
|
|
|
|
data[i++] = (unsigned int)ntohl(ss->fw_stats->dropped_pause);
|
|
|
|
data[i++] = (unsigned int)ntohl(ss->fw_stats->dropped_bad_phy);
|
|
|
|
data[i++] = (unsigned int)ntohl(ss->fw_stats->dropped_bad_crc32);
|
|
|
|
data[i++] = (unsigned int)ntohl(ss->fw_stats->dropped_unicast_filtered);
|
2006-08-22 05:36:56 +08:00
|
|
|
data[i++] =
|
2008-05-09 08:20:03 +08:00
|
|
|
(unsigned int)ntohl(ss->fw_stats->dropped_multicast_filtered);
|
|
|
|
data[i++] = (unsigned int)ntohl(ss->fw_stats->dropped_runt);
|
|
|
|
data[i++] = (unsigned int)ntohl(ss->fw_stats->dropped_overrun);
|
|
|
|
data[i++] = (unsigned int)ntohl(ss->fw_stats->dropped_no_small_buffer);
|
|
|
|
data[i++] = (unsigned int)ntohl(ss->fw_stats->dropped_no_big_buffer);
|
|
|
|
|
2008-05-09 08:21:49 +08:00
|
|
|
for (slice = 0; slice < mgp->num_slices; slice++) {
|
|
|
|
ss = &mgp->ss[slice];
|
|
|
|
data[i++] = slice;
|
|
|
|
data[i++] = (unsigned int)ss->tx.pkt_start;
|
|
|
|
data[i++] = (unsigned int)ss->tx.pkt_done;
|
|
|
|
data[i++] = (unsigned int)ss->tx.req;
|
|
|
|
data[i++] = (unsigned int)ss->tx.done;
|
|
|
|
data[i++] = (unsigned int)ss->rx_small.cnt;
|
|
|
|
data[i++] = (unsigned int)ss->rx_big.cnt;
|
|
|
|
data[i++] = (unsigned int)ss->tx.wake_queue;
|
|
|
|
data[i++] = (unsigned int)ss->tx.stop_queue;
|
|
|
|
data[i++] = (unsigned int)ss->tx.linearized;
|
|
|
|
}
|
2006-05-23 18:10:15 +08:00
|
|
|
}
|
|
|
|
|
2006-08-22 05:36:49 +08:00
|
|
|
static void myri10ge_set_msglevel(struct net_device *netdev, u32 value)
|
|
|
|
{
|
|
|
|
struct myri10ge_priv *mgp = netdev_priv(netdev);
|
|
|
|
mgp->msg_enable = value;
|
|
|
|
}
|
|
|
|
|
|
|
|
static u32 myri10ge_get_msglevel(struct net_device *netdev)
|
|
|
|
{
|
|
|
|
struct myri10ge_priv *mgp = netdev_priv(netdev);
|
|
|
|
return mgp->msg_enable;
|
|
|
|
}
|
|
|
|
|
2011-06-27 13:05:04 +08:00
|
|
|
/*
|
|
|
|
* Use a low-level command to change the LED behavior. Rather than
|
|
|
|
* blinking (which is the normal case), when identify is used, the
|
|
|
|
* yellow LED turns solid.
|
|
|
|
*/
|
|
|
|
static int myri10ge_led(struct myri10ge_priv *mgp, int on)
|
|
|
|
{
|
|
|
|
struct mcp_gen_header *hdr;
|
|
|
|
struct device *dev = &mgp->pdev->dev;
|
|
|
|
size_t hdr_off, pattern_off, hdr_len;
|
|
|
|
u32 pattern = 0xfffffffe;
|
|
|
|
|
|
|
|
/* find running firmware header */
|
|
|
|
hdr_off = swab32(readl(mgp->sram + MCP_HEADER_PTR_OFFSET));
|
|
|
|
if ((hdr_off & 3) || hdr_off + sizeof(*hdr) > mgp->sram_size) {
|
|
|
|
dev_err(dev, "Running firmware has bad header offset (%d)\n",
|
|
|
|
(int)hdr_off);
|
|
|
|
return -EIO;
|
|
|
|
}
|
|
|
|
hdr_len = swab32(readl(mgp->sram + hdr_off +
|
|
|
|
offsetof(struct mcp_gen_header, header_length)));
|
|
|
|
pattern_off = hdr_off + offsetof(struct mcp_gen_header, led_pattern);
|
|
|
|
if (pattern_off >= (hdr_len + hdr_off)) {
|
|
|
|
dev_info(dev, "Firmware does not support LED identification\n");
|
|
|
|
return -EINVAL;
|
|
|
|
}
|
|
|
|
if (!on)
|
|
|
|
pattern = swab32(readl(mgp->sram + pattern_off + 4));
|
2012-12-04 18:17:15 +08:00
|
|
|
writel(swab32(pattern), mgp->sram + pattern_off);
|
2011-06-27 13:05:04 +08:00
|
|
|
return 0;
|
|
|
|
}
|
|
|
|
|
|
|
|
static int
|
|
|
|
myri10ge_phys_id(struct net_device *netdev, enum ethtool_phys_id_state state)
|
|
|
|
{
|
|
|
|
struct myri10ge_priv *mgp = netdev_priv(netdev);
|
|
|
|
int rc;
|
|
|
|
|
|
|
|
switch (state) {
|
|
|
|
case ETHTOOL_ID_ACTIVE:
|
|
|
|
rc = myri10ge_led(mgp, 1);
|
|
|
|
break;
|
|
|
|
|
|
|
|
case ETHTOOL_ID_INACTIVE:
|
|
|
|
rc = myri10ge_led(mgp, 0);
|
|
|
|
break;
|
|
|
|
|
|
|
|
default:
|
|
|
|
rc = -EINVAL;
|
|
|
|
}
|
|
|
|
|
|
|
|
return rc;
|
|
|
|
}
|
|
|
|
|
2006-09-14 02:30:00 +08:00
|
|
|
static const struct ethtool_ops myri10ge_ethtool_ops = {
|
2006-05-23 18:10:15 +08:00
|
|
|
.get_drvinfo = myri10ge_get_drvinfo,
|
|
|
|
.get_coalesce = myri10ge_get_coalesce,
|
|
|
|
.set_coalesce = myri10ge_set_coalesce,
|
|
|
|
.get_pauseparam = myri10ge_get_pauseparam,
|
|
|
|
.set_pauseparam = myri10ge_set_pauseparam,
|
|
|
|
.get_ringparam = myri10ge_get_ringparam,
|
2007-05-31 03:13:59 +08:00
|
|
|
.get_link = ethtool_op_get_link,
|
2006-05-23 18:10:15 +08:00
|
|
|
.get_strings = myri10ge_get_strings,
|
2007-10-04 09:07:32 +08:00
|
|
|
.get_sset_count = myri10ge_get_sset_count,
|
2006-08-22 05:36:49 +08:00
|
|
|
.get_ethtool_stats = myri10ge_get_ethtool_stats,
|
|
|
|
.set_msglevel = myri10ge_set_msglevel,
|
2009-05-19 18:15:32 +08:00
|
|
|
.get_msglevel = myri10ge_get_msglevel,
|
2011-06-27 13:05:04 +08:00
|
|
|
.set_phys_id = myri10ge_phys_id,
|
2017-02-10 06:17:23 +08:00
|
|
|
.get_link_ksettings = myri10ge_get_link_ksettings,
|
2006-05-23 18:10:15 +08:00
|
|
|
};
|
|
|
|
|
2008-05-09 08:20:03 +08:00
|
|
|
static int myri10ge_allocate_rings(struct myri10ge_slice_state *ss)
|
2006-05-23 18:10:15 +08:00
|
|
|
{
|
2008-05-09 08:20:03 +08:00
|
|
|
struct myri10ge_priv *mgp = ss->mgp;
|
2006-05-23 18:10:15 +08:00
|
|
|
struct myri10ge_cmd cmd;
|
2008-05-09 08:20:03 +08:00
|
|
|
struct net_device *dev = mgp->dev;
|
2006-05-23 18:10:15 +08:00
|
|
|
int tx_ring_size, rx_ring_size;
|
|
|
|
int tx_ring_entries, rx_ring_entries;
|
2008-05-09 08:21:49 +08:00
|
|
|
int i, slice, status;
|
2006-05-23 18:10:15 +08:00
|
|
|
size_t bytes;
|
|
|
|
|
|
|
|
/* get ring sizes */
|
2008-05-09 08:21:49 +08:00
|
|
|
slice = ss - mgp->ss;
|
|
|
|
cmd.data0 = slice;
|
2006-05-23 18:10:15 +08:00
|
|
|
status = myri10ge_send_cmd(mgp, MXGEFW_CMD_GET_SEND_RING_SIZE, &cmd, 0);
|
|
|
|
tx_ring_size = cmd.data0;
|
2008-05-09 08:21:49 +08:00
|
|
|
cmd.data0 = slice;
|
2006-05-23 18:10:15 +08:00
|
|
|
status |= myri10ge_send_cmd(mgp, MXGEFW_CMD_GET_RX_RING_SIZE, &cmd, 0);
|
2007-03-08 02:59:52 +08:00
|
|
|
if (status != 0)
|
|
|
|
return status;
|
2006-05-23 18:10:15 +08:00
|
|
|
rx_ring_size = cmd.data0;
|
|
|
|
|
|
|
|
tx_ring_entries = tx_ring_size / sizeof(struct mcp_kreq_ether_send);
|
|
|
|
rx_ring_entries = rx_ring_size / sizeof(struct mcp_dma_addr);
|
2008-05-09 08:20:03 +08:00
|
|
|
ss->tx.mask = tx_ring_entries - 1;
|
|
|
|
ss->rx_small.mask = ss->rx_big.mask = rx_ring_entries - 1;
|
2006-05-23 18:10:15 +08:00
|
|
|
|
2007-03-08 02:59:52 +08:00
|
|
|
status = -ENOMEM;
|
|
|
|
|
2006-05-23 18:10:15 +08:00
|
|
|
/* allocate the host shadow rings */
|
|
|
|
|
|
|
|
bytes = 8 + (MYRI10GE_MAX_SEND_DESC_TSO + 4)
|
2008-05-09 08:20:03 +08:00
|
|
|
* sizeof(*ss->tx.req_list);
|
|
|
|
ss->tx.req_bytes = kzalloc(bytes, GFP_KERNEL);
|
|
|
|
if (ss->tx.req_bytes == NULL)
|
2006-05-23 18:10:15 +08:00
|
|
|
goto abort_with_nothing;
|
|
|
|
|
|
|
|
/* ensure req_list entries are aligned to 8 bytes */
|
2008-05-09 08:20:03 +08:00
|
|
|
ss->tx.req_list = (struct mcp_kreq_ether_send *)
|
|
|
|
ALIGN((unsigned long)ss->tx.req_bytes, 8);
|
2008-09-28 23:34:21 +08:00
|
|
|
ss->tx.queue_active = 0;
|
2006-05-23 18:10:15 +08:00
|
|
|
|
2008-05-09 08:20:03 +08:00
|
|
|
bytes = rx_ring_entries * sizeof(*ss->rx_small.shadow);
|
|
|
|
ss->rx_small.shadow = kzalloc(bytes, GFP_KERNEL);
|
|
|
|
if (ss->rx_small.shadow == NULL)
|
2006-05-23 18:10:15 +08:00
|
|
|
goto abort_with_tx_req_bytes;
|
|
|
|
|
2008-05-09 08:20:03 +08:00
|
|
|
bytes = rx_ring_entries * sizeof(*ss->rx_big.shadow);
|
|
|
|
ss->rx_big.shadow = kzalloc(bytes, GFP_KERNEL);
|
|
|
|
if (ss->rx_big.shadow == NULL)
|
2006-05-23 18:10:15 +08:00
|
|
|
goto abort_with_rx_small_shadow;
|
|
|
|
|
|
|
|
/* allocate the host info rings */
|
|
|
|
|
2008-05-09 08:20:03 +08:00
|
|
|
bytes = tx_ring_entries * sizeof(*ss->tx.info);
|
|
|
|
ss->tx.info = kzalloc(bytes, GFP_KERNEL);
|
|
|
|
if (ss->tx.info == NULL)
|
2006-05-23 18:10:15 +08:00
|
|
|
goto abort_with_rx_big_shadow;
|
|
|
|
|
2008-05-09 08:20:03 +08:00
|
|
|
bytes = rx_ring_entries * sizeof(*ss->rx_small.info);
|
|
|
|
ss->rx_small.info = kzalloc(bytes, GFP_KERNEL);
|
|
|
|
if (ss->rx_small.info == NULL)
|
2006-05-23 18:10:15 +08:00
|
|
|
goto abort_with_tx_info;
|
|
|
|
|
2008-05-09 08:20:03 +08:00
|
|
|
bytes = rx_ring_entries * sizeof(*ss->rx_big.info);
|
|
|
|
ss->rx_big.info = kzalloc(bytes, GFP_KERNEL);
|
|
|
|
if (ss->rx_big.info == NULL)
|
2006-05-23 18:10:15 +08:00
|
|
|
goto abort_with_rx_small_info;
|
|
|
|
|
|
|
|
/* Fill the receive rings */
|
2008-05-09 08:20:03 +08:00
|
|
|
ss->rx_big.cnt = 0;
|
|
|
|
ss->rx_small.cnt = 0;
|
|
|
|
ss->rx_big.fill_cnt = 0;
|
|
|
|
ss->rx_small.fill_cnt = 0;
|
|
|
|
ss->rx_small.page_offset = MYRI10GE_ALLOC_SIZE;
|
|
|
|
ss->rx_big.page_offset = MYRI10GE_ALLOC_SIZE;
|
|
|
|
ss->rx_small.watchdog_needed = 0;
|
|
|
|
ss->rx_big.watchdog_needed = 0;
|
2011-06-27 13:05:03 +08:00
|
|
|
if (mgp->small_bytes == 0) {
|
|
|
|
ss->rx_small.fill_cnt = ss->rx_small.mask + 1;
|
|
|
|
} else {
|
|
|
|
myri10ge_alloc_rx_pages(mgp, &ss->rx_small,
|
|
|
|
mgp->small_bytes + MXGEFW_PAD, 0);
|
|
|
|
}
|
2006-05-23 18:10:15 +08:00
|
|
|
|
2008-05-09 08:20:03 +08:00
|
|
|
if (ss->rx_small.fill_cnt < ss->rx_small.mask + 1) {
|
2010-02-23 00:56:58 +08:00
|
|
|
netdev_err(dev, "slice-%d: alloced only %d small bufs\n",
|
|
|
|
slice, ss->rx_small.fill_cnt);
|
2006-12-11 18:25:42 +08:00
|
|
|
goto abort_with_rx_small_ring;
|
2006-05-23 18:10:15 +08:00
|
|
|
}
|
|
|
|
|
2008-05-09 08:20:03 +08:00
|
|
|
myri10ge_alloc_rx_pages(mgp, &ss->rx_big, mgp->big_bytes, 0);
|
|
|
|
if (ss->rx_big.fill_cnt < ss->rx_big.mask + 1) {
|
2010-02-23 00:56:58 +08:00
|
|
|
netdev_err(dev, "slice-%d: alloced only %d big bufs\n",
|
|
|
|
slice, ss->rx_big.fill_cnt);
|
2006-12-11 18:25:42 +08:00
|
|
|
goto abort_with_rx_big_ring;
|
2006-05-23 18:10:15 +08:00
|
|
|
}
|
|
|
|
|
|
|
|
return 0;
|
|
|
|
|
|
|
|
abort_with_rx_big_ring:
|
2008-05-09 08:20:03 +08:00
|
|
|
for (i = ss->rx_big.cnt; i < ss->rx_big.fill_cnt; i++) {
|
|
|
|
int idx = i & ss->rx_big.mask;
|
|
|
|
myri10ge_unmap_rx_page(mgp->pdev, &ss->rx_big.info[idx],
|
2006-12-11 18:25:42 +08:00
|
|
|
mgp->big_bytes);
|
2008-05-09 08:20:03 +08:00
|
|
|
put_page(ss->rx_big.info[idx].page);
|
2006-05-23 18:10:15 +08:00
|
|
|
}
|
|
|
|
|
|
|
|
abort_with_rx_small_ring:
|
2011-06-27 13:05:03 +08:00
|
|
|
if (mgp->small_bytes == 0)
|
|
|
|
ss->rx_small.fill_cnt = ss->rx_small.cnt;
|
2008-05-09 08:20:03 +08:00
|
|
|
for (i = ss->rx_small.cnt; i < ss->rx_small.fill_cnt; i++) {
|
|
|
|
int idx = i & ss->rx_small.mask;
|
|
|
|
myri10ge_unmap_rx_page(mgp->pdev, &ss->rx_small.info[idx],
|
2006-12-11 18:25:42 +08:00
|
|
|
mgp->small_bytes + MXGEFW_PAD);
|
2008-05-09 08:20:03 +08:00
|
|
|
put_page(ss->rx_small.info[idx].page);
|
2006-05-23 18:10:15 +08:00
|
|
|
}
|
2006-12-11 18:25:42 +08:00
|
|
|
|
2008-05-09 08:20:03 +08:00
|
|
|
kfree(ss->rx_big.info);
|
2006-05-23 18:10:15 +08:00
|
|
|
|
|
|
|
abort_with_rx_small_info:
|
2008-05-09 08:20:03 +08:00
|
|
|
kfree(ss->rx_small.info);
|
2006-05-23 18:10:15 +08:00
|
|
|
|
|
|
|
abort_with_tx_info:
|
2008-05-09 08:20:03 +08:00
|
|
|
kfree(ss->tx.info);
|
2006-05-23 18:10:15 +08:00
|
|
|
|
|
|
|
abort_with_rx_big_shadow:
|
2008-05-09 08:20:03 +08:00
|
|
|
kfree(ss->rx_big.shadow);
|
2006-05-23 18:10:15 +08:00
|
|
|
|
|
|
|
abort_with_rx_small_shadow:
|
2008-05-09 08:20:03 +08:00
|
|
|
kfree(ss->rx_small.shadow);
|
2006-05-23 18:10:15 +08:00
|
|
|
|
|
|
|
abort_with_tx_req_bytes:
|
2008-05-09 08:20:03 +08:00
|
|
|
kfree(ss->tx.req_bytes);
|
|
|
|
ss->tx.req_bytes = NULL;
|
|
|
|
ss->tx.req_list = NULL;
|
2006-05-23 18:10:15 +08:00
|
|
|
|
|
|
|
abort_with_nothing:
|
|
|
|
return status;
|
|
|
|
}
|
|
|
|
|
2008-05-09 08:20:03 +08:00
|
|
|
static void myri10ge_free_rings(struct myri10ge_slice_state *ss)
|
2006-05-23 18:10:15 +08:00
|
|
|
{
|
2008-05-09 08:20:03 +08:00
|
|
|
struct myri10ge_priv *mgp = ss->mgp;
|
2006-05-23 18:10:15 +08:00
|
|
|
struct sk_buff *skb;
|
|
|
|
struct myri10ge_tx_buf *tx;
|
|
|
|
int i, len, idx;
|
|
|
|
|
2008-05-09 08:21:49 +08:00
|
|
|
/* If not allocated, skip it */
|
|
|
|
if (ss->tx.req_list == NULL)
|
|
|
|
return;
|
|
|
|
|
2008-05-09 08:20:03 +08:00
|
|
|
for (i = ss->rx_big.cnt; i < ss->rx_big.fill_cnt; i++) {
|
|
|
|
idx = i & ss->rx_big.mask;
|
|
|
|
if (i == ss->rx_big.fill_cnt - 1)
|
|
|
|
ss->rx_big.info[idx].page_offset = MYRI10GE_ALLOC_SIZE;
|
|
|
|
myri10ge_unmap_rx_page(mgp->pdev, &ss->rx_big.info[idx],
|
2006-12-11 18:25:42 +08:00
|
|
|
mgp->big_bytes);
|
2008-05-09 08:20:03 +08:00
|
|
|
put_page(ss->rx_big.info[idx].page);
|
2006-05-23 18:10:15 +08:00
|
|
|
}
|
|
|
|
|
2011-06-27 13:05:03 +08:00
|
|
|
if (mgp->small_bytes == 0)
|
|
|
|
ss->rx_small.fill_cnt = ss->rx_small.cnt;
|
2008-05-09 08:20:03 +08:00
|
|
|
for (i = ss->rx_small.cnt; i < ss->rx_small.fill_cnt; i++) {
|
|
|
|
idx = i & ss->rx_small.mask;
|
|
|
|
if (i == ss->rx_small.fill_cnt - 1)
|
|
|
|
ss->rx_small.info[idx].page_offset =
|
2006-12-11 18:25:42 +08:00
|
|
|
MYRI10GE_ALLOC_SIZE;
|
2008-05-09 08:20:03 +08:00
|
|
|
myri10ge_unmap_rx_page(mgp->pdev, &ss->rx_small.info[idx],
|
2006-12-11 18:25:42 +08:00
|
|
|
mgp->small_bytes + MXGEFW_PAD);
|
2008-05-09 08:20:03 +08:00
|
|
|
put_page(ss->rx_small.info[idx].page);
|
2006-12-11 18:25:42 +08:00
|
|
|
}
|
2008-05-09 08:20:03 +08:00
|
|
|
tx = &ss->tx;
|
2006-05-23 18:10:15 +08:00
|
|
|
while (tx->done != tx->req) {
|
|
|
|
idx = tx->done & tx->mask;
|
|
|
|
skb = tx->info[idx].skb;
|
|
|
|
|
|
|
|
/* Mark as free */
|
|
|
|
tx->info[idx].skb = NULL;
|
|
|
|
tx->done++;
|
2010-04-12 22:32:10 +08:00
|
|
|
len = dma_unmap_len(&tx->info[idx], len);
|
|
|
|
dma_unmap_len_set(&tx->info[idx], len, 0);
|
2006-05-23 18:10:15 +08:00
|
|
|
if (skb) {
|
2008-05-09 08:20:03 +08:00
|
|
|
ss->stats.tx_dropped++;
|
2006-05-23 18:10:15 +08:00
|
|
|
dev_kfree_skb_any(skb);
|
|
|
|
if (len)
|
|
|
|
pci_unmap_single(mgp->pdev,
|
2010-04-12 22:32:10 +08:00
|
|
|
dma_unmap_addr(&tx->info[idx],
|
2006-05-23 18:10:15 +08:00
|
|
|
bus), len,
|
|
|
|
PCI_DMA_TODEVICE);
|
|
|
|
} else {
|
|
|
|
if (len)
|
|
|
|
pci_unmap_page(mgp->pdev,
|
2010-04-12 22:32:10 +08:00
|
|
|
dma_unmap_addr(&tx->info[idx],
|
2006-05-23 18:10:15 +08:00
|
|
|
bus), len,
|
|
|
|
PCI_DMA_TODEVICE);
|
|
|
|
}
|
|
|
|
}
|
2008-05-09 08:20:03 +08:00
|
|
|
kfree(ss->rx_big.info);
|
2006-05-23 18:10:15 +08:00
|
|
|
|
2008-05-09 08:20:03 +08:00
|
|
|
kfree(ss->rx_small.info);
|
2006-05-23 18:10:15 +08:00
|
|
|
|
2008-05-09 08:20:03 +08:00
|
|
|
kfree(ss->tx.info);
|
2006-05-23 18:10:15 +08:00
|
|
|
|
2008-05-09 08:20:03 +08:00
|
|
|
kfree(ss->rx_big.shadow);
|
2006-05-23 18:10:15 +08:00
|
|
|
|
2008-05-09 08:20:03 +08:00
|
|
|
kfree(ss->rx_small.shadow);
|
2006-05-23 18:10:15 +08:00
|
|
|
|
2008-05-09 08:20:03 +08:00
|
|
|
kfree(ss->tx.req_bytes);
|
|
|
|
ss->tx.req_bytes = NULL;
|
|
|
|
ss->tx.req_list = NULL;
|
2006-05-23 18:10:15 +08:00
|
|
|
}
|
|
|
|
|
2006-12-18 18:50:40 +08:00
|
|
|
static int myri10ge_request_irq(struct myri10ge_priv *mgp)
|
|
|
|
{
|
|
|
|
struct pci_dev *pdev = mgp->pdev;
|
2008-05-09 08:21:49 +08:00
|
|
|
struct myri10ge_slice_state *ss;
|
|
|
|
struct net_device *netdev = mgp->dev;
|
|
|
|
int i;
|
2006-12-18 18:50:40 +08:00
|
|
|
int status;
|
|
|
|
|
2008-05-09 08:21:49 +08:00
|
|
|
mgp->msi_enabled = 0;
|
|
|
|
mgp->msix_enabled = 0;
|
|
|
|
status = 0;
|
2006-12-18 18:50:40 +08:00
|
|
|
if (myri10ge_msi) {
|
2008-05-09 08:21:49 +08:00
|
|
|
if (mgp->num_slices > 1) {
|
2014-02-18 18:11:49 +08:00
|
|
|
status = pci_enable_msix_range(pdev, mgp->msix_vectors,
|
|
|
|
mgp->num_slices, mgp->num_slices);
|
|
|
|
if (status < 0) {
|
2008-05-09 08:21:49 +08:00
|
|
|
dev_err(&pdev->dev,
|
|
|
|
"Error %d setting up MSI-X\n", status);
|
|
|
|
return status;
|
|
|
|
}
|
2014-02-18 18:11:49 +08:00
|
|
|
mgp->msix_enabled = 1;
|
2008-05-09 08:21:49 +08:00
|
|
|
}
|
|
|
|
if (mgp->msix_enabled == 0) {
|
|
|
|
status = pci_enable_msi(pdev);
|
|
|
|
if (status != 0) {
|
|
|
|
dev_err(&pdev->dev,
|
|
|
|
"Error %d setting up MSI; falling back to xPIC\n",
|
|
|
|
status);
|
|
|
|
} else {
|
|
|
|
mgp->msi_enabled = 1;
|
|
|
|
}
|
|
|
|
}
|
2006-12-18 18:50:40 +08:00
|
|
|
}
|
2008-05-09 08:21:49 +08:00
|
|
|
if (mgp->msix_enabled) {
|
|
|
|
for (i = 0; i < mgp->num_slices; i++) {
|
|
|
|
ss = &mgp->ss[i];
|
|
|
|
snprintf(ss->irq_desc, sizeof(ss->irq_desc),
|
|
|
|
"%s:slice-%d", netdev->name, i);
|
|
|
|
status = request_irq(mgp->msix_vectors[i].vector,
|
|
|
|
myri10ge_intr, 0, ss->irq_desc,
|
|
|
|
ss);
|
|
|
|
if (status != 0) {
|
|
|
|
dev_err(&pdev->dev,
|
|
|
|
"slice %d failed to allocate IRQ\n", i);
|
|
|
|
i--;
|
|
|
|
while (i >= 0) {
|
|
|
|
free_irq(mgp->msix_vectors[i].vector,
|
|
|
|
&mgp->ss[i]);
|
|
|
|
i--;
|
|
|
|
}
|
|
|
|
pci_disable_msix(pdev);
|
|
|
|
return status;
|
|
|
|
}
|
|
|
|
}
|
|
|
|
} else {
|
|
|
|
status = request_irq(pdev->irq, myri10ge_intr, IRQF_SHARED,
|
|
|
|
mgp->dev->name, &mgp->ss[0]);
|
|
|
|
if (status != 0) {
|
|
|
|
dev_err(&pdev->dev, "failed to allocate IRQ\n");
|
|
|
|
if (mgp->msi_enabled)
|
|
|
|
pci_disable_msi(pdev);
|
|
|
|
}
|
2006-12-18 18:50:40 +08:00
|
|
|
}
|
|
|
|
return status;
|
|
|
|
}
|
|
|
|
|
|
|
|
static void myri10ge_free_irq(struct myri10ge_priv *mgp)
|
|
|
|
{
|
|
|
|
struct pci_dev *pdev = mgp->pdev;
|
2008-05-09 08:21:49 +08:00
|
|
|
int i;
|
2006-12-18 18:50:40 +08:00
|
|
|
|
2008-05-09 08:21:49 +08:00
|
|
|
if (mgp->msix_enabled) {
|
|
|
|
for (i = 0; i < mgp->num_slices; i++)
|
|
|
|
free_irq(mgp->msix_vectors[i].vector, &mgp->ss[i]);
|
|
|
|
} else {
|
|
|
|
free_irq(pdev->irq, &mgp->ss[0]);
|
|
|
|
}
|
2006-12-18 18:50:40 +08:00
|
|
|
if (mgp->msi_enabled)
|
|
|
|
pci_disable_msi(pdev);
|
2008-05-09 08:21:49 +08:00
|
|
|
if (mgp->msix_enabled)
|
|
|
|
pci_disable_msix(pdev);
|
2006-12-18 18:50:40 +08:00
|
|
|
}
|
|
|
|
|
2008-05-09 08:21:10 +08:00
|
|
|
static int myri10ge_get_txrx(struct myri10ge_priv *mgp, int slice)
|
|
|
|
{
|
|
|
|
struct myri10ge_cmd cmd;
|
|
|
|
struct myri10ge_slice_state *ss;
|
|
|
|
int status;
|
|
|
|
|
|
|
|
ss = &mgp->ss[slice];
|
2008-09-28 23:34:21 +08:00
|
|
|
status = 0;
|
|
|
|
if (slice == 0 || (mgp->dev->real_num_tx_queues > 1)) {
|
|
|
|
cmd.data0 = slice;
|
|
|
|
status = myri10ge_send_cmd(mgp, MXGEFW_CMD_GET_SEND_OFFSET,
|
|
|
|
&cmd, 0);
|
|
|
|
ss->tx.lanai = (struct mcp_kreq_ether_send __iomem *)
|
|
|
|
(mgp->sram + cmd.data0);
|
|
|
|
}
|
2008-05-09 08:21:10 +08:00
|
|
|
cmd.data0 = slice;
|
|
|
|
status |= myri10ge_send_cmd(mgp, MXGEFW_CMD_GET_SMALL_RX_OFFSET,
|
|
|
|
&cmd, 0);
|
|
|
|
ss->rx_small.lanai = (struct mcp_kreq_ether_recv __iomem *)
|
|
|
|
(mgp->sram + cmd.data0);
|
|
|
|
|
|
|
|
cmd.data0 = slice;
|
|
|
|
status |= myri10ge_send_cmd(mgp, MXGEFW_CMD_GET_BIG_RX_OFFSET, &cmd, 0);
|
|
|
|
ss->rx_big.lanai = (struct mcp_kreq_ether_recv __iomem *)
|
|
|
|
(mgp->sram + cmd.data0);
|
|
|
|
|
2008-09-28 23:34:21 +08:00
|
|
|
ss->tx.send_go = (__iomem __be32 *)
|
|
|
|
(mgp->sram + MXGEFW_ETH_SEND_GO + 64 * slice);
|
|
|
|
ss->tx.send_stop = (__iomem __be32 *)
|
|
|
|
(mgp->sram + MXGEFW_ETH_SEND_STOP + 64 * slice);
|
2008-05-09 08:21:10 +08:00
|
|
|
return status;
|
|
|
|
|
|
|
|
}
|
|
|
|
|
|
|
|
static int myri10ge_set_stats(struct myri10ge_priv *mgp, int slice)
|
|
|
|
{
|
|
|
|
struct myri10ge_cmd cmd;
|
|
|
|
struct myri10ge_slice_state *ss;
|
|
|
|
int status;
|
|
|
|
|
|
|
|
ss = &mgp->ss[slice];
|
|
|
|
cmd.data0 = MYRI10GE_LOWPART_TO_U32(ss->fw_stats_bus);
|
|
|
|
cmd.data1 = MYRI10GE_HIGHPART_TO_U32(ss->fw_stats_bus);
|
2008-09-28 23:34:21 +08:00
|
|
|
cmd.data2 = sizeof(struct mcp_irq_data) | (slice << 16);
|
2008-05-09 08:21:10 +08:00
|
|
|
status = myri10ge_send_cmd(mgp, MXGEFW_CMD_SET_STATS_DMA_V2, &cmd, 0);
|
|
|
|
if (status == -ENOSYS) {
|
|
|
|
dma_addr_t bus = ss->fw_stats_bus;
|
|
|
|
if (slice != 0)
|
|
|
|
return -EINVAL;
|
|
|
|
bus += offsetof(struct mcp_irq_data, send_done_count);
|
|
|
|
cmd.data0 = MYRI10GE_LOWPART_TO_U32(bus);
|
|
|
|
cmd.data1 = MYRI10GE_HIGHPART_TO_U32(bus);
|
|
|
|
status = myri10ge_send_cmd(mgp,
|
|
|
|
MXGEFW_CMD_SET_STATS_DMA_OBSOLETE,
|
|
|
|
&cmd, 0);
|
|
|
|
/* Firmware cannot support multicast without STATS_DMA_V2 */
|
|
|
|
mgp->fw_multicast_support = 0;
|
|
|
|
} else {
|
|
|
|
mgp->fw_multicast_support = 1;
|
|
|
|
}
|
|
|
|
return 0;
|
|
|
|
}
|
|
|
|
|
2006-05-23 18:10:15 +08:00
|
|
|
static int myri10ge_open(struct net_device *dev)
|
|
|
|
{
|
2008-05-09 08:21:49 +08:00
|
|
|
struct myri10ge_slice_state *ss;
|
2008-05-09 08:20:03 +08:00
|
|
|
struct myri10ge_priv *mgp = netdev_priv(dev);
|
2006-05-23 18:10:15 +08:00
|
|
|
struct myri10ge_cmd cmd;
|
2008-05-09 08:21:49 +08:00
|
|
|
int i, status, big_pow2, slice;
|
2012-12-04 18:17:15 +08:00
|
|
|
u8 __iomem *itable;
|
2006-05-23 18:10:15 +08:00
|
|
|
|
|
|
|
if (mgp->running != MYRI10GE_ETH_STOPPED)
|
|
|
|
return -EBUSY;
|
|
|
|
|
|
|
|
mgp->running = MYRI10GE_ETH_STARTING;
|
|
|
|
status = myri10ge_reset(mgp);
|
|
|
|
if (status != 0) {
|
2010-02-23 00:56:58 +08:00
|
|
|
netdev_err(dev, "failed reset\n");
|
2006-12-18 18:50:40 +08:00
|
|
|
goto abort_with_nothing;
|
2006-05-23 18:10:15 +08:00
|
|
|
}
|
|
|
|
|
2008-05-09 08:21:49 +08:00
|
|
|
if (mgp->num_slices > 1) {
|
|
|
|
cmd.data0 = mgp->num_slices;
|
2008-09-28 23:34:21 +08:00
|
|
|
cmd.data1 = MXGEFW_SLICE_INTR_MODE_ONE_PER_SLICE;
|
|
|
|
if (mgp->dev->real_num_tx_queues > 1)
|
|
|
|
cmd.data1 |= MXGEFW_SLICE_ENABLE_MULTIPLE_TX_QUEUES;
|
2008-05-09 08:21:49 +08:00
|
|
|
status = myri10ge_send_cmd(mgp, MXGEFW_CMD_ENABLE_RSS_QUEUES,
|
|
|
|
&cmd, 0);
|
|
|
|
if (status != 0) {
|
2010-02-23 00:56:58 +08:00
|
|
|
netdev_err(dev, "failed to set number of slices\n");
|
2008-05-09 08:21:49 +08:00
|
|
|
goto abort_with_nothing;
|
|
|
|
}
|
|
|
|
/* setup the indirection table */
|
|
|
|
cmd.data0 = mgp->num_slices;
|
|
|
|
status = myri10ge_send_cmd(mgp, MXGEFW_CMD_SET_RSS_TABLE_SIZE,
|
|
|
|
&cmd, 0);
|
|
|
|
|
|
|
|
status |= myri10ge_send_cmd(mgp,
|
|
|
|
MXGEFW_CMD_GET_RSS_TABLE_OFFSET,
|
|
|
|
&cmd, 0);
|
|
|
|
if (status != 0) {
|
2010-02-23 00:56:58 +08:00
|
|
|
netdev_err(dev, "failed to setup rss tables\n");
|
2008-09-28 23:34:21 +08:00
|
|
|
goto abort_with_nothing;
|
2008-05-09 08:21:49 +08:00
|
|
|
}
|
|
|
|
|
|
|
|
/* just enable an identity mapping */
|
|
|
|
itable = mgp->sram + cmd.data0;
|
|
|
|
for (i = 0; i < mgp->num_slices; i++)
|
|
|
|
__raw_writeb(i, &itable[i]);
|
|
|
|
|
|
|
|
cmd.data0 = 1;
|
|
|
|
cmd.data1 = myri10ge_rss_hash;
|
|
|
|
status = myri10ge_send_cmd(mgp, MXGEFW_CMD_SET_RSS_ENABLE,
|
|
|
|
&cmd, 0);
|
|
|
|
if (status != 0) {
|
2010-02-23 00:56:58 +08:00
|
|
|
netdev_err(dev, "failed to enable slices\n");
|
2008-05-09 08:21:49 +08:00
|
|
|
goto abort_with_nothing;
|
|
|
|
}
|
|
|
|
}
|
|
|
|
|
2006-12-18 18:50:40 +08:00
|
|
|
status = myri10ge_request_irq(mgp);
|
|
|
|
if (status != 0)
|
|
|
|
goto abort_with_nothing;
|
|
|
|
|
2006-05-23 18:10:15 +08:00
|
|
|
/* decide what small buffer size to use. For good TCP rx
|
|
|
|
* performance, it is important to not receive 1514 byte
|
|
|
|
* frames into jumbo buffers, as it confuses the socket buffer
|
|
|
|
* accounting code, leading to drops and erratic performance.
|
|
|
|
*/
|
|
|
|
|
|
|
|
if (dev->mtu <= ETH_DATA_LEN)
|
2006-12-11 18:25:42 +08:00
|
|
|
/* enough for a TCP header */
|
|
|
|
mgp->small_bytes = (128 > SMP_CACHE_BYTES)
|
|
|
|
? (128 - MXGEFW_PAD)
|
|
|
|
: (SMP_CACHE_BYTES - MXGEFW_PAD);
|
2006-05-23 18:10:15 +08:00
|
|
|
else
|
2006-12-11 18:26:38 +08:00
|
|
|
/* enough for a vlan encapsulated ETH_DATA_LEN frame */
|
|
|
|
mgp->small_bytes = VLAN_ETH_FRAME_LEN;
|
2006-05-23 18:10:15 +08:00
|
|
|
|
|
|
|
/* Override the small buffer size? */
|
2011-06-27 13:05:03 +08:00
|
|
|
if (myri10ge_small_bytes >= 0)
|
2006-05-23 18:10:15 +08:00
|
|
|
mgp->small_bytes = myri10ge_small_bytes;
|
|
|
|
|
|
|
|
/* Firmware needs the big buff size as a power of 2. Lie and
|
|
|
|
* tell him the buffer is larger, because we only use 1
|
|
|
|
* buffer/pkt, and the mtu will prevent overruns.
|
|
|
|
*/
|
2006-12-11 18:27:19 +08:00
|
|
|
big_pow2 = dev->mtu + ETH_HLEN + VLAN_HLEN + MXGEFW_PAD;
|
2006-12-11 18:25:42 +08:00
|
|
|
if (big_pow2 < MYRI10GE_ALLOC_SIZE / 2) {
|
2007-07-10 02:50:22 +08:00
|
|
|
while (!is_power_of_2(big_pow2))
|
2006-12-11 18:25:42 +08:00
|
|
|
big_pow2++;
|
2006-12-11 18:27:19 +08:00
|
|
|
mgp->big_bytes = dev->mtu + ETH_HLEN + VLAN_HLEN + MXGEFW_PAD;
|
2006-12-11 18:25:42 +08:00
|
|
|
} else {
|
|
|
|
big_pow2 = MYRI10GE_ALLOC_SIZE;
|
|
|
|
mgp->big_bytes = big_pow2;
|
|
|
|
}
|
|
|
|
|
2008-05-09 08:21:49 +08:00
|
|
|
/* setup the per-slice data structures */
|
|
|
|
for (slice = 0; slice < mgp->num_slices; slice++) {
|
|
|
|
ss = &mgp->ss[slice];
|
|
|
|
|
|
|
|
status = myri10ge_get_txrx(mgp, slice);
|
|
|
|
if (status != 0) {
|
2010-02-23 00:56:58 +08:00
|
|
|
netdev_err(dev, "failed to get ring sizes or locations\n");
|
2008-05-09 08:21:49 +08:00
|
|
|
goto abort_with_rings;
|
|
|
|
}
|
|
|
|
status = myri10ge_allocate_rings(ss);
|
|
|
|
if (status != 0)
|
|
|
|
goto abort_with_rings;
|
2008-09-28 23:34:21 +08:00
|
|
|
|
|
|
|
/* only firmware which supports multiple TX queues
|
|
|
|
* supports setting up the tx stats on non-zero
|
|
|
|
* slices */
|
|
|
|
if (slice == 0 || mgp->dev->real_num_tx_queues > 1)
|
2008-05-09 08:21:49 +08:00
|
|
|
status = myri10ge_set_stats(mgp, slice);
|
|
|
|
if (status) {
|
2010-02-23 00:56:58 +08:00
|
|
|
netdev_err(dev, "Couldn't set stats DMA\n");
|
2008-05-09 08:21:49 +08:00
|
|
|
goto abort_with_rings;
|
|
|
|
}
|
|
|
|
|
|
|
|
/* must happen prior to any irq */
|
|
|
|
napi_enable(&(ss)->napi);
|
|
|
|
}
|
2006-05-23 18:10:15 +08:00
|
|
|
|
|
|
|
/* now give firmware buffers sizes, and MTU */
|
|
|
|
cmd.data0 = dev->mtu + ETH_HLEN + VLAN_HLEN;
|
|
|
|
status = myri10ge_send_cmd(mgp, MXGEFW_CMD_SET_MTU, &cmd, 0);
|
|
|
|
cmd.data0 = mgp->small_bytes;
|
|
|
|
status |=
|
|
|
|
myri10ge_send_cmd(mgp, MXGEFW_CMD_SET_SMALL_BUFFER_SIZE, &cmd, 0);
|
|
|
|
cmd.data0 = big_pow2;
|
|
|
|
status |=
|
|
|
|
myri10ge_send_cmd(mgp, MXGEFW_CMD_SET_BIG_BUFFER_SIZE, &cmd, 0);
|
|
|
|
if (status) {
|
2010-02-23 00:56:58 +08:00
|
|
|
netdev_err(dev, "Couldn't set buffer sizes\n");
|
2006-05-23 18:10:15 +08:00
|
|
|
goto abort_with_rings;
|
|
|
|
}
|
|
|
|
|
2008-05-09 08:21:49 +08:00
|
|
|
/*
|
|
|
|
* Set Linux style TSO mode; this is needed only on newer
|
|
|
|
* firmware versions. Older versions default to Linux
|
|
|
|
* style TSO
|
|
|
|
*/
|
|
|
|
cmd.data0 = 0;
|
|
|
|
status = myri10ge_send_cmd(mgp, MXGEFW_CMD_SET_TSO_MODE, &cmd, 0);
|
|
|
|
if (status && status != -ENOSYS) {
|
2010-02-23 00:56:58 +08:00
|
|
|
netdev_err(dev, "Couldn't set TSO mode\n");
|
2006-05-23 18:10:15 +08:00
|
|
|
goto abort_with_rings;
|
|
|
|
}
|
|
|
|
|
2007-12-23 02:56:43 +08:00
|
|
|
mgp->link_state = ~0U;
|
2006-05-23 18:10:15 +08:00
|
|
|
mgp->rdma_tags_available = 15;
|
|
|
|
|
|
|
|
status = myri10ge_send_cmd(mgp, MXGEFW_CMD_ETHERNET_UP, &cmd, 0);
|
|
|
|
if (status) {
|
2010-02-23 00:56:58 +08:00
|
|
|
netdev_err(dev, "Couldn't bring up link\n");
|
2006-05-23 18:10:15 +08:00
|
|
|
goto abort_with_rings;
|
|
|
|
}
|
|
|
|
|
|
|
|
mgp->running = MYRI10GE_ETH_RUNNING;
|
|
|
|
mgp->watchdog_timer.expires = jiffies + myri10ge_watchdog_timeout * HZ;
|
|
|
|
add_timer(&mgp->watchdog_timer);
|
2008-09-28 23:34:21 +08:00
|
|
|
netif_tx_wake_all_queues(dev);
|
|
|
|
|
2006-05-23 18:10:15 +08:00
|
|
|
return 0;
|
|
|
|
|
|
|
|
abort_with_rings:
|
2008-10-20 19:54:12 +08:00
|
|
|
while (slice) {
|
|
|
|
slice--;
|
|
|
|
napi_disable(&mgp->ss[slice].napi);
|
|
|
|
}
|
2008-05-09 08:21:49 +08:00
|
|
|
for (i = 0; i < mgp->num_slices; i++)
|
|
|
|
myri10ge_free_rings(&mgp->ss[i]);
|
2006-05-23 18:10:15 +08:00
|
|
|
|
2006-12-18 18:50:40 +08:00
|
|
|
myri10ge_free_irq(mgp);
|
|
|
|
|
2006-05-23 18:10:15 +08:00
|
|
|
abort_with_nothing:
|
|
|
|
mgp->running = MYRI10GE_ETH_STOPPED;
|
|
|
|
return -ENOMEM;
|
|
|
|
}
|
|
|
|
|
|
|
|
static int myri10ge_close(struct net_device *dev)
|
|
|
|
{
|
2008-05-09 08:20:03 +08:00
|
|
|
struct myri10ge_priv *mgp = netdev_priv(dev);
|
2006-05-23 18:10:15 +08:00
|
|
|
struct myri10ge_cmd cmd;
|
|
|
|
int status, old_down_cnt;
|
2008-05-09 08:21:49 +08:00
|
|
|
int i;
|
2006-05-23 18:10:15 +08:00
|
|
|
|
|
|
|
if (mgp->running != MYRI10GE_ETH_RUNNING)
|
|
|
|
return 0;
|
|
|
|
|
2008-05-09 08:21:49 +08:00
|
|
|
if (mgp->ss[0].tx.req_bytes == NULL)
|
2006-05-23 18:10:15 +08:00
|
|
|
return 0;
|
|
|
|
|
|
|
|
del_timer_sync(&mgp->watchdog_timer);
|
|
|
|
mgp->running = MYRI10GE_ETH_STOPPING;
|
2017-02-03 02:50:48 +08:00
|
|
|
for (i = 0; i < mgp->num_slices; i++)
|
2008-05-09 08:21:49 +08:00
|
|
|
napi_disable(&mgp->ss[i].napi);
|
2017-02-03 02:50:48 +08:00
|
|
|
|
2006-05-23 18:10:15 +08:00
|
|
|
netif_carrier_off(dev);
|
2008-09-28 23:34:21 +08:00
|
|
|
|
|
|
|
netif_tx_stop_all_queues(dev);
|
2009-08-07 18:44:22 +08:00
|
|
|
if (mgp->rebooted == 0) {
|
|
|
|
old_down_cnt = mgp->down_cnt;
|
|
|
|
mb();
|
|
|
|
status =
|
|
|
|
myri10ge_send_cmd(mgp, MXGEFW_CMD_ETHERNET_DOWN, &cmd, 0);
|
|
|
|
if (status)
|
2010-02-23 00:56:58 +08:00
|
|
|
netdev_err(dev, "Couldn't bring down link\n");
|
2006-05-23 18:10:15 +08:00
|
|
|
|
2009-08-07 18:44:22 +08:00
|
|
|
wait_event_timeout(mgp->down_wq, old_down_cnt != mgp->down_cnt,
|
|
|
|
HZ);
|
|
|
|
if (old_down_cnt == mgp->down_cnt)
|
2010-02-23 00:56:58 +08:00
|
|
|
netdev_err(dev, "never got down irq\n");
|
2009-08-07 18:44:22 +08:00
|
|
|
}
|
2006-05-23 18:10:15 +08:00
|
|
|
netif_tx_disable(dev);
|
2006-12-18 18:50:40 +08:00
|
|
|
myri10ge_free_irq(mgp);
|
2008-05-09 08:21:49 +08:00
|
|
|
for (i = 0; i < mgp->num_slices; i++)
|
|
|
|
myri10ge_free_rings(&mgp->ss[i]);
|
2006-05-23 18:10:15 +08:00
|
|
|
|
|
|
|
mgp->running = MYRI10GE_ETH_STOPPED;
|
|
|
|
return 0;
|
|
|
|
}
|
|
|
|
|
|
|
|
/* copy an array of struct mcp_kreq_ether_send's to the mcp. Copy
|
|
|
|
* backwards one at a time and handle ring wraps */
|
|
|
|
|
|
|
|
static inline void
|
|
|
|
myri10ge_submit_req_backwards(struct myri10ge_tx_buf *tx,
|
|
|
|
struct mcp_kreq_ether_send *src, int cnt)
|
|
|
|
{
|
|
|
|
int idx, starting_slot;
|
|
|
|
starting_slot = tx->req;
|
|
|
|
while (cnt > 1) {
|
|
|
|
cnt--;
|
|
|
|
idx = (starting_slot + cnt) & tx->mask;
|
|
|
|
myri10ge_pio_copy(&tx->lanai[idx], &src[cnt], sizeof(*src));
|
|
|
|
mb();
|
|
|
|
}
|
|
|
|
}
|
|
|
|
|
|
|
|
/*
|
|
|
|
* copy an array of struct mcp_kreq_ether_send's to the mcp. Copy
|
|
|
|
* at most 32 bytes at a time, so as to avoid involving the software
|
|
|
|
* pio handler in the nic. We re-write the first segment's flags
|
|
|
|
* to mark them valid only after writing the entire chain.
|
|
|
|
*/
|
|
|
|
|
|
|
|
static inline void
|
|
|
|
myri10ge_submit_req(struct myri10ge_tx_buf *tx, struct mcp_kreq_ether_send *src,
|
|
|
|
int cnt)
|
|
|
|
{
|
|
|
|
int idx, i;
|
|
|
|
struct mcp_kreq_ether_send __iomem *dstp, *dst;
|
|
|
|
struct mcp_kreq_ether_send *srcp;
|
|
|
|
u8 last_flags;
|
|
|
|
|
|
|
|
idx = tx->req & tx->mask;
|
|
|
|
|
|
|
|
last_flags = src->flags;
|
|
|
|
src->flags = 0;
|
|
|
|
mb();
|
|
|
|
dst = dstp = &tx->lanai[idx];
|
|
|
|
srcp = src;
|
|
|
|
|
|
|
|
if ((idx + cnt) < tx->mask) {
|
|
|
|
for (i = 0; i < (cnt - 1); i += 2) {
|
|
|
|
myri10ge_pio_copy(dstp, srcp, 2 * sizeof(*src));
|
|
|
|
mb(); /* force write every 32 bytes */
|
|
|
|
srcp += 2;
|
|
|
|
dstp += 2;
|
|
|
|
}
|
|
|
|
} else {
|
|
|
|
/* submit all but the first request, and ensure
|
|
|
|
* that it is submitted below */
|
|
|
|
myri10ge_submit_req_backwards(tx, src, cnt);
|
|
|
|
i = 0;
|
|
|
|
}
|
|
|
|
if (i < cnt) {
|
|
|
|
/* submit the first request */
|
|
|
|
myri10ge_pio_copy(dstp, srcp, sizeof(*src));
|
|
|
|
mb(); /* barrier before setting valid flag */
|
|
|
|
}
|
|
|
|
|
|
|
|
/* re-write the last 32-bits with the valid flags */
|
|
|
|
src->flags = last_flags;
|
2006-11-21 02:48:32 +08:00
|
|
|
put_be32(*((__be32 *) src + 3), (__be32 __iomem *) dst + 3);
|
2006-05-23 18:10:15 +08:00
|
|
|
tx->req += cnt;
|
|
|
|
mb();
|
|
|
|
}
|
|
|
|
|
2014-08-12 16:35:19 +08:00
|
|
|
static void myri10ge_unmap_tx_dma(struct myri10ge_priv *mgp,
|
|
|
|
struct myri10ge_tx_buf *tx, int idx)
|
|
|
|
{
|
|
|
|
unsigned int len;
|
|
|
|
int last_idx;
|
|
|
|
|
|
|
|
/* Free any DMA resources we've alloced and clear out the skb slot */
|
|
|
|
last_idx = (idx + 1) & tx->mask;
|
|
|
|
idx = tx->req & tx->mask;
|
|
|
|
do {
|
|
|
|
len = dma_unmap_len(&tx->info[idx], len);
|
|
|
|
if (len) {
|
|
|
|
if (tx->info[idx].skb != NULL)
|
|
|
|
pci_unmap_single(mgp->pdev,
|
|
|
|
dma_unmap_addr(&tx->info[idx],
|
|
|
|
bus), len,
|
|
|
|
PCI_DMA_TODEVICE);
|
|
|
|
else
|
|
|
|
pci_unmap_page(mgp->pdev,
|
|
|
|
dma_unmap_addr(&tx->info[idx],
|
|
|
|
bus), len,
|
|
|
|
PCI_DMA_TODEVICE);
|
|
|
|
dma_unmap_len_set(&tx->info[idx], len, 0);
|
|
|
|
tx->info[idx].skb = NULL;
|
|
|
|
}
|
|
|
|
idx = (idx + 1) & tx->mask;
|
|
|
|
} while (idx != last_idx);
|
|
|
|
}
|
|
|
|
|
2006-05-23 18:10:15 +08:00
|
|
|
/*
|
|
|
|
* Transmit a packet. We need to split the packet so that a single
|
2008-05-09 08:20:03 +08:00
|
|
|
* segment does not cross myri10ge->tx_boundary, so this makes segment
|
2006-05-23 18:10:15 +08:00
|
|
|
* counting tricky. So rather than try to count segments up front, we
|
|
|
|
* just give up if there are too few segments to hold a reasonably
|
|
|
|
* fragmented packet currently available. If we run
|
|
|
|
* out of segments while preparing a packet for DMA, we just linearize
|
|
|
|
* it and try again.
|
|
|
|
*/
|
|
|
|
|
2009-09-01 03:50:58 +08:00
|
|
|
static netdev_tx_t myri10ge_xmit(struct sk_buff *skb,
|
|
|
|
struct net_device *dev)
|
2006-05-23 18:10:15 +08:00
|
|
|
{
|
|
|
|
struct myri10ge_priv *mgp = netdev_priv(dev);
|
2008-05-09 08:20:03 +08:00
|
|
|
struct myri10ge_slice_state *ss;
|
2006-05-23 18:10:15 +08:00
|
|
|
struct mcp_kreq_ether_send *req;
|
2008-05-09 08:20:03 +08:00
|
|
|
struct myri10ge_tx_buf *tx;
|
2019-07-23 11:08:25 +08:00
|
|
|
skb_frag_t *frag;
|
2008-09-28 23:34:21 +08:00
|
|
|
struct netdev_queue *netdev_queue;
|
2006-05-23 18:10:15 +08:00
|
|
|
dma_addr_t bus;
|
2006-11-21 02:48:32 +08:00
|
|
|
u32 low;
|
|
|
|
__be32 high_swapped;
|
2006-05-23 18:10:15 +08:00
|
|
|
unsigned int len;
|
2014-08-12 16:35:19 +08:00
|
|
|
int idx, avail, frag_cnt, frag_idx, count, mss, max_segments;
|
2008-09-28 23:34:21 +08:00
|
|
|
u16 pseudo_hdr_offset, cksum_offset, queue;
|
2006-05-23 18:10:15 +08:00
|
|
|
int cum_len, seglen, boundary, rdma_count;
|
|
|
|
u8 flags, odd_flag;
|
|
|
|
|
2008-09-28 23:34:21 +08:00
|
|
|
queue = skb_get_queue_mapping(skb);
|
|
|
|
ss = &mgp->ss[queue];
|
|
|
|
netdev_queue = netdev_get_tx_queue(mgp->dev, queue);
|
2008-05-09 08:20:03 +08:00
|
|
|
tx = &ss->tx;
|
2008-09-28 23:34:21 +08:00
|
|
|
|
2006-05-23 18:10:15 +08:00
|
|
|
again:
|
|
|
|
req = tx->req_list;
|
|
|
|
avail = tx->mask - 1 - (tx->req - tx->done);
|
|
|
|
|
|
|
|
mss = 0;
|
|
|
|
max_segments = MXGEFW_MAX_SEND_DESC;
|
|
|
|
|
2007-03-28 03:54:53 +08:00
|
|
|
if (skb_is_gso(skb)) {
|
2006-06-22 17:40:14 +08:00
|
|
|
mss = skb_shinfo(skb)->gso_size;
|
2007-03-28 03:54:53 +08:00
|
|
|
max_segments = MYRI10GE_MAX_SEND_DESC_TSO;
|
2006-05-23 18:10:15 +08:00
|
|
|
}
|
|
|
|
|
|
|
|
if ((unlikely(avail < max_segments))) {
|
|
|
|
/* we are out of transmit resources */
|
2008-05-09 08:20:03 +08:00
|
|
|
tx->stop_queue++;
|
2008-09-28 23:34:21 +08:00
|
|
|
netif_tx_stop_queue(netdev_queue);
|
2009-06-12 14:22:29 +08:00
|
|
|
return NETDEV_TX_BUSY;
|
2006-05-23 18:10:15 +08:00
|
|
|
}
|
|
|
|
|
|
|
|
/* Setup checksum offloading, if needed */
|
|
|
|
cksum_offset = 0;
|
|
|
|
pseudo_hdr_offset = 0;
|
|
|
|
odd_flag = 0;
|
|
|
|
flags = (MXGEFW_FLAGS_NO_TSO | MXGEFW_FLAGS_FIRST);
|
2006-08-30 07:44:56 +08:00
|
|
|
if (likely(skb->ip_summed == CHECKSUM_PARTIAL)) {
|
2010-12-14 23:24:08 +08:00
|
|
|
cksum_offset = skb_checksum_start_offset(skb);
|
2006-11-21 10:07:29 +08:00
|
|
|
pseudo_hdr_offset = cksum_offset + skb->csum_offset;
|
2006-05-23 18:10:15 +08:00
|
|
|
/* If the headers are excessively large, then we must
|
|
|
|
* fall back to a software checksum */
|
2007-10-13 18:34:01 +08:00
|
|
|
if (unlikely(!mss && (cksum_offset > 255 ||
|
|
|
|
pseudo_hdr_offset > 127))) {
|
2006-08-30 07:44:56 +08:00
|
|
|
if (skb_checksum_help(skb))
|
2006-05-23 18:10:15 +08:00
|
|
|
goto drop;
|
|
|
|
cksum_offset = 0;
|
|
|
|
pseudo_hdr_offset = 0;
|
|
|
|
} else {
|
|
|
|
odd_flag = MXGEFW_FLAGS_ALIGN_ODD;
|
|
|
|
flags |= MXGEFW_FLAGS_CKSUM;
|
|
|
|
}
|
|
|
|
}
|
|
|
|
|
|
|
|
cum_len = 0;
|
|
|
|
|
|
|
|
if (mss) { /* TSO */
|
|
|
|
/* this removes any CKSUM flag from before */
|
|
|
|
flags = (MXGEFW_FLAGS_TSO_HDR | MXGEFW_FLAGS_FIRST);
|
|
|
|
|
|
|
|
/* negative cum_len signifies to the
|
|
|
|
* send loop that we are still in the
|
|
|
|
* header portion of the TSO packet.
|
2007-10-13 18:34:01 +08:00
|
|
|
* TSO header can be at most 1KB long */
|
2007-03-19 08:43:48 +08:00
|
|
|
cum_len = -(skb_transport_offset(skb) + tcp_hdrlen(skb));
|
2006-05-23 18:10:15 +08:00
|
|
|
|
2007-10-13 18:34:01 +08:00
|
|
|
/* for IPv6 TSO, the checksum offset stores the
|
|
|
|
* TCP header length, to save the firmware from
|
|
|
|
* the need to parse the headers */
|
|
|
|
if (skb_is_gso_v6(skb)) {
|
|
|
|
cksum_offset = tcp_hdrlen(skb);
|
|
|
|
/* Can only handle headers <= max_tso6 long */
|
|
|
|
if (unlikely(-cum_len > mgp->max_tso6))
|
|
|
|
return myri10ge_sw_tso(skb, dev);
|
|
|
|
}
|
2006-05-23 18:10:15 +08:00
|
|
|
/* for TSO, pseudo_hdr_offset holds mss.
|
|
|
|
* The firmware figures out where to put
|
|
|
|
* the checksum by parsing the header. */
|
2006-11-21 02:48:32 +08:00
|
|
|
pseudo_hdr_offset = mss;
|
2006-05-23 18:10:15 +08:00
|
|
|
} else
|
|
|
|
/* Mark small packets, and pad out tiny packets */
|
|
|
|
if (skb->len <= MXGEFW_SEND_SMALL_SIZE) {
|
|
|
|
flags |= MXGEFW_FLAGS_SMALL;
|
|
|
|
|
|
|
|
/* pad frames to at least ETH_ZLEN bytes */
|
2014-12-04 00:17:58 +08:00
|
|
|
if (eth_skb_pad(skb)) {
|
|
|
|
/* The packet is gone, so we must
|
|
|
|
* return 0 */
|
|
|
|
ss->stats.tx_dropped += 1;
|
|
|
|
return NETDEV_TX_OK;
|
2006-05-23 18:10:15 +08:00
|
|
|
}
|
|
|
|
}
|
|
|
|
|
|
|
|
/* map the skb for DMA */
|
2010-04-15 06:59:40 +08:00
|
|
|
len = skb_headlen(skb);
|
2014-08-12 16:35:19 +08:00
|
|
|
bus = pci_map_single(mgp->pdev, skb->data, len, PCI_DMA_TODEVICE);
|
|
|
|
if (unlikely(pci_dma_mapping_error(mgp->pdev, bus)))
|
|
|
|
goto drop;
|
|
|
|
|
2006-05-23 18:10:15 +08:00
|
|
|
idx = tx->req & tx->mask;
|
|
|
|
tx->info[idx].skb = skb;
|
2010-04-12 22:32:10 +08:00
|
|
|
dma_unmap_addr_set(&tx->info[idx], bus, bus);
|
|
|
|
dma_unmap_len_set(&tx->info[idx], len, len);
|
2006-05-23 18:10:15 +08:00
|
|
|
|
|
|
|
frag_cnt = skb_shinfo(skb)->nr_frags;
|
|
|
|
frag_idx = 0;
|
|
|
|
count = 0;
|
|
|
|
rdma_count = 0;
|
|
|
|
|
|
|
|
/* "rdma_count" is the number of RDMAs belonging to the
|
|
|
|
* current packet BEFORE the current send request. For
|
|
|
|
* non-TSO packets, this is equal to "count".
|
|
|
|
* For TSO packets, rdma_count needs to be reset
|
|
|
|
* to 0 after a segment cut.
|
|
|
|
*
|
|
|
|
* The rdma_count field of the send request is
|
|
|
|
* the number of RDMAs of the packet starting at
|
|
|
|
* that request. For TSO send requests with one ore more cuts
|
|
|
|
* in the middle, this is the number of RDMAs starting
|
|
|
|
* after the last cut in the request. All previous
|
|
|
|
* segments before the last cut implicitly have 1 RDMA.
|
|
|
|
*
|
|
|
|
* Since the number of RDMAs is not known beforehand,
|
|
|
|
* it must be filled-in retroactively - after each
|
|
|
|
* segmentation cut or at the end of the entire packet.
|
|
|
|
*/
|
|
|
|
|
|
|
|
while (1) {
|
|
|
|
/* Break the SKB or Fragment up into pieces which
|
2008-05-09 08:20:03 +08:00
|
|
|
* do not cross mgp->tx_boundary */
|
2006-05-23 18:10:15 +08:00
|
|
|
low = MYRI10GE_LOWPART_TO_U32(bus);
|
|
|
|
high_swapped = htonl(MYRI10GE_HIGHPART_TO_U32(bus));
|
|
|
|
while (len) {
|
|
|
|
u8 flags_next;
|
|
|
|
int cum_len_next;
|
|
|
|
|
|
|
|
if (unlikely(count == max_segments))
|
|
|
|
goto abort_linearize;
|
|
|
|
|
2008-05-09 08:20:03 +08:00
|
|
|
boundary =
|
|
|
|
(low + mgp->tx_boundary) & ~(mgp->tx_boundary - 1);
|
2006-05-23 18:10:15 +08:00
|
|
|
seglen = boundary - low;
|
|
|
|
if (seglen > len)
|
|
|
|
seglen = len;
|
|
|
|
flags_next = flags & ~MXGEFW_FLAGS_FIRST;
|
|
|
|
cum_len_next = cum_len + seglen;
|
|
|
|
if (mss) { /* TSO */
|
|
|
|
(req - rdma_count)->rdma_count = rdma_count + 1;
|
|
|
|
|
|
|
|
if (likely(cum_len >= 0)) { /* payload */
|
|
|
|
int next_is_first, chop;
|
|
|
|
|
|
|
|
chop = (cum_len_next > mss);
|
|
|
|
cum_len_next = cum_len_next % mss;
|
|
|
|
next_is_first = (cum_len_next == 0);
|
|
|
|
flags |= chop * MXGEFW_FLAGS_TSO_CHOP;
|
|
|
|
flags_next |= next_is_first *
|
|
|
|
MXGEFW_FLAGS_FIRST;
|
|
|
|
rdma_count |= -(chop | next_is_first);
|
2012-12-04 18:17:15 +08:00
|
|
|
rdma_count += chop & ~next_is_first;
|
2006-05-23 18:10:15 +08:00
|
|
|
} else if (likely(cum_len_next >= 0)) { /* header ends */
|
|
|
|
int small;
|
|
|
|
|
|
|
|
rdma_count = -1;
|
|
|
|
cum_len_next = 0;
|
|
|
|
seglen = -cum_len;
|
|
|
|
small = (mss <= MXGEFW_SEND_SMALL_SIZE);
|
|
|
|
flags_next = MXGEFW_FLAGS_TSO_PLD |
|
|
|
|
MXGEFW_FLAGS_FIRST |
|
|
|
|
(small * MXGEFW_FLAGS_SMALL);
|
|
|
|
}
|
|
|
|
}
|
|
|
|
req->addr_high = high_swapped;
|
|
|
|
req->addr_low = htonl(low);
|
2006-11-21 02:48:32 +08:00
|
|
|
req->pseudo_hdr_offset = htons(pseudo_hdr_offset);
|
2006-05-23 18:10:15 +08:00
|
|
|
req->pad = 0; /* complete solid 16-byte block; does this matter? */
|
|
|
|
req->rdma_count = 1;
|
|
|
|
req->length = htons(seglen);
|
|
|
|
req->cksum_offset = cksum_offset;
|
|
|
|
req->flags = flags | ((cum_len & 1) * odd_flag);
|
|
|
|
|
|
|
|
low += seglen;
|
|
|
|
len -= seglen;
|
|
|
|
cum_len = cum_len_next;
|
|
|
|
flags = flags_next;
|
|
|
|
req++;
|
|
|
|
count++;
|
|
|
|
rdma_count++;
|
2007-10-13 18:34:01 +08:00
|
|
|
if (cksum_offset != 0 && !(mss && skb_is_gso_v6(skb))) {
|
|
|
|
if (unlikely(cksum_offset > seglen))
|
|
|
|
cksum_offset -= seglen;
|
|
|
|
else
|
|
|
|
cksum_offset = 0;
|
|
|
|
}
|
2006-05-23 18:10:15 +08:00
|
|
|
}
|
|
|
|
if (frag_idx == frag_cnt)
|
|
|
|
break;
|
|
|
|
|
|
|
|
/* map next fragment for DMA */
|
|
|
|
frag = &skb_shinfo(skb)->frags[frag_idx];
|
|
|
|
frag_idx++;
|
2011-10-19 05:00:24 +08:00
|
|
|
len = skb_frag_size(frag);
|
2011-10-05 08:28:50 +08:00
|
|
|
bus = skb_frag_dma_map(&mgp->pdev->dev, frag, 0, len,
|
2011-10-06 18:10:48 +08:00
|
|
|
DMA_TO_DEVICE);
|
2014-08-12 16:35:19 +08:00
|
|
|
if (unlikely(pci_dma_mapping_error(mgp->pdev, bus))) {
|
|
|
|
myri10ge_unmap_tx_dma(mgp, tx, idx);
|
|
|
|
goto drop;
|
|
|
|
}
|
|
|
|
idx = (count + tx->req) & tx->mask;
|
2010-04-12 22:32:10 +08:00
|
|
|
dma_unmap_addr_set(&tx->info[idx], bus, bus);
|
|
|
|
dma_unmap_len_set(&tx->info[idx], len, len);
|
2006-05-23 18:10:15 +08:00
|
|
|
}
|
|
|
|
|
|
|
|
(req - rdma_count)->rdma_count = rdma_count;
|
|
|
|
if (mss)
|
|
|
|
do {
|
|
|
|
req--;
|
|
|
|
req->flags |= MXGEFW_FLAGS_TSO_LAST;
|
|
|
|
} while (!(req->flags & (MXGEFW_FLAGS_TSO_CHOP |
|
|
|
|
MXGEFW_FLAGS_FIRST)));
|
|
|
|
idx = ((count - 1) + tx->req) & tx->mask;
|
|
|
|
tx->info[idx].last = 1;
|
2008-07-21 16:25:50 +08:00
|
|
|
myri10ge_submit_req(tx, tx->req_list, count);
|
2008-09-28 23:34:21 +08:00
|
|
|
/* if using multiple tx queues, make sure NIC polls the
|
|
|
|
* current slice */
|
|
|
|
if ((mgp->dev->real_num_tx_queues > 1) && tx->queue_active == 0) {
|
|
|
|
tx->queue_active = 1;
|
|
|
|
put_be32(htonl(1), tx->send_go);
|
2008-11-10 20:58:41 +08:00
|
|
|
mb();
|
2008-09-28 23:34:21 +08:00
|
|
|
}
|
2006-05-23 18:10:15 +08:00
|
|
|
tx->pkt_start++;
|
|
|
|
if ((avail - count) < MXGEFW_MAX_SEND_DESC) {
|
2008-05-09 08:20:03 +08:00
|
|
|
tx->stop_queue++;
|
2008-09-28 23:34:21 +08:00
|
|
|
netif_tx_stop_queue(netdev_queue);
|
2006-05-23 18:10:15 +08:00
|
|
|
}
|
2009-06-23 14:03:08 +08:00
|
|
|
return NETDEV_TX_OK;
|
2006-05-23 18:10:15 +08:00
|
|
|
|
|
|
|
abort_linearize:
|
2014-08-12 16:35:19 +08:00
|
|
|
myri10ge_unmap_tx_dma(mgp, tx, idx);
|
2006-05-23 18:10:15 +08:00
|
|
|
|
2006-07-09 04:34:32 +08:00
|
|
|
if (skb_is_gso(skb)) {
|
2010-02-23 00:56:58 +08:00
|
|
|
netdev_err(mgp->dev, "TSO but wanted to linearize?!?!?\n");
|
2006-05-23 18:10:15 +08:00
|
|
|
goto drop;
|
|
|
|
}
|
|
|
|
|
2006-06-23 05:47:19 +08:00
|
|
|
if (skb_linearize(skb))
|
2006-05-23 18:10:15 +08:00
|
|
|
goto drop;
|
|
|
|
|
2008-05-09 08:20:03 +08:00
|
|
|
tx->linearized++;
|
2006-05-23 18:10:15 +08:00
|
|
|
goto again;
|
|
|
|
|
|
|
|
drop:
|
|
|
|
dev_kfree_skb_any(skb);
|
2008-05-09 08:20:03 +08:00
|
|
|
ss->stats.tx_dropped += 1;
|
2009-06-23 14:03:08 +08:00
|
|
|
return NETDEV_TX_OK;
|
2006-05-23 18:10:15 +08:00
|
|
|
|
|
|
|
}
|
|
|
|
|
2009-09-01 03:50:58 +08:00
|
|
|
static netdev_tx_t myri10ge_sw_tso(struct sk_buff *skb,
|
|
|
|
struct net_device *dev)
|
2007-10-13 18:34:01 +08:00
|
|
|
{
|
2020-01-09 05:59:08 +08:00
|
|
|
struct sk_buff *segs, *curr, *next;
|
2008-05-09 08:20:03 +08:00
|
|
|
struct myri10ge_priv *mgp = netdev_priv(dev);
|
2008-11-20 17:50:04 +08:00
|
|
|
struct myri10ge_slice_state *ss;
|
2009-09-01 03:50:58 +08:00
|
|
|
netdev_tx_t status;
|
2007-10-13 18:34:01 +08:00
|
|
|
|
|
|
|
segs = skb_gso_segment(skb, dev->features & ~NETIF_F_TSO6);
|
2008-04-29 16:03:09 +08:00
|
|
|
if (IS_ERR(segs))
|
2007-10-13 18:34:01 +08:00
|
|
|
goto drop;
|
|
|
|
|
2020-01-09 05:59:08 +08:00
|
|
|
skb_list_walk_safe(segs, curr, next) {
|
|
|
|
skb_mark_not_on_list(curr);
|
2007-10-13 18:34:01 +08:00
|
|
|
status = myri10ge_xmit(curr, dev);
|
|
|
|
if (status != 0) {
|
|
|
|
dev_kfree_skb_any(curr);
|
|
|
|
if (segs != NULL) {
|
|
|
|
curr = segs;
|
|
|
|
segs = segs->next;
|
|
|
|
curr->next = NULL;
|
|
|
|
dev_kfree_skb_any(segs);
|
|
|
|
}
|
|
|
|
goto drop;
|
|
|
|
}
|
|
|
|
}
|
|
|
|
dev_kfree_skb_any(skb);
|
2009-07-06 10:23:38 +08:00
|
|
|
return NETDEV_TX_OK;
|
2007-10-13 18:34:01 +08:00
|
|
|
|
|
|
|
drop:
|
2008-11-20 17:50:04 +08:00
|
|
|
ss = &mgp->ss[skb_get_queue_mapping(skb)];
|
2007-10-13 18:34:01 +08:00
|
|
|
dev_kfree_skb_any(skb);
|
2008-11-20 17:50:04 +08:00
|
|
|
ss->stats.tx_dropped += 1;
|
2009-07-06 10:23:38 +08:00
|
|
|
return NETDEV_TX_OK;
|
2007-10-13 18:34:01 +08:00
|
|
|
}
|
|
|
|
|
2017-01-07 11:12:52 +08:00
|
|
|
static void myri10ge_get_stats(struct net_device *dev,
|
|
|
|
struct rtnl_link_stats64 *stats)
|
2006-05-23 18:10:15 +08:00
|
|
|
{
|
2011-06-20 04:07:46 +08:00
|
|
|
const struct myri10ge_priv *mgp = netdev_priv(dev);
|
|
|
|
const struct myri10ge_slice_netstats *slice_stats;
|
2008-05-09 08:21:49 +08:00
|
|
|
int i;
|
|
|
|
|
|
|
|
for (i = 0; i < mgp->num_slices; i++) {
|
|
|
|
slice_stats = &mgp->ss[i].stats;
|
|
|
|
stats->rx_packets += slice_stats->rx_packets;
|
|
|
|
stats->tx_packets += slice_stats->tx_packets;
|
|
|
|
stats->rx_bytes += slice_stats->rx_bytes;
|
|
|
|
stats->tx_bytes += slice_stats->tx_bytes;
|
|
|
|
stats->rx_dropped += slice_stats->rx_dropped;
|
|
|
|
stats->tx_dropped += slice_stats->tx_dropped;
|
|
|
|
}
|
2006-05-23 18:10:15 +08:00
|
|
|
}
|
|
|
|
|
|
|
|
static void myri10ge_set_multicast_list(struct net_device *dev)
|
|
|
|
{
|
2008-05-09 08:20:03 +08:00
|
|
|
struct myri10ge_priv *mgp = netdev_priv(dev);
|
2006-08-22 05:36:56 +08:00
|
|
|
struct myri10ge_cmd cmd;
|
2010-04-02 05:22:57 +08:00
|
|
|
struct netdev_hw_addr *ha;
|
2006-12-11 18:24:37 +08:00
|
|
|
__be32 data[2] = { 0, 0 };
|
2006-08-22 05:36:56 +08:00
|
|
|
int err;
|
|
|
|
|
2006-05-23 18:10:15 +08:00
|
|
|
/* can be called from atomic contexts,
|
|
|
|
* pass 1 to force atomicity in myri10ge_send_cmd() */
|
2006-08-22 05:36:56 +08:00
|
|
|
myri10ge_change_promisc(mgp, dev->flags & IFF_PROMISC, 1);
|
|
|
|
|
|
|
|
/* This firmware is known to not support multicast */
|
2007-05-08 05:50:37 +08:00
|
|
|
if (!mgp->fw_multicast_support)
|
2006-08-22 05:36:56 +08:00
|
|
|
return;
|
|
|
|
|
|
|
|
/* Disable multicast filtering */
|
|
|
|
|
|
|
|
err = myri10ge_send_cmd(mgp, MXGEFW_ENABLE_ALLMULTI, &cmd, 1);
|
|
|
|
if (err != 0) {
|
2010-02-23 00:56:58 +08:00
|
|
|
netdev_err(dev, "Failed MXGEFW_ENABLE_ALLMULTI, error status: %d\n",
|
|
|
|
err);
|
2006-08-22 05:36:56 +08:00
|
|
|
goto abort;
|
|
|
|
}
|
|
|
|
|
2007-05-08 05:50:37 +08:00
|
|
|
if ((dev->flags & IFF_ALLMULTI) || mgp->adopted_rx_filter_bug) {
|
2006-08-22 05:36:56 +08:00
|
|
|
/* request to disable multicast filtering, so quit here */
|
|
|
|
return;
|
|
|
|
}
|
|
|
|
|
|
|
|
/* Flush the filters */
|
|
|
|
|
|
|
|
err = myri10ge_send_cmd(mgp, MXGEFW_LEAVE_ALL_MULTICAST_GROUPS,
|
|
|
|
&cmd, 1);
|
|
|
|
if (err != 0) {
|
2010-02-23 00:56:58 +08:00
|
|
|
netdev_err(dev, "Failed MXGEFW_LEAVE_ALL_MULTICAST_GROUPS, error status: %d\n",
|
|
|
|
err);
|
2006-08-22 05:36:56 +08:00
|
|
|
goto abort;
|
|
|
|
}
|
|
|
|
|
|
|
|
/* Walk the multicast list, and add each address */
|
2010-04-02 05:22:57 +08:00
|
|
|
netdev_for_each_mc_addr(ha, dev) {
|
2013-10-02 10:04:40 +08:00
|
|
|
memcpy(data, &ha->addr, ETH_ALEN);
|
2006-11-21 02:48:32 +08:00
|
|
|
cmd.data0 = ntohl(data[0]);
|
|
|
|
cmd.data1 = ntohl(data[1]);
|
2006-08-22 05:36:56 +08:00
|
|
|
err = myri10ge_send_cmd(mgp, MXGEFW_JOIN_MULTICAST_GROUP,
|
|
|
|
&cmd, 1);
|
|
|
|
|
|
|
|
if (err != 0) {
|
2010-02-23 00:56:58 +08:00
|
|
|
netdev_err(dev, "Failed MXGEFW_JOIN_MULTICAST_GROUP, error status:%d %pM\n",
|
2010-04-02 05:22:57 +08:00
|
|
|
err, ha->addr);
|
2006-08-22 05:36:56 +08:00
|
|
|
goto abort;
|
|
|
|
}
|
|
|
|
}
|
|
|
|
/* Enable multicast filtering */
|
|
|
|
err = myri10ge_send_cmd(mgp, MXGEFW_DISABLE_ALLMULTI, &cmd, 1);
|
|
|
|
if (err != 0) {
|
2010-02-23 00:56:58 +08:00
|
|
|
netdev_err(dev, "Failed MXGEFW_DISABLE_ALLMULTI, error status: %d\n",
|
|
|
|
err);
|
2006-08-22 05:36:56 +08:00
|
|
|
goto abort;
|
|
|
|
}
|
|
|
|
|
|
|
|
return;
|
|
|
|
|
|
|
|
abort:
|
|
|
|
return;
|
2006-05-23 18:10:15 +08:00
|
|
|
}
|
|
|
|
|
|
|
|
static int myri10ge_set_mac_address(struct net_device *dev, void *addr)
|
|
|
|
{
|
|
|
|
struct sockaddr *sa = addr;
|
|
|
|
struct myri10ge_priv *mgp = netdev_priv(dev);
|
|
|
|
int status;
|
|
|
|
|
|
|
|
if (!is_valid_ether_addr(sa->sa_data))
|
|
|
|
return -EADDRNOTAVAIL;
|
|
|
|
|
|
|
|
status = myri10ge_update_mac_address(mgp, sa->sa_data);
|
|
|
|
if (status != 0) {
|
2010-02-23 00:56:58 +08:00
|
|
|
netdev_err(dev, "changing mac address failed with %d\n",
|
|
|
|
status);
|
2006-05-23 18:10:15 +08:00
|
|
|
return status;
|
|
|
|
}
|
|
|
|
|
|
|
|
/* change the dev structure */
|
2013-10-02 10:04:40 +08:00
|
|
|
memcpy(dev->dev_addr, sa->sa_data, ETH_ALEN);
|
2006-05-23 18:10:15 +08:00
|
|
|
return 0;
|
|
|
|
}
|
|
|
|
|
|
|
|
static int myri10ge_change_mtu(struct net_device *dev, int new_mtu)
|
|
|
|
{
|
|
|
|
struct myri10ge_priv *mgp = netdev_priv(dev);
|
|
|
|
|
2010-02-23 00:56:58 +08:00
|
|
|
netdev_info(dev, "changing mtu from %d to %d\n", dev->mtu, new_mtu);
|
2006-05-23 18:10:15 +08:00
|
|
|
if (mgp->running) {
|
|
|
|
/* if we change the mtu on an active device, we must
|
|
|
|
* reset the device so the firmware sees the change */
|
|
|
|
myri10ge_close(dev);
|
|
|
|
dev->mtu = new_mtu;
|
|
|
|
myri10ge_open(dev);
|
|
|
|
} else
|
|
|
|
dev->mtu = new_mtu;
|
|
|
|
|
2019-07-31 16:53:46 +08:00
|
|
|
return 0;
|
2006-05-23 18:10:15 +08:00
|
|
|
}
|
|
|
|
|
|
|
|
/*
|
|
|
|
* Enable ECRC to align PCI-E Completion packets on an 8-byte boundary.
|
|
|
|
* Only do it if the bridge is a root port since we don't want to disturb
|
|
|
|
* any other device, except if forced with myri10ge_ecrc_enable > 1.
|
|
|
|
*/
|
|
|
|
|
|
|
|
static void myri10ge_enable_ecrc(struct myri10ge_priv *mgp)
|
|
|
|
{
|
|
|
|
struct pci_dev *bridge = mgp->pdev->bus->self;
|
|
|
|
struct device *dev = &mgp->pdev->dev;
|
2011-06-27 13:05:05 +08:00
|
|
|
int cap;
|
2006-05-23 18:10:15 +08:00
|
|
|
unsigned err_cap;
|
|
|
|
int ret;
|
|
|
|
|
|
|
|
if (!myri10ge_ecrc_enable || !bridge)
|
|
|
|
return;
|
|
|
|
|
|
|
|
/* check that the bridge is a root port */
|
2012-07-24 17:20:22 +08:00
|
|
|
if (pci_pcie_type(bridge) != PCI_EXP_TYPE_ROOT_PORT) {
|
2006-05-23 18:10:15 +08:00
|
|
|
if (myri10ge_ecrc_enable > 1) {
|
2008-05-09 08:19:29 +08:00
|
|
|
struct pci_dev *prev_bridge, *old_bridge = bridge;
|
2006-05-23 18:10:15 +08:00
|
|
|
|
|
|
|
/* Walk the hierarchy up to the root port
|
|
|
|
* where ECRC has to be enabled */
|
|
|
|
do {
|
2008-05-09 08:19:29 +08:00
|
|
|
prev_bridge = bridge;
|
2006-05-23 18:10:15 +08:00
|
|
|
bridge = bridge->bus->self;
|
2008-05-09 08:19:29 +08:00
|
|
|
if (!bridge || prev_bridge == bridge) {
|
2006-05-23 18:10:15 +08:00
|
|
|
dev_err(dev,
|
|
|
|
"Failed to find root port"
|
|
|
|
" to force ECRC\n");
|
|
|
|
return;
|
|
|
|
}
|
2012-07-24 17:20:22 +08:00
|
|
|
} while (pci_pcie_type(bridge) !=
|
|
|
|
PCI_EXP_TYPE_ROOT_PORT);
|
2006-05-23 18:10:15 +08:00
|
|
|
|
|
|
|
dev_info(dev,
|
|
|
|
"Forcing ECRC on non-root port %s"
|
|
|
|
" (enabling on root port %s)\n",
|
|
|
|
pci_name(old_bridge), pci_name(bridge));
|
|
|
|
} else {
|
|
|
|
dev_err(dev,
|
|
|
|
"Not enabling ECRC on non-root port %s\n",
|
|
|
|
pci_name(bridge));
|
|
|
|
return;
|
|
|
|
}
|
|
|
|
}
|
|
|
|
|
|
|
|
cap = pci_find_ext_capability(bridge, PCI_EXT_CAP_ID_ERR);
|
|
|
|
if (!cap)
|
|
|
|
return;
|
|
|
|
|
|
|
|
ret = pci_read_config_dword(bridge, cap + PCI_ERR_CAP, &err_cap);
|
|
|
|
if (ret) {
|
|
|
|
dev_err(dev, "failed reading ext-conf-space of %s\n",
|
|
|
|
pci_name(bridge));
|
|
|
|
dev_err(dev, "\t pci=nommconf in use? "
|
|
|
|
"or buggy/incomplete/absent ACPI MCFG attr?\n");
|
|
|
|
return;
|
|
|
|
}
|
|
|
|
if (!(err_cap & PCI_ERR_CAP_ECRC_GENC))
|
|
|
|
return;
|
|
|
|
|
|
|
|
err_cap |= PCI_ERR_CAP_ECRC_GENE;
|
|
|
|
pci_write_config_dword(bridge, cap + PCI_ERR_CAP, err_cap);
|
|
|
|
dev_info(dev, "Enabled ECRC on upstream bridge %s\n", pci_name(bridge));
|
|
|
|
}
|
|
|
|
|
|
|
|
/*
|
|
|
|
* The Lanai Z8E PCI-E interface achieves higher Read-DMA throughput
|
|
|
|
* when the PCI-E Completion packets are aligned on an 8-byte
|
|
|
|
* boundary. Some PCI-E chip sets always align Completion packets; on
|
|
|
|
* the ones that do not, the alignment can be enforced by enabling
|
|
|
|
* ECRC generation (if supported).
|
|
|
|
*
|
|
|
|
* When PCI-E Completion packets are not aligned, it is actually more
|
|
|
|
* efficient to limit Read-DMA transactions to 2KB, rather than 4KB.
|
|
|
|
*
|
|
|
|
* If the driver can neither enable ECRC nor verify that it has
|
|
|
|
* already been enabled, then it must use a firmware image which works
|
2008-05-09 08:21:49 +08:00
|
|
|
* around unaligned completion packets (myri10ge_rss_ethp_z8e.dat), and it
|
2006-05-23 18:10:15 +08:00
|
|
|
* should also ensure that it never gives the device a Read-DMA which is
|
2008-05-09 08:20:03 +08:00
|
|
|
* larger than 2KB by setting the tx_boundary to 2KB. If ECRC is
|
2008-05-09 08:21:49 +08:00
|
|
|
* enabled, then the driver should use the aligned (myri10ge_rss_eth_z8e.dat)
|
2008-05-09 08:20:03 +08:00
|
|
|
* firmware image, and set tx_boundary to 4KB.
|
2006-05-23 18:10:15 +08:00
|
|
|
*/
|
|
|
|
|
2007-05-08 05:52:22 +08:00
|
|
|
static void myri10ge_firmware_probe(struct myri10ge_priv *mgp)
|
2006-05-23 18:10:15 +08:00
|
|
|
{
|
2007-05-08 05:52:22 +08:00
|
|
|
struct pci_dev *pdev = mgp->pdev;
|
|
|
|
struct device *dev = &pdev->dev;
|
2007-08-24 14:57:17 +08:00
|
|
|
int status;
|
2006-05-23 18:10:15 +08:00
|
|
|
|
2008-05-09 08:20:03 +08:00
|
|
|
mgp->tx_boundary = 4096;
|
2007-05-08 05:52:22 +08:00
|
|
|
/*
|
|
|
|
* Verify the max read request size was set to 4KB
|
|
|
|
* before trying the test with 4KB.
|
|
|
|
*/
|
2007-08-24 14:57:17 +08:00
|
|
|
status = pcie_get_readrq(pdev);
|
|
|
|
if (status < 0) {
|
2007-05-08 05:52:22 +08:00
|
|
|
dev_err(dev, "Couldn't read max read req size: %d\n", status);
|
|
|
|
goto abort;
|
|
|
|
}
|
2007-08-24 14:57:17 +08:00
|
|
|
if (status != 4096) {
|
|
|
|
dev_warn(dev, "Max Read Request size != 4096 (%d)\n", status);
|
2008-05-09 08:20:03 +08:00
|
|
|
mgp->tx_boundary = 2048;
|
2007-05-08 05:52:22 +08:00
|
|
|
}
|
|
|
|
/*
|
|
|
|
* load the optimized firmware (which assumes aligned PCIe
|
|
|
|
* completions) in order to see if it works on this host.
|
|
|
|
*/
|
2010-08-12 13:04:31 +08:00
|
|
|
set_fw_name(mgp, myri10ge_fw_aligned, false);
|
2008-05-09 08:21:49 +08:00
|
|
|
status = myri10ge_load_firmware(mgp, 1);
|
2007-05-08 05:52:22 +08:00
|
|
|
if (status != 0) {
|
|
|
|
goto abort;
|
|
|
|
}
|
|
|
|
|
|
|
|
/*
|
|
|
|
* Enable ECRC if possible
|
|
|
|
*/
|
|
|
|
myri10ge_enable_ecrc(mgp);
|
|
|
|
|
|
|
|
/*
|
|
|
|
* Run a DMA test which watches for unaligned completions and
|
|
|
|
* aborts on the first one seen.
|
|
|
|
*/
|
|
|
|
|
|
|
|
status = myri10ge_dma_test(mgp, MXGEFW_CMD_UNALIGNED_TEST);
|
|
|
|
if (status == 0)
|
|
|
|
return; /* keep the aligned firmware */
|
|
|
|
|
|
|
|
if (status != -E2BIG)
|
|
|
|
dev_warn(dev, "DMA test failed: %d\n", status);
|
|
|
|
if (status == -ENOSYS)
|
|
|
|
dev_warn(dev, "Falling back to ethp! "
|
|
|
|
"Please install up to date fw\n");
|
|
|
|
abort:
|
|
|
|
/* fall back to using the unaligned firmware */
|
2008-05-09 08:20:03 +08:00
|
|
|
mgp->tx_boundary = 2048;
|
2010-08-12 13:04:31 +08:00
|
|
|
set_fw_name(mgp, myri10ge_fw_unaligned, false);
|
2007-05-08 05:52:22 +08:00
|
|
|
}
|
|
|
|
|
|
|
|
static void myri10ge_select_firmware(struct myri10ge_priv *mgp)
|
|
|
|
{
|
2009-04-16 10:24:59 +08:00
|
|
|
int overridden = 0;
|
|
|
|
|
2006-05-23 18:10:15 +08:00
|
|
|
if (myri10ge_force_firmware == 0) {
|
2012-07-24 17:20:22 +08:00
|
|
|
int link_width;
|
2006-08-31 13:32:59 +08:00
|
|
|
u16 lnk;
|
|
|
|
|
2012-07-24 17:20:22 +08:00
|
|
|
pcie_capability_read_word(mgp->pdev, PCI_EXP_LNKSTA, &lnk);
|
2006-08-31 13:32:59 +08:00
|
|
|
link_width = (lnk >> 4) & 0x3f;
|
|
|
|
|
|
|
|
/* Check to see if Link is less than 8 or if the
|
|
|
|
* upstream bridge is known to provide aligned
|
|
|
|
* completions */
|
|
|
|
if (link_width < 8) {
|
|
|
|
dev_info(&mgp->pdev->dev, "PCIE x%d Link\n",
|
|
|
|
link_width);
|
2008-05-09 08:20:03 +08:00
|
|
|
mgp->tx_boundary = 4096;
|
2010-08-12 13:04:31 +08:00
|
|
|
set_fw_name(mgp, myri10ge_fw_aligned, false);
|
2007-05-08 05:52:22 +08:00
|
|
|
} else {
|
|
|
|
myri10ge_firmware_probe(mgp);
|
2006-05-23 18:10:15 +08:00
|
|
|
}
|
|
|
|
} else {
|
|
|
|
if (myri10ge_force_firmware == 1) {
|
|
|
|
dev_info(&mgp->pdev->dev,
|
|
|
|
"Assuming aligned completions (forced)\n");
|
2008-05-09 08:20:03 +08:00
|
|
|
mgp->tx_boundary = 4096;
|
2010-08-12 13:04:31 +08:00
|
|
|
set_fw_name(mgp, myri10ge_fw_aligned, false);
|
2006-05-23 18:10:15 +08:00
|
|
|
} else {
|
|
|
|
dev_info(&mgp->pdev->dev,
|
|
|
|
"Assuming unaligned completions (forced)\n");
|
2008-05-09 08:20:03 +08:00
|
|
|
mgp->tx_boundary = 2048;
|
2010-08-12 13:04:31 +08:00
|
|
|
set_fw_name(mgp, myri10ge_fw_unaligned, false);
|
2006-05-23 18:10:15 +08:00
|
|
|
}
|
|
|
|
}
|
2010-08-12 13:04:31 +08:00
|
|
|
|
module: add per-module param_lock
Add a "param_lock" mutex to each module, and update params.c to use
the correct built-in or module mutex while locking kernel params.
Remove the kparam_block_sysfs_r/w() macros, replace them with direct
calls to kernel_param_[un]lock(module).
The kernel param code currently uses a single mutex to protect
modification of any and all kernel params. While this generally works,
there is one specific problem with it; a module callback function
cannot safely load another module, i.e. with request_module() or even
with indirect calls such as crypto_has_alg(). If the module to be
loaded has any of its params configured (e.g. with a /etc/modprobe.d/*
config file), then the attempt will result in a deadlock between the
first module param callback waiting for modprobe, and modprobe trying to
lock the single kernel param mutex to set the new module's param.
This fixes that by using per-module mutexes, so that each individual module
is protected against concurrent changes in its own kernel params, but is
not blocked by changes to other module params. All built-in modules
continue to use the built-in mutex, since they will always be loaded at
runtime and references (e.g. request_module(), crypto_has_alg()) to them
will never cause load-time param changing.
This also simplifies the interface used by modules to block sysfs access
to their params; while there are currently functions to block and unblock
sysfs param access which are split up by read and write and expect a single
kernel param to be passed, their actual operation is identical and applies
to all params, not just the one passed to them; they simply lock and unlock
the global param mutex. They are replaced with direct calls to
kernel_param_[un]lock(THIS_MODULE), which locks THIS_MODULE's param_lock, or
if the module is built-in, it locks the built-in mutex.
Suggested-by: Rusty Russell <rusty@rustcorp.com.au>
Signed-off-by: Dan Streetman <ddstreet@ieee.org>
Signed-off-by: Rusty Russell <rusty@rustcorp.com.au>
2015-06-17 04:48:52 +08:00
|
|
|
kernel_param_lock(THIS_MODULE);
|
2006-05-23 18:10:15 +08:00
|
|
|
if (myri10ge_fw_name != NULL) {
|
2010-08-12 13:04:31 +08:00
|
|
|
char *fw_name = kstrdup(myri10ge_fw_name, GFP_KERNEL);
|
|
|
|
if (fw_name) {
|
|
|
|
overridden = 1;
|
|
|
|
set_fw_name(mgp, fw_name, true);
|
|
|
|
}
|
2006-05-23 18:10:15 +08:00
|
|
|
}
|
module: add per-module param_lock
Add a "param_lock" mutex to each module, and update params.c to use
the correct built-in or module mutex while locking kernel params.
Remove the kparam_block_sysfs_r/w() macros, replace them with direct
calls to kernel_param_[un]lock(module).
The kernel param code currently uses a single mutex to protect
modification of any and all kernel params. While this generally works,
there is one specific problem with it; a module callback function
cannot safely load another module, i.e. with request_module() or even
with indirect calls such as crypto_has_alg(). If the module to be
loaded has any of its params configured (e.g. with a /etc/modprobe.d/*
config file), then the attempt will result in a deadlock between the
first module param callback waiting for modprobe, and modprobe trying to
lock the single kernel param mutex to set the new module's param.
This fixes that by using per-module mutexes, so that each individual module
is protected against concurrent changes in its own kernel params, but is
not blocked by changes to other module params. All built-in modules
continue to use the built-in mutex, since they will always be loaded at
runtime and references (e.g. request_module(), crypto_has_alg()) to them
will never cause load-time param changing.
This also simplifies the interface used by modules to block sysfs access
to their params; while there are currently functions to block and unblock
sysfs param access which are split up by read and write and expect a single
kernel param to be passed, their actual operation is identical and applies
to all params, not just the one passed to them; they simply lock and unlock
the global param mutex. They are replaced with direct calls to
kernel_param_[un]lock(THIS_MODULE), which locks THIS_MODULE's param_lock, or
if the module is built-in, it locks the built-in mutex.
Suggested-by: Rusty Russell <rusty@rustcorp.com.au>
Signed-off-by: Dan Streetman <ddstreet@ieee.org>
Signed-off-by: Rusty Russell <rusty@rustcorp.com.au>
2015-06-17 04:48:52 +08:00
|
|
|
kernel_param_unlock(THIS_MODULE);
|
2010-08-12 13:04:31 +08:00
|
|
|
|
2009-04-16 10:24:59 +08:00
|
|
|
if (mgp->board_number < MYRI10GE_MAX_BOARDS &&
|
|
|
|
myri10ge_fw_names[mgp->board_number] != NULL &&
|
|
|
|
strlen(myri10ge_fw_names[mgp->board_number])) {
|
2010-08-12 13:04:31 +08:00
|
|
|
set_fw_name(mgp, myri10ge_fw_names[mgp->board_number], false);
|
2009-04-16 10:24:59 +08:00
|
|
|
overridden = 1;
|
|
|
|
}
|
|
|
|
if (overridden)
|
|
|
|
dev_info(&mgp->pdev->dev, "overriding firmware to %s\n",
|
|
|
|
mgp->fw_name);
|
2006-05-23 18:10:15 +08:00
|
|
|
}
|
|
|
|
|
2011-06-27 13:05:01 +08:00
|
|
|
static void myri10ge_mask_surprise_down(struct pci_dev *pdev)
|
|
|
|
{
|
|
|
|
struct pci_dev *bridge = pdev->bus->self;
|
|
|
|
int cap;
|
|
|
|
u32 mask;
|
|
|
|
|
|
|
|
if (bridge == NULL)
|
|
|
|
return;
|
|
|
|
|
|
|
|
cap = pci_find_ext_capability(bridge, PCI_EXT_CAP_ID_ERR);
|
|
|
|
if (cap) {
|
|
|
|
/* a sram parity error can cause a surprise link
|
|
|
|
* down; since we expect and can recover from sram
|
|
|
|
* parity errors, mask surprise link down events */
|
|
|
|
pci_read_config_dword(bridge, cap + PCI_ERR_UNCOR_MASK, &mask);
|
|
|
|
mask |= 0x20;
|
|
|
|
pci_write_config_dword(bridge, cap + PCI_ERR_UNCOR_MASK, mask);
|
|
|
|
}
|
|
|
|
}
|
|
|
|
|
2006-05-23 18:10:15 +08:00
|
|
|
#ifdef CONFIG_PM
|
|
|
|
static int myri10ge_suspend(struct pci_dev *pdev, pm_message_t state)
|
|
|
|
{
|
|
|
|
struct myri10ge_priv *mgp;
|
|
|
|
struct net_device *netdev;
|
|
|
|
|
|
|
|
mgp = pci_get_drvdata(pdev);
|
|
|
|
if (mgp == NULL)
|
|
|
|
return -EINVAL;
|
|
|
|
netdev = mgp->dev;
|
|
|
|
|
|
|
|
netif_device_detach(netdev);
|
|
|
|
if (netif_running(netdev)) {
|
2010-02-23 00:56:58 +08:00
|
|
|
netdev_info(netdev, "closing\n");
|
2006-05-23 18:10:15 +08:00
|
|
|
rtnl_lock();
|
|
|
|
myri10ge_close(netdev);
|
|
|
|
rtnl_unlock();
|
|
|
|
}
|
|
|
|
myri10ge_dummy_rdma(mgp, 0);
|
2006-12-18 18:52:02 +08:00
|
|
|
pci_save_state(pdev);
|
2006-05-23 18:10:15 +08:00
|
|
|
pci_disable_device(pdev);
|
2006-12-18 18:52:34 +08:00
|
|
|
|
|
|
|
return pci_set_power_state(pdev, pci_choose_state(pdev, state));
|
2006-05-23 18:10:15 +08:00
|
|
|
}
|
|
|
|
|
|
|
|
static int myri10ge_resume(struct pci_dev *pdev)
|
|
|
|
{
|
|
|
|
struct myri10ge_priv *mgp;
|
|
|
|
struct net_device *netdev;
|
|
|
|
int status;
|
|
|
|
u16 vendor;
|
|
|
|
|
|
|
|
mgp = pci_get_drvdata(pdev);
|
|
|
|
if (mgp == NULL)
|
|
|
|
return -EINVAL;
|
|
|
|
netdev = mgp->dev;
|
2013-06-27 20:53:42 +08:00
|
|
|
pci_set_power_state(pdev, PCI_D0); /* zeros conf space as a side effect */
|
2006-05-23 18:10:15 +08:00
|
|
|
msleep(5); /* give card time to respond */
|
|
|
|
pci_read_config_word(mgp->pdev, PCI_VENDOR_ID, &vendor);
|
|
|
|
if (vendor == 0xffff) {
|
2010-02-23 00:56:58 +08:00
|
|
|
netdev_err(mgp->dev, "device disappeared!\n");
|
2006-05-23 18:10:15 +08:00
|
|
|
return -EIO;
|
|
|
|
}
|
2006-12-18 18:52:02 +08:00
|
|
|
|
2010-12-01 07:43:26 +08:00
|
|
|
pci_restore_state(pdev);
|
2006-07-10 09:10:18 +08:00
|
|
|
|
|
|
|
status = pci_enable_device(pdev);
|
2006-12-18 18:52:34 +08:00
|
|
|
if (status) {
|
2006-07-10 09:10:18 +08:00
|
|
|
dev_err(&pdev->dev, "failed to enable device\n");
|
2006-12-18 18:52:34 +08:00
|
|
|
return status;
|
2006-07-10 09:10:18 +08:00
|
|
|
}
|
|
|
|
|
2006-05-23 18:10:15 +08:00
|
|
|
pci_set_master(pdev);
|
|
|
|
|
|
|
|
myri10ge_reset(mgp);
|
2006-08-09 12:07:53 +08:00
|
|
|
myri10ge_dummy_rdma(mgp, 1);
|
2006-05-23 18:10:15 +08:00
|
|
|
|
|
|
|
/* Save configuration space to be restored if the
|
|
|
|
* nic resets due to a parity error */
|
2006-12-18 18:52:02 +08:00
|
|
|
pci_save_state(pdev);
|
2006-05-23 18:10:15 +08:00
|
|
|
|
|
|
|
if (netif_running(netdev)) {
|
|
|
|
rtnl_lock();
|
2006-12-18 18:50:40 +08:00
|
|
|
status = myri10ge_open(netdev);
|
2006-05-23 18:10:15 +08:00
|
|
|
rtnl_unlock();
|
2006-12-18 18:50:40 +08:00
|
|
|
if (status != 0)
|
|
|
|
goto abort_with_enabled;
|
|
|
|
|
2006-05-23 18:10:15 +08:00
|
|
|
}
|
|
|
|
netif_device_attach(netdev);
|
|
|
|
|
|
|
|
return 0;
|
|
|
|
|
2006-07-10 09:10:18 +08:00
|
|
|
abort_with_enabled:
|
|
|
|
pci_disable_device(pdev);
|
2006-05-23 18:10:15 +08:00
|
|
|
return -EIO;
|
|
|
|
|
|
|
|
}
|
|
|
|
#endif /* CONFIG_PM */
|
|
|
|
|
|
|
|
static u32 myri10ge_read_reboot(struct myri10ge_priv *mgp)
|
|
|
|
{
|
|
|
|
struct pci_dev *pdev = mgp->pdev;
|
|
|
|
int vs = mgp->vendor_specific_offset;
|
|
|
|
u32 reboot;
|
|
|
|
|
|
|
|
/*enter read32 mode */
|
|
|
|
pci_write_config_byte(pdev, vs + 0x10, 0x3);
|
|
|
|
|
|
|
|
/*read REBOOT_STATUS (0xfffffff0) */
|
|
|
|
pci_write_config_dword(pdev, vs + 0x18, 0xfffffff0);
|
|
|
|
pci_read_config_dword(pdev, vs + 0x14, &reboot);
|
|
|
|
return reboot;
|
|
|
|
}
|
|
|
|
|
2011-06-28 01:57:28 +08:00
|
|
|
static void
|
|
|
|
myri10ge_check_slice(struct myri10ge_slice_state *ss, int *reset_needed,
|
|
|
|
int *busy_slice_cnt, u32 rx_pause_cnt)
|
|
|
|
{
|
|
|
|
struct myri10ge_priv *mgp = ss->mgp;
|
|
|
|
int slice = ss - mgp->ss;
|
|
|
|
|
|
|
|
if (ss->tx.req != ss->tx.done &&
|
|
|
|
ss->tx.done == ss->watchdog_tx_done &&
|
|
|
|
ss->watchdog_tx_req != ss->watchdog_tx_done) {
|
|
|
|
/* nic seems like it might be stuck.. */
|
|
|
|
if (rx_pause_cnt != mgp->watchdog_pause) {
|
|
|
|
if (net_ratelimit())
|
|
|
|
netdev_warn(mgp->dev, "slice %d: TX paused, "
|
|
|
|
"check link partner\n", slice);
|
|
|
|
} else {
|
|
|
|
netdev_warn(mgp->dev,
|
|
|
|
"slice %d: TX stuck %d %d %d %d %d %d\n",
|
|
|
|
slice, ss->tx.queue_active, ss->tx.req,
|
|
|
|
ss->tx.done, ss->tx.pkt_start,
|
|
|
|
ss->tx.pkt_done,
|
|
|
|
(int)ntohl(mgp->ss[slice].fw_stats->
|
|
|
|
send_done_count));
|
|
|
|
*reset_needed = 1;
|
|
|
|
ss->stuck = 1;
|
|
|
|
}
|
|
|
|
}
|
|
|
|
if (ss->watchdog_tx_done != ss->tx.done ||
|
|
|
|
ss->watchdog_rx_done != ss->rx_done.cnt) {
|
|
|
|
*busy_slice_cnt += 1;
|
|
|
|
}
|
|
|
|
ss->watchdog_tx_done = ss->tx.done;
|
|
|
|
ss->watchdog_tx_req = ss->tx.req;
|
|
|
|
ss->watchdog_rx_done = ss->rx_done.cnt;
|
|
|
|
}
|
|
|
|
|
2006-05-23 18:10:15 +08:00
|
|
|
/*
|
|
|
|
* This watchdog is used to check whether the board has suffered
|
|
|
|
* from a parity error and needs to be recovered.
|
|
|
|
*/
|
2006-11-22 22:57:56 +08:00
|
|
|
static void myri10ge_watchdog(struct work_struct *work)
|
2006-05-23 18:10:15 +08:00
|
|
|
{
|
2006-11-22 22:57:56 +08:00
|
|
|
struct myri10ge_priv *mgp =
|
2006-12-11 18:24:37 +08:00
|
|
|
container_of(work, struct myri10ge_priv, watchdog_work);
|
2011-06-28 01:57:28 +08:00
|
|
|
struct myri10ge_slice_state *ss;
|
|
|
|
u32 reboot, rx_pause_cnt;
|
2009-08-07 18:44:22 +08:00
|
|
|
int status, rebooted;
|
2008-05-09 08:21:49 +08:00
|
|
|
int i;
|
2011-06-28 01:57:28 +08:00
|
|
|
int reset_needed = 0;
|
|
|
|
int busy_slice_cnt = 0;
|
2006-05-23 18:10:15 +08:00
|
|
|
u16 cmd, vendor;
|
|
|
|
|
|
|
|
mgp->watchdog_resets++;
|
|
|
|
pci_read_config_word(mgp->pdev, PCI_COMMAND, &cmd);
|
2009-08-07 18:44:22 +08:00
|
|
|
rebooted = 0;
|
2006-05-23 18:10:15 +08:00
|
|
|
if ((cmd & PCI_COMMAND_MASTER) == 0) {
|
|
|
|
/* Bus master DMA disabled? Check to see
|
|
|
|
* if the card rebooted due to a parity error
|
|
|
|
* For now, just report it */
|
|
|
|
reboot = myri10ge_read_reboot(mgp);
|
2010-02-23 00:56:58 +08:00
|
|
|
netdev_err(mgp->dev, "NIC rebooted (0x%x),%s resetting\n",
|
2011-06-28 01:57:28 +08:00
|
|
|
reboot, myri10ge_reset_recover ? "" : " not");
|
2007-06-12 02:26:31 +08:00
|
|
|
if (myri10ge_reset_recover == 0)
|
|
|
|
return;
|
2009-08-07 18:44:22 +08:00
|
|
|
rtnl_lock();
|
|
|
|
mgp->rebooted = 1;
|
|
|
|
rebooted = 1;
|
|
|
|
myri10ge_close(mgp->dev);
|
2007-06-12 02:26:31 +08:00
|
|
|
myri10ge_reset_recover--;
|
2009-08-07 18:44:22 +08:00
|
|
|
mgp->rebooted = 0;
|
2006-05-23 18:10:15 +08:00
|
|
|
/*
|
|
|
|
* A rebooted nic will come back with config space as
|
|
|
|
* it was after power was applied to PCIe bus.
|
|
|
|
* Attempt to restore config space which was saved
|
|
|
|
* when the driver was loaded, or the last time the
|
|
|
|
* nic was resumed from power saving mode.
|
|
|
|
*/
|
2006-12-18 18:52:02 +08:00
|
|
|
pci_restore_state(mgp->pdev);
|
2006-12-18 18:50:00 +08:00
|
|
|
|
|
|
|
/* save state again for accounting reasons */
|
2006-12-18 18:52:02 +08:00
|
|
|
pci_save_state(mgp->pdev);
|
2006-12-18 18:50:00 +08:00
|
|
|
|
2006-05-23 18:10:15 +08:00
|
|
|
} else {
|
|
|
|
/* if we get back -1's from our slot, perhaps somebody
|
|
|
|
* powered off our card. Don't try to reset it in
|
|
|
|
* this case */
|
|
|
|
if (cmd == 0xffff) {
|
|
|
|
pci_read_config_word(mgp->pdev, PCI_VENDOR_ID, &vendor);
|
|
|
|
if (vendor == 0xffff) {
|
2010-02-23 00:56:58 +08:00
|
|
|
netdev_err(mgp->dev, "device disappeared!\n");
|
2006-05-23 18:10:15 +08:00
|
|
|
return;
|
|
|
|
}
|
|
|
|
}
|
2011-06-28 01:57:28 +08:00
|
|
|
/* Perhaps it is a software error. See if stuck slice
|
|
|
|
* has recovered, reset if not */
|
|
|
|
rx_pause_cnt = ntohl(mgp->ss[0].fw_stats->dropped_pause);
|
2008-05-09 08:21:49 +08:00
|
|
|
for (i = 0; i < mgp->num_slices; i++) {
|
2011-06-28 01:57:28 +08:00
|
|
|
ss = mgp->ss;
|
|
|
|
if (ss->stuck) {
|
|
|
|
myri10ge_check_slice(ss, &reset_needed,
|
|
|
|
&busy_slice_cnt,
|
|
|
|
rx_pause_cnt);
|
|
|
|
ss->stuck = 0;
|
|
|
|
}
|
2008-05-09 08:21:49 +08:00
|
|
|
}
|
2011-06-28 01:57:28 +08:00
|
|
|
if (!reset_needed) {
|
|
|
|
netdev_dbg(mgp->dev, "not resetting\n");
|
|
|
|
return;
|
|
|
|
}
|
|
|
|
|
|
|
|
netdev_err(mgp->dev, "device timeout, resetting\n");
|
2006-05-23 18:10:15 +08:00
|
|
|
}
|
2008-09-28 23:34:21 +08:00
|
|
|
|
2009-08-07 18:44:22 +08:00
|
|
|
if (!rebooted) {
|
|
|
|
rtnl_lock();
|
|
|
|
myri10ge_close(mgp->dev);
|
|
|
|
}
|
2008-05-09 08:21:49 +08:00
|
|
|
status = myri10ge_load_firmware(mgp, 1);
|
2006-05-23 18:10:15 +08:00
|
|
|
if (status != 0)
|
2010-02-23 00:56:58 +08:00
|
|
|
netdev_err(mgp->dev, "failed to load firmware\n");
|
2006-05-23 18:10:15 +08:00
|
|
|
else
|
|
|
|
myri10ge_open(mgp->dev);
|
|
|
|
rtnl_unlock();
|
|
|
|
}
|
|
|
|
|
|
|
|
/*
|
|
|
|
* We use our own timer routine rather than relying upon
|
|
|
|
* netdev->tx_timeout because we have a very large hardware transmit
|
|
|
|
* queue. Due to the large queue, the netdev->tx_timeout function
|
|
|
|
* cannot detect a NIC with a parity error in a timely fashion if the
|
|
|
|
* NIC is lightly loaded.
|
|
|
|
*/
|
treewide: setup_timer() -> timer_setup()
This converts all remaining cases of the old setup_timer() API into using
timer_setup(), where the callback argument is the structure already
holding the struct timer_list. These should have no behavioral changes,
since they just change which pointer is passed into the callback with
the same available pointers after conversion. It handles the following
examples, in addition to some other variations.
Casting from unsigned long:
void my_callback(unsigned long data)
{
struct something *ptr = (struct something *)data;
...
}
...
setup_timer(&ptr->my_timer, my_callback, ptr);
and forced object casts:
void my_callback(struct something *ptr)
{
...
}
...
setup_timer(&ptr->my_timer, my_callback, (unsigned long)ptr);
become:
void my_callback(struct timer_list *t)
{
struct something *ptr = from_timer(ptr, t, my_timer);
...
}
...
timer_setup(&ptr->my_timer, my_callback, 0);
Direct function assignments:
void my_callback(unsigned long data)
{
struct something *ptr = (struct something *)data;
...
}
...
ptr->my_timer.function = my_callback;
have a temporary cast added, along with converting the args:
void my_callback(struct timer_list *t)
{
struct something *ptr = from_timer(ptr, t, my_timer);
...
}
...
ptr->my_timer.function = (TIMER_FUNC_TYPE)my_callback;
And finally, callbacks without a data assignment:
void my_callback(unsigned long data)
{
...
}
...
setup_timer(&ptr->my_timer, my_callback, 0);
have their argument renamed to verify they're unused during conversion:
void my_callback(struct timer_list *unused)
{
...
}
...
timer_setup(&ptr->my_timer, my_callback, 0);
The conversion is done with the following Coccinelle script:
spatch --very-quiet --all-includes --include-headers \
-I ./arch/x86/include -I ./arch/x86/include/generated \
-I ./include -I ./arch/x86/include/uapi \
-I ./arch/x86/include/generated/uapi -I ./include/uapi \
-I ./include/generated/uapi --include ./include/linux/kconfig.h \
--dir . \
--cocci-file ~/src/data/timer_setup.cocci
@fix_address_of@
expression e;
@@
setup_timer(
-&(e)
+&e
, ...)
// Update any raw setup_timer() usages that have a NULL callback, but
// would otherwise match change_timer_function_usage, since the latter
// will update all function assignments done in the face of a NULL
// function initialization in setup_timer().
@change_timer_function_usage_NULL@
expression _E;
identifier _timer;
type _cast_data;
@@
(
-setup_timer(&_E->_timer, NULL, _E);
+timer_setup(&_E->_timer, NULL, 0);
|
-setup_timer(&_E->_timer, NULL, (_cast_data)_E);
+timer_setup(&_E->_timer, NULL, 0);
|
-setup_timer(&_E._timer, NULL, &_E);
+timer_setup(&_E._timer, NULL, 0);
|
-setup_timer(&_E._timer, NULL, (_cast_data)&_E);
+timer_setup(&_E._timer, NULL, 0);
)
@change_timer_function_usage@
expression _E;
identifier _timer;
struct timer_list _stl;
identifier _callback;
type _cast_func, _cast_data;
@@
(
-setup_timer(&_E->_timer, _callback, _E);
+timer_setup(&_E->_timer, _callback, 0);
|
-setup_timer(&_E->_timer, &_callback, _E);
+timer_setup(&_E->_timer, _callback, 0);
|
-setup_timer(&_E->_timer, _callback, (_cast_data)_E);
+timer_setup(&_E->_timer, _callback, 0);
|
-setup_timer(&_E->_timer, &_callback, (_cast_data)_E);
+timer_setup(&_E->_timer, _callback, 0);
|
-setup_timer(&_E->_timer, (_cast_func)_callback, _E);
+timer_setup(&_E->_timer, _callback, 0);
|
-setup_timer(&_E->_timer, (_cast_func)&_callback, _E);
+timer_setup(&_E->_timer, _callback, 0);
|
-setup_timer(&_E->_timer, (_cast_func)_callback, (_cast_data)_E);
+timer_setup(&_E->_timer, _callback, 0);
|
-setup_timer(&_E->_timer, (_cast_func)&_callback, (_cast_data)_E);
+timer_setup(&_E->_timer, _callback, 0);
|
-setup_timer(&_E._timer, _callback, (_cast_data)_E);
+timer_setup(&_E._timer, _callback, 0);
|
-setup_timer(&_E._timer, _callback, (_cast_data)&_E);
+timer_setup(&_E._timer, _callback, 0);
|
-setup_timer(&_E._timer, &_callback, (_cast_data)_E);
+timer_setup(&_E._timer, _callback, 0);
|
-setup_timer(&_E._timer, &_callback, (_cast_data)&_E);
+timer_setup(&_E._timer, _callback, 0);
|
-setup_timer(&_E._timer, (_cast_func)_callback, (_cast_data)_E);
+timer_setup(&_E._timer, _callback, 0);
|
-setup_timer(&_E._timer, (_cast_func)_callback, (_cast_data)&_E);
+timer_setup(&_E._timer, _callback, 0);
|
-setup_timer(&_E._timer, (_cast_func)&_callback, (_cast_data)_E);
+timer_setup(&_E._timer, _callback, 0);
|
-setup_timer(&_E._timer, (_cast_func)&_callback, (_cast_data)&_E);
+timer_setup(&_E._timer, _callback, 0);
|
_E->_timer@_stl.function = _callback;
|
_E->_timer@_stl.function = &_callback;
|
_E->_timer@_stl.function = (_cast_func)_callback;
|
_E->_timer@_stl.function = (_cast_func)&_callback;
|
_E._timer@_stl.function = _callback;
|
_E._timer@_stl.function = &_callback;
|
_E._timer@_stl.function = (_cast_func)_callback;
|
_E._timer@_stl.function = (_cast_func)&_callback;
)
// callback(unsigned long arg)
@change_callback_handle_cast
depends on change_timer_function_usage@
identifier change_timer_function_usage._callback;
identifier change_timer_function_usage._timer;
type _origtype;
identifier _origarg;
type _handletype;
identifier _handle;
@@
void _callback(
-_origtype _origarg
+struct timer_list *t
)
{
(
... when != _origarg
_handletype *_handle =
-(_handletype *)_origarg;
+from_timer(_handle, t, _timer);
... when != _origarg
|
... when != _origarg
_handletype *_handle =
-(void *)_origarg;
+from_timer(_handle, t, _timer);
... when != _origarg
|
... when != _origarg
_handletype *_handle;
... when != _handle
_handle =
-(_handletype *)_origarg;
+from_timer(_handle, t, _timer);
... when != _origarg
|
... when != _origarg
_handletype *_handle;
... when != _handle
_handle =
-(void *)_origarg;
+from_timer(_handle, t, _timer);
... when != _origarg
)
}
// callback(unsigned long arg) without existing variable
@change_callback_handle_cast_no_arg
depends on change_timer_function_usage &&
!change_callback_handle_cast@
identifier change_timer_function_usage._callback;
identifier change_timer_function_usage._timer;
type _origtype;
identifier _origarg;
type _handletype;
@@
void _callback(
-_origtype _origarg
+struct timer_list *t
)
{
+ _handletype *_origarg = from_timer(_origarg, t, _timer);
+
... when != _origarg
- (_handletype *)_origarg
+ _origarg
... when != _origarg
}
// Avoid already converted callbacks.
@match_callback_converted
depends on change_timer_function_usage &&
!change_callback_handle_cast &&
!change_callback_handle_cast_no_arg@
identifier change_timer_function_usage._callback;
identifier t;
@@
void _callback(struct timer_list *t)
{ ... }
// callback(struct something *handle)
@change_callback_handle_arg
depends on change_timer_function_usage &&
!match_callback_converted &&
!change_callback_handle_cast &&
!change_callback_handle_cast_no_arg@
identifier change_timer_function_usage._callback;
identifier change_timer_function_usage._timer;
type _handletype;
identifier _handle;
@@
void _callback(
-_handletype *_handle
+struct timer_list *t
)
{
+ _handletype *_handle = from_timer(_handle, t, _timer);
...
}
// If change_callback_handle_arg ran on an empty function, remove
// the added handler.
@unchange_callback_handle_arg
depends on change_timer_function_usage &&
change_callback_handle_arg@
identifier change_timer_function_usage._callback;
identifier change_timer_function_usage._timer;
type _handletype;
identifier _handle;
identifier t;
@@
void _callback(struct timer_list *t)
{
- _handletype *_handle = from_timer(_handle, t, _timer);
}
// We only want to refactor the setup_timer() data argument if we've found
// the matching callback. This undoes changes in change_timer_function_usage.
@unchange_timer_function_usage
depends on change_timer_function_usage &&
!change_callback_handle_cast &&
!change_callback_handle_cast_no_arg &&
!change_callback_handle_arg@
expression change_timer_function_usage._E;
identifier change_timer_function_usage._timer;
identifier change_timer_function_usage._callback;
type change_timer_function_usage._cast_data;
@@
(
-timer_setup(&_E->_timer, _callback, 0);
+setup_timer(&_E->_timer, _callback, (_cast_data)_E);
|
-timer_setup(&_E._timer, _callback, 0);
+setup_timer(&_E._timer, _callback, (_cast_data)&_E);
)
// If we fixed a callback from a .function assignment, fix the
// assignment cast now.
@change_timer_function_assignment
depends on change_timer_function_usage &&
(change_callback_handle_cast ||
change_callback_handle_cast_no_arg ||
change_callback_handle_arg)@
expression change_timer_function_usage._E;
identifier change_timer_function_usage._timer;
identifier change_timer_function_usage._callback;
type _cast_func;
typedef TIMER_FUNC_TYPE;
@@
(
_E->_timer.function =
-_callback
+(TIMER_FUNC_TYPE)_callback
;
|
_E->_timer.function =
-&_callback
+(TIMER_FUNC_TYPE)_callback
;
|
_E->_timer.function =
-(_cast_func)_callback;
+(TIMER_FUNC_TYPE)_callback
;
|
_E->_timer.function =
-(_cast_func)&_callback
+(TIMER_FUNC_TYPE)_callback
;
|
_E._timer.function =
-_callback
+(TIMER_FUNC_TYPE)_callback
;
|
_E._timer.function =
-&_callback;
+(TIMER_FUNC_TYPE)_callback
;
|
_E._timer.function =
-(_cast_func)_callback
+(TIMER_FUNC_TYPE)_callback
;
|
_E._timer.function =
-(_cast_func)&_callback
+(TIMER_FUNC_TYPE)_callback
;
)
// Sometimes timer functions are called directly. Replace matched args.
@change_timer_function_calls
depends on change_timer_function_usage &&
(change_callback_handle_cast ||
change_callback_handle_cast_no_arg ||
change_callback_handle_arg)@
expression _E;
identifier change_timer_function_usage._timer;
identifier change_timer_function_usage._callback;
type _cast_data;
@@
_callback(
(
-(_cast_data)_E
+&_E->_timer
|
-(_cast_data)&_E
+&_E._timer
|
-_E
+&_E->_timer
)
)
// If a timer has been configured without a data argument, it can be
// converted without regard to the callback argument, since it is unused.
@match_timer_function_unused_data@
expression _E;
identifier _timer;
identifier _callback;
@@
(
-setup_timer(&_E->_timer, _callback, 0);
+timer_setup(&_E->_timer, _callback, 0);
|
-setup_timer(&_E->_timer, _callback, 0L);
+timer_setup(&_E->_timer, _callback, 0);
|
-setup_timer(&_E->_timer, _callback, 0UL);
+timer_setup(&_E->_timer, _callback, 0);
|
-setup_timer(&_E._timer, _callback, 0);
+timer_setup(&_E._timer, _callback, 0);
|
-setup_timer(&_E._timer, _callback, 0L);
+timer_setup(&_E._timer, _callback, 0);
|
-setup_timer(&_E._timer, _callback, 0UL);
+timer_setup(&_E._timer, _callback, 0);
|
-setup_timer(&_timer, _callback, 0);
+timer_setup(&_timer, _callback, 0);
|
-setup_timer(&_timer, _callback, 0L);
+timer_setup(&_timer, _callback, 0);
|
-setup_timer(&_timer, _callback, 0UL);
+timer_setup(&_timer, _callback, 0);
|
-setup_timer(_timer, _callback, 0);
+timer_setup(_timer, _callback, 0);
|
-setup_timer(_timer, _callback, 0L);
+timer_setup(_timer, _callback, 0);
|
-setup_timer(_timer, _callback, 0UL);
+timer_setup(_timer, _callback, 0);
)
@change_callback_unused_data
depends on match_timer_function_unused_data@
identifier match_timer_function_unused_data._callback;
type _origtype;
identifier _origarg;
@@
void _callback(
-_origtype _origarg
+struct timer_list *unused
)
{
... when != _origarg
}
Signed-off-by: Kees Cook <keescook@chromium.org>
2017-10-17 05:43:17 +08:00
|
|
|
static void myri10ge_watchdog_timer(struct timer_list *t)
|
2006-05-23 18:10:15 +08:00
|
|
|
{
|
|
|
|
struct myri10ge_priv *mgp;
|
2008-05-09 08:20:03 +08:00
|
|
|
struct myri10ge_slice_state *ss;
|
2009-08-07 18:44:22 +08:00
|
|
|
int i, reset_needed, busy_slice_cnt;
|
2007-08-09 15:02:14 +08:00
|
|
|
u32 rx_pause_cnt;
|
2009-08-07 18:44:22 +08:00
|
|
|
u16 cmd;
|
2006-05-23 18:10:15 +08:00
|
|
|
|
treewide: setup_timer() -> timer_setup()
This converts all remaining cases of the old setup_timer() API into using
timer_setup(), where the callback argument is the structure already
holding the struct timer_list. These should have no behavioral changes,
since they just change which pointer is passed into the callback with
the same available pointers after conversion. It handles the following
examples, in addition to some other variations.
Casting from unsigned long:
void my_callback(unsigned long data)
{
struct something *ptr = (struct something *)data;
...
}
...
setup_timer(&ptr->my_timer, my_callback, ptr);
and forced object casts:
void my_callback(struct something *ptr)
{
...
}
...
setup_timer(&ptr->my_timer, my_callback, (unsigned long)ptr);
become:
void my_callback(struct timer_list *t)
{
struct something *ptr = from_timer(ptr, t, my_timer);
...
}
...
timer_setup(&ptr->my_timer, my_callback, 0);
Direct function assignments:
void my_callback(unsigned long data)
{
struct something *ptr = (struct something *)data;
...
}
...
ptr->my_timer.function = my_callback;
have a temporary cast added, along with converting the args:
void my_callback(struct timer_list *t)
{
struct something *ptr = from_timer(ptr, t, my_timer);
...
}
...
ptr->my_timer.function = (TIMER_FUNC_TYPE)my_callback;
And finally, callbacks without a data assignment:
void my_callback(unsigned long data)
{
...
}
...
setup_timer(&ptr->my_timer, my_callback, 0);
have their argument renamed to verify they're unused during conversion:
void my_callback(struct timer_list *unused)
{
...
}
...
timer_setup(&ptr->my_timer, my_callback, 0);
The conversion is done with the following Coccinelle script:
spatch --very-quiet --all-includes --include-headers \
-I ./arch/x86/include -I ./arch/x86/include/generated \
-I ./include -I ./arch/x86/include/uapi \
-I ./arch/x86/include/generated/uapi -I ./include/uapi \
-I ./include/generated/uapi --include ./include/linux/kconfig.h \
--dir . \
--cocci-file ~/src/data/timer_setup.cocci
@fix_address_of@
expression e;
@@
setup_timer(
-&(e)
+&e
, ...)
// Update any raw setup_timer() usages that have a NULL callback, but
// would otherwise match change_timer_function_usage, since the latter
// will update all function assignments done in the face of a NULL
// function initialization in setup_timer().
@change_timer_function_usage_NULL@
expression _E;
identifier _timer;
type _cast_data;
@@
(
-setup_timer(&_E->_timer, NULL, _E);
+timer_setup(&_E->_timer, NULL, 0);
|
-setup_timer(&_E->_timer, NULL, (_cast_data)_E);
+timer_setup(&_E->_timer, NULL, 0);
|
-setup_timer(&_E._timer, NULL, &_E);
+timer_setup(&_E._timer, NULL, 0);
|
-setup_timer(&_E._timer, NULL, (_cast_data)&_E);
+timer_setup(&_E._timer, NULL, 0);
)
@change_timer_function_usage@
expression _E;
identifier _timer;
struct timer_list _stl;
identifier _callback;
type _cast_func, _cast_data;
@@
(
-setup_timer(&_E->_timer, _callback, _E);
+timer_setup(&_E->_timer, _callback, 0);
|
-setup_timer(&_E->_timer, &_callback, _E);
+timer_setup(&_E->_timer, _callback, 0);
|
-setup_timer(&_E->_timer, _callback, (_cast_data)_E);
+timer_setup(&_E->_timer, _callback, 0);
|
-setup_timer(&_E->_timer, &_callback, (_cast_data)_E);
+timer_setup(&_E->_timer, _callback, 0);
|
-setup_timer(&_E->_timer, (_cast_func)_callback, _E);
+timer_setup(&_E->_timer, _callback, 0);
|
-setup_timer(&_E->_timer, (_cast_func)&_callback, _E);
+timer_setup(&_E->_timer, _callback, 0);
|
-setup_timer(&_E->_timer, (_cast_func)_callback, (_cast_data)_E);
+timer_setup(&_E->_timer, _callback, 0);
|
-setup_timer(&_E->_timer, (_cast_func)&_callback, (_cast_data)_E);
+timer_setup(&_E->_timer, _callback, 0);
|
-setup_timer(&_E._timer, _callback, (_cast_data)_E);
+timer_setup(&_E._timer, _callback, 0);
|
-setup_timer(&_E._timer, _callback, (_cast_data)&_E);
+timer_setup(&_E._timer, _callback, 0);
|
-setup_timer(&_E._timer, &_callback, (_cast_data)_E);
+timer_setup(&_E._timer, _callback, 0);
|
-setup_timer(&_E._timer, &_callback, (_cast_data)&_E);
+timer_setup(&_E._timer, _callback, 0);
|
-setup_timer(&_E._timer, (_cast_func)_callback, (_cast_data)_E);
+timer_setup(&_E._timer, _callback, 0);
|
-setup_timer(&_E._timer, (_cast_func)_callback, (_cast_data)&_E);
+timer_setup(&_E._timer, _callback, 0);
|
-setup_timer(&_E._timer, (_cast_func)&_callback, (_cast_data)_E);
+timer_setup(&_E._timer, _callback, 0);
|
-setup_timer(&_E._timer, (_cast_func)&_callback, (_cast_data)&_E);
+timer_setup(&_E._timer, _callback, 0);
|
_E->_timer@_stl.function = _callback;
|
_E->_timer@_stl.function = &_callback;
|
_E->_timer@_stl.function = (_cast_func)_callback;
|
_E->_timer@_stl.function = (_cast_func)&_callback;
|
_E._timer@_stl.function = _callback;
|
_E._timer@_stl.function = &_callback;
|
_E._timer@_stl.function = (_cast_func)_callback;
|
_E._timer@_stl.function = (_cast_func)&_callback;
)
// callback(unsigned long arg)
@change_callback_handle_cast
depends on change_timer_function_usage@
identifier change_timer_function_usage._callback;
identifier change_timer_function_usage._timer;
type _origtype;
identifier _origarg;
type _handletype;
identifier _handle;
@@
void _callback(
-_origtype _origarg
+struct timer_list *t
)
{
(
... when != _origarg
_handletype *_handle =
-(_handletype *)_origarg;
+from_timer(_handle, t, _timer);
... when != _origarg
|
... when != _origarg
_handletype *_handle =
-(void *)_origarg;
+from_timer(_handle, t, _timer);
... when != _origarg
|
... when != _origarg
_handletype *_handle;
... when != _handle
_handle =
-(_handletype *)_origarg;
+from_timer(_handle, t, _timer);
... when != _origarg
|
... when != _origarg
_handletype *_handle;
... when != _handle
_handle =
-(void *)_origarg;
+from_timer(_handle, t, _timer);
... when != _origarg
)
}
// callback(unsigned long arg) without existing variable
@change_callback_handle_cast_no_arg
depends on change_timer_function_usage &&
!change_callback_handle_cast@
identifier change_timer_function_usage._callback;
identifier change_timer_function_usage._timer;
type _origtype;
identifier _origarg;
type _handletype;
@@
void _callback(
-_origtype _origarg
+struct timer_list *t
)
{
+ _handletype *_origarg = from_timer(_origarg, t, _timer);
+
... when != _origarg
- (_handletype *)_origarg
+ _origarg
... when != _origarg
}
// Avoid already converted callbacks.
@match_callback_converted
depends on change_timer_function_usage &&
!change_callback_handle_cast &&
!change_callback_handle_cast_no_arg@
identifier change_timer_function_usage._callback;
identifier t;
@@
void _callback(struct timer_list *t)
{ ... }
// callback(struct something *handle)
@change_callback_handle_arg
depends on change_timer_function_usage &&
!match_callback_converted &&
!change_callback_handle_cast &&
!change_callback_handle_cast_no_arg@
identifier change_timer_function_usage._callback;
identifier change_timer_function_usage._timer;
type _handletype;
identifier _handle;
@@
void _callback(
-_handletype *_handle
+struct timer_list *t
)
{
+ _handletype *_handle = from_timer(_handle, t, _timer);
...
}
// If change_callback_handle_arg ran on an empty function, remove
// the added handler.
@unchange_callback_handle_arg
depends on change_timer_function_usage &&
change_callback_handle_arg@
identifier change_timer_function_usage._callback;
identifier change_timer_function_usage._timer;
type _handletype;
identifier _handle;
identifier t;
@@
void _callback(struct timer_list *t)
{
- _handletype *_handle = from_timer(_handle, t, _timer);
}
// We only want to refactor the setup_timer() data argument if we've found
// the matching callback. This undoes changes in change_timer_function_usage.
@unchange_timer_function_usage
depends on change_timer_function_usage &&
!change_callback_handle_cast &&
!change_callback_handle_cast_no_arg &&
!change_callback_handle_arg@
expression change_timer_function_usage._E;
identifier change_timer_function_usage._timer;
identifier change_timer_function_usage._callback;
type change_timer_function_usage._cast_data;
@@
(
-timer_setup(&_E->_timer, _callback, 0);
+setup_timer(&_E->_timer, _callback, (_cast_data)_E);
|
-timer_setup(&_E._timer, _callback, 0);
+setup_timer(&_E._timer, _callback, (_cast_data)&_E);
)
// If we fixed a callback from a .function assignment, fix the
// assignment cast now.
@change_timer_function_assignment
depends on change_timer_function_usage &&
(change_callback_handle_cast ||
change_callback_handle_cast_no_arg ||
change_callback_handle_arg)@
expression change_timer_function_usage._E;
identifier change_timer_function_usage._timer;
identifier change_timer_function_usage._callback;
type _cast_func;
typedef TIMER_FUNC_TYPE;
@@
(
_E->_timer.function =
-_callback
+(TIMER_FUNC_TYPE)_callback
;
|
_E->_timer.function =
-&_callback
+(TIMER_FUNC_TYPE)_callback
;
|
_E->_timer.function =
-(_cast_func)_callback;
+(TIMER_FUNC_TYPE)_callback
;
|
_E->_timer.function =
-(_cast_func)&_callback
+(TIMER_FUNC_TYPE)_callback
;
|
_E._timer.function =
-_callback
+(TIMER_FUNC_TYPE)_callback
;
|
_E._timer.function =
-&_callback;
+(TIMER_FUNC_TYPE)_callback
;
|
_E._timer.function =
-(_cast_func)_callback
+(TIMER_FUNC_TYPE)_callback
;
|
_E._timer.function =
-(_cast_func)&_callback
+(TIMER_FUNC_TYPE)_callback
;
)
// Sometimes timer functions are called directly. Replace matched args.
@change_timer_function_calls
depends on change_timer_function_usage &&
(change_callback_handle_cast ||
change_callback_handle_cast_no_arg ||
change_callback_handle_arg)@
expression _E;
identifier change_timer_function_usage._timer;
identifier change_timer_function_usage._callback;
type _cast_data;
@@
_callback(
(
-(_cast_data)_E
+&_E->_timer
|
-(_cast_data)&_E
+&_E._timer
|
-_E
+&_E->_timer
)
)
// If a timer has been configured without a data argument, it can be
// converted without regard to the callback argument, since it is unused.
@match_timer_function_unused_data@
expression _E;
identifier _timer;
identifier _callback;
@@
(
-setup_timer(&_E->_timer, _callback, 0);
+timer_setup(&_E->_timer, _callback, 0);
|
-setup_timer(&_E->_timer, _callback, 0L);
+timer_setup(&_E->_timer, _callback, 0);
|
-setup_timer(&_E->_timer, _callback, 0UL);
+timer_setup(&_E->_timer, _callback, 0);
|
-setup_timer(&_E._timer, _callback, 0);
+timer_setup(&_E._timer, _callback, 0);
|
-setup_timer(&_E._timer, _callback, 0L);
+timer_setup(&_E._timer, _callback, 0);
|
-setup_timer(&_E._timer, _callback, 0UL);
+timer_setup(&_E._timer, _callback, 0);
|
-setup_timer(&_timer, _callback, 0);
+timer_setup(&_timer, _callback, 0);
|
-setup_timer(&_timer, _callback, 0L);
+timer_setup(&_timer, _callback, 0);
|
-setup_timer(&_timer, _callback, 0UL);
+timer_setup(&_timer, _callback, 0);
|
-setup_timer(_timer, _callback, 0);
+timer_setup(_timer, _callback, 0);
|
-setup_timer(_timer, _callback, 0L);
+timer_setup(_timer, _callback, 0);
|
-setup_timer(_timer, _callback, 0UL);
+timer_setup(_timer, _callback, 0);
)
@change_callback_unused_data
depends on match_timer_function_unused_data@
identifier match_timer_function_unused_data._callback;
type _origtype;
identifier _origarg;
@@
void _callback(
-_origtype _origarg
+struct timer_list *unused
)
{
... when != _origarg
}
Signed-off-by: Kees Cook <keescook@chromium.org>
2017-10-17 05:43:17 +08:00
|
|
|
mgp = from_timer(mgp, t, watchdog_timer);
|
2006-12-11 18:25:42 +08:00
|
|
|
|
2008-05-09 08:21:49 +08:00
|
|
|
rx_pause_cnt = ntohl(mgp->ss[0].fw_stats->dropped_pause);
|
2009-08-07 18:44:22 +08:00
|
|
|
busy_slice_cnt = 0;
|
2008-05-09 08:21:49 +08:00
|
|
|
for (i = 0, reset_needed = 0;
|
|
|
|
i < mgp->num_slices && reset_needed == 0; ++i) {
|
2008-05-09 08:20:03 +08:00
|
|
|
|
2008-05-09 08:21:49 +08:00
|
|
|
ss = &mgp->ss[i];
|
|
|
|
if (ss->rx_small.watchdog_needed) {
|
|
|
|
myri10ge_alloc_rx_pages(mgp, &ss->rx_small,
|
|
|
|
mgp->small_bytes + MXGEFW_PAD,
|
|
|
|
1);
|
|
|
|
if (ss->rx_small.fill_cnt - ss->rx_small.cnt >=
|
|
|
|
myri10ge_fill_thresh)
|
|
|
|
ss->rx_small.watchdog_needed = 0;
|
|
|
|
}
|
|
|
|
if (ss->rx_big.watchdog_needed) {
|
|
|
|
myri10ge_alloc_rx_pages(mgp, &ss->rx_big,
|
|
|
|
mgp->big_bytes, 1);
|
|
|
|
if (ss->rx_big.fill_cnt - ss->rx_big.cnt >=
|
|
|
|
myri10ge_fill_thresh)
|
|
|
|
ss->rx_big.watchdog_needed = 0;
|
|
|
|
}
|
2011-06-28 01:57:28 +08:00
|
|
|
myri10ge_check_slice(ss, &reset_needed, &busy_slice_cnt,
|
|
|
|
rx_pause_cnt);
|
2009-08-07 18:44:22 +08:00
|
|
|
}
|
|
|
|
/* if we've sent or received no traffic, poll the NIC to
|
|
|
|
* ensure it is still there. Otherwise, we risk not noticing
|
|
|
|
* an error in a timely fashion */
|
|
|
|
if (busy_slice_cnt == 0) {
|
|
|
|
pci_read_config_word(mgp->pdev, PCI_COMMAND, &cmd);
|
|
|
|
if ((cmd & PCI_COMMAND_MASTER) == 0) {
|
|
|
|
reset_needed = 1;
|
|
|
|
}
|
2007-08-09 15:02:14 +08:00
|
|
|
}
|
|
|
|
mgp->watchdog_pause = rx_pause_cnt;
|
2008-05-09 08:21:49 +08:00
|
|
|
|
|
|
|
if (reset_needed) {
|
|
|
|
schedule_work(&mgp->watchdog_work);
|
|
|
|
} else {
|
|
|
|
/* rearm timer */
|
|
|
|
mod_timer(&mgp->watchdog_timer,
|
|
|
|
jiffies + myri10ge_watchdog_timeout * HZ);
|
|
|
|
}
|
2006-05-23 18:10:15 +08:00
|
|
|
}
|
|
|
|
|
2008-05-09 08:21:10 +08:00
|
|
|
static void myri10ge_free_slices(struct myri10ge_priv *mgp)
|
|
|
|
{
|
|
|
|
struct myri10ge_slice_state *ss;
|
|
|
|
struct pci_dev *pdev = mgp->pdev;
|
|
|
|
size_t bytes;
|
|
|
|
int i;
|
|
|
|
|
|
|
|
if (mgp->ss == NULL)
|
|
|
|
return;
|
|
|
|
|
|
|
|
for (i = 0; i < mgp->num_slices; i++) {
|
|
|
|
ss = &mgp->ss[i];
|
|
|
|
if (ss->rx_done.entry != NULL) {
|
|
|
|
bytes = mgp->max_intr_slots *
|
|
|
|
sizeof(*ss->rx_done.entry);
|
|
|
|
dma_free_coherent(&pdev->dev, bytes,
|
|
|
|
ss->rx_done.entry, ss->rx_done.bus);
|
|
|
|
ss->rx_done.entry = NULL;
|
|
|
|
}
|
|
|
|
if (ss->fw_stats != NULL) {
|
|
|
|
bytes = sizeof(*ss->fw_stats);
|
|
|
|
dma_free_coherent(&pdev->dev, bytes,
|
|
|
|
ss->fw_stats, ss->fw_stats_bus);
|
|
|
|
ss->fw_stats = NULL;
|
|
|
|
}
|
2013-08-19 17:02:19 +08:00
|
|
|
napi_hash_del(&ss->napi);
|
2011-06-27 18:56:41 +08:00
|
|
|
netif_napi_del(&ss->napi);
|
2008-05-09 08:21:10 +08:00
|
|
|
}
|
2013-08-19 17:02:19 +08:00
|
|
|
/* Wait till napi structs are no longer used, and then free ss. */
|
|
|
|
synchronize_rcu();
|
2008-05-09 08:21:10 +08:00
|
|
|
kfree(mgp->ss);
|
|
|
|
mgp->ss = NULL;
|
|
|
|
}
|
|
|
|
|
|
|
|
static int myri10ge_alloc_slices(struct myri10ge_priv *mgp)
|
|
|
|
{
|
|
|
|
struct myri10ge_slice_state *ss;
|
|
|
|
struct pci_dev *pdev = mgp->pdev;
|
|
|
|
size_t bytes;
|
|
|
|
int i;
|
|
|
|
|
|
|
|
bytes = sizeof(*mgp->ss) * mgp->num_slices;
|
|
|
|
mgp->ss = kzalloc(bytes, GFP_KERNEL);
|
|
|
|
if (mgp->ss == NULL) {
|
|
|
|
return -ENOMEM;
|
|
|
|
}
|
|
|
|
|
|
|
|
for (i = 0; i < mgp->num_slices; i++) {
|
|
|
|
ss = &mgp->ss[i];
|
|
|
|
bytes = mgp->max_intr_slots * sizeof(*ss->rx_done.entry);
|
cross-tree: phase out dma_zalloc_coherent()
We already need to zero out memory for dma_alloc_coherent(), as such
using dma_zalloc_coherent() is superflous. Phase it out.
This change was generated with the following Coccinelle SmPL patch:
@ replace_dma_zalloc_coherent @
expression dev, size, data, handle, flags;
@@
-dma_zalloc_coherent(dev, size, handle, flags)
+dma_alloc_coherent(dev, size, handle, flags)
Suggested-by: Christoph Hellwig <hch@lst.de>
Signed-off-by: Luis Chamberlain <mcgrof@kernel.org>
[hch: re-ran the script on the latest tree]
Signed-off-by: Christoph Hellwig <hch@lst.de>
2019-01-04 16:23:09 +08:00
|
|
|
ss->rx_done.entry = dma_alloc_coherent(&pdev->dev, bytes,
|
|
|
|
&ss->rx_done.bus,
|
|
|
|
GFP_KERNEL);
|
2008-05-09 08:21:10 +08:00
|
|
|
if (ss->rx_done.entry == NULL)
|
|
|
|
goto abort;
|
|
|
|
bytes = sizeof(*ss->fw_stats);
|
|
|
|
ss->fw_stats = dma_alloc_coherent(&pdev->dev, bytes,
|
|
|
|
&ss->fw_stats_bus,
|
|
|
|
GFP_KERNEL);
|
|
|
|
if (ss->fw_stats == NULL)
|
|
|
|
goto abort;
|
|
|
|
ss->mgp = mgp;
|
|
|
|
ss->dev = mgp->dev;
|
|
|
|
netif_napi_add(ss->dev, &ss->napi, myri10ge_poll,
|
|
|
|
myri10ge_napi_weight);
|
|
|
|
}
|
|
|
|
return 0;
|
|
|
|
abort:
|
|
|
|
myri10ge_free_slices(mgp);
|
|
|
|
return -ENOMEM;
|
|
|
|
}
|
|
|
|
|
|
|
|
/*
|
|
|
|
* This function determines the number of slices supported.
|
2011-03-31 09:57:33 +08:00
|
|
|
* The number slices is the minimum of the number of CPUS,
|
2008-05-09 08:21:10 +08:00
|
|
|
* the number of MSI-X irqs supported, the number of slices
|
|
|
|
* supported by the firmware
|
|
|
|
*/
|
|
|
|
static void myri10ge_probe_slices(struct myri10ge_priv *mgp)
|
|
|
|
{
|
|
|
|
struct myri10ge_cmd cmd;
|
|
|
|
struct pci_dev *pdev = mgp->pdev;
|
|
|
|
char *old_fw;
|
2010-08-12 13:04:31 +08:00
|
|
|
bool old_allocated;
|
2013-08-08 21:02:44 +08:00
|
|
|
int i, status, ncpus;
|
2008-05-09 08:21:10 +08:00
|
|
|
|
|
|
|
mgp->num_slices = 1;
|
2012-07-01 11:18:56 +08:00
|
|
|
ncpus = netif_get_num_default_rss_queues();
|
2008-05-09 08:21:10 +08:00
|
|
|
|
2013-08-08 21:02:44 +08:00
|
|
|
if (myri10ge_max_slices == 1 || !pdev->msix_cap ||
|
2008-05-09 08:21:10 +08:00
|
|
|
(myri10ge_max_slices == -1 && ncpus < 2))
|
|
|
|
return;
|
|
|
|
|
|
|
|
/* try to load the slice aware rss firmware */
|
|
|
|
old_fw = mgp->fw_name;
|
2010-08-12 13:04:31 +08:00
|
|
|
old_allocated = mgp->fw_name_allocated;
|
|
|
|
/* don't free old_fw if we override it. */
|
|
|
|
mgp->fw_name_allocated = false;
|
|
|
|
|
2008-08-14 03:05:52 +08:00
|
|
|
if (myri10ge_fw_name != NULL) {
|
|
|
|
dev_info(&mgp->pdev->dev, "overriding rss firmware to %s\n",
|
|
|
|
myri10ge_fw_name);
|
2010-08-12 13:04:31 +08:00
|
|
|
set_fw_name(mgp, myri10ge_fw_name, false);
|
2008-08-14 03:05:52 +08:00
|
|
|
} else if (old_fw == myri10ge_fw_aligned)
|
2010-08-12 13:04:31 +08:00
|
|
|
set_fw_name(mgp, myri10ge_fw_rss_aligned, false);
|
2008-05-09 08:21:10 +08:00
|
|
|
else
|
2010-08-12 13:04:31 +08:00
|
|
|
set_fw_name(mgp, myri10ge_fw_rss_unaligned, false);
|
2008-05-09 08:21:10 +08:00
|
|
|
status = myri10ge_load_firmware(mgp, 0);
|
|
|
|
if (status != 0) {
|
|
|
|
dev_info(&pdev->dev, "Rss firmware not found\n");
|
2010-08-12 13:04:31 +08:00
|
|
|
if (old_allocated)
|
|
|
|
kfree(old_fw);
|
2008-05-09 08:21:10 +08:00
|
|
|
return;
|
|
|
|
}
|
|
|
|
|
|
|
|
/* hit the board with a reset to ensure it is alive */
|
|
|
|
memset(&cmd, 0, sizeof(cmd));
|
|
|
|
status = myri10ge_send_cmd(mgp, MXGEFW_CMD_RESET, &cmd, 0);
|
|
|
|
if (status != 0) {
|
|
|
|
dev_err(&mgp->pdev->dev, "failed reset\n");
|
|
|
|
goto abort_with_fw;
|
|
|
|
}
|
|
|
|
|
|
|
|
mgp->max_intr_slots = cmd.data0 / sizeof(struct mcp_slot);
|
|
|
|
|
|
|
|
/* tell it the size of the interrupt queues */
|
|
|
|
cmd.data0 = mgp->max_intr_slots * sizeof(struct mcp_slot);
|
|
|
|
status = myri10ge_send_cmd(mgp, MXGEFW_CMD_SET_INTRQ_SIZE, &cmd, 0);
|
|
|
|
if (status != 0) {
|
|
|
|
dev_err(&mgp->pdev->dev, "failed MXGEFW_CMD_SET_INTRQ_SIZE\n");
|
|
|
|
goto abort_with_fw;
|
|
|
|
}
|
|
|
|
|
|
|
|
/* ask the maximum number of slices it supports */
|
|
|
|
status = myri10ge_send_cmd(mgp, MXGEFW_CMD_GET_MAX_RSS_QUEUES, &cmd, 0);
|
|
|
|
if (status != 0)
|
|
|
|
goto abort_with_fw;
|
|
|
|
else
|
|
|
|
mgp->num_slices = cmd.data0;
|
|
|
|
|
|
|
|
/* Only allow multiple slices if MSI-X is usable */
|
|
|
|
if (!myri10ge_msi) {
|
|
|
|
goto abort_with_fw;
|
|
|
|
}
|
|
|
|
|
|
|
|
/* if the admin did not specify a limit to how many
|
|
|
|
* slices we should use, cap it automatically to the
|
|
|
|
* number of CPUs currently online */
|
|
|
|
if (myri10ge_max_slices == -1)
|
|
|
|
myri10ge_max_slices = ncpus;
|
|
|
|
|
|
|
|
if (mgp->num_slices > myri10ge_max_slices)
|
|
|
|
mgp->num_slices = myri10ge_max_slices;
|
|
|
|
|
|
|
|
/* Now try to allocate as many MSI-X vectors as we have
|
|
|
|
* slices. We give up on MSI-X if we can only get a single
|
|
|
|
* vector. */
|
|
|
|
|
2010-08-11 15:02:48 +08:00
|
|
|
mgp->msix_vectors = kcalloc(mgp->num_slices, sizeof(*mgp->msix_vectors),
|
|
|
|
GFP_KERNEL);
|
2008-05-09 08:21:10 +08:00
|
|
|
if (mgp->msix_vectors == NULL)
|
2014-02-18 18:11:49 +08:00
|
|
|
goto no_msix;
|
2008-05-09 08:21:10 +08:00
|
|
|
for (i = 0; i < mgp->num_slices; i++) {
|
|
|
|
mgp->msix_vectors[i].entry = i;
|
|
|
|
}
|
|
|
|
|
|
|
|
while (mgp->num_slices > 1) {
|
2014-02-18 18:11:49 +08:00
|
|
|
mgp->num_slices = rounddown_pow_of_two(mgp->num_slices);
|
2008-05-09 08:21:10 +08:00
|
|
|
if (mgp->num_slices == 1)
|
2014-02-18 18:11:49 +08:00
|
|
|
goto no_msix;
|
|
|
|
status = pci_enable_msix_range(pdev,
|
|
|
|
mgp->msix_vectors,
|
|
|
|
mgp->num_slices,
|
|
|
|
mgp->num_slices);
|
|
|
|
if (status < 0)
|
|
|
|
goto no_msix;
|
|
|
|
|
|
|
|
pci_disable_msix(pdev);
|
|
|
|
|
|
|
|
if (status == mgp->num_slices) {
|
2010-08-12 13:04:31 +08:00
|
|
|
if (old_allocated)
|
|
|
|
kfree(old_fw);
|
2008-05-09 08:21:10 +08:00
|
|
|
return;
|
2014-02-18 18:11:49 +08:00
|
|
|
} else {
|
2008-05-09 08:21:10 +08:00
|
|
|
mgp->num_slices = status;
|
2014-02-18 18:11:49 +08:00
|
|
|
}
|
2008-05-09 08:21:10 +08:00
|
|
|
}
|
|
|
|
|
2014-02-18 18:11:49 +08:00
|
|
|
no_msix:
|
2008-05-09 08:21:10 +08:00
|
|
|
if (mgp->msix_vectors != NULL) {
|
|
|
|
kfree(mgp->msix_vectors);
|
|
|
|
mgp->msix_vectors = NULL;
|
|
|
|
}
|
|
|
|
|
|
|
|
abort_with_fw:
|
|
|
|
mgp->num_slices = 1;
|
2010-08-12 13:04:31 +08:00
|
|
|
set_fw_name(mgp, old_fw, old_allocated);
|
2008-05-09 08:21:10 +08:00
|
|
|
myri10ge_load_firmware(mgp, 0);
|
|
|
|
}
|
|
|
|
|
2008-11-22 09:30:35 +08:00
|
|
|
static const struct net_device_ops myri10ge_netdev_ops = {
|
|
|
|
.ndo_open = myri10ge_open,
|
|
|
|
.ndo_stop = myri10ge_close,
|
|
|
|
.ndo_start_xmit = myri10ge_xmit,
|
2011-06-08 22:54:03 +08:00
|
|
|
.ndo_get_stats64 = myri10ge_get_stats,
|
2008-11-22 09:30:35 +08:00
|
|
|
.ndo_validate_addr = eth_validate_addr,
|
|
|
|
.ndo_change_mtu = myri10ge_change_mtu,
|
2011-08-16 14:29:01 +08:00
|
|
|
.ndo_set_rx_mode = myri10ge_set_multicast_list,
|
2008-11-22 09:30:35 +08:00
|
|
|
.ndo_set_mac_address = myri10ge_set_mac_address,
|
|
|
|
};
|
|
|
|
|
2006-05-23 18:10:15 +08:00
|
|
|
static int myri10ge_probe(struct pci_dev *pdev, const struct pci_device_id *ent)
|
|
|
|
{
|
|
|
|
struct net_device *netdev;
|
|
|
|
struct myri10ge_priv *mgp;
|
|
|
|
struct device *dev = &pdev->dev;
|
|
|
|
int i;
|
|
|
|
int status = -ENXIO;
|
|
|
|
int dac_enabled;
|
2008-11-20 17:50:28 +08:00
|
|
|
unsigned hdr_offset, ss_offset;
|
2009-04-16 10:24:59 +08:00
|
|
|
static int board_number;
|
2006-05-23 18:10:15 +08:00
|
|
|
|
2008-09-28 23:34:21 +08:00
|
|
|
netdev = alloc_etherdev_mq(sizeof(*mgp), MYRI10GE_MAX_SLICES);
|
2012-01-29 21:47:52 +08:00
|
|
|
if (netdev == NULL)
|
2006-05-23 18:10:15 +08:00
|
|
|
return -ENOMEM;
|
|
|
|
|
2007-06-28 23:07:26 +08:00
|
|
|
SET_NETDEV_DEV(netdev, &pdev->dev);
|
|
|
|
|
2006-05-23 18:10:15 +08:00
|
|
|
mgp = netdev_priv(netdev);
|
|
|
|
mgp->dev = netdev;
|
|
|
|
mgp->pdev = pdev;
|
|
|
|
mgp->pause = myri10ge_flow_control;
|
|
|
|
mgp->intr_coal_delay = myri10ge_intr_coal_delay;
|
2006-08-22 05:36:49 +08:00
|
|
|
mgp->msg_enable = netif_msg_init(myri10ge_debug, MYRI10GE_MSG_DEFAULT);
|
2009-04-16 10:24:59 +08:00
|
|
|
mgp->board_number = board_number;
|
2006-05-23 18:10:15 +08:00
|
|
|
init_waitqueue_head(&mgp->down_wq);
|
|
|
|
|
|
|
|
if (pci_enable_device(pdev)) {
|
|
|
|
dev_err(&pdev->dev, "pci_enable_device call failed\n");
|
|
|
|
status = -ENODEV;
|
|
|
|
goto abort_with_netdev;
|
|
|
|
}
|
|
|
|
|
|
|
|
/* Find the vendor-specific cap so we can check
|
|
|
|
* the reboot register later on */
|
|
|
|
mgp->vendor_specific_offset
|
|
|
|
= pci_find_capability(pdev, PCI_CAP_ID_VNDR);
|
|
|
|
|
|
|
|
/* Set our max read request to 4KB */
|
2007-08-24 14:57:17 +08:00
|
|
|
status = pcie_set_readrq(pdev, 4096);
|
2006-05-23 18:10:15 +08:00
|
|
|
if (status != 0) {
|
|
|
|
dev_err(&pdev->dev, "Error %d writing PCI_EXP_DEVCTL\n",
|
|
|
|
status);
|
2009-01-17 16:27:19 +08:00
|
|
|
goto abort_with_enabled;
|
2006-05-23 18:10:15 +08:00
|
|
|
}
|
|
|
|
|
2011-06-27 13:05:01 +08:00
|
|
|
myri10ge_mask_surprise_down(pdev);
|
2006-05-23 18:10:15 +08:00
|
|
|
pci_set_master(pdev);
|
|
|
|
dac_enabled = 1;
|
2009-04-07 10:01:13 +08:00
|
|
|
status = pci_set_dma_mask(pdev, DMA_BIT_MASK(64));
|
2006-05-23 18:10:15 +08:00
|
|
|
if (status != 0) {
|
|
|
|
dac_enabled = 0;
|
|
|
|
dev_err(&pdev->dev,
|
2007-10-18 18:06:30 +08:00
|
|
|
"64-bit pci address mask was refused, "
|
|
|
|
"trying 32-bit\n");
|
2009-04-07 10:01:15 +08:00
|
|
|
status = pci_set_dma_mask(pdev, DMA_BIT_MASK(32));
|
2006-05-23 18:10:15 +08:00
|
|
|
}
|
|
|
|
if (status != 0) {
|
|
|
|
dev_err(&pdev->dev, "Error %d setting DMA mask\n", status);
|
2009-01-17 16:27:19 +08:00
|
|
|
goto abort_with_enabled;
|
2006-05-23 18:10:15 +08:00
|
|
|
}
|
2009-04-07 10:01:13 +08:00
|
|
|
(void)pci_set_consistent_dma_mask(pdev, DMA_BIT_MASK(64));
|
2006-06-08 22:25:00 +08:00
|
|
|
mgp->cmd = dma_alloc_coherent(&pdev->dev, sizeof(*mgp->cmd),
|
|
|
|
&mgp->cmd_bus, GFP_KERNEL);
|
2014-12-30 01:04:37 +08:00
|
|
|
if (!mgp->cmd) {
|
|
|
|
status = -ENOMEM;
|
2009-01-17 16:27:19 +08:00
|
|
|
goto abort_with_enabled;
|
2014-12-30 01:04:37 +08:00
|
|
|
}
|
2006-05-23 18:10:15 +08:00
|
|
|
|
|
|
|
mgp->board_span = pci_resource_len(pdev, 0);
|
|
|
|
mgp->iomem_base = pci_resource_start(pdev, 0);
|
2015-04-22 04:09:45 +08:00
|
|
|
mgp->wc_cookie = arch_phys_wc_add(mgp->iomem_base, mgp->board_span);
|
2008-07-21 16:26:25 +08:00
|
|
|
mgp->sram = ioremap_wc(mgp->iomem_base, mgp->board_span);
|
2006-05-23 18:10:15 +08:00
|
|
|
if (mgp->sram == NULL) {
|
|
|
|
dev_err(&pdev->dev, "ioremap failed for %ld bytes at 0x%lx\n",
|
|
|
|
mgp->board_span, mgp->iomem_base);
|
|
|
|
status = -ENXIO;
|
2008-07-21 16:26:25 +08:00
|
|
|
goto abort_with_mtrr;
|
2006-05-23 18:10:15 +08:00
|
|
|
}
|
2008-11-20 17:50:28 +08:00
|
|
|
hdr_offset =
|
2012-12-04 18:17:15 +08:00
|
|
|
swab32(readl(mgp->sram + MCP_HEADER_PTR_OFFSET)) & 0xffffc;
|
2008-11-20 17:50:28 +08:00
|
|
|
ss_offset = hdr_offset + offsetof(struct mcp_gen_header, string_specs);
|
2012-12-04 18:17:15 +08:00
|
|
|
mgp->sram_size = swab32(readl(mgp->sram + ss_offset));
|
2008-11-20 17:50:28 +08:00
|
|
|
if (mgp->sram_size > mgp->board_span ||
|
|
|
|
mgp->sram_size <= MYRI10GE_FW_OFFSET) {
|
|
|
|
dev_err(&pdev->dev,
|
|
|
|
"invalid sram_size %dB or board span %ldB\n",
|
|
|
|
mgp->sram_size, mgp->board_span);
|
|
|
|
goto abort_with_ioremap;
|
|
|
|
}
|
2006-05-23 18:10:15 +08:00
|
|
|
memcpy_fromio(mgp->eeprom_strings,
|
2008-11-20 17:50:28 +08:00
|
|
|
mgp->sram + mgp->sram_size, MYRI10GE_EEPROM_STRINGS_SIZE);
|
2006-05-23 18:10:15 +08:00
|
|
|
memset(mgp->eeprom_strings + MYRI10GE_EEPROM_STRINGS_SIZE - 2, 0, 2);
|
|
|
|
status = myri10ge_read_mac_addr(mgp);
|
|
|
|
if (status)
|
|
|
|
goto abort_with_ioremap;
|
|
|
|
|
|
|
|
for (i = 0; i < ETH_ALEN; i++)
|
|
|
|
netdev->dev_addr[i] = mgp->mac_addr[i];
|
|
|
|
|
2007-05-08 05:52:22 +08:00
|
|
|
myri10ge_select_firmware(mgp);
|
|
|
|
|
2008-05-09 08:21:49 +08:00
|
|
|
status = myri10ge_load_firmware(mgp, 1);
|
2006-05-23 18:10:15 +08:00
|
|
|
if (status != 0) {
|
|
|
|
dev_err(&pdev->dev, "failed to load firmware\n");
|
2008-05-09 08:21:49 +08:00
|
|
|
goto abort_with_ioremap;
|
|
|
|
}
|
|
|
|
myri10ge_probe_slices(mgp);
|
|
|
|
status = myri10ge_alloc_slices(mgp);
|
|
|
|
if (status != 0) {
|
|
|
|
dev_err(&pdev->dev, "failed to alloc slice state\n");
|
|
|
|
goto abort_with_firmware;
|
2006-05-23 18:10:15 +08:00
|
|
|
}
|
2010-09-27 16:30:34 +08:00
|
|
|
netif_set_real_num_tx_queues(netdev, mgp->num_slices);
|
|
|
|
netif_set_real_num_rx_queues(netdev, mgp->num_slices);
|
2006-05-23 18:10:15 +08:00
|
|
|
status = myri10ge_reset(mgp);
|
|
|
|
if (status != 0) {
|
|
|
|
dev_err(&pdev->dev, "failed reset\n");
|
2008-05-09 08:21:49 +08:00
|
|
|
goto abort_with_slices;
|
2006-05-23 18:10:15 +08:00
|
|
|
}
|
2008-10-16 17:09:31 +08:00
|
|
|
#ifdef CONFIG_MYRI10GE_DCA
|
2008-05-09 08:22:16 +08:00
|
|
|
myri10ge_setup_dca(mgp);
|
|
|
|
#endif
|
2006-05-23 18:10:15 +08:00
|
|
|
pci_set_drvdata(pdev, mgp);
|
2008-11-22 09:30:35 +08:00
|
|
|
|
ethernet: use core min/max MTU checking
et131x: min_mtu 64, max_mtu 9216
altera_tse: min_mtu 64, max_mtu 1500
amd8111e: min_mtu 60, max_mtu 9000
bnad: min_mtu 46, max_mtu 9000
macb: min_mtu 68, max_mtu 1500 or 10240 depending on hardware capability
xgmac: min_mtu 46, max_mtu 9000
cxgb2: min_mtu 68, max_mtu 9582 (pm3393) or 9600 (vsc7326)
enic: min_mtu 68, max_mtu 9000
gianfar: min_mtu 50, max_mu 9586
hns_enet: min_mtu 68, max_mtu 9578 (v1) or 9706 (v2)
ksz884x: min_mtu 60, max_mtu 1894
myri10ge: min_mtu 68, max_mtu 9000
natsemi: min_mtu 64, max_mtu 2024
nfp: min_mtu 68, max_mtu hardware-specific
forcedeth: min_mtu 64, max_mtu 1500 or 9100, depending on hardware
pch_gbe: min_mtu 46, max_mtu 10300
pasemi_mac: min_mtu 64, max_mtu 9000
qcaspi: min_mtu 46, max_mtu 1500
- remove qcaspi_netdev_change_mtu as it is now redundant
rocker: min_mtu 68, max_mtu 9000
sxgbe: min_mtu 68, max_mtu 9000
stmmac: min_mtu 46, max_mtu depends on hardware
tehuti: min_mtu 60, max_mtu 16384
- driver had no max mtu checking, but product docs say 16k jumbo packets
are supported by the hardware
netcp: min_mtu 68, max_mtu 9486
- remove netcp_ndo_change_mtu as it is now redundant
via-velocity: min_mtu 64, max_mtu 9000
octeon: min_mtu 46, max_mtu 65370
CC: netdev@vger.kernel.org
CC: Mark Einon <mark.einon@gmail.com>
CC: Vince Bridgers <vbridger@opensource.altera.com>
CC: Rasesh Mody <rasesh.mody@qlogic.com>
CC: Nicolas Ferre <nicolas.ferre@atmel.com>
CC: Santosh Raspatur <santosh@chelsio.com>
CC: Hariprasad S <hariprasad@chelsio.com>
CC: Christian Benvenuti <benve@cisco.com>
CC: Sujith Sankar <ssujith@cisco.com>
CC: Govindarajulu Varadarajan <_govind@gmx.com>
CC: Neel Patel <neepatel@cisco.com>
CC: Claudiu Manoil <claudiu.manoil@freescale.com>
CC: Yisen Zhuang <yisen.zhuang@huawei.com>
CC: Salil Mehta <salil.mehta@huawei.com>
CC: Hyong-Youb Kim <hykim@myri.com>
CC: Jakub Kicinski <jakub.kicinski@netronome.com>
CC: Olof Johansson <olof@lixom.net>
CC: Jiri Pirko <jiri@resnulli.us>
CC: Byungho An <bh74.an@samsung.com>
CC: Girish K S <ks.giri@samsung.com>
CC: Vipul Pandya <vipul.pandya@samsung.com>
CC: Giuseppe Cavallaro <peppe.cavallaro@st.com>
CC: Alexandre Torgue <alexandre.torgue@st.com>
CC: Andy Gospodarek <andy@greyhouse.net>
CC: Wingman Kwok <w-kwok2@ti.com>
CC: Murali Karicheri <m-karicheri2@ti.com>
CC: Francois Romieu <romieu@fr.zoreil.com>
Signed-off-by: Jarod Wilson <jarod@redhat.com>
Signed-off-by: David S. Miller <davem@davemloft.net>
2016-10-18 03:54:17 +08:00
|
|
|
/* MTU range: 68 - 9000 */
|
|
|
|
netdev->min_mtu = ETH_MIN_MTU;
|
|
|
|
netdev->max_mtu = MYRI10GE_MAX_ETHER_MTU - ETH_HLEN;
|
|
|
|
|
|
|
|
if (myri10ge_initial_mtu > netdev->max_mtu)
|
|
|
|
myri10ge_initial_mtu = netdev->max_mtu;
|
|
|
|
if (myri10ge_initial_mtu < netdev->min_mtu)
|
|
|
|
myri10ge_initial_mtu = netdev->min_mtu;
|
|
|
|
|
2006-05-23 18:10:15 +08:00
|
|
|
netdev->mtu = myri10ge_initial_mtu;
|
ethernet: use core min/max MTU checking
et131x: min_mtu 64, max_mtu 9216
altera_tse: min_mtu 64, max_mtu 1500
amd8111e: min_mtu 60, max_mtu 9000
bnad: min_mtu 46, max_mtu 9000
macb: min_mtu 68, max_mtu 1500 or 10240 depending on hardware capability
xgmac: min_mtu 46, max_mtu 9000
cxgb2: min_mtu 68, max_mtu 9582 (pm3393) or 9600 (vsc7326)
enic: min_mtu 68, max_mtu 9000
gianfar: min_mtu 50, max_mu 9586
hns_enet: min_mtu 68, max_mtu 9578 (v1) or 9706 (v2)
ksz884x: min_mtu 60, max_mtu 1894
myri10ge: min_mtu 68, max_mtu 9000
natsemi: min_mtu 64, max_mtu 2024
nfp: min_mtu 68, max_mtu hardware-specific
forcedeth: min_mtu 64, max_mtu 1500 or 9100, depending on hardware
pch_gbe: min_mtu 46, max_mtu 10300
pasemi_mac: min_mtu 64, max_mtu 9000
qcaspi: min_mtu 46, max_mtu 1500
- remove qcaspi_netdev_change_mtu as it is now redundant
rocker: min_mtu 68, max_mtu 9000
sxgbe: min_mtu 68, max_mtu 9000
stmmac: min_mtu 46, max_mtu depends on hardware
tehuti: min_mtu 60, max_mtu 16384
- driver had no max mtu checking, but product docs say 16k jumbo packets
are supported by the hardware
netcp: min_mtu 68, max_mtu 9486
- remove netcp_ndo_change_mtu as it is now redundant
via-velocity: min_mtu 64, max_mtu 9000
octeon: min_mtu 46, max_mtu 65370
CC: netdev@vger.kernel.org
CC: Mark Einon <mark.einon@gmail.com>
CC: Vince Bridgers <vbridger@opensource.altera.com>
CC: Rasesh Mody <rasesh.mody@qlogic.com>
CC: Nicolas Ferre <nicolas.ferre@atmel.com>
CC: Santosh Raspatur <santosh@chelsio.com>
CC: Hariprasad S <hariprasad@chelsio.com>
CC: Christian Benvenuti <benve@cisco.com>
CC: Sujith Sankar <ssujith@cisco.com>
CC: Govindarajulu Varadarajan <_govind@gmx.com>
CC: Neel Patel <neepatel@cisco.com>
CC: Claudiu Manoil <claudiu.manoil@freescale.com>
CC: Yisen Zhuang <yisen.zhuang@huawei.com>
CC: Salil Mehta <salil.mehta@huawei.com>
CC: Hyong-Youb Kim <hykim@myri.com>
CC: Jakub Kicinski <jakub.kicinski@netronome.com>
CC: Olof Johansson <olof@lixom.net>
CC: Jiri Pirko <jiri@resnulli.us>
CC: Byungho An <bh74.an@samsung.com>
CC: Girish K S <ks.giri@samsung.com>
CC: Vipul Pandya <vipul.pandya@samsung.com>
CC: Giuseppe Cavallaro <peppe.cavallaro@st.com>
CC: Alexandre Torgue <alexandre.torgue@st.com>
CC: Andy Gospodarek <andy@greyhouse.net>
CC: Wingman Kwok <w-kwok2@ti.com>
CC: Murali Karicheri <m-karicheri2@ti.com>
CC: Francois Romieu <romieu@fr.zoreil.com>
Signed-off-by: Jarod Wilson <jarod@redhat.com>
Signed-off-by: David S. Miller <davem@davemloft.net>
2016-10-18 03:54:17 +08:00
|
|
|
|
|
|
|
netdev->netdev_ops = &myri10ge_netdev_ops;
|
2012-11-30 16:31:58 +08:00
|
|
|
netdev->hw_features = mgp->features | NETIF_F_RXCSUM;
|
2012-11-30 16:31:59 +08:00
|
|
|
|
2013-04-19 10:04:27 +08:00
|
|
|
/* fake NETIF_F_HW_VLAN_CTAG_RX for good GRO performance */
|
|
|
|
netdev->hw_features |= NETIF_F_HW_VLAN_CTAG_RX;
|
2012-11-30 16:31:59 +08:00
|
|
|
|
2011-04-15 12:50:50 +08:00
|
|
|
netdev->features = netdev->hw_features;
|
2008-09-28 23:34:21 +08:00
|
|
|
|
2006-05-23 18:10:15 +08:00
|
|
|
if (dac_enabled)
|
|
|
|
netdev->features |= NETIF_F_HIGHDMA;
|
|
|
|
|
2009-05-24 13:27:59 +08:00
|
|
|
netdev->vlan_features |= mgp->features;
|
|
|
|
if (mgp->fw_ver_tiny < 37)
|
|
|
|
netdev->vlan_features &= ~NETIF_F_TSO6;
|
|
|
|
if (mgp->fw_ver_tiny < 32)
|
|
|
|
netdev->vlan_features &= ~NETIF_F_TSO;
|
|
|
|
|
2007-01-10 04:05:04 +08:00
|
|
|
/* make sure we can get an irq, and that MSI can be
|
2012-03-24 01:51:20 +08:00
|
|
|
* setup (if available). */
|
2007-01-10 04:05:04 +08:00
|
|
|
status = myri10ge_request_irq(mgp);
|
|
|
|
if (status != 0)
|
2019-08-14 14:38:39 +08:00
|
|
|
goto abort_with_slices;
|
2007-01-10 04:05:04 +08:00
|
|
|
myri10ge_free_irq(mgp);
|
|
|
|
|
2006-05-23 18:10:15 +08:00
|
|
|
/* Save configuration space to be restored if the
|
|
|
|
* nic resets due to a parity error */
|
2006-12-18 18:52:02 +08:00
|
|
|
pci_save_state(pdev);
|
2006-05-23 18:10:15 +08:00
|
|
|
|
|
|
|
/* Setup the watchdog timer */
|
treewide: setup_timer() -> timer_setup()
This converts all remaining cases of the old setup_timer() API into using
timer_setup(), where the callback argument is the structure already
holding the struct timer_list. These should have no behavioral changes,
since they just change which pointer is passed into the callback with
the same available pointers after conversion. It handles the following
examples, in addition to some other variations.
Casting from unsigned long:
void my_callback(unsigned long data)
{
struct something *ptr = (struct something *)data;
...
}
...
setup_timer(&ptr->my_timer, my_callback, ptr);
and forced object casts:
void my_callback(struct something *ptr)
{
...
}
...
setup_timer(&ptr->my_timer, my_callback, (unsigned long)ptr);
become:
void my_callback(struct timer_list *t)
{
struct something *ptr = from_timer(ptr, t, my_timer);
...
}
...
timer_setup(&ptr->my_timer, my_callback, 0);
Direct function assignments:
void my_callback(unsigned long data)
{
struct something *ptr = (struct something *)data;
...
}
...
ptr->my_timer.function = my_callback;
have a temporary cast added, along with converting the args:
void my_callback(struct timer_list *t)
{
struct something *ptr = from_timer(ptr, t, my_timer);
...
}
...
ptr->my_timer.function = (TIMER_FUNC_TYPE)my_callback;
And finally, callbacks without a data assignment:
void my_callback(unsigned long data)
{
...
}
...
setup_timer(&ptr->my_timer, my_callback, 0);
have their argument renamed to verify they're unused during conversion:
void my_callback(struct timer_list *unused)
{
...
}
...
timer_setup(&ptr->my_timer, my_callback, 0);
The conversion is done with the following Coccinelle script:
spatch --very-quiet --all-includes --include-headers \
-I ./arch/x86/include -I ./arch/x86/include/generated \
-I ./include -I ./arch/x86/include/uapi \
-I ./arch/x86/include/generated/uapi -I ./include/uapi \
-I ./include/generated/uapi --include ./include/linux/kconfig.h \
--dir . \
--cocci-file ~/src/data/timer_setup.cocci
@fix_address_of@
expression e;
@@
setup_timer(
-&(e)
+&e
, ...)
// Update any raw setup_timer() usages that have a NULL callback, but
// would otherwise match change_timer_function_usage, since the latter
// will update all function assignments done in the face of a NULL
// function initialization in setup_timer().
@change_timer_function_usage_NULL@
expression _E;
identifier _timer;
type _cast_data;
@@
(
-setup_timer(&_E->_timer, NULL, _E);
+timer_setup(&_E->_timer, NULL, 0);
|
-setup_timer(&_E->_timer, NULL, (_cast_data)_E);
+timer_setup(&_E->_timer, NULL, 0);
|
-setup_timer(&_E._timer, NULL, &_E);
+timer_setup(&_E._timer, NULL, 0);
|
-setup_timer(&_E._timer, NULL, (_cast_data)&_E);
+timer_setup(&_E._timer, NULL, 0);
)
@change_timer_function_usage@
expression _E;
identifier _timer;
struct timer_list _stl;
identifier _callback;
type _cast_func, _cast_data;
@@
(
-setup_timer(&_E->_timer, _callback, _E);
+timer_setup(&_E->_timer, _callback, 0);
|
-setup_timer(&_E->_timer, &_callback, _E);
+timer_setup(&_E->_timer, _callback, 0);
|
-setup_timer(&_E->_timer, _callback, (_cast_data)_E);
+timer_setup(&_E->_timer, _callback, 0);
|
-setup_timer(&_E->_timer, &_callback, (_cast_data)_E);
+timer_setup(&_E->_timer, _callback, 0);
|
-setup_timer(&_E->_timer, (_cast_func)_callback, _E);
+timer_setup(&_E->_timer, _callback, 0);
|
-setup_timer(&_E->_timer, (_cast_func)&_callback, _E);
+timer_setup(&_E->_timer, _callback, 0);
|
-setup_timer(&_E->_timer, (_cast_func)_callback, (_cast_data)_E);
+timer_setup(&_E->_timer, _callback, 0);
|
-setup_timer(&_E->_timer, (_cast_func)&_callback, (_cast_data)_E);
+timer_setup(&_E->_timer, _callback, 0);
|
-setup_timer(&_E._timer, _callback, (_cast_data)_E);
+timer_setup(&_E._timer, _callback, 0);
|
-setup_timer(&_E._timer, _callback, (_cast_data)&_E);
+timer_setup(&_E._timer, _callback, 0);
|
-setup_timer(&_E._timer, &_callback, (_cast_data)_E);
+timer_setup(&_E._timer, _callback, 0);
|
-setup_timer(&_E._timer, &_callback, (_cast_data)&_E);
+timer_setup(&_E._timer, _callback, 0);
|
-setup_timer(&_E._timer, (_cast_func)_callback, (_cast_data)_E);
+timer_setup(&_E._timer, _callback, 0);
|
-setup_timer(&_E._timer, (_cast_func)_callback, (_cast_data)&_E);
+timer_setup(&_E._timer, _callback, 0);
|
-setup_timer(&_E._timer, (_cast_func)&_callback, (_cast_data)_E);
+timer_setup(&_E._timer, _callback, 0);
|
-setup_timer(&_E._timer, (_cast_func)&_callback, (_cast_data)&_E);
+timer_setup(&_E._timer, _callback, 0);
|
_E->_timer@_stl.function = _callback;
|
_E->_timer@_stl.function = &_callback;
|
_E->_timer@_stl.function = (_cast_func)_callback;
|
_E->_timer@_stl.function = (_cast_func)&_callback;
|
_E._timer@_stl.function = _callback;
|
_E._timer@_stl.function = &_callback;
|
_E._timer@_stl.function = (_cast_func)_callback;
|
_E._timer@_stl.function = (_cast_func)&_callback;
)
// callback(unsigned long arg)
@change_callback_handle_cast
depends on change_timer_function_usage@
identifier change_timer_function_usage._callback;
identifier change_timer_function_usage._timer;
type _origtype;
identifier _origarg;
type _handletype;
identifier _handle;
@@
void _callback(
-_origtype _origarg
+struct timer_list *t
)
{
(
... when != _origarg
_handletype *_handle =
-(_handletype *)_origarg;
+from_timer(_handle, t, _timer);
... when != _origarg
|
... when != _origarg
_handletype *_handle =
-(void *)_origarg;
+from_timer(_handle, t, _timer);
... when != _origarg
|
... when != _origarg
_handletype *_handle;
... when != _handle
_handle =
-(_handletype *)_origarg;
+from_timer(_handle, t, _timer);
... when != _origarg
|
... when != _origarg
_handletype *_handle;
... when != _handle
_handle =
-(void *)_origarg;
+from_timer(_handle, t, _timer);
... when != _origarg
)
}
// callback(unsigned long arg) without existing variable
@change_callback_handle_cast_no_arg
depends on change_timer_function_usage &&
!change_callback_handle_cast@
identifier change_timer_function_usage._callback;
identifier change_timer_function_usage._timer;
type _origtype;
identifier _origarg;
type _handletype;
@@
void _callback(
-_origtype _origarg
+struct timer_list *t
)
{
+ _handletype *_origarg = from_timer(_origarg, t, _timer);
+
... when != _origarg
- (_handletype *)_origarg
+ _origarg
... when != _origarg
}
// Avoid already converted callbacks.
@match_callback_converted
depends on change_timer_function_usage &&
!change_callback_handle_cast &&
!change_callback_handle_cast_no_arg@
identifier change_timer_function_usage._callback;
identifier t;
@@
void _callback(struct timer_list *t)
{ ... }
// callback(struct something *handle)
@change_callback_handle_arg
depends on change_timer_function_usage &&
!match_callback_converted &&
!change_callback_handle_cast &&
!change_callback_handle_cast_no_arg@
identifier change_timer_function_usage._callback;
identifier change_timer_function_usage._timer;
type _handletype;
identifier _handle;
@@
void _callback(
-_handletype *_handle
+struct timer_list *t
)
{
+ _handletype *_handle = from_timer(_handle, t, _timer);
...
}
// If change_callback_handle_arg ran on an empty function, remove
// the added handler.
@unchange_callback_handle_arg
depends on change_timer_function_usage &&
change_callback_handle_arg@
identifier change_timer_function_usage._callback;
identifier change_timer_function_usage._timer;
type _handletype;
identifier _handle;
identifier t;
@@
void _callback(struct timer_list *t)
{
- _handletype *_handle = from_timer(_handle, t, _timer);
}
// We only want to refactor the setup_timer() data argument if we've found
// the matching callback. This undoes changes in change_timer_function_usage.
@unchange_timer_function_usage
depends on change_timer_function_usage &&
!change_callback_handle_cast &&
!change_callback_handle_cast_no_arg &&
!change_callback_handle_arg@
expression change_timer_function_usage._E;
identifier change_timer_function_usage._timer;
identifier change_timer_function_usage._callback;
type change_timer_function_usage._cast_data;
@@
(
-timer_setup(&_E->_timer, _callback, 0);
+setup_timer(&_E->_timer, _callback, (_cast_data)_E);
|
-timer_setup(&_E._timer, _callback, 0);
+setup_timer(&_E._timer, _callback, (_cast_data)&_E);
)
// If we fixed a callback from a .function assignment, fix the
// assignment cast now.
@change_timer_function_assignment
depends on change_timer_function_usage &&
(change_callback_handle_cast ||
change_callback_handle_cast_no_arg ||
change_callback_handle_arg)@
expression change_timer_function_usage._E;
identifier change_timer_function_usage._timer;
identifier change_timer_function_usage._callback;
type _cast_func;
typedef TIMER_FUNC_TYPE;
@@
(
_E->_timer.function =
-_callback
+(TIMER_FUNC_TYPE)_callback
;
|
_E->_timer.function =
-&_callback
+(TIMER_FUNC_TYPE)_callback
;
|
_E->_timer.function =
-(_cast_func)_callback;
+(TIMER_FUNC_TYPE)_callback
;
|
_E->_timer.function =
-(_cast_func)&_callback
+(TIMER_FUNC_TYPE)_callback
;
|
_E._timer.function =
-_callback
+(TIMER_FUNC_TYPE)_callback
;
|
_E._timer.function =
-&_callback;
+(TIMER_FUNC_TYPE)_callback
;
|
_E._timer.function =
-(_cast_func)_callback
+(TIMER_FUNC_TYPE)_callback
;
|
_E._timer.function =
-(_cast_func)&_callback
+(TIMER_FUNC_TYPE)_callback
;
)
// Sometimes timer functions are called directly. Replace matched args.
@change_timer_function_calls
depends on change_timer_function_usage &&
(change_callback_handle_cast ||
change_callback_handle_cast_no_arg ||
change_callback_handle_arg)@
expression _E;
identifier change_timer_function_usage._timer;
identifier change_timer_function_usage._callback;
type _cast_data;
@@
_callback(
(
-(_cast_data)_E
+&_E->_timer
|
-(_cast_data)&_E
+&_E._timer
|
-_E
+&_E->_timer
)
)
// If a timer has been configured without a data argument, it can be
// converted without regard to the callback argument, since it is unused.
@match_timer_function_unused_data@
expression _E;
identifier _timer;
identifier _callback;
@@
(
-setup_timer(&_E->_timer, _callback, 0);
+timer_setup(&_E->_timer, _callback, 0);
|
-setup_timer(&_E->_timer, _callback, 0L);
+timer_setup(&_E->_timer, _callback, 0);
|
-setup_timer(&_E->_timer, _callback, 0UL);
+timer_setup(&_E->_timer, _callback, 0);
|
-setup_timer(&_E._timer, _callback, 0);
+timer_setup(&_E._timer, _callback, 0);
|
-setup_timer(&_E._timer, _callback, 0L);
+timer_setup(&_E._timer, _callback, 0);
|
-setup_timer(&_E._timer, _callback, 0UL);
+timer_setup(&_E._timer, _callback, 0);
|
-setup_timer(&_timer, _callback, 0);
+timer_setup(&_timer, _callback, 0);
|
-setup_timer(&_timer, _callback, 0L);
+timer_setup(&_timer, _callback, 0);
|
-setup_timer(&_timer, _callback, 0UL);
+timer_setup(&_timer, _callback, 0);
|
-setup_timer(_timer, _callback, 0);
+timer_setup(_timer, _callback, 0);
|
-setup_timer(_timer, _callback, 0L);
+timer_setup(_timer, _callback, 0);
|
-setup_timer(_timer, _callback, 0UL);
+timer_setup(_timer, _callback, 0);
)
@change_callback_unused_data
depends on match_timer_function_unused_data@
identifier match_timer_function_unused_data._callback;
type _origtype;
identifier _origarg;
@@
void _callback(
-_origtype _origarg
+struct timer_list *unused
)
{
... when != _origarg
}
Signed-off-by: Kees Cook <keescook@chromium.org>
2017-10-17 05:43:17 +08:00
|
|
|
timer_setup(&mgp->watchdog_timer, myri10ge_watchdog_timer, 0);
|
2006-05-23 18:10:15 +08:00
|
|
|
|
2014-05-11 08:12:32 +08:00
|
|
|
netdev->ethtool_ops = &myri10ge_ethtool_ops;
|
2006-11-22 22:57:56 +08:00
|
|
|
INIT_WORK(&mgp->watchdog_work, myri10ge_watchdog);
|
2006-05-23 18:10:15 +08:00
|
|
|
status = register_netdev(netdev);
|
|
|
|
if (status != 0) {
|
|
|
|
dev_err(&pdev->dev, "register_netdev failed: %d\n", status);
|
2006-12-18 18:50:00 +08:00
|
|
|
goto abort_with_state;
|
2006-05-23 18:10:15 +08:00
|
|
|
}
|
2008-05-09 08:21:49 +08:00
|
|
|
if (mgp->msix_enabled)
|
2015-04-22 04:09:45 +08:00
|
|
|
dev_info(dev, "%d MSI-X IRQs, tx bndry %d, fw %s, MTRR %s, WC Enabled\n",
|
2008-05-09 08:21:49 +08:00
|
|
|
mgp->num_slices, mgp->tx_boundary, mgp->fw_name,
|
2015-04-22 04:09:45 +08:00
|
|
|
(mgp->wc_cookie > 0 ? "Enabled" : "Disabled"));
|
2008-05-09 08:21:49 +08:00
|
|
|
else
|
2015-04-22 04:09:45 +08:00
|
|
|
dev_info(dev, "%s IRQ %d, tx bndry %d, fw %s, MTRR %s, WC Enabled\n",
|
2008-05-09 08:21:49 +08:00
|
|
|
mgp->msi_enabled ? "MSI" : "xPIC",
|
2012-03-24 01:51:20 +08:00
|
|
|
pdev->irq, mgp->tx_boundary, mgp->fw_name,
|
2015-04-22 04:09:45 +08:00
|
|
|
(mgp->wc_cookie > 0 ? "Enabled" : "Disabled"));
|
2006-05-23 18:10:15 +08:00
|
|
|
|
2009-04-16 10:24:59 +08:00
|
|
|
board_number++;
|
2006-05-23 18:10:15 +08:00
|
|
|
return 0;
|
|
|
|
|
2006-12-18 18:50:00 +08:00
|
|
|
abort_with_state:
|
2006-12-18 18:52:02 +08:00
|
|
|
pci_restore_state(pdev);
|
2006-05-23 18:10:15 +08:00
|
|
|
|
2008-05-09 08:21:49 +08:00
|
|
|
abort_with_slices:
|
|
|
|
myri10ge_free_slices(mgp);
|
|
|
|
|
2006-05-23 18:10:15 +08:00
|
|
|
abort_with_firmware:
|
|
|
|
myri10ge_dummy_rdma(mgp, 0);
|
|
|
|
|
|
|
|
abort_with_ioremap:
|
2009-01-06 10:16:14 +08:00
|
|
|
if (mgp->mac_addr_string != NULL)
|
|
|
|
dev_err(&pdev->dev,
|
|
|
|
"myri10ge_probe() failed: MAC=%s, SN=%ld\n",
|
|
|
|
mgp->mac_addr_string, mgp->serial_number);
|
2006-05-23 18:10:15 +08:00
|
|
|
iounmap(mgp->sram);
|
|
|
|
|
2008-07-21 16:26:25 +08:00
|
|
|
abort_with_mtrr:
|
2015-04-22 04:09:45 +08:00
|
|
|
arch_phys_wc_del(mgp->wc_cookie);
|
2006-06-08 22:25:00 +08:00
|
|
|
dma_free_coherent(&pdev->dev, sizeof(*mgp->cmd),
|
|
|
|
mgp->cmd, mgp->cmd_bus);
|
2006-05-23 18:10:15 +08:00
|
|
|
|
2009-01-17 16:27:19 +08:00
|
|
|
abort_with_enabled:
|
|
|
|
pci_disable_device(pdev);
|
2006-05-23 18:10:15 +08:00
|
|
|
|
2009-01-17 16:27:19 +08:00
|
|
|
abort_with_netdev:
|
2010-08-12 13:04:31 +08:00
|
|
|
set_fw_name(mgp, NULL, false);
|
2006-05-23 18:10:15 +08:00
|
|
|
free_netdev(netdev);
|
|
|
|
return status;
|
|
|
|
}
|
|
|
|
|
|
|
|
/*
|
|
|
|
* myri10ge_remove
|
|
|
|
*
|
|
|
|
* Does what is necessary to shutdown one Myrinet device. Called
|
|
|
|
* once for each Myrinet card by the kernel when a module is
|
|
|
|
* unloaded.
|
|
|
|
*/
|
|
|
|
static void myri10ge_remove(struct pci_dev *pdev)
|
|
|
|
{
|
|
|
|
struct myri10ge_priv *mgp;
|
|
|
|
struct net_device *netdev;
|
|
|
|
|
|
|
|
mgp = pci_get_drvdata(pdev);
|
|
|
|
if (mgp == NULL)
|
|
|
|
return;
|
|
|
|
|
2010-12-12 23:45:14 +08:00
|
|
|
cancel_work_sync(&mgp->watchdog_work);
|
2006-05-23 18:10:15 +08:00
|
|
|
netdev = mgp->dev;
|
|
|
|
unregister_netdev(netdev);
|
|
|
|
|
2008-10-16 17:09:31 +08:00
|
|
|
#ifdef CONFIG_MYRI10GE_DCA
|
2008-05-09 08:22:16 +08:00
|
|
|
myri10ge_teardown_dca(mgp);
|
|
|
|
#endif
|
2006-05-23 18:10:15 +08:00
|
|
|
myri10ge_dummy_rdma(mgp, 0);
|
|
|
|
|
2006-12-18 18:50:00 +08:00
|
|
|
/* avoid a memory leak */
|
2006-12-18 18:52:02 +08:00
|
|
|
pci_restore_state(pdev);
|
2006-12-18 18:50:00 +08:00
|
|
|
|
2006-05-23 18:10:15 +08:00
|
|
|
iounmap(mgp->sram);
|
2015-04-22 04:09:45 +08:00
|
|
|
arch_phys_wc_del(mgp->wc_cookie);
|
2008-05-09 08:21:49 +08:00
|
|
|
myri10ge_free_slices(mgp);
|
2015-02-04 19:32:14 +08:00
|
|
|
kfree(mgp->msix_vectors);
|
2006-06-08 22:25:00 +08:00
|
|
|
dma_free_coherent(&pdev->dev, sizeof(*mgp->cmd),
|
|
|
|
mgp->cmd, mgp->cmd_bus);
|
2006-05-23 18:10:15 +08:00
|
|
|
|
2010-08-12 13:04:31 +08:00
|
|
|
set_fw_name(mgp, NULL, false);
|
2006-05-23 18:10:15 +08:00
|
|
|
free_netdev(netdev);
|
2009-01-17 16:27:19 +08:00
|
|
|
pci_disable_device(pdev);
|
2006-05-23 18:10:15 +08:00
|
|
|
}
|
|
|
|
|
2006-06-08 22:25:00 +08:00
|
|
|
#define PCI_DEVICE_ID_MYRICOM_MYRI10GE_Z8E 0x0008
|
2007-09-14 06:40:14 +08:00
|
|
|
#define PCI_DEVICE_ID_MYRICOM_MYRI10GE_Z8E_9 0x0009
|
2006-05-23 18:10:15 +08:00
|
|
|
|
2014-08-08 21:56:03 +08:00
|
|
|
static const struct pci_device_id myri10ge_pci_tbl[] = {
|
2006-06-08 22:25:00 +08:00
|
|
|
{PCI_DEVICE(PCI_VENDOR_ID_MYRICOM, PCI_DEVICE_ID_MYRICOM_MYRI10GE_Z8E)},
|
2007-09-14 06:40:14 +08:00
|
|
|
{PCI_DEVICE
|
|
|
|
(PCI_VENDOR_ID_MYRICOM, PCI_DEVICE_ID_MYRICOM_MYRI10GE_Z8E_9)},
|
2006-05-23 18:10:15 +08:00
|
|
|
{0},
|
|
|
|
};
|
|
|
|
|
2009-04-16 10:29:22 +08:00
|
|
|
MODULE_DEVICE_TABLE(pci, myri10ge_pci_tbl);
|
|
|
|
|
2006-05-23 18:10:15 +08:00
|
|
|
static struct pci_driver myri10ge_driver = {
|
|
|
|
.name = "myri10ge",
|
|
|
|
.probe = myri10ge_probe,
|
|
|
|
.remove = myri10ge_remove,
|
|
|
|
.id_table = myri10ge_pci_tbl,
|
|
|
|
#ifdef CONFIG_PM
|
|
|
|
.suspend = myri10ge_suspend,
|
|
|
|
.resume = myri10ge_resume,
|
|
|
|
#endif
|
|
|
|
};
|
|
|
|
|
2008-10-16 17:09:31 +08:00
|
|
|
#ifdef CONFIG_MYRI10GE_DCA
|
2008-05-09 08:22:16 +08:00
|
|
|
static int
|
|
|
|
myri10ge_notify_dca(struct notifier_block *nb, unsigned long event, void *p)
|
|
|
|
{
|
|
|
|
int err = driver_for_each_device(&myri10ge_driver.driver,
|
|
|
|
NULL, &event,
|
|
|
|
myri10ge_notify_dca_device);
|
|
|
|
|
|
|
|
if (err)
|
|
|
|
return NOTIFY_BAD;
|
|
|
|
return NOTIFY_DONE;
|
|
|
|
}
|
|
|
|
|
|
|
|
static struct notifier_block myri10ge_dca_notifier = {
|
|
|
|
.notifier_call = myri10ge_notify_dca,
|
|
|
|
.next = NULL,
|
|
|
|
.priority = 0,
|
|
|
|
};
|
2008-11-24 07:49:28 +08:00
|
|
|
#endif /* CONFIG_MYRI10GE_DCA */
|
2008-05-09 08:22:16 +08:00
|
|
|
|
2006-05-23 18:10:15 +08:00
|
|
|
static __init int myri10ge_init_module(void)
|
|
|
|
{
|
2010-02-23 00:56:58 +08:00
|
|
|
pr_info("Version %s\n", MYRI10GE_VERSION_STR);
|
2008-05-09 08:21:49 +08:00
|
|
|
|
2008-09-28 23:34:21 +08:00
|
|
|
if (myri10ge_rss_hash > MXGEFW_RSS_HASH_TYPE_MAX) {
|
2010-02-23 00:56:58 +08:00
|
|
|
pr_err("Illegal rssh hash type %d, defaulting to source port\n",
|
|
|
|
myri10ge_rss_hash);
|
2008-05-09 08:21:49 +08:00
|
|
|
myri10ge_rss_hash = MXGEFW_RSS_HASH_TYPE_SRC_PORT;
|
|
|
|
}
|
2008-10-16 17:09:31 +08:00
|
|
|
#ifdef CONFIG_MYRI10GE_DCA
|
2008-05-09 08:22:16 +08:00
|
|
|
dca_register_notify(&myri10ge_dca_notifier);
|
|
|
|
#endif
|
2008-09-28 23:34:21 +08:00
|
|
|
if (myri10ge_max_slices > MYRI10GE_MAX_SLICES)
|
|
|
|
myri10ge_max_slices = MYRI10GE_MAX_SLICES;
|
2008-05-09 08:21:49 +08:00
|
|
|
|
2006-05-23 18:10:15 +08:00
|
|
|
return pci_register_driver(&myri10ge_driver);
|
|
|
|
}
|
|
|
|
|
|
|
|
module_init(myri10ge_init_module);
|
|
|
|
|
|
|
|
static __exit void myri10ge_cleanup_module(void)
|
|
|
|
{
|
2008-10-16 17:09:31 +08:00
|
|
|
#ifdef CONFIG_MYRI10GE_DCA
|
2008-05-09 08:22:16 +08:00
|
|
|
dca_unregister_notify(&myri10ge_dca_notifier);
|
|
|
|
#endif
|
2006-05-23 18:10:15 +08:00
|
|
|
pci_unregister_driver(&myri10ge_driver);
|
|
|
|
}
|
|
|
|
|
|
|
|
module_exit(myri10ge_cleanup_module);
|