2007-01-19 06:04:14 +03:00
/*
2007-01-31 06:44:35 +03:00
* Copyright ( c ) 2003 - 2007 Chelsio , Inc . All rights reserved .
2007-01-19 06:04:14 +03:00
*
2007-01-31 06:44:35 +03:00
* This software is available to you under a choice of one of two
* licenses . You may choose to be licensed under the terms of the GNU
* General Public License ( GPL ) Version 2 , available from the file
* COPYING in the main directory of this source tree , or the
* OpenIB . org BSD license below :
2007-01-19 06:04:14 +03:00
*
2007-01-31 06:44:35 +03:00
* Redistribution and use in source and binary forms , with or
* without modification , are permitted provided that the following
* conditions are met :
*
* - Redistributions of source code must retain the above
* copyright notice , this list of conditions and the following
* disclaimer .
*
* - 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 .
*
* THE SOFTWARE IS PROVIDED " AS IS " , WITHOUT WARRANTY OF ANY KIND ,
* EXPRESS OR IMPLIED , INCLUDING BUT NOT LIMITED TO THE WARRANTIES OF
* MERCHANTABILITY , FITNESS FOR A PARTICULAR PURPOSE AND
* NONINFRINGEMENT . IN NO EVENT SHALL THE AUTHORS OR COPYRIGHT HOLDERS
* BE LIABLE FOR ANY CLAIM , DAMAGES OR OTHER LIABILITY , WHETHER IN AN
* ACTION OF CONTRACT , TORT OR OTHERWISE , ARISING FROM , OUT OF OR IN
* CONNECTION WITH THE SOFTWARE OR THE USE OR OTHER DEALINGS IN THE
* SOFTWARE .
2007-01-19 06:04:14 +03:00
*/
# include <linux/module.h>
# include <linux/moduleparam.h>
# include <linux/init.h>
# include <linux/pci.h>
# include <linux/dma-mapping.h>
# include <linux/netdevice.h>
# include <linux/etherdevice.h>
# include <linux/if_vlan.h>
# include <linux/mii.h>
# include <linux/sockios.h>
# include <linux/workqueue.h>
# include <linux/proc_fs.h>
# include <linux/rtnetlink.h>
2007-03-18 23:10:06 +03:00
# include <linux/firmware.h>
2007-01-19 06:04:14 +03:00
# include <asm/uaccess.h>
# include "common.h"
# include "cxgb3_ioctl.h"
# include "regs.h"
# include "cxgb3_offload.h"
# include "version.h"
# include "cxgb3_ctl_defs.h"
# include "t3_cpl.h"
# include "firmware_exports.h"
enum {
MAX_TXQ_ENTRIES = 16384 ,
MAX_CTRL_TXQ_ENTRIES = 1024 ,
MAX_RSPQ_ENTRIES = 16384 ,
MAX_RX_BUFFERS = 16384 ,
MAX_RX_JUMBO_BUFFERS = 16384 ,
MIN_TXQ_ENTRIES = 4 ,
MIN_CTRL_TXQ_ENTRIES = 4 ,
MIN_RSPQ_ENTRIES = 32 ,
MIN_FL_ENTRIES = 32
} ;
# define PORT_MASK ((1 << MAX_NPORTS) - 1)
# define DFLT_MSG_ENABLE (NETIF_MSG_DRV | NETIF_MSG_PROBE | NETIF_MSG_LINK | \
NETIF_MSG_TIMER | NETIF_MSG_IFDOWN | NETIF_MSG_IFUP | \
NETIF_MSG_RX_ERR | NETIF_MSG_TX_ERR )
# define EEPROM_MAGIC 0x38E2F10C
# define CH_DEVICE(devid, ssid, idx) \
{ PCI_VENDOR_ID_CHELSIO , devid , PCI_ANY_ID , ssid , 0 , 0 , idx }
static const struct pci_device_id cxgb3_pci_tbl [ ] = {
CH_DEVICE ( 0x20 , 1 , 0 ) , /* PE9000 */
CH_DEVICE ( 0x21 , 1 , 1 ) , /* T302E */
CH_DEVICE ( 0x22 , 1 , 2 ) , /* T310E */
CH_DEVICE ( 0x23 , 1 , 3 ) , /* T320X */
CH_DEVICE ( 0x24 , 1 , 1 ) , /* T302X */
CH_DEVICE ( 0x25 , 1 , 3 ) , /* T320E */
CH_DEVICE ( 0x26 , 1 , 2 ) , /* T310X */
CH_DEVICE ( 0x30 , 1 , 2 ) , /* T3B10 */
CH_DEVICE ( 0x31 , 1 , 3 ) , /* T3B20 */
CH_DEVICE ( 0x32 , 1 , 1 ) , /* T3B02 */
{ 0 , }
} ;
MODULE_DESCRIPTION ( DRV_DESC ) ;
MODULE_AUTHOR ( " Chelsio Communications " ) ;
2007-01-31 06:44:35 +03:00
MODULE_LICENSE ( " Dual BSD/GPL " ) ;
2007-01-19 06:04:14 +03:00
MODULE_VERSION ( DRV_VERSION ) ;
MODULE_DEVICE_TABLE ( pci , cxgb3_pci_tbl ) ;
static int dflt_msg_enable = DFLT_MSG_ENABLE ;
module_param ( dflt_msg_enable , int , 0644 ) ;
MODULE_PARM_DESC ( dflt_msg_enable , " Chelsio T3 default message enable bitmap " ) ;
/*
* The driver uses the best interrupt scheme available on a platform in the
* order MSI - X , MSI , legacy pin interrupts . This parameter determines which
* of these schemes the driver may consider as follows :
*
* msi = 2 : choose from among all three options
* msi = 1 : only consider MSI and pin interrupts
* msi = 0 : force pin interrupts
*/
static int msi = 2 ;
module_param ( msi , int , 0644 ) ;
MODULE_PARM_DESC ( msi , " whether to use MSI or MSI-X " ) ;
/*
* The driver enables offload as a default .
* To disable it , use ofld_disable = 1.
*/
static int ofld_disable = 0 ;
module_param ( ofld_disable , int , 0644 ) ;
MODULE_PARM_DESC ( ofld_disable , " whether to enable offload at init time or not " ) ;
/*
* We have work elements that we need to cancel when an interface is taken
* down . Normally the work elements would be executed by keventd but that
* can deadlock because of linkwatch . If our close method takes the rtnl
* lock and linkwatch is ahead of our work elements in keventd , linkwatch
* will block keventd as it needs the rtnl lock , and we ' ll deadlock waiting
* for our work to complete . Get our own work queue to solve this .
*/
static struct workqueue_struct * cxgb3_wq ;
/**
* link_report - show link status and link speed / duplex
* @ p : the port whose settings are to be reported
*
* Shows the link status , speed , and duplex of a port .
*/
static void link_report ( struct net_device * dev )
{
if ( ! netif_carrier_ok ( dev ) )
printk ( KERN_INFO " %s: link down \n " , dev - > name ) ;
else {
const char * s = " 10Mbps " ;
const struct port_info * p = netdev_priv ( dev ) ;
switch ( p - > link_config . speed ) {
case SPEED_10000 :
s = " 10Gbps " ;
break ;
case SPEED_1000 :
s = " 1000Mbps " ;
break ;
case SPEED_100 :
s = " 100Mbps " ;
break ;
}
printk ( KERN_INFO " %s: link up, %s, %s-duplex \n " , dev - > name , s ,
p - > link_config . duplex = = DUPLEX_FULL ? " full " : " half " ) ;
}
}
/**
* t3_os_link_changed - handle link status changes
* @ adapter : the adapter associated with the link change
* @ port_id : the port index whose limk status has changed
* @ link_stat : the new status of the link
* @ speed : the new speed setting
* @ duplex : the new duplex setting
* @ pause : the new flow - control setting
*
* This is the OS - dependent handler for link status changes . The OS
* neutral handler takes care of most of the processing for these events ,
* then calls this handler for any OS - specific processing .
*/
void t3_os_link_changed ( struct adapter * adapter , int port_id , int link_stat ,
int speed , int duplex , int pause )
{
struct net_device * dev = adapter - > port [ port_id ] ;
/* Skip changes from disabled ports. */
if ( ! netif_running ( dev ) )
return ;
if ( link_stat ! = netif_carrier_ok ( dev ) ) {
if ( link_stat )
netif_carrier_on ( dev ) ;
else
netif_carrier_off ( dev ) ;
link_report ( dev ) ;
}
}
static void cxgb_set_rxmode ( struct net_device * dev )
{
struct t3_rx_mode rm ;
struct port_info * pi = netdev_priv ( dev ) ;
init_rx_mode ( & rm , dev , dev - > mc_list ) ;
t3_mac_set_rx_mode ( & pi - > mac , & rm ) ;
}
/**
* link_start - enable a port
* @ dev : the device to enable
*
* Performs the MAC and PHY actions needed to enable a port .
*/
static void link_start ( struct net_device * dev )
{
struct t3_rx_mode rm ;
struct port_info * pi = netdev_priv ( dev ) ;
struct cmac * mac = & pi - > mac ;
init_rx_mode ( & rm , dev , dev - > mc_list ) ;
t3_mac_reset ( mac ) ;
t3_mac_set_mtu ( mac , dev - > mtu ) ;
t3_mac_set_address ( mac , 0 , dev - > dev_addr ) ;
t3_mac_set_rx_mode ( mac , & rm ) ;
t3_link_start ( & pi - > phy , mac , & pi - > link_config ) ;
t3_mac_enable ( mac , MAC_DIRECTION_RX | MAC_DIRECTION_TX ) ;
}
static inline void cxgb_disable_msi ( struct adapter * adapter )
{
if ( adapter - > flags & USING_MSIX ) {
pci_disable_msix ( adapter - > pdev ) ;
adapter - > flags & = ~ USING_MSIX ;
} else if ( adapter - > flags & USING_MSI ) {
pci_disable_msi ( adapter - > pdev ) ;
adapter - > flags & = ~ USING_MSI ;
}
}
/*
* Interrupt handler for asynchronous events used with MSI - X .
*/
static irqreturn_t t3_async_intr_handler ( int irq , void * cookie )
{
t3_slow_intr_handler ( cookie ) ;
return IRQ_HANDLED ;
}
/*
* Name the MSI - X interrupts .
*/
static void name_msix_vecs ( struct adapter * adap )
{
int i , j , msi_idx = 1 , n = sizeof ( adap - > msix_info [ 0 ] . desc ) - 1 ;
snprintf ( adap - > msix_info [ 0 ] . desc , n , " %s " , adap - > name ) ;
adap - > msix_info [ 0 ] . desc [ n ] = 0 ;
for_each_port ( adap , j ) {
struct net_device * d = adap - > port [ j ] ;
const struct port_info * pi = netdev_priv ( d ) ;
for ( i = 0 ; i < pi - > nqsets ; i + + , msi_idx + + ) {
snprintf ( adap - > msix_info [ msi_idx ] . desc , n ,
" %s (queue %d) " , d - > name , i ) ;
adap - > msix_info [ msi_idx ] . desc [ n ] = 0 ;
}
}
}
static int request_msix_data_irqs ( struct adapter * adap )
{
int i , j , err , qidx = 0 ;
for_each_port ( adap , i ) {
int nqsets = adap2pinfo ( adap , i ) - > nqsets ;
for ( j = 0 ; j < nqsets ; + + j ) {
err = request_irq ( adap - > msix_info [ qidx + 1 ] . vec ,
t3_intr_handler ( adap ,
adap - > sge . qs [ qidx ] .
rspq . polling ) , 0 ,
adap - > msix_info [ qidx + 1 ] . desc ,
& adap - > sge . qs [ qidx ] ) ;
if ( err ) {
while ( - - qidx > = 0 )
free_irq ( adap - > msix_info [ qidx + 1 ] . vec ,
& adap - > sge . qs [ qidx ] ) ;
return err ;
}
qidx + + ;
}
}
return 0 ;
}
/**
* setup_rss - configure RSS
* @ adap : the adapter
*
* Sets up RSS to distribute packets to multiple receive queues . We
* configure the RSS CPU lookup table to distribute to the number of HW
* receive queues , and the response queue lookup table to narrow that
* down to the response queues actually configured for each port .
* We always configure the RSS mapping for two ports since the mapping
* table has plenty of entries .
*/
static void setup_rss ( struct adapter * adap )
{
int i ;
unsigned int nq0 = adap2pinfo ( adap , 0 ) - > nqsets ;
unsigned int nq1 = adap - > port [ 1 ] ? adap2pinfo ( adap , 1 ) - > nqsets : 1 ;
u8 cpus [ SGE_QSETS + 1 ] ;
u16 rspq_map [ RSS_TABLE_SIZE ] ;
for ( i = 0 ; i < SGE_QSETS ; + + i )
cpus [ i ] = i ;
cpus [ SGE_QSETS ] = 0xff ; /* terminator */
for ( i = 0 ; i < RSS_TABLE_SIZE / 2 ; + + i ) {
rspq_map [ i ] = i % nq0 ;
rspq_map [ i + RSS_TABLE_SIZE / 2 ] = ( i % nq1 ) + nq0 ;
}
t3_config_rss ( adap , F_RQFEEDBACKENABLE | F_TNLLKPEN | F_TNLMAPEN |
F_TNLPRTEN | F_TNL2TUPEN | F_TNL4TUPEN |
V_RRCPLCPUSIZE ( 6 ) , cpus , rspq_map ) ;
}
/*
* If we have multiple receive queues per port serviced by NAPI we need one
* netdevice per queue as NAPI operates on netdevices . We already have one
* netdevice , namely the one associated with the interface , so we use dummy
* ones for any additional queues . Note that these netdevices exist purely
* so that NAPI has something to work with , they do not represent network
* ports and are not registered .
*/
static int init_dummy_netdevs ( struct adapter * adap )
{
int i , j , dummy_idx = 0 ;
struct net_device * nd ;
for_each_port ( adap , i ) {
struct net_device * dev = adap - > port [ i ] ;
const struct port_info * pi = netdev_priv ( dev ) ;
for ( j = 0 ; j < pi - > nqsets - 1 ; j + + ) {
if ( ! adap - > dummy_netdev [ dummy_idx ] ) {
nd = alloc_netdev ( 0 , " " , ether_setup ) ;
if ( ! nd )
goto free_all ;
nd - > priv = adap ;
nd - > weight = 64 ;
set_bit ( __LINK_STATE_START , & nd - > state ) ;
adap - > dummy_netdev [ dummy_idx ] = nd ;
}
strcpy ( adap - > dummy_netdev [ dummy_idx ] - > name , dev - > name ) ;
dummy_idx + + ;
}
}
return 0 ;
free_all :
while ( - - dummy_idx > = 0 ) {
free_netdev ( adap - > dummy_netdev [ dummy_idx ] ) ;
adap - > dummy_netdev [ dummy_idx ] = NULL ;
}
return - ENOMEM ;
}
/*
* Wait until all NAPI handlers are descheduled . This includes the handlers of
* both netdevices representing interfaces and the dummy ones for the extra
* queues .
*/
static void quiesce_rx ( struct adapter * adap )
{
int i ;
struct net_device * dev ;
for_each_port ( adap , i ) {
dev = adap - > port [ i ] ;
while ( test_bit ( __LINK_STATE_RX_SCHED , & dev - > state ) )
msleep ( 1 ) ;
}
for ( i = 0 ; i < ARRAY_SIZE ( adap - > dummy_netdev ) ; i + + ) {
dev = adap - > dummy_netdev [ i ] ;
if ( dev )
while ( test_bit ( __LINK_STATE_RX_SCHED , & dev - > state ) )
msleep ( 1 ) ;
}
}
/**
* setup_sge_qsets - configure SGE Tx / Rx / response queues
* @ adap : the adapter
*
* Determines how many sets of SGE queues to use and initializes them .
* We support multiple queue sets per port if we have MSI - X , otherwise
* just one queue set per port .
*/
static int setup_sge_qsets ( struct adapter * adap )
{
int i , j , err , irq_idx = 0 , qset_idx = 0 , dummy_dev_idx = 0 ;
unsigned int ntxq = is_offload ( adap ) ? SGE_TXQ_PER_SET : 1 ;
if ( adap - > params . rev > 0 & & ! ( adap - > flags & USING_MSI ) )
irq_idx = - 1 ;
for_each_port ( adap , i ) {
struct net_device * dev = adap - > port [ i ] ;
const struct port_info * pi = netdev_priv ( dev ) ;
for ( j = 0 ; j < pi - > nqsets ; + + j , + + qset_idx ) {
err = t3_sge_alloc_qset ( adap , qset_idx , 1 ,
( adap - > flags & USING_MSIX ) ? qset_idx + 1 :
irq_idx ,
& adap - > params . sge . qset [ qset_idx ] , ntxq ,
j = = 0 ? dev :
adap - > dummy_netdev [ dummy_dev_idx + + ] ) ;
if ( err ) {
t3_free_sge_resources ( adap ) ;
return err ;
}
}
}
return 0 ;
}
2007-02-09 03:55:59 +03:00
static ssize_t attr_show ( struct device * d , struct device_attribute * attr ,
char * buf ,
2007-02-25 03:43:50 +03:00
ssize_t ( * format ) ( struct net_device * , char * ) )
2007-01-19 06:04:14 +03:00
{
ssize_t len ;
/* Synchronize with ioctls that may shut down the device */
rtnl_lock ( ) ;
2007-02-25 03:43:50 +03:00
len = ( * format ) ( to_net_dev ( d ) , buf ) ;
2007-01-19 06:04:14 +03:00
rtnl_unlock ( ) ;
return len ;
}
2007-02-09 03:55:59 +03:00
static ssize_t attr_store ( struct device * d , struct device_attribute * attr ,
const char * buf , size_t len ,
2007-02-25 03:43:50 +03:00
ssize_t ( * set ) ( struct net_device * , unsigned int ) ,
2007-01-19 06:04:14 +03:00
unsigned int min_val , unsigned int max_val )
{
char * endp ;
ssize_t ret ;
unsigned int val ;
if ( ! capable ( CAP_NET_ADMIN ) )
return - EPERM ;
val = simple_strtoul ( buf , & endp , 0 ) ;
if ( endp = = buf | | val < min_val | | val > max_val )
return - EINVAL ;
rtnl_lock ( ) ;
2007-02-25 03:43:50 +03:00
ret = ( * set ) ( to_net_dev ( d ) , val ) ;
2007-01-19 06:04:14 +03:00
if ( ! ret )
ret = len ;
rtnl_unlock ( ) ;
return ret ;
}
# define CXGB3_SHOW(name, val_expr) \
2007-02-25 03:43:50 +03:00
static ssize_t format_ # # name ( struct net_device * dev , char * buf ) \
2007-01-19 06:04:14 +03:00
{ \
2007-02-25 03:43:50 +03:00
struct adapter * adap = dev - > priv ; \
2007-01-19 06:04:14 +03:00
return sprintf ( buf , " %u \n " , val_expr ) ; \
} \
2007-02-09 03:55:59 +03:00
static ssize_t show_ # # name ( struct device * d , struct device_attribute * attr , \
char * buf ) \
2007-01-19 06:04:14 +03:00
{ \
2007-02-09 03:55:59 +03:00
return attr_show ( d , attr , buf , format_ # # name ) ; \
2007-01-19 06:04:14 +03:00
}
2007-02-25 03:43:50 +03:00
static ssize_t set_nfilters ( struct net_device * dev , unsigned int val )
2007-01-19 06:04:14 +03:00
{
2007-02-25 03:43:50 +03:00
struct adapter * adap = dev - > priv ;
2007-03-31 11:23:13 +04:00
int min_tids = is_offload ( adap ) ? MC5_MIN_TIDS : 0 ;
2007-02-25 03:43:50 +03:00
2007-01-19 06:04:14 +03:00
if ( adap - > flags & FULL_INIT_DONE )
return - EBUSY ;
if ( val & & adap - > params . rev = = 0 )
return - EINVAL ;
2007-03-31 11:23:13 +04:00
if ( val > t3_mc5_size ( & adap - > mc5 ) - adap - > params . mc5 . nservers -
min_tids )
2007-01-19 06:04:14 +03:00
return - EINVAL ;
adap - > params . mc5 . nfilters = val ;
return 0 ;
}
2007-02-09 03:55:59 +03:00
static ssize_t store_nfilters ( struct device * d , struct device_attribute * attr ,
const char * buf , size_t len )
2007-01-19 06:04:14 +03:00
{
2007-02-09 03:55:59 +03:00
return attr_store ( d , attr , buf , len , set_nfilters , 0 , ~ 0 ) ;
2007-01-19 06:04:14 +03:00
}
2007-02-25 03:43:50 +03:00
static ssize_t set_nservers ( struct net_device * dev , unsigned int val )
2007-01-19 06:04:14 +03:00
{
2007-02-25 03:43:50 +03:00
struct adapter * adap = dev - > priv ;
2007-01-19 06:04:14 +03:00
if ( adap - > flags & FULL_INIT_DONE )
return - EBUSY ;
2007-03-31 11:23:13 +04:00
if ( val > t3_mc5_size ( & adap - > mc5 ) - adap - > params . mc5 . nfilters -
MC5_MIN_TIDS )
2007-01-19 06:04:14 +03:00
return - EINVAL ;
adap - > params . mc5 . nservers = val ;
return 0 ;
}
2007-02-09 03:55:59 +03:00
static ssize_t store_nservers ( struct device * d , struct device_attribute * attr ,
const char * buf , size_t len )
2007-01-19 06:04:14 +03:00
{
2007-02-09 03:55:59 +03:00
return attr_store ( d , attr , buf , len , set_nservers , 0 , ~ 0 ) ;
2007-01-19 06:04:14 +03:00
}
# define CXGB3_ATTR_R(name, val_expr) \
CXGB3_SHOW ( name , val_expr ) \
2007-02-09 03:55:59 +03:00
static DEVICE_ATTR ( name , S_IRUGO , show_ # # name , NULL )
2007-01-19 06:04:14 +03:00
# define CXGB3_ATTR_RW(name, val_expr, store_method) \
CXGB3_SHOW ( name , val_expr ) \
2007-02-09 03:55:59 +03:00
static DEVICE_ATTR ( name , S_IRUGO | S_IWUSR , show_ # # name , store_method )
2007-01-19 06:04:14 +03:00
CXGB3_ATTR_R ( cam_size , t3_mc5_size ( & adap - > mc5 ) ) ;
CXGB3_ATTR_RW ( nfilters , adap - > params . mc5 . nfilters , store_nfilters ) ;
CXGB3_ATTR_RW ( nservers , adap - > params . mc5 . nservers , store_nservers ) ;
static struct attribute * cxgb3_attrs [ ] = {
2007-02-09 03:55:59 +03:00
& dev_attr_cam_size . attr ,
& dev_attr_nfilters . attr ,
& dev_attr_nservers . attr ,
2007-01-19 06:04:14 +03:00
NULL
} ;
static struct attribute_group cxgb3_attr_group = { . attrs = cxgb3_attrs } ;
2007-02-09 03:55:59 +03:00
static ssize_t tm_attr_show ( struct device * d , struct device_attribute * attr ,
char * buf , int sched )
2007-01-19 06:04:14 +03:00
{
ssize_t len ;
unsigned int v , addr , bpt , cpt ;
2007-02-09 03:55:59 +03:00
struct adapter * adap = to_net_dev ( d ) - > priv ;
2007-01-19 06:04:14 +03:00
addr = A_TP_TX_MOD_Q1_Q0_RATE_LIMIT - sched / 2 ;
rtnl_lock ( ) ;
t3_write_reg ( adap , A_TP_TM_PIO_ADDR , addr ) ;
v = t3_read_reg ( adap , A_TP_TM_PIO_DATA ) ;
if ( sched & 1 )
v > > = 16 ;
bpt = ( v > > 8 ) & 0xff ;
cpt = v & 0xff ;
if ( ! cpt )
len = sprintf ( buf , " disabled \n " ) ;
else {
v = ( adap - > params . vpd . cclk * 1000 ) / cpt ;
len = sprintf ( buf , " %u Kbps \n " , ( v * bpt ) / 125 ) ;
}
rtnl_unlock ( ) ;
return len ;
}
2007-02-09 03:55:59 +03:00
static ssize_t tm_attr_store ( struct device * d , struct device_attribute * attr ,
const char * buf , size_t len , int sched )
2007-01-19 06:04:14 +03:00
{
char * endp ;
ssize_t ret ;
unsigned int val ;
2007-02-09 03:55:59 +03:00
struct adapter * adap = to_net_dev ( d ) - > priv ;
2007-01-19 06:04:14 +03:00
if ( ! capable ( CAP_NET_ADMIN ) )
return - EPERM ;
val = simple_strtoul ( buf , & endp , 0 ) ;
if ( endp = = buf | | val > 10000000 )
return - EINVAL ;
rtnl_lock ( ) ;
ret = t3_config_sched ( adap , val , sched ) ;
if ( ! ret )
ret = len ;
rtnl_unlock ( ) ;
return ret ;
}
# define TM_ATTR(name, sched) \
2007-02-09 03:55:59 +03:00
static ssize_t show_ # # name ( struct device * d , struct device_attribute * attr , \
char * buf ) \
2007-01-19 06:04:14 +03:00
{ \
2007-02-09 03:55:59 +03:00
return tm_attr_show ( d , attr , buf , sched ) ; \
2007-01-19 06:04:14 +03:00
} \
2007-02-09 03:55:59 +03:00
static ssize_t store_ # # name ( struct device * d , struct device_attribute * attr , \
const char * buf , size_t len ) \
2007-01-19 06:04:14 +03:00
{ \
2007-02-09 03:55:59 +03:00
return tm_attr_store ( d , attr , buf , len , sched ) ; \
2007-01-19 06:04:14 +03:00
} \
2007-02-09 03:55:59 +03:00
static DEVICE_ATTR ( name , S_IRUGO | S_IWUSR , show_ # # name , store_ # # name )
2007-01-19 06:04:14 +03:00
TM_ATTR ( sched0 , 0 ) ;
TM_ATTR ( sched1 , 1 ) ;
TM_ATTR ( sched2 , 2 ) ;
TM_ATTR ( sched3 , 3 ) ;
TM_ATTR ( sched4 , 4 ) ;
TM_ATTR ( sched5 , 5 ) ;
TM_ATTR ( sched6 , 6 ) ;
TM_ATTR ( sched7 , 7 ) ;
static struct attribute * offload_attrs [ ] = {
2007-02-09 03:55:59 +03:00
& dev_attr_sched0 . attr ,
& dev_attr_sched1 . attr ,
& dev_attr_sched2 . attr ,
& dev_attr_sched3 . attr ,
& dev_attr_sched4 . attr ,
& dev_attr_sched5 . attr ,
& dev_attr_sched6 . attr ,
& dev_attr_sched7 . attr ,
2007-01-19 06:04:14 +03:00
NULL
} ;
static struct attribute_group offload_attr_group = { . attrs = offload_attrs } ;
/*
* Sends an sk_buff to an offload queue driver
* after dealing with any active network taps .
*/
static inline int offload_tx ( struct t3cdev * tdev , struct sk_buff * skb )
{
int ret ;
local_bh_disable ( ) ;
ret = t3_offload_tx ( tdev , skb ) ;
local_bh_enable ( ) ;
return ret ;
}
static int write_smt_entry ( struct adapter * adapter , int idx )
{
struct cpl_smt_write_req * req ;
struct sk_buff * skb = alloc_skb ( sizeof ( * req ) , GFP_KERNEL ) ;
if ( ! skb )
return - ENOMEM ;
req = ( struct cpl_smt_write_req * ) __skb_put ( skb , sizeof ( * req ) ) ;
req - > wr . wr_hi = htonl ( V_WR_OP ( FW_WROPCODE_FORWARD ) ) ;
OPCODE_TID ( req ) = htonl ( MK_OPCODE_TID ( CPL_SMT_WRITE_REQ , idx ) ) ;
req - > mtu_idx = NMTUS - 1 ; /* should be 0 but there's a T3 bug */
req - > iff = idx ;
memset ( req - > src_mac1 , 0 , sizeof ( req - > src_mac1 ) ) ;
memcpy ( req - > src_mac0 , adapter - > port [ idx ] - > dev_addr , ETH_ALEN ) ;
skb - > priority = 1 ;
offload_tx ( & adapter - > tdev , skb ) ;
return 0 ;
}
static int init_smt ( struct adapter * adapter )
{
int i ;
for_each_port ( adapter , i )
write_smt_entry ( adapter , i ) ;
return 0 ;
}
static void init_port_mtus ( struct adapter * adapter )
{
unsigned int mtus = adapter - > port [ 0 ] - > mtu ;
if ( adapter - > port [ 1 ] )
mtus | = adapter - > port [ 1 ] - > mtu < < 16 ;
t3_write_reg ( adapter , A_TP_MTU_PORT_TABLE , mtus ) ;
}
2007-01-31 06:43:50 +03:00
static void send_pktsched_cmd ( struct adapter * adap , int sched , int qidx , int lo ,
int hi , int port )
{
struct sk_buff * skb ;
struct mngt_pktsched_wr * req ;
skb = alloc_skb ( sizeof ( * req ) , GFP_KERNEL | __GFP_NOFAIL ) ;
req = ( struct mngt_pktsched_wr * ) skb_put ( skb , sizeof ( * req ) ) ;
req - > wr_hi = htonl ( V_WR_OP ( FW_WROPCODE_MNGT ) ) ;
req - > mngt_opcode = FW_MNGTOPCODE_PKTSCHED_SET ;
req - > sched = sched ;
req - > idx = qidx ;
req - > min = lo ;
req - > max = hi ;
req - > binding = port ;
t3_mgmt_tx ( adap , skb ) ;
}
static void bind_qsets ( struct adapter * adap )
{
int i , j ;
for_each_port ( adap , i ) {
const struct port_info * pi = adap2pinfo ( adap , i ) ;
for ( j = 0 ; j < pi - > nqsets ; + + j )
send_pktsched_cmd ( adap , 1 , pi - > first_qset + j , - 1 ,
- 1 , i ) ;
}
}
2007-03-18 23:10:06 +03:00
# define FW_FNAME "t3fw-%d.%d.bin"
static int upgrade_fw ( struct adapter * adap )
{
int ret ;
char buf [ 64 ] ;
const struct firmware * fw ;
struct device * dev = & adap - > pdev - > dev ;
snprintf ( buf , sizeof ( buf ) , FW_FNAME , FW_VERSION_MAJOR ,
FW_VERSION_MINOR ) ;
ret = request_firmware ( & fw , buf , dev ) ;
if ( ret < 0 ) {
dev_err ( dev , " could not upgrade firmware: unable to load %s \n " ,
buf ) ;
return ret ;
}
ret = t3_load_fw ( adap , fw - > data , fw - > size ) ;
release_firmware ( fw ) ;
return ret ;
}
2007-01-19 06:04:14 +03:00
/**
* cxgb_up - enable the adapter
* @ adapter : adapter being enabled
*
* Called when the first port is enabled , this function performs the
* actions necessary to make an adapter operational , such as completing
* the initialization of HW modules , and enabling interrupts .
*
* Must be called with the rtnl lock held .
*/
static int cxgb_up ( struct adapter * adap )
{
int err = 0 ;
if ( ! ( adap - > flags & FULL_INIT_DONE ) ) {
err = t3_check_fw_version ( adap ) ;
2007-03-18 23:10:06 +03:00
if ( err = = - EINVAL )
err = upgrade_fw ( adap ) ;
2007-01-31 06:43:45 +03:00
if ( err )
2007-01-19 06:04:14 +03:00
goto out ;
err = init_dummy_netdevs ( adap ) ;
if ( err )
goto out ;
err = t3_init_hw ( adap , 0 ) ;
if ( err )
goto out ;
err = setup_sge_qsets ( adap ) ;
if ( err )
goto out ;
setup_rss ( adap ) ;
adap - > flags | = FULL_INIT_DONE ;
}
t3_intr_clear ( adap ) ;
if ( adap - > flags & USING_MSIX ) {
name_msix_vecs ( adap ) ;
err = request_irq ( adap - > msix_info [ 0 ] . vec ,
t3_async_intr_handler , 0 ,
adap - > msix_info [ 0 ] . desc , adap ) ;
if ( err )
goto irq_err ;
if ( request_msix_data_irqs ( adap ) ) {
free_irq ( adap - > msix_info [ 0 ] . vec , adap ) ;
goto irq_err ;
}
} else if ( ( err = request_irq ( adap - > pdev - > irq ,
t3_intr_handler ( adap ,
adap - > sge . qs [ 0 ] . rspq .
polling ) ,
2007-02-14 11:33:20 +03:00
( adap - > flags & USING_MSI ) ?
0 : IRQF_SHARED ,
2007-01-19 06:04:14 +03:00
adap - > name , adap ) ) )
goto irq_err ;
t3_sge_start ( adap ) ;
t3_intr_enable ( adap ) ;
2007-01-31 06:43:50 +03:00
if ( ( adap - > flags & ( USING_MSIX | QUEUES_BOUND ) ) = = USING_MSIX )
bind_qsets ( adap ) ;
adap - > flags | = QUEUES_BOUND ;
2007-01-19 06:04:14 +03:00
out :
return err ;
irq_err :
CH_ERR ( adap , " request_irq failed, err %d \n " , err ) ;
goto out ;
}
/*
* Release resources when all the ports and offloading have been stopped .
*/
static void cxgb_down ( struct adapter * adapter )
{
t3_sge_stop ( adapter ) ;
spin_lock_irq ( & adapter - > work_lock ) ; /* sync with PHY intr task */
t3_intr_disable ( adapter ) ;
spin_unlock_irq ( & adapter - > work_lock ) ;
if ( adapter - > flags & USING_MSIX ) {
int i , n = 0 ;
free_irq ( adapter - > msix_info [ 0 ] . vec , adapter ) ;
for_each_port ( adapter , i )
n + = adap2pinfo ( adapter , i ) - > nqsets ;
for ( i = 0 ; i < n ; + + i )
free_irq ( adapter - > msix_info [ i + 1 ] . vec ,
& adapter - > sge . qs [ i ] ) ;
} else
free_irq ( adapter - > pdev - > irq , adapter ) ;
flush_workqueue ( cxgb3_wq ) ; /* wait for external IRQ handler */
quiesce_rx ( adapter ) ;
}
static void schedule_chk_task ( struct adapter * adap )
{
unsigned int timeo ;
timeo = adap - > params . linkpoll_period ?
( HZ * adap - > params . linkpoll_period ) / 10 :
adap - > params . stats_update_period * HZ ;
if ( timeo )
queue_delayed_work ( cxgb3_wq , & adap - > adap_check_task , timeo ) ;
}
static int offload_open ( struct net_device * dev )
{
struct adapter * adapter = dev - > priv ;
struct t3cdev * tdev = T3CDEV ( dev ) ;
int adap_up = adapter - > open_device_map & PORT_MASK ;
int err = 0 ;
if ( test_and_set_bit ( OFFLOAD_DEVMAP_BIT , & adapter - > open_device_map ) )
return 0 ;
if ( ! adap_up & & ( err = cxgb_up ( adapter ) ) < 0 )
return err ;
t3_tp_set_offload_mode ( adapter , 1 ) ;
tdev - > lldev = adapter - > port [ 0 ] ;
err = cxgb3_offload_activate ( adapter ) ;
if ( err )
goto out ;
init_port_mtus ( adapter ) ;
t3_load_mtus ( adapter , adapter - > params . mtus , adapter - > params . a_wnd ,
adapter - > params . b_wnd ,
adapter - > params . rev = = 0 ?
adapter - > port [ 0 ] - > mtu : 0xffff ) ;
init_smt ( adapter ) ;
/* Never mind if the next step fails */
2007-02-09 03:55:59 +03:00
sysfs_create_group ( & tdev - > lldev - > dev . kobj , & offload_attr_group ) ;
2007-01-19 06:04:14 +03:00
/* Call back all registered clients */
cxgb3_add_clients ( tdev ) ;
out :
/* restore them in case the offload module has changed them */
if ( err ) {
t3_tp_set_offload_mode ( adapter , 0 ) ;
clear_bit ( OFFLOAD_DEVMAP_BIT , & adapter - > open_device_map ) ;
cxgb3_set_dummy_ops ( tdev ) ;
}
return err ;
}
static int offload_close ( struct t3cdev * tdev )
{
struct adapter * adapter = tdev2adap ( tdev ) ;
if ( ! test_bit ( OFFLOAD_DEVMAP_BIT , & adapter - > open_device_map ) )
return 0 ;
/* Call back all registered clients */
cxgb3_remove_clients ( tdev ) ;
2007-02-09 03:55:59 +03:00
sysfs_remove_group ( & tdev - > lldev - > dev . kobj , & offload_attr_group ) ;
2007-01-19 06:04:14 +03:00
tdev - > lldev = NULL ;
cxgb3_set_dummy_ops ( tdev ) ;
t3_tp_set_offload_mode ( adapter , 0 ) ;
clear_bit ( OFFLOAD_DEVMAP_BIT , & adapter - > open_device_map ) ;
if ( ! adapter - > open_device_map )
cxgb_down ( adapter ) ;
cxgb3_offload_deactivate ( adapter ) ;
return 0 ;
}
static int cxgb_open ( struct net_device * dev )
{
int err ;
struct adapter * adapter = dev - > priv ;
struct port_info * pi = netdev_priv ( dev ) ;
int other_ports = adapter - > open_device_map & PORT_MASK ;
if ( ! adapter - > open_device_map & & ( err = cxgb_up ( adapter ) ) < 0 )
return err ;
set_bit ( pi - > port_id , & adapter - > open_device_map ) ;
if ( ! ofld_disable ) {
err = offload_open ( dev ) ;
if ( err )
printk ( KERN_WARNING
" Could not initialize offload capabilities \n " ) ;
}
link_start ( dev ) ;
t3_port_intr_enable ( adapter , pi - > port_id ) ;
netif_start_queue ( dev ) ;
if ( ! other_ports )
schedule_chk_task ( adapter ) ;
return 0 ;
}
static int cxgb_close ( struct net_device * dev )
{
struct adapter * adapter = dev - > priv ;
struct port_info * p = netdev_priv ( dev ) ;
t3_port_intr_disable ( adapter , p - > port_id ) ;
netif_stop_queue ( dev ) ;
p - > phy . ops - > power_down ( & p - > phy , 1 ) ;
netif_carrier_off ( dev ) ;
t3_mac_disable ( & p - > mac , MAC_DIRECTION_TX | MAC_DIRECTION_RX ) ;
spin_lock ( & adapter - > work_lock ) ; /* sync with update task */
clear_bit ( p - > port_id , & adapter - > open_device_map ) ;
spin_unlock ( & adapter - > work_lock ) ;
if ( ! ( adapter - > open_device_map & PORT_MASK ) )
cancel_rearming_delayed_workqueue ( cxgb3_wq ,
& adapter - > adap_check_task ) ;
if ( ! adapter - > open_device_map )
cxgb_down ( adapter ) ;
return 0 ;
}
static struct net_device_stats * cxgb_get_stats ( struct net_device * dev )
{
struct adapter * adapter = dev - > priv ;
struct port_info * p = netdev_priv ( dev ) ;
struct net_device_stats * ns = & p - > netstats ;
const struct mac_stats * pstats ;
spin_lock ( & adapter - > stats_lock ) ;
pstats = t3_mac_update_stats ( & p - > mac ) ;
spin_unlock ( & adapter - > stats_lock ) ;
ns - > tx_bytes = pstats - > tx_octets ;
ns - > tx_packets = pstats - > tx_frames ;
ns - > rx_bytes = pstats - > rx_octets ;
ns - > rx_packets = pstats - > rx_frames ;
ns - > multicast = pstats - > rx_mcast_frames ;
ns - > tx_errors = pstats - > tx_underrun ;
ns - > rx_errors = pstats - > rx_symbol_errs + pstats - > rx_fcs_errs +
pstats - > rx_too_long + pstats - > rx_jabber + pstats - > rx_short +
pstats - > rx_fifo_ovfl ;
/* detailed rx_errors */
ns - > rx_length_errors = pstats - > rx_jabber + pstats - > rx_too_long ;
ns - > rx_over_errors = 0 ;
ns - > rx_crc_errors = pstats - > rx_fcs_errs ;
ns - > rx_frame_errors = pstats - > rx_symbol_errs ;
ns - > rx_fifo_errors = pstats - > rx_fifo_ovfl ;
ns - > rx_missed_errors = pstats - > rx_cong_drops ;
/* detailed tx_errors */
ns - > tx_aborted_errors = 0 ;
ns - > tx_carrier_errors = 0 ;
ns - > tx_fifo_errors = pstats - > tx_underrun ;
ns - > tx_heartbeat_errors = 0 ;
ns - > tx_window_errors = 0 ;
return ns ;
}
static u32 get_msglevel ( struct net_device * dev )
{
struct adapter * adapter = dev - > priv ;
return adapter - > msg_enable ;
}
static void set_msglevel ( struct net_device * dev , u32 val )
{
struct adapter * adapter = dev - > priv ;
adapter - > msg_enable = val ;
}
static char stats_strings [ ] [ ETH_GSTRING_LEN ] = {
" TxOctetsOK " ,
" TxFramesOK " ,
" TxMulticastFramesOK " ,
" TxBroadcastFramesOK " ,
" TxPauseFrames " ,
" TxUnderrun " ,
" TxExtUnderrun " ,
" TxFrames64 " ,
" TxFrames65To127 " ,
" TxFrames128To255 " ,
" TxFrames256To511 " ,
" TxFrames512To1023 " ,
" TxFrames1024To1518 " ,
" TxFrames1519ToMax " ,
" RxOctetsOK " ,
" RxFramesOK " ,
" RxMulticastFramesOK " ,
" RxBroadcastFramesOK " ,
" RxPauseFrames " ,
" RxFCSErrors " ,
" RxSymbolErrors " ,
" RxShortErrors " ,
" RxJabberErrors " ,
" RxLengthErrors " ,
" RxFIFOoverflow " ,
" RxFrames64 " ,
" RxFrames65To127 " ,
" RxFrames128To255 " ,
" RxFrames256To511 " ,
" RxFrames512To1023 " ,
" RxFrames1024To1518 " ,
" RxFrames1519ToMax " ,
" PhyFIFOErrors " ,
" TSO " ,
" VLANextractions " ,
" VLANinsertions " ,
" TxCsumOffload " ,
" RxCsumGood " ,
2007-03-18 23:10:12 +03:00
" RxDrops " ,
" CheckTXEnToggled " ,
" CheckResets " ,
2007-01-19 06:04:14 +03:00
} ;
static int get_stats_count ( struct net_device * dev )
{
return ARRAY_SIZE ( stats_strings ) ;
}
# define T3_REGMAP_SIZE (3 * 1024)
static int get_regs_len ( struct net_device * dev )
{
return T3_REGMAP_SIZE ;
}
static int get_eeprom_len ( struct net_device * dev )
{
return EEPROMSIZE ;
}
static void get_drvinfo ( struct net_device * dev , struct ethtool_drvinfo * info )
{
u32 fw_vers = 0 ;
struct adapter * adapter = dev - > priv ;
t3_get_fw_version ( adapter , & fw_vers ) ;
strcpy ( info - > driver , DRV_NAME ) ;
strcpy ( info - > version , DRV_VERSION ) ;
strcpy ( info - > bus_info , pci_name ( adapter - > pdev ) ) ;
if ( ! fw_vers )
strcpy ( info - > fw_version , " N/A " ) ;
2007-01-31 06:43:45 +03:00
else {
2007-01-19 06:04:14 +03:00
snprintf ( info - > fw_version , sizeof ( info - > fw_version ) ,
2007-01-31 06:43:45 +03:00
" %s %u.%u.%u " ,
G_FW_VERSION_TYPE ( fw_vers ) ? " T " : " N " ,
G_FW_VERSION_MAJOR ( fw_vers ) ,
G_FW_VERSION_MINOR ( fw_vers ) ,
G_FW_VERSION_MICRO ( fw_vers ) ) ;
}
2007-01-19 06:04:14 +03:00
}
static void get_strings ( struct net_device * dev , u32 stringset , u8 * data )
{
if ( stringset = = ETH_SS_STATS )
memcpy ( data , stats_strings , sizeof ( stats_strings ) ) ;
}
static unsigned long collect_sge_port_stats ( struct adapter * adapter ,
struct port_info * p , int idx )
{
int i ;
unsigned long tot = 0 ;
for ( i = 0 ; i < p - > nqsets ; + + i )
tot + = adapter - > sge . qs [ i + p - > first_qset ] . port_stats [ idx ] ;
return tot ;
}
static void get_stats ( struct net_device * dev , struct ethtool_stats * stats ,
u64 * data )
{
struct adapter * adapter = dev - > priv ;
struct port_info * pi = netdev_priv ( dev ) ;
const struct mac_stats * s ;
spin_lock ( & adapter - > stats_lock ) ;
s = t3_mac_update_stats ( & pi - > mac ) ;
spin_unlock ( & adapter - > stats_lock ) ;
* data + + = s - > tx_octets ;
* data + + = s - > tx_frames ;
* data + + = s - > tx_mcast_frames ;
* data + + = s - > tx_bcast_frames ;
* data + + = s - > tx_pause ;
* data + + = s - > tx_underrun ;
* data + + = s - > tx_fifo_urun ;
* data + + = s - > tx_frames_64 ;
* data + + = s - > tx_frames_65_127 ;
* data + + = s - > tx_frames_128_255 ;
* data + + = s - > tx_frames_256_511 ;
* data + + = s - > tx_frames_512_1023 ;
* data + + = s - > tx_frames_1024_1518 ;
* data + + = s - > tx_frames_1519_max ;
* data + + = s - > rx_octets ;
* data + + = s - > rx_frames ;
* data + + = s - > rx_mcast_frames ;
* data + + = s - > rx_bcast_frames ;
* data + + = s - > rx_pause ;
* data + + = s - > rx_fcs_errs ;
* data + + = s - > rx_symbol_errs ;
* data + + = s - > rx_short ;
* data + + = s - > rx_jabber ;
* data + + = s - > rx_too_long ;
* data + + = s - > rx_fifo_ovfl ;
* data + + = s - > rx_frames_64 ;
* data + + = s - > rx_frames_65_127 ;
* data + + = s - > rx_frames_128_255 ;
* data + + = s - > rx_frames_256_511 ;
* data + + = s - > rx_frames_512_1023 ;
* data + + = s - > rx_frames_1024_1518 ;
* data + + = s - > rx_frames_1519_max ;
* data + + = pi - > phy . fifo_errors ;
* data + + = collect_sge_port_stats ( adapter , pi , SGE_PSTAT_TSO ) ;
* data + + = collect_sge_port_stats ( adapter , pi , SGE_PSTAT_VLANEX ) ;
* data + + = collect_sge_port_stats ( adapter , pi , SGE_PSTAT_VLANINS ) ;
* data + + = collect_sge_port_stats ( adapter , pi , SGE_PSTAT_TX_CSUM ) ;
* data + + = collect_sge_port_stats ( adapter , pi , SGE_PSTAT_RX_CSUM_GOOD ) ;
* data + + = s - > rx_cong_drops ;
2007-03-18 23:10:12 +03:00
* data + + = s - > num_toggled ;
* data + + = s - > num_resets ;
2007-01-19 06:04:14 +03:00
}
static inline void reg_block_dump ( struct adapter * ap , void * buf ,
unsigned int start , unsigned int end )
{
u32 * p = buf + start ;
for ( ; start < = end ; start + = sizeof ( u32 ) )
* p + + = t3_read_reg ( ap , start ) ;
}
static void get_regs ( struct net_device * dev , struct ethtool_regs * regs ,
void * buf )
{
struct adapter * ap = dev - > priv ;
/*
* Version scheme :
* bits 0. .9 : chip version
* bits 10. .15 : chip revision
* bit 31 : set for PCIe cards
*/
regs - > version = 3 | ( ap - > params . rev < < 10 ) | ( is_pcie ( ap ) < < 31 ) ;
/*
* We skip the MAC statistics registers because they are clear - on - read .
* Also reading multi - register stats would need to synchronize with the
* periodic mac stats accumulation . Hard to justify the complexity .
*/
memset ( buf , 0 , T3_REGMAP_SIZE ) ;
reg_block_dump ( ap , buf , 0 , A_SG_RSPQ_CREDIT_RETURN ) ;
reg_block_dump ( ap , buf , A_SG_HI_DRB_HI_THRSH , A_ULPRX_PBL_ULIMIT ) ;
reg_block_dump ( ap , buf , A_ULPTX_CONFIG , A_MPS_INT_CAUSE ) ;
reg_block_dump ( ap , buf , A_CPL_SWITCH_CNTRL , A_CPL_MAP_TBL_DATA ) ;
reg_block_dump ( ap , buf , A_SMB_GLOBAL_TIME_CFG , A_XGM_SERDES_STAT3 ) ;
reg_block_dump ( ap , buf , A_XGM_SERDES_STATUS0 ,
XGM_REG ( A_XGM_SERDES_STAT3 , 1 ) ) ;
reg_block_dump ( ap , buf , XGM_REG ( A_XGM_SERDES_STATUS0 , 1 ) ,
XGM_REG ( A_XGM_RX_SPI4_SOP_EOP_CNT , 1 ) ) ;
}
static int restart_autoneg ( struct net_device * dev )
{
struct port_info * p = netdev_priv ( dev ) ;
if ( ! netif_running ( dev ) )
return - EAGAIN ;
if ( p - > link_config . autoneg ! = AUTONEG_ENABLE )
return - EINVAL ;
p - > phy . ops - > autoneg_restart ( & p - > phy ) ;
return 0 ;
}
static int cxgb3_phys_id ( struct net_device * dev , u32 data )
{
int i ;
struct adapter * adapter = dev - > priv ;
if ( data = = 0 )
data = 2 ;
for ( i = 0 ; i < data * 2 ; i + + ) {
t3_set_reg_field ( adapter , A_T3DBG_GPIO_EN , F_GPIO0_OUT_VAL ,
( i & 1 ) ? F_GPIO0_OUT_VAL : 0 ) ;
if ( msleep_interruptible ( 500 ) )
break ;
}
t3_set_reg_field ( adapter , A_T3DBG_GPIO_EN , F_GPIO0_OUT_VAL ,
F_GPIO0_OUT_VAL ) ;
return 0 ;
}
static int get_settings ( struct net_device * dev , struct ethtool_cmd * cmd )
{
struct port_info * p = netdev_priv ( dev ) ;
cmd - > supported = p - > link_config . supported ;
cmd - > advertising = p - > link_config . advertising ;
if ( netif_carrier_ok ( dev ) ) {
cmd - > speed = p - > link_config . speed ;
cmd - > duplex = p - > link_config . duplex ;
} else {
cmd - > speed = - 1 ;
cmd - > duplex = - 1 ;
}
cmd - > port = ( cmd - > supported & SUPPORTED_TP ) ? PORT_TP : PORT_FIBRE ;
cmd - > phy_address = p - > phy . addr ;
cmd - > transceiver = XCVR_EXTERNAL ;
cmd - > autoneg = p - > link_config . autoneg ;
cmd - > maxtxpkt = 0 ;
cmd - > maxrxpkt = 0 ;
return 0 ;
}
static int speed_duplex_to_caps ( int speed , int duplex )
{
int cap = 0 ;
switch ( speed ) {
case SPEED_10 :
if ( duplex = = DUPLEX_FULL )
cap = SUPPORTED_10baseT_Full ;
else
cap = SUPPORTED_10baseT_Half ;
break ;
case SPEED_100 :
if ( duplex = = DUPLEX_FULL )
cap = SUPPORTED_100baseT_Full ;
else
cap = SUPPORTED_100baseT_Half ;
break ;
case SPEED_1000 :
if ( duplex = = DUPLEX_FULL )
cap = SUPPORTED_1000baseT_Full ;
else
cap = SUPPORTED_1000baseT_Half ;
break ;
case SPEED_10000 :
if ( duplex = = DUPLEX_FULL )
cap = SUPPORTED_10000baseT_Full ;
}
return cap ;
}
# define ADVERTISED_MASK (ADVERTISED_10baseT_Half | ADVERTISED_10baseT_Full | \
ADVERTISED_100baseT_Half | ADVERTISED_100baseT_Full | \
ADVERTISED_1000baseT_Half | ADVERTISED_1000baseT_Full | \
ADVERTISED_10000baseT_Full )
static int set_settings ( struct net_device * dev , struct ethtool_cmd * cmd )
{
struct port_info * p = netdev_priv ( dev ) ;
struct link_config * lc = & p - > link_config ;
if ( ! ( lc - > supported & SUPPORTED_Autoneg ) )
return - EOPNOTSUPP ; /* can't change speed/duplex */
if ( cmd - > autoneg = = AUTONEG_DISABLE ) {
int cap = speed_duplex_to_caps ( cmd - > speed , cmd - > duplex ) ;
if ( ! ( lc - > supported & cap ) | | cmd - > speed = = SPEED_1000 )
return - EINVAL ;
lc - > requested_speed = cmd - > speed ;
lc - > requested_duplex = cmd - > duplex ;
lc - > advertising = 0 ;
} else {
cmd - > advertising & = ADVERTISED_MASK ;
cmd - > advertising & = lc - > supported ;
if ( ! cmd - > advertising )
return - EINVAL ;
lc - > requested_speed = SPEED_INVALID ;
lc - > requested_duplex = DUPLEX_INVALID ;
lc - > advertising = cmd - > advertising | ADVERTISED_Autoneg ;
}
lc - > autoneg = cmd - > autoneg ;
if ( netif_running ( dev ) )
t3_link_start ( & p - > phy , & p - > mac , lc ) ;
return 0 ;
}
static void get_pauseparam ( struct net_device * dev ,
struct ethtool_pauseparam * epause )
{
struct port_info * p = netdev_priv ( dev ) ;
epause - > autoneg = ( p - > link_config . requested_fc & PAUSE_AUTONEG ) ! = 0 ;
epause - > rx_pause = ( p - > link_config . fc & PAUSE_RX ) ! = 0 ;
epause - > tx_pause = ( p - > link_config . fc & PAUSE_TX ) ! = 0 ;
}
static int set_pauseparam ( struct net_device * dev ,
struct ethtool_pauseparam * epause )
{
struct port_info * p = netdev_priv ( dev ) ;
struct link_config * lc = & p - > link_config ;
if ( epause - > autoneg = = AUTONEG_DISABLE )
lc - > requested_fc = 0 ;
else if ( lc - > supported & SUPPORTED_Autoneg )
lc - > requested_fc = PAUSE_AUTONEG ;
else
return - EINVAL ;
if ( epause - > rx_pause )
lc - > requested_fc | = PAUSE_RX ;
if ( epause - > tx_pause )
lc - > requested_fc | = PAUSE_TX ;
if ( lc - > autoneg = = AUTONEG_ENABLE ) {
if ( netif_running ( dev ) )
t3_link_start ( & p - > phy , & p - > mac , lc ) ;
} else {
lc - > fc = lc - > requested_fc & ( PAUSE_RX | PAUSE_TX ) ;
if ( netif_running ( dev ) )
t3_mac_set_speed_duplex_fc ( & p - > mac , - 1 , - 1 , lc - > fc ) ;
}
return 0 ;
}
static u32 get_rx_csum ( struct net_device * dev )
{
struct port_info * p = netdev_priv ( dev ) ;
return p - > rx_csum_offload ;
}
static int set_rx_csum ( struct net_device * dev , u32 data )
{
struct port_info * p = netdev_priv ( dev ) ;
p - > rx_csum_offload = data ;
return 0 ;
}
static void get_sge_param ( struct net_device * dev , struct ethtool_ringparam * e )
{
2007-03-18 23:10:01 +03:00
const struct adapter * adapter = dev - > priv ;
const struct port_info * pi = netdev_priv ( dev ) ;
const struct qset_params * q = & adapter - > params . sge . qset [ pi - > first_qset ] ;
2007-01-19 06:04:14 +03:00
e - > rx_max_pending = MAX_RX_BUFFERS ;
e - > rx_mini_max_pending = 0 ;
e - > rx_jumbo_max_pending = MAX_RX_JUMBO_BUFFERS ;
e - > tx_max_pending = MAX_TXQ_ENTRIES ;
2007-03-18 23:10:01 +03:00
e - > rx_pending = q - > fl_size ;
e - > rx_mini_pending = q - > rspq_size ;
e - > rx_jumbo_pending = q - > jumbo_size ;
e - > tx_pending = q - > txq_size [ 0 ] ;
2007-01-19 06:04:14 +03:00
}
static int set_sge_param ( struct net_device * dev , struct ethtool_ringparam * e )
{
int i ;
2007-03-18 23:10:01 +03:00
struct qset_params * q ;
2007-01-19 06:04:14 +03:00
struct adapter * adapter = dev - > priv ;
2007-03-18 23:10:01 +03:00
const struct port_info * pi = netdev_priv ( dev ) ;
2007-01-19 06:04:14 +03:00
if ( e - > rx_pending > MAX_RX_BUFFERS | |
e - > rx_jumbo_pending > MAX_RX_JUMBO_BUFFERS | |
e - > tx_pending > MAX_TXQ_ENTRIES | |
e - > rx_mini_pending > MAX_RSPQ_ENTRIES | |
e - > rx_mini_pending < MIN_RSPQ_ENTRIES | |
e - > rx_pending < MIN_FL_ENTRIES | |
e - > rx_jumbo_pending < MIN_FL_ENTRIES | |
e - > tx_pending < adapter - > params . nports * MIN_TXQ_ENTRIES )
return - EINVAL ;
if ( adapter - > flags & FULL_INIT_DONE )
return - EBUSY ;
2007-03-18 23:10:01 +03:00
q = & adapter - > params . sge . qset [ pi - > first_qset ] ;
for ( i = 0 ; i < pi - > nqsets ; + + i , + + q ) {
2007-01-19 06:04:14 +03:00
q - > rspq_size = e - > rx_mini_pending ;
q - > fl_size = e - > rx_pending ;
q - > jumbo_size = e - > rx_jumbo_pending ;
q - > txq_size [ 0 ] = e - > tx_pending ;
q - > txq_size [ 1 ] = e - > tx_pending ;
q - > txq_size [ 2 ] = e - > tx_pending ;
}
return 0 ;
}
static int set_coalesce ( struct net_device * dev , struct ethtool_coalesce * c )
{
struct adapter * adapter = dev - > priv ;
struct qset_params * qsp = & adapter - > params . sge . qset [ 0 ] ;
struct sge_qset * qs = & adapter - > sge . qs [ 0 ] ;
if ( c - > rx_coalesce_usecs * 10 > M_NEWTIMER )
return - EINVAL ;
qsp - > coalesce_usecs = c - > rx_coalesce_usecs ;
t3_update_qset_coalesce ( qs , qsp ) ;
return 0 ;
}
static int get_coalesce ( struct net_device * dev , struct ethtool_coalesce * c )
{
struct adapter * adapter = dev - > priv ;
struct qset_params * q = adapter - > params . sge . qset ;
c - > rx_coalesce_usecs = q - > coalesce_usecs ;
return 0 ;
}
static int get_eeprom ( struct net_device * dev , struct ethtool_eeprom * e ,
u8 * data )
{
int i , err = 0 ;
struct adapter * adapter = dev - > priv ;
u8 * buf = kmalloc ( EEPROMSIZE , GFP_KERNEL ) ;
if ( ! buf )
return - ENOMEM ;
e - > magic = EEPROM_MAGIC ;
for ( i = e - > offset & ~ 3 ; ! err & & i < e - > offset + e - > len ; i + = 4 )
err = t3_seeprom_read ( adapter , i , ( u32 * ) & buf [ i ] ) ;
if ( ! err )
memcpy ( data , buf + e - > offset , e - > len ) ;
kfree ( buf ) ;
return err ;
}
static int set_eeprom ( struct net_device * dev , struct ethtool_eeprom * eeprom ,
u8 * data )
{
u8 * buf ;
int err = 0 ;
u32 aligned_offset , aligned_len , * p ;
struct adapter * adapter = dev - > priv ;
if ( eeprom - > magic ! = EEPROM_MAGIC )
return - EINVAL ;
aligned_offset = eeprom - > offset & ~ 3 ;
aligned_len = ( eeprom - > len + ( eeprom - > offset & 3 ) + 3 ) & ~ 3 ;
if ( aligned_offset ! = eeprom - > offset | | aligned_len ! = eeprom - > len ) {
buf = kmalloc ( aligned_len , GFP_KERNEL ) ;
if ( ! buf )
return - ENOMEM ;
err = t3_seeprom_read ( adapter , aligned_offset , ( u32 * ) buf ) ;
if ( ! err & & aligned_len > 4 )
err = t3_seeprom_read ( adapter ,
aligned_offset + aligned_len - 4 ,
( u32 * ) & buf [ aligned_len - 4 ] ) ;
if ( err )
goto out ;
memcpy ( buf + ( eeprom - > offset & 3 ) , data , eeprom - > len ) ;
} else
buf = data ;
err = t3_seeprom_wp ( adapter , 0 ) ;
if ( err )
goto out ;
for ( p = ( u32 * ) buf ; ! err & & aligned_len ; aligned_len - = 4 , p + + ) {
err = t3_seeprom_write ( adapter , aligned_offset , * p ) ;
aligned_offset + = 4 ;
}
if ( ! err )
err = t3_seeprom_wp ( adapter , 1 ) ;
out :
if ( buf ! = data )
kfree ( buf ) ;
return err ;
}
static void get_wol ( struct net_device * dev , struct ethtool_wolinfo * wol )
{
wol - > supported = 0 ;
wol - > wolopts = 0 ;
memset ( & wol - > sopass , 0 , sizeof ( wol - > sopass ) ) ;
}
static const struct ethtool_ops cxgb_ethtool_ops = {
. get_settings = get_settings ,
. set_settings = set_settings ,
. get_drvinfo = get_drvinfo ,
. get_msglevel = get_msglevel ,
. set_msglevel = set_msglevel ,
. get_ringparam = get_sge_param ,
. set_ringparam = set_sge_param ,
. get_coalesce = get_coalesce ,
. set_coalesce = set_coalesce ,
. get_eeprom_len = get_eeprom_len ,
. get_eeprom = get_eeprom ,
. set_eeprom = set_eeprom ,
. get_pauseparam = get_pauseparam ,
. set_pauseparam = set_pauseparam ,
. get_rx_csum = get_rx_csum ,
. set_rx_csum = set_rx_csum ,
. get_tx_csum = ethtool_op_get_tx_csum ,
. set_tx_csum = ethtool_op_set_tx_csum ,
. get_sg = ethtool_op_get_sg ,
. set_sg = ethtool_op_set_sg ,
. get_link = ethtool_op_get_link ,
. get_strings = get_strings ,
. phys_id = cxgb3_phys_id ,
. nway_reset = restart_autoneg ,
. get_stats_count = get_stats_count ,
. get_ethtool_stats = get_stats ,
. get_regs_len = get_regs_len ,
. get_regs = get_regs ,
. get_wol = get_wol ,
. get_tso = ethtool_op_get_tso ,
. set_tso = ethtool_op_set_tso ,
. get_perm_addr = ethtool_op_get_perm_addr
} ;
static int in_range ( int val , int lo , int hi )
{
return val < 0 | | ( val < = hi & & val > = lo ) ;
}
static int cxgb_extension_ioctl ( struct net_device * dev , void __user * useraddr )
{
int ret ;
u32 cmd ;
struct adapter * adapter = dev - > priv ;
if ( copy_from_user ( & cmd , useraddr , sizeof ( cmd ) ) )
return - EFAULT ;
switch ( cmd ) {
case CHELSIO_SET_QSET_PARAMS : {
int i ;
struct qset_params * q ;
struct ch_qset_params t ;
if ( ! capable ( CAP_NET_ADMIN ) )
return - EPERM ;
if ( copy_from_user ( & t , useraddr , sizeof ( t ) ) )
return - EFAULT ;
if ( t . qset_idx > = SGE_QSETS )
return - EINVAL ;
if ( ! in_range ( t . intr_lat , 0 , M_NEWTIMER ) | |
! in_range ( t . cong_thres , 0 , 255 ) | |
! in_range ( t . txq_size [ 0 ] , MIN_TXQ_ENTRIES ,
MAX_TXQ_ENTRIES ) | |
! in_range ( t . txq_size [ 1 ] , MIN_TXQ_ENTRIES ,
MAX_TXQ_ENTRIES ) | |
! in_range ( t . txq_size [ 2 ] , MIN_CTRL_TXQ_ENTRIES ,
MAX_CTRL_TXQ_ENTRIES ) | |
! in_range ( t . fl_size [ 0 ] , MIN_FL_ENTRIES ,
MAX_RX_BUFFERS )
| | ! in_range ( t . fl_size [ 1 ] , MIN_FL_ENTRIES ,
MAX_RX_JUMBO_BUFFERS )
| | ! in_range ( t . rspq_size , MIN_RSPQ_ENTRIES ,
MAX_RSPQ_ENTRIES ) )
return - EINVAL ;
if ( ( adapter - > flags & FULL_INIT_DONE ) & &
( t . rspq_size > = 0 | | t . fl_size [ 0 ] > = 0 | |
t . fl_size [ 1 ] > = 0 | | t . txq_size [ 0 ] > = 0 | |
t . txq_size [ 1 ] > = 0 | | t . txq_size [ 2 ] > = 0 | |
t . polling > = 0 | | t . cong_thres > = 0 ) )
return - EBUSY ;
q = & adapter - > params . sge . qset [ t . qset_idx ] ;
if ( t . rspq_size > = 0 )
q - > rspq_size = t . rspq_size ;
if ( t . fl_size [ 0 ] > = 0 )
q - > fl_size = t . fl_size [ 0 ] ;
if ( t . fl_size [ 1 ] > = 0 )
q - > jumbo_size = t . fl_size [ 1 ] ;
if ( t . txq_size [ 0 ] > = 0 )
q - > txq_size [ 0 ] = t . txq_size [ 0 ] ;
if ( t . txq_size [ 1 ] > = 0 )
q - > txq_size [ 1 ] = t . txq_size [ 1 ] ;
if ( t . txq_size [ 2 ] > = 0 )
q - > txq_size [ 2 ] = t . txq_size [ 2 ] ;
if ( t . cong_thres > = 0 )
q - > cong_thres = t . cong_thres ;
if ( t . intr_lat > = 0 ) {
struct sge_qset * qs =
& adapter - > sge . qs [ t . qset_idx ] ;
q - > coalesce_usecs = t . intr_lat ;
t3_update_qset_coalesce ( qs , q ) ;
}
if ( t . polling > = 0 ) {
if ( adapter - > flags & USING_MSIX )
q - > polling = t . polling ;
else {
/* No polling with INTx for T3A */
if ( adapter - > params . rev = = 0 & &
! ( adapter - > flags & USING_MSI ) )
t . polling = 0 ;
for ( i = 0 ; i < SGE_QSETS ; i + + ) {
q = & adapter - > params . sge .
qset [ i ] ;
q - > polling = t . polling ;
}
}
}
break ;
}
case CHELSIO_GET_QSET_PARAMS : {
struct qset_params * q ;
struct ch_qset_params t ;
if ( copy_from_user ( & t , useraddr , sizeof ( t ) ) )
return - EFAULT ;
if ( t . qset_idx > = SGE_QSETS )
return - EINVAL ;
q = & adapter - > params . sge . qset [ t . qset_idx ] ;
t . rspq_size = q - > rspq_size ;
t . txq_size [ 0 ] = q - > txq_size [ 0 ] ;
t . txq_size [ 1 ] = q - > txq_size [ 1 ] ;
t . txq_size [ 2 ] = q - > txq_size [ 2 ] ;
t . fl_size [ 0 ] = q - > fl_size ;
t . fl_size [ 1 ] = q - > jumbo_size ;
t . polling = q - > polling ;
t . intr_lat = q - > coalesce_usecs ;
t . cong_thres = q - > cong_thres ;
if ( copy_to_user ( useraddr , & t , sizeof ( t ) ) )
return - EFAULT ;
break ;
}
case CHELSIO_SET_QSET_NUM : {
struct ch_reg edata ;
struct port_info * pi = netdev_priv ( dev ) ;
unsigned int i , first_qset = 0 , other_qsets = 0 ;
if ( ! capable ( CAP_NET_ADMIN ) )
return - EPERM ;
if ( adapter - > flags & FULL_INIT_DONE )
return - EBUSY ;
if ( copy_from_user ( & edata , useraddr , sizeof ( edata ) ) )
return - EFAULT ;
if ( edata . val < 1 | |
( edata . val > 1 & & ! ( adapter - > flags & USING_MSIX ) ) )
return - EINVAL ;
for_each_port ( adapter , i )
if ( adapter - > port [ i ] & & adapter - > port [ i ] ! = dev )
other_qsets + = adap2pinfo ( adapter , i ) - > nqsets ;
if ( edata . val + other_qsets > SGE_QSETS )
return - EINVAL ;
pi - > nqsets = edata . val ;
for_each_port ( adapter , i )
if ( adapter - > port [ i ] ) {
pi = adap2pinfo ( adapter , i ) ;
pi - > first_qset = first_qset ;
first_qset + = pi - > nqsets ;
}
break ;
}
case CHELSIO_GET_QSET_NUM : {
struct ch_reg edata ;
struct port_info * pi = netdev_priv ( dev ) ;
edata . cmd = CHELSIO_GET_QSET_NUM ;
edata . val = pi - > nqsets ;
if ( copy_to_user ( useraddr , & edata , sizeof ( edata ) ) )
return - EFAULT ;
break ;
}
case CHELSIO_LOAD_FW : {
u8 * fw_data ;
struct ch_mem_range t ;
if ( ! capable ( CAP_NET_ADMIN ) )
return - EPERM ;
if ( copy_from_user ( & t , useraddr , sizeof ( t ) ) )
return - EFAULT ;
fw_data = kmalloc ( t . len , GFP_KERNEL ) ;
if ( ! fw_data )
return - ENOMEM ;
if ( copy_from_user
( fw_data , useraddr + sizeof ( t ) , t . len ) ) {
kfree ( fw_data ) ;
return - EFAULT ;
}
ret = t3_load_fw ( adapter , fw_data , t . len ) ;
kfree ( fw_data ) ;
if ( ret )
return ret ;
break ;
}
case CHELSIO_SETMTUTAB : {
struct ch_mtus m ;
int i ;
if ( ! is_offload ( adapter ) )
return - EOPNOTSUPP ;
if ( ! capable ( CAP_NET_ADMIN ) )
return - EPERM ;
if ( offload_running ( adapter ) )
return - EBUSY ;
if ( copy_from_user ( & m , useraddr , sizeof ( m ) ) )
return - EFAULT ;
if ( m . nmtus ! = NMTUS )
return - EINVAL ;
if ( m . mtus [ 0 ] < 81 ) /* accommodate SACK */
return - EINVAL ;
/* MTUs must be in ascending order */
for ( i = 1 ; i < NMTUS ; + + i )
if ( m . mtus [ i ] < m . mtus [ i - 1 ] )
return - EINVAL ;
memcpy ( adapter - > params . mtus , m . mtus ,
sizeof ( adapter - > params . mtus ) ) ;
break ;
}
case CHELSIO_GET_PM : {
struct tp_params * p = & adapter - > params . tp ;
struct ch_pm m = { . cmd = CHELSIO_GET_PM } ;
if ( ! is_offload ( adapter ) )
return - EOPNOTSUPP ;
m . tx_pg_sz = p - > tx_pg_size ;
m . tx_num_pg = p - > tx_num_pgs ;
m . rx_pg_sz = p - > rx_pg_size ;
m . rx_num_pg = p - > rx_num_pgs ;
m . pm_total = p - > pmtx_size + p - > chan_rx_size * p - > nchan ;
if ( copy_to_user ( useraddr , & m , sizeof ( m ) ) )
return - EFAULT ;
break ;
}
case CHELSIO_SET_PM : {
struct ch_pm m ;
struct tp_params * p = & adapter - > params . tp ;
if ( ! is_offload ( adapter ) )
return - EOPNOTSUPP ;
if ( ! capable ( CAP_NET_ADMIN ) )
return - EPERM ;
if ( adapter - > flags & FULL_INIT_DONE )
return - EBUSY ;
if ( copy_from_user ( & m , useraddr , sizeof ( m ) ) )
return - EFAULT ;
if ( ! m . rx_pg_sz | | ( m . rx_pg_sz & ( m . rx_pg_sz - 1 ) ) | |
! m . tx_pg_sz | | ( m . tx_pg_sz & ( m . tx_pg_sz - 1 ) ) )
return - EINVAL ; /* not power of 2 */
if ( ! ( m . rx_pg_sz & 0x14000 ) )
return - EINVAL ; /* not 16KB or 64KB */
if ( ! ( m . tx_pg_sz & 0x1554000 ) )
return - EINVAL ;
if ( m . tx_num_pg = = - 1 )
m . tx_num_pg = p - > tx_num_pgs ;
if ( m . rx_num_pg = = - 1 )
m . rx_num_pg = p - > rx_num_pgs ;
if ( m . tx_num_pg % 24 | | m . rx_num_pg % 24 )
return - EINVAL ;
if ( m . rx_num_pg * m . rx_pg_sz > p - > chan_rx_size | |
m . tx_num_pg * m . tx_pg_sz > p - > chan_tx_size )
return - EINVAL ;
p - > rx_pg_size = m . rx_pg_sz ;
p - > tx_pg_size = m . tx_pg_sz ;
p - > rx_num_pgs = m . rx_num_pg ;
p - > tx_num_pgs = m . tx_num_pg ;
break ;
}
case CHELSIO_GET_MEM : {
struct ch_mem_range t ;
struct mc7 * mem ;
u64 buf [ 32 ] ;
if ( ! is_offload ( adapter ) )
return - EOPNOTSUPP ;
if ( ! ( adapter - > flags & FULL_INIT_DONE ) )
return - EIO ; /* need the memory controllers */
if ( copy_from_user ( & t , useraddr , sizeof ( t ) ) )
return - EFAULT ;
if ( ( t . addr & 7 ) | | ( t . len & 7 ) )
return - EINVAL ;
if ( t . mem_id = = MEM_CM )
mem = & adapter - > cm ;
else if ( t . mem_id = = MEM_PMRX )
mem = & adapter - > pmrx ;
else if ( t . mem_id = = MEM_PMTX )
mem = & adapter - > pmtx ;
else
return - EINVAL ;
/*
2007-02-25 03:43:56 +03:00
* Version scheme :
* bits 0. .9 : chip version
* bits 10. .15 : chip revision
*/
2007-01-19 06:04:14 +03:00
t . version = 3 | ( adapter - > params . rev < < 10 ) ;
if ( copy_to_user ( useraddr , & t , sizeof ( t ) ) )
return - EFAULT ;
/*
* Read 256 bytes at a time as len can be large and we don ' t
* want to use huge intermediate buffers .
*/
useraddr + = sizeof ( t ) ; /* advance to start of buffer */
while ( t . len ) {
unsigned int chunk =
min_t ( unsigned int , t . len , sizeof ( buf ) ) ;
ret =
t3_mc7_bd_read ( mem , t . addr / 8 , chunk / 8 ,
buf ) ;
if ( ret )
return ret ;
if ( copy_to_user ( useraddr , buf , chunk ) )
return - EFAULT ;
useraddr + = chunk ;
t . addr + = chunk ;
t . len - = chunk ;
}
break ;
}
case CHELSIO_SET_TRACE_FILTER : {
struct ch_trace t ;
const struct trace_params * tp ;
if ( ! capable ( CAP_NET_ADMIN ) )
return - EPERM ;
if ( ! offload_running ( adapter ) )
return - EAGAIN ;
if ( copy_from_user ( & t , useraddr , sizeof ( t ) ) )
return - EFAULT ;
tp = ( const struct trace_params * ) & t . sip ;
if ( t . config_tx )
t3_config_trace_filter ( adapter , tp , 0 ,
t . invert_match ,
t . trace_tx ) ;
if ( t . config_rx )
t3_config_trace_filter ( adapter , tp , 1 ,
t . invert_match ,
t . trace_rx ) ;
break ;
}
default :
return - EOPNOTSUPP ;
}
return 0 ;
}
static int cxgb_ioctl ( struct net_device * dev , struct ifreq * req , int cmd )
{
int ret , mmd ;
struct adapter * adapter = dev - > priv ;
struct port_info * pi = netdev_priv ( dev ) ;
struct mii_ioctl_data * data = if_mii ( req ) ;
switch ( cmd ) {
case SIOCGMIIPHY :
data - > phy_id = pi - > phy . addr ;
/* FALLTHRU */
case SIOCGMIIREG : {
u32 val ;
struct cphy * phy = & pi - > phy ;
if ( ! phy - > mdio_read )
return - EOPNOTSUPP ;
if ( is_10G ( adapter ) ) {
mmd = data - > phy_id > > 8 ;
if ( ! mmd )
mmd = MDIO_DEV_PCS ;
else if ( mmd > MDIO_DEV_XGXS )
return - EINVAL ;
ret =
phy - > mdio_read ( adapter , data - > phy_id & 0x1f ,
mmd , data - > reg_num , & val ) ;
} else
ret =
phy - > mdio_read ( adapter , data - > phy_id & 0x1f ,
0 , data - > reg_num & 0x1f ,
& val ) ;
if ( ! ret )
data - > val_out = val ;
break ;
}
case SIOCSMIIREG : {
struct cphy * phy = & pi - > phy ;
if ( ! capable ( CAP_NET_ADMIN ) )
return - EPERM ;
if ( ! phy - > mdio_write )
return - EOPNOTSUPP ;
if ( is_10G ( adapter ) ) {
mmd = data - > phy_id > > 8 ;
if ( ! mmd )
mmd = MDIO_DEV_PCS ;
else if ( mmd > MDIO_DEV_XGXS )
return - EINVAL ;
ret =
phy - > mdio_write ( adapter ,
data - > phy_id & 0x1f , mmd ,
data - > reg_num ,
data - > val_in ) ;
} else
ret =
phy - > mdio_write ( adapter ,
data - > phy_id & 0x1f , 0 ,
data - > reg_num & 0x1f ,
data - > val_in ) ;
break ;
}
case SIOCCHIOCTL :
return cxgb_extension_ioctl ( dev , req - > ifr_data ) ;
default :
return - EOPNOTSUPP ;
}
return ret ;
}
static int cxgb_change_mtu ( struct net_device * dev , int new_mtu )
{
int ret ;
struct adapter * adapter = dev - > priv ;
struct port_info * pi = netdev_priv ( dev ) ;
if ( new_mtu < 81 ) /* accommodate SACK */
return - EINVAL ;
if ( ( ret = t3_mac_set_mtu ( & pi - > mac , new_mtu ) ) )
return ret ;
dev - > mtu = new_mtu ;
init_port_mtus ( adapter ) ;
if ( adapter - > params . rev = = 0 & & offload_running ( adapter ) )
t3_load_mtus ( adapter , adapter - > params . mtus ,
adapter - > params . a_wnd , adapter - > params . b_wnd ,
adapter - > port [ 0 ] - > mtu ) ;
return 0 ;
}
static int cxgb_set_mac_addr ( struct net_device * dev , void * p )
{
struct adapter * adapter = dev - > priv ;
struct port_info * pi = netdev_priv ( dev ) ;
struct sockaddr * addr = p ;
if ( ! is_valid_ether_addr ( addr - > sa_data ) )
return - EINVAL ;
memcpy ( dev - > dev_addr , addr - > sa_data , dev - > addr_len ) ;
t3_mac_set_address ( & pi - > mac , 0 , dev - > dev_addr ) ;
if ( offload_running ( adapter ) )
write_smt_entry ( adapter , pi - > port_id ) ;
return 0 ;
}
/**
* t3_synchronize_rx - wait for current Rx processing on a port to complete
* @ adap : the adapter
* @ p : the port
*
* Ensures that current Rx processing on any of the queues associated with
* the given port completes before returning . We do this by acquiring and
* releasing the locks of the response queues associated with the port .
*/
static void t3_synchronize_rx ( struct adapter * adap , const struct port_info * p )
{
int i ;
for ( i = 0 ; i < p - > nqsets ; i + + ) {
struct sge_rspq * q = & adap - > sge . qs [ i + p - > first_qset ] . rspq ;
spin_lock_irq ( & q - > lock ) ;
spin_unlock_irq ( & q - > lock ) ;
}
}
static void vlan_rx_register ( struct net_device * dev , struct vlan_group * grp )
{
struct adapter * adapter = dev - > priv ;
struct port_info * pi = netdev_priv ( dev ) ;
pi - > vlan_grp = grp ;
if ( adapter - > params . rev > 0 )
t3_set_vlan_accel ( adapter , 1 < < pi - > port_id , grp ! = NULL ) ;
else {
/* single control for all ports */
unsigned int i , have_vlans = 0 ;
for_each_port ( adapter , i )
have_vlans | = adap2pinfo ( adapter , i ) - > vlan_grp ! = NULL ;
t3_set_vlan_accel ( adapter , 1 , have_vlans ) ;
}
t3_synchronize_rx ( adapter , pi ) ;
}
static void vlan_rx_kill_vid ( struct net_device * dev , unsigned short vid )
{
/* nothing */
}
# ifdef CONFIG_NET_POLL_CONTROLLER
static void cxgb_netpoll ( struct net_device * dev )
{
struct adapter * adapter = dev - > priv ;
struct sge_qset * qs = dev2qset ( dev ) ;
t3_intr_handler ( adapter , qs - > rspq . polling ) ( adapter - > pdev - > irq ,
adapter ) ;
}
# endif
/*
* Periodic accumulation of MAC statistics .
*/
static void mac_stats_update ( struct adapter * adapter )
{
int i ;
for_each_port ( adapter , i ) {
struct net_device * dev = adapter - > port [ i ] ;
struct port_info * p = netdev_priv ( dev ) ;
if ( netif_running ( dev ) ) {
spin_lock ( & adapter - > stats_lock ) ;
t3_mac_update_stats ( & p - > mac ) ;
spin_unlock ( & adapter - > stats_lock ) ;
}
}
}
static void check_link_status ( struct adapter * adapter )
{
int i ;
for_each_port ( adapter , i ) {
struct net_device * dev = adapter - > port [ i ] ;
struct port_info * p = netdev_priv ( dev ) ;
if ( ! ( p - > port_type - > caps & SUPPORTED_IRQ ) & & netif_running ( dev ) )
t3_link_changed ( adapter , i ) ;
}
}
2007-03-18 23:10:12 +03:00
static void check_t3b2_mac ( struct adapter * adapter )
{
int i ;
rtnl_lock ( ) ; /* synchronize with ifdown */
for_each_port ( adapter , i ) {
struct net_device * dev = adapter - > port [ i ] ;
struct port_info * p = netdev_priv ( dev ) ;
int status ;
if ( ! netif_running ( dev ) )
continue ;
status = 0 ;
if ( netif_running ( dev ) )
status = t3b2_mac_watchdog_task ( & p - > mac ) ;
if ( status = = 1 )
p - > mac . stats . num_toggled + + ;
else if ( status = = 2 ) {
struct cmac * mac = & p - > mac ;
t3_mac_set_mtu ( mac , dev - > mtu ) ;
t3_mac_set_address ( mac , 0 , dev - > dev_addr ) ;
cxgb_set_rxmode ( dev ) ;
t3_link_start ( & p - > phy , mac , & p - > link_config ) ;
t3_mac_enable ( mac , MAC_DIRECTION_RX | MAC_DIRECTION_TX ) ;
t3_port_intr_enable ( adapter , p - > port_id ) ;
p - > mac . stats . num_resets + + ;
}
}
rtnl_unlock ( ) ;
}
2007-01-19 06:04:14 +03:00
static void t3_adap_check_task ( struct work_struct * work )
{
struct adapter * adapter = container_of ( work , struct adapter ,
adap_check_task . work ) ;
const struct adapter_params * p = & adapter - > params ;
adapter - > check_task_cnt + + ;
/* Check link status for PHYs without interrupts */
if ( p - > linkpoll_period )
check_link_status ( adapter ) ;
/* Accumulate MAC stats if needed */
if ( ! p - > linkpoll_period | |
( adapter - > check_task_cnt * p - > linkpoll_period ) / 10 > =
p - > stats_update_period ) {
mac_stats_update ( adapter ) ;
adapter - > check_task_cnt = 0 ;
}
2007-03-18 23:10:12 +03:00
if ( p - > rev = = T3_REV_B2 )
check_t3b2_mac ( adapter ) ;
2007-01-19 06:04:14 +03:00
/* Schedule the next check update if any port is active. */
spin_lock ( & adapter - > work_lock ) ;
if ( adapter - > open_device_map & PORT_MASK )
schedule_chk_task ( adapter ) ;
spin_unlock ( & adapter - > work_lock ) ;
}
/*
* Processes external ( PHY ) interrupts in process context .
*/
static void ext_intr_task ( struct work_struct * work )
{
struct adapter * adapter = container_of ( work , struct adapter ,
ext_intr_handler_task ) ;
t3_phy_intr_handler ( adapter ) ;
/* Now reenable external interrupts */
spin_lock_irq ( & adapter - > work_lock ) ;
if ( adapter - > slow_intr_mask ) {
adapter - > slow_intr_mask | = F_T3DBG ;
t3_write_reg ( adapter , A_PL_INT_CAUSE0 , F_T3DBG ) ;
t3_write_reg ( adapter , A_PL_INT_ENABLE0 ,
adapter - > slow_intr_mask ) ;
}
spin_unlock_irq ( & adapter - > work_lock ) ;
}
/*
* Interrupt - context handler for external ( PHY ) interrupts .
*/
void t3_os_ext_intr_handler ( struct adapter * adapter )
{
/*
* Schedule a task to handle external interrupts as they may be slow
* and we use a mutex to protect MDIO registers . We disable PHY
* interrupts in the meantime and let the task reenable them when
* it ' s done .
*/
spin_lock ( & adapter - > work_lock ) ;
if ( adapter - > slow_intr_mask ) {
adapter - > slow_intr_mask & = ~ F_T3DBG ;
t3_write_reg ( adapter , A_PL_INT_ENABLE0 ,
adapter - > slow_intr_mask ) ;
queue_work ( cxgb3_wq , & adapter - > ext_intr_handler_task ) ;
}
spin_unlock ( & adapter - > work_lock ) ;
}
void t3_fatal_err ( struct adapter * adapter )
{
unsigned int fw_status [ 4 ] ;
if ( adapter - > flags & FULL_INIT_DONE ) {
t3_sge_stop ( adapter ) ;
t3_intr_disable ( adapter ) ;
}
CH_ALERT ( adapter , " encountered fatal error, operation suspended \n " ) ;
if ( ! t3_cim_ctl_blk_read ( adapter , 0xa0 , 4 , fw_status ) )
CH_ALERT ( adapter , " FW status: 0x%x, 0x%x, 0x%x, 0x%x \n " ,
fw_status [ 0 ] , fw_status [ 1 ] ,
fw_status [ 2 ] , fw_status [ 3 ] ) ;
}
static int __devinit cxgb_enable_msix ( struct adapter * adap )
{
struct msix_entry entries [ SGE_QSETS + 1 ] ;
int i , err ;
for ( i = 0 ; i < ARRAY_SIZE ( entries ) ; + + i )
entries [ i ] . entry = i ;
err = pci_enable_msix ( adap - > pdev , entries , ARRAY_SIZE ( entries ) ) ;
if ( ! err ) {
for ( i = 0 ; i < ARRAY_SIZE ( entries ) ; + + i )
adap - > msix_info [ i ] . vec = entries [ i ] . vector ;
} else if ( err > 0 )
dev_info ( & adap - > pdev - > dev ,
" only %d MSI-X vectors left, not using MSI-X \n " , err ) ;
return err ;
}
static void __devinit print_port_info ( struct adapter * adap ,
const struct adapter_info * ai )
{
static const char * pci_variant [ ] = {
" PCI " , " PCI-X " , " PCI-X ECC " , " PCI-X 266 " , " PCI Express "
} ;
int i ;
char buf [ 80 ] ;
if ( is_pcie ( adap ) )
snprintf ( buf , sizeof ( buf ) , " %s x%d " ,
pci_variant [ adap - > params . pci . variant ] ,
adap - > params . pci . width ) ;
else
snprintf ( buf , sizeof ( buf ) , " %s %dMHz/%d-bit " ,
pci_variant [ adap - > params . pci . variant ] ,
adap - > params . pci . speed , adap - > params . pci . width ) ;
for_each_port ( adap , i ) {
struct net_device * dev = adap - > port [ i ] ;
const struct port_info * pi = netdev_priv ( dev ) ;
if ( ! test_bit ( i , & adap - > registered_device_map ) )
continue ;
printk ( KERN_INFO " %s: %s %s RNIC (rev %d) %s%s \n " ,
dev - > name , ai - > desc , pi - > port_type - > desc ,
adap - > params . rev , buf ,
( adap - > flags & USING_MSIX ) ? " MSI-X " :
( adap - > flags & USING_MSI ) ? " MSI " : " " ) ;
if ( adap - > name = = dev - > name & & adap - > params . vpd . mclk )
printk ( KERN_INFO " %s: %uMB CM, %uMB PMTX, %uMB PMRX \n " ,
adap - > name , t3_mc7_size ( & adap - > cm ) > > 20 ,
t3_mc7_size ( & adap - > pmtx ) > > 20 ,
t3_mc7_size ( & adap - > pmrx ) > > 20 ) ;
}
}
static int __devinit init_one ( struct pci_dev * pdev ,
const struct pci_device_id * ent )
{
static int version_printed ;
int i , err , pci_using_dac = 0 ;
unsigned long mmio_start , mmio_len ;
const struct adapter_info * ai ;
struct adapter * adapter = NULL ;
struct port_info * pi ;
if ( ! version_printed ) {
printk ( KERN_INFO " %s - version %s \n " , DRV_DESC , DRV_VERSION ) ;
+ + version_printed ;
}
if ( ! cxgb3_wq ) {
cxgb3_wq = create_singlethread_workqueue ( DRV_NAME ) ;
if ( ! cxgb3_wq ) {
printk ( KERN_ERR DRV_NAME
" : cannot initialize work queue \n " ) ;
return - ENOMEM ;
}
}
err = pci_request_regions ( pdev , DRV_NAME ) ;
if ( err ) {
/* Just info, some other driver may have claimed the device. */
dev_info ( & pdev - > dev , " cannot obtain PCI resources \n " ) ;
return err ;
}
err = pci_enable_device ( pdev ) ;
if ( err ) {
dev_err ( & pdev - > dev , " cannot enable PCI device \n " ) ;
goto out_release_regions ;
}
if ( ! pci_set_dma_mask ( pdev , DMA_64BIT_MASK ) ) {
pci_using_dac = 1 ;
err = pci_set_consistent_dma_mask ( pdev , DMA_64BIT_MASK ) ;
if ( err ) {
dev_err ( & pdev - > dev , " unable to obtain 64-bit DMA for "
" coherent allocations \n " ) ;
goto out_disable_device ;
}
} else if ( ( err = pci_set_dma_mask ( pdev , DMA_32BIT_MASK ) ) ! = 0 ) {
dev_err ( & pdev - > dev , " no usable DMA configuration \n " ) ;
goto out_disable_device ;
}
pci_set_master ( pdev ) ;
mmio_start = pci_resource_start ( pdev , 0 ) ;
mmio_len = pci_resource_len ( pdev , 0 ) ;
ai = t3_get_adapter_info ( ent - > driver_data ) ;
adapter = kzalloc ( sizeof ( * adapter ) , GFP_KERNEL ) ;
if ( ! adapter ) {
err = - ENOMEM ;
goto out_disable_device ;
}
adapter - > regs = ioremap_nocache ( mmio_start , mmio_len ) ;
if ( ! adapter - > regs ) {
dev_err ( & pdev - > dev , " cannot map device registers \n " ) ;
err = - ENOMEM ;
goto out_free_adapter ;
}
adapter - > pdev = pdev ;
adapter - > name = pci_name ( pdev ) ;
adapter - > msg_enable = dflt_msg_enable ;
adapter - > mmio_len = mmio_len ;
mutex_init ( & adapter - > mdio_lock ) ;
spin_lock_init ( & adapter - > work_lock ) ;
spin_lock_init ( & adapter - > stats_lock ) ;
INIT_LIST_HEAD ( & adapter - > adapter_list ) ;
INIT_WORK ( & adapter - > ext_intr_handler_task , ext_intr_task ) ;
INIT_DELAYED_WORK ( & adapter - > adap_check_task , t3_adap_check_task ) ;
for ( i = 0 ; i < ai - > nports ; + + i ) {
struct net_device * netdev ;
netdev = alloc_etherdev ( sizeof ( struct port_info ) ) ;
if ( ! netdev ) {
err = - ENOMEM ;
goto out_free_dev ;
}
SET_MODULE_OWNER ( netdev ) ;
SET_NETDEV_DEV ( netdev , & pdev - > dev ) ;
adapter - > port [ i ] = netdev ;
pi = netdev_priv ( netdev ) ;
pi - > rx_csum_offload = 1 ;
pi - > nqsets = 1 ;
pi - > first_qset = i ;
pi - > activity = 0 ;
pi - > port_id = i ;
netif_carrier_off ( netdev ) ;
netdev - > irq = pdev - > irq ;
netdev - > mem_start = mmio_start ;
netdev - > mem_end = mmio_start + mmio_len - 1 ;
netdev - > priv = adapter ;
netdev - > features | = NETIF_F_SG | NETIF_F_IP_CSUM | NETIF_F_TSO ;
netdev - > features | = NETIF_F_LLTX ;
if ( pci_using_dac )
netdev - > features | = NETIF_F_HIGHDMA ;
netdev - > features | = NETIF_F_HW_VLAN_TX | NETIF_F_HW_VLAN_RX ;
netdev - > vlan_rx_register = vlan_rx_register ;
netdev - > vlan_rx_kill_vid = vlan_rx_kill_vid ;
netdev - > open = cxgb_open ;
netdev - > stop = cxgb_close ;
netdev - > hard_start_xmit = t3_eth_xmit ;
netdev - > get_stats = cxgb_get_stats ;
netdev - > set_multicast_list = cxgb_set_rxmode ;
netdev - > do_ioctl = cxgb_ioctl ;
netdev - > change_mtu = cxgb_change_mtu ;
netdev - > set_mac_address = cxgb_set_mac_addr ;
# ifdef CONFIG_NET_POLL_CONTROLLER
netdev - > poll_controller = cxgb_netpoll ;
# endif
netdev - > weight = 64 ;
SET_ETHTOOL_OPS ( netdev , & cxgb_ethtool_ops ) ;
}
pci_set_drvdata ( pdev , adapter - > port [ 0 ] ) ;
if ( t3_prep_adapter ( adapter , ai , 1 ) < 0 ) {
err = - ENODEV ;
goto out_free_dev ;
}
/*
* The card is now ready to go . If any errors occur during device
* registration we do not fail the whole card but rather proceed only
* with the ports we manage to register successfully . However we must
* register at least one net device .
*/
for_each_port ( adapter , i ) {
err = register_netdev ( adapter - > port [ i ] ) ;
if ( err )
dev_warn ( & pdev - > dev ,
" cannot register net device %s, skipping \n " ,
adapter - > port [ i ] - > name ) ;
else {
/*
* Change the name we use for messages to the name of
* the first successfully registered interface .
*/
if ( ! adapter - > registered_device_map )
adapter - > name = adapter - > port [ i ] - > name ;
__set_bit ( i , & adapter - > registered_device_map ) ;
}
}
if ( ! adapter - > registered_device_map ) {
dev_err ( & pdev - > dev , " could not register any net devices \n " ) ;
goto out_free_dev ;
}
/* Driver's ready. Reflect it on LEDs */
t3_led_ready ( adapter ) ;
if ( is_offload ( adapter ) ) {
__set_bit ( OFFLOAD_DEVMAP_BIT , & adapter - > registered_device_map ) ;
cxgb3_adapter_ofld ( adapter ) ;
}
/* See what interrupts we'll be using */
if ( msi > 1 & & cxgb_enable_msix ( adapter ) = = 0 )
adapter - > flags | = USING_MSIX ;
else if ( msi > 0 & & pci_enable_msi ( pdev ) = = 0 )
adapter - > flags | = USING_MSI ;
2007-02-09 03:55:59 +03:00
err = sysfs_create_group ( & adapter - > port [ 0 ] - > dev . kobj ,
2007-01-19 06:04:14 +03:00
& cxgb3_attr_group ) ;
print_port_info ( adapter , ai ) ;
return 0 ;
out_free_dev :
iounmap ( adapter - > regs ) ;
for ( i = ai - > nports - 1 ; i > = 0 ; - - i )
if ( adapter - > port [ i ] )
free_netdev ( adapter - > port [ i ] ) ;
out_free_adapter :
kfree ( adapter ) ;
out_disable_device :
pci_disable_device ( pdev ) ;
out_release_regions :
pci_release_regions ( pdev ) ;
pci_set_drvdata ( pdev , NULL ) ;
return err ;
}
static void __devexit remove_one ( struct pci_dev * pdev )
{
struct net_device * dev = pci_get_drvdata ( pdev ) ;
if ( dev ) {
int i ;
struct adapter * adapter = dev - > priv ;
t3_sge_stop ( adapter ) ;
2007-02-09 03:55:59 +03:00
sysfs_remove_group ( & adapter - > port [ 0 ] - > dev . kobj ,
2007-01-19 06:04:14 +03:00
& cxgb3_attr_group ) ;
for_each_port ( adapter , i )
if ( test_bit ( i , & adapter - > registered_device_map ) )
unregister_netdev ( adapter - > port [ i ] ) ;
if ( is_offload ( adapter ) ) {
cxgb3_adapter_unofld ( adapter ) ;
if ( test_bit ( OFFLOAD_DEVMAP_BIT ,
& adapter - > open_device_map ) )
offload_close ( & adapter - > tdev ) ;
}
t3_free_sge_resources ( adapter ) ;
cxgb_disable_msi ( adapter ) ;
for ( i = 0 ; i < ARRAY_SIZE ( adapter - > dummy_netdev ) ; i + + )
if ( adapter - > dummy_netdev [ i ] ) {
free_netdev ( adapter - > dummy_netdev [ i ] ) ;
adapter - > dummy_netdev [ i ] = NULL ;
}
for_each_port ( adapter , i )
if ( adapter - > port [ i ] )
free_netdev ( adapter - > port [ i ] ) ;
iounmap ( adapter - > regs ) ;
kfree ( adapter ) ;
pci_release_regions ( pdev ) ;
pci_disable_device ( pdev ) ;
pci_set_drvdata ( pdev , NULL ) ;
}
}
static struct pci_driver driver = {
. name = DRV_NAME ,
. id_table = cxgb3_pci_tbl ,
. probe = init_one ,
. remove = __devexit_p ( remove_one ) ,
} ;
static int __init cxgb3_init_module ( void )
{
int ret ;
cxgb3_offload_init ( ) ;
ret = pci_register_driver ( & driver ) ;
return ret ;
}
static void __exit cxgb3_cleanup_module ( void )
{
pci_unregister_driver ( & driver ) ;
if ( cxgb3_wq )
destroy_workqueue ( cxgb3_wq ) ;
}
module_init ( cxgb3_init_module ) ;
module_exit ( cxgb3_cleanup_module ) ;