2008-09-15 20:17:11 +04:00
/*
2010-06-24 14:52:26 +04:00
* Copyright 2008 - 2010 Cisco Systems , Inc . All rights reserved .
2008-09-15 20:17:11 +04:00
* Copyright 2007 Nuova Systems , Inc . All rights reserved .
*
* This program is free software ; you may redistribute it and / or modify
* it under the terms of the GNU General Public License as published by
* the Free Software Foundation ; version 2 of the License .
*
* 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 .
*
*/
# include <linux/module.h>
# include <linux/kernel.h>
# include <linux/string.h>
# include <linux/errno.h>
# include <linux/types.h>
# include <linux/init.h>
# include <linux/workqueue.h>
# include <linux/pci.h>
# include <linux/netdevice.h>
# include <linux/etherdevice.h>
# include <linux/if_ether.h>
# include <linux/if_vlan.h>
# include <linux/ethtool.h>
# include <linux/in.h>
# include <linux/ip.h>
# include <linux/ipv6.h>
# include <linux/tcp.h>
2010-06-24 14:52:26 +04:00
# include <linux/rtnetlink.h>
2008-10-14 05:41:01 +04:00
# include <net/ip6_checksum.h>
2008-09-15 20:17:11 +04:00
# include "cq_enet_desc.h"
# include "vnic_dev.h"
# include "vnic_intr.h"
# include "vnic_stats.h"
2010-05-18 09:50:19 +04:00
# include "vnic_vic.h"
2008-09-15 20:17:11 +04:00
# include "enic_res.h"
# include "enic.h"
# define ENIC_NOTIFY_TIMER_PERIOD (2 * HZ)
2009-09-03 21:02:03 +04:00
# define WQ_ENET_MAX_DESC_LEN (1 << WQ_ENET_LEN_BITS)
# define MAX_TSO (1 << 16)
# define ENIC_DESC_MAX_SPLITS (MAX_TSO / WQ_ENET_MAX_DESC_LEN + 1)
# define PCI_DEVICE_ID_CISCO_VIC_ENET 0x0043 /* ethernet vnic */
2010-05-18 09:50:19 +04:00
# define PCI_DEVICE_ID_CISCO_VIC_ENET_DYN 0x0044 /* enet dynamic vnic */
2008-09-15 20:17:11 +04:00
/* Supported devices */
2010-01-07 14:58:11 +03:00
static DEFINE_PCI_DEVICE_TABLE ( enic_id_table ) = {
2009-09-03 21:02:03 +04:00
{ PCI_VDEVICE ( CISCO , PCI_DEVICE_ID_CISCO_VIC_ENET ) } ,
2010-05-18 09:50:19 +04:00
{ PCI_VDEVICE ( CISCO , PCI_DEVICE_ID_CISCO_VIC_ENET_DYN ) } ,
2008-09-15 20:17:11 +04:00
{ 0 , } /* end of table */
} ;
MODULE_DESCRIPTION ( DRV_DESCRIPTION ) ;
MODULE_AUTHOR ( " Scott Feldman <scofeldm@cisco.com> " ) ;
MODULE_LICENSE ( " GPL " ) ;
MODULE_VERSION ( DRV_VERSION ) ;
MODULE_DEVICE_TABLE ( pci , enic_id_table ) ;
struct enic_stat {
char name [ ETH_GSTRING_LEN ] ;
unsigned int offset ;
} ;
# define ENIC_TX_STAT(stat) \
{ . name = # stat , . offset = offsetof ( struct vnic_tx_stats , stat ) / 8 }
# define ENIC_RX_STAT(stat) \
{ . name = # stat , . offset = offsetof ( struct vnic_rx_stats , stat ) / 8 }
static const struct enic_stat enic_tx_stats [ ] = {
ENIC_TX_STAT ( tx_frames_ok ) ,
ENIC_TX_STAT ( tx_unicast_frames_ok ) ,
ENIC_TX_STAT ( tx_multicast_frames_ok ) ,
ENIC_TX_STAT ( tx_broadcast_frames_ok ) ,
ENIC_TX_STAT ( tx_bytes_ok ) ,
ENIC_TX_STAT ( tx_unicast_bytes_ok ) ,
ENIC_TX_STAT ( tx_multicast_bytes_ok ) ,
ENIC_TX_STAT ( tx_broadcast_bytes_ok ) ,
ENIC_TX_STAT ( tx_drops ) ,
ENIC_TX_STAT ( tx_errors ) ,
ENIC_TX_STAT ( tx_tso ) ,
} ;
static const struct enic_stat enic_rx_stats [ ] = {
ENIC_RX_STAT ( rx_frames_ok ) ,
ENIC_RX_STAT ( rx_frames_total ) ,
ENIC_RX_STAT ( rx_unicast_frames_ok ) ,
ENIC_RX_STAT ( rx_multicast_frames_ok ) ,
ENIC_RX_STAT ( rx_broadcast_frames_ok ) ,
ENIC_RX_STAT ( rx_bytes_ok ) ,
ENIC_RX_STAT ( rx_unicast_bytes_ok ) ,
ENIC_RX_STAT ( rx_multicast_bytes_ok ) ,
ENIC_RX_STAT ( rx_broadcast_bytes_ok ) ,
ENIC_RX_STAT ( rx_drop ) ,
ENIC_RX_STAT ( rx_no_bufs ) ,
ENIC_RX_STAT ( rx_errors ) ,
ENIC_RX_STAT ( rx_rss ) ,
ENIC_RX_STAT ( rx_crc_errors ) ,
ENIC_RX_STAT ( rx_frames_64 ) ,
ENIC_RX_STAT ( rx_frames_127 ) ,
ENIC_RX_STAT ( rx_frames_255 ) ,
ENIC_RX_STAT ( rx_frames_511 ) ,
ENIC_RX_STAT ( rx_frames_1023 ) ,
ENIC_RX_STAT ( rx_frames_1518 ) ,
ENIC_RX_STAT ( rx_frames_to_max ) ,
} ;
static const unsigned int enic_n_tx_stats = ARRAY_SIZE ( enic_tx_stats ) ;
static const unsigned int enic_n_rx_stats = ARRAY_SIZE ( enic_rx_stats ) ;
2010-05-18 09:50:19 +04:00
static int enic_is_dynamic ( struct enic * enic )
{
return enic - > pdev - > device = = PCI_DEVICE_ID_CISCO_VIC_ENET_DYN ;
}
2010-10-20 14:16:59 +04:00
static inline unsigned int enic_cq_rq ( struct enic * enic , unsigned int rq )
{
return rq ;
}
static inline unsigned int enic_cq_wq ( struct enic * enic , unsigned int wq )
{
return enic - > rq_count + wq ;
}
static inline unsigned int enic_legacy_io_intr ( void )
{
return 0 ;
}
static inline unsigned int enic_legacy_err_intr ( void )
{
return 1 ;
}
static inline unsigned int enic_legacy_notify_intr ( void )
{
return 2 ;
}
static inline unsigned int enic_msix_rq_intr ( struct enic * enic , unsigned int rq )
{
return rq ;
}
static inline unsigned int enic_msix_wq_intr ( struct enic * enic , unsigned int wq )
{
return enic - > rq_count + wq ;
}
static inline unsigned int enic_msix_err_intr ( struct enic * enic )
{
return enic - > rq_count + enic - > wq_count ;
}
static inline unsigned int enic_msix_notify_intr ( struct enic * enic )
{
return enic - > rq_count + enic - > wq_count + 1 ;
}
2008-09-15 20:17:11 +04:00
static int enic_get_settings ( struct net_device * netdev ,
struct ethtool_cmd * ecmd )
{
struct enic * enic = netdev_priv ( netdev ) ;
ecmd - > supported = ( SUPPORTED_10000baseT_Full | SUPPORTED_FIBRE ) ;
ecmd - > advertising = ( ADVERTISED_10000baseT_Full | ADVERTISED_FIBRE ) ;
ecmd - > port = PORT_FIBRE ;
ecmd - > transceiver = XCVR_EXTERNAL ;
if ( netif_carrier_ok ( netdev ) ) {
ecmd - > speed = vnic_dev_port_speed ( enic - > vdev ) ;
ecmd - > duplex = DUPLEX_FULL ;
} else {
ecmd - > speed = - 1 ;
ecmd - > duplex = - 1 ;
}
ecmd - > autoneg = AUTONEG_DISABLE ;
return 0 ;
}
2010-06-24 14:50:12 +04:00
static int enic_dev_fw_info ( struct enic * enic ,
struct vnic_devcmd_fw_info * * fw_info )
{
int err ;
spin_lock ( & enic - > devcmd_lock ) ;
err = vnic_dev_fw_info ( enic - > vdev , fw_info ) ;
spin_unlock ( & enic - > devcmd_lock ) ;
return err ;
}
2008-09-15 20:17:11 +04:00
static void enic_get_drvinfo ( struct net_device * netdev ,
struct ethtool_drvinfo * drvinfo )
{
struct enic * enic = netdev_priv ( netdev ) ;
struct vnic_devcmd_fw_info * fw_info ;
2010-06-24 14:50:12 +04:00
enic_dev_fw_info ( enic , & fw_info ) ;
2008-09-15 20:17:11 +04:00
strncpy ( drvinfo - > driver , DRV_NAME , sizeof ( drvinfo - > driver ) ) ;
strncpy ( drvinfo - > version , DRV_VERSION , sizeof ( drvinfo - > version ) ) ;
strncpy ( drvinfo - > fw_version , fw_info - > fw_version ,
sizeof ( drvinfo - > fw_version ) ) ;
strncpy ( drvinfo - > bus_info , pci_name ( enic - > pdev ) ,
sizeof ( drvinfo - > bus_info ) ) ;
}
static void enic_get_strings ( struct net_device * netdev , u32 stringset , u8 * data )
{
unsigned int i ;
switch ( stringset ) {
case ETH_SS_STATS :
for ( i = 0 ; i < enic_n_tx_stats ; i + + ) {
memcpy ( data , enic_tx_stats [ i ] . name , ETH_GSTRING_LEN ) ;
data + = ETH_GSTRING_LEN ;
}
for ( i = 0 ; i < enic_n_rx_stats ; i + + ) {
memcpy ( data , enic_rx_stats [ i ] . name , ETH_GSTRING_LEN ) ;
data + = ETH_GSTRING_LEN ;
}
break ;
}
}
2008-09-24 22:23:32 +04:00
static int enic_get_sset_count ( struct net_device * netdev , int sset )
2008-09-15 20:17:11 +04:00
{
2008-09-24 22:23:32 +04:00
switch ( sset ) {
case ETH_SS_STATS :
return enic_n_tx_stats + enic_n_rx_stats ;
default :
return - EOPNOTSUPP ;
}
2008-09-15 20:17:11 +04:00
}
2010-06-24 14:50:12 +04:00
static int enic_dev_stats_dump ( struct enic * enic , struct vnic_stats * * vstats )
{
int err ;
spin_lock ( & enic - > devcmd_lock ) ;
err = vnic_dev_stats_dump ( enic - > vdev , vstats ) ;
spin_unlock ( & enic - > devcmd_lock ) ;
return err ;
}
2008-09-15 20:17:11 +04:00
static void enic_get_ethtool_stats ( struct net_device * netdev ,
struct ethtool_stats * stats , u64 * data )
{
struct enic * enic = netdev_priv ( netdev ) ;
struct vnic_stats * vstats ;
unsigned int i ;
2010-06-24 14:50:12 +04:00
enic_dev_stats_dump ( enic , & vstats ) ;
2008-09-15 20:17:11 +04:00
for ( i = 0 ; i < enic_n_tx_stats ; i + + )
* ( data + + ) = ( ( u64 * ) & vstats - > tx ) [ enic_tx_stats [ i ] . offset ] ;
for ( i = 0 ; i < enic_n_rx_stats ; i + + )
* ( data + + ) = ( ( u64 * ) & vstats - > rx ) [ enic_rx_stats [ i ] . offset ] ;
}
static u32 enic_get_rx_csum ( struct net_device * netdev )
{
struct enic * enic = netdev_priv ( netdev ) ;
return enic - > csum_rx_enabled ;
}
static int enic_set_rx_csum ( struct net_device * netdev , u32 data )
{
struct enic * enic = netdev_priv ( netdev ) ;
2008-09-24 22:23:32 +04:00
if ( data & & ! ENIC_SETTING ( enic , RXCSUM ) )
return - EINVAL ;
enic - > csum_rx_enabled = ! ! data ;
2008-09-15 20:17:11 +04:00
return 0 ;
}
static int enic_set_tx_csum ( struct net_device * netdev , u32 data )
{
struct enic * enic = netdev_priv ( netdev ) ;
2008-09-24 22:23:32 +04:00
if ( data & & ! ENIC_SETTING ( enic , TXCSUM ) )
return - EINVAL ;
if ( data )
2008-09-15 20:17:11 +04:00
netdev - > features | = NETIF_F_HW_CSUM ;
else
netdev - > features & = ~ NETIF_F_HW_CSUM ;
return 0 ;
}
static int enic_set_tso ( struct net_device * netdev , u32 data )
{
struct enic * enic = netdev_priv ( netdev ) ;
2008-09-24 22:23:32 +04:00
if ( data & & ! ENIC_SETTING ( enic , TSO ) )
return - EINVAL ;
if ( data )
2008-09-15 20:17:11 +04:00
netdev - > features | =
NETIF_F_TSO | NETIF_F_TSO6 | NETIF_F_TSO_ECN ;
else
netdev - > features & =
~ ( NETIF_F_TSO | NETIF_F_TSO6 | NETIF_F_TSO_ECN ) ;
return 0 ;
}
static u32 enic_get_msglevel ( struct net_device * netdev )
{
struct enic * enic = netdev_priv ( netdev ) ;
return enic - > msg_enable ;
}
static void enic_set_msglevel ( struct net_device * netdev , u32 value )
{
struct enic * enic = netdev_priv ( netdev ) ;
enic - > msg_enable = value ;
}
2009-12-23 16:27:54 +03:00
static int enic_get_coalesce ( struct net_device * netdev ,
struct ethtool_coalesce * ecmd )
{
struct enic * enic = netdev_priv ( netdev ) ;
ecmd - > tx_coalesce_usecs = enic - > tx_coalesce_usecs ;
ecmd - > rx_coalesce_usecs = enic - > rx_coalesce_usecs ;
return 0 ;
}
static int enic_set_coalesce ( struct net_device * netdev ,
struct ethtool_coalesce * ecmd )
{
struct enic * enic = netdev_priv ( netdev ) ;
u32 tx_coalesce_usecs ;
u32 rx_coalesce_usecs ;
2010-10-20 14:16:59 +04:00
unsigned int i , intr ;
2009-12-23 16:27:54 +03:00
tx_coalesce_usecs = min_t ( u32 ,
INTR_COALESCE_HW_TO_USEC ( VNIC_INTR_TIMER_MAX ) ,
ecmd - > tx_coalesce_usecs ) ;
rx_coalesce_usecs = min_t ( u32 ,
INTR_COALESCE_HW_TO_USEC ( VNIC_INTR_TIMER_MAX ) ,
ecmd - > rx_coalesce_usecs ) ;
switch ( vnic_dev_get_intr_mode ( enic - > vdev ) ) {
case VNIC_DEV_INTR_MODE_INTX :
if ( tx_coalesce_usecs ! = rx_coalesce_usecs )
return - EINVAL ;
2010-10-20 14:16:59 +04:00
intr = enic_legacy_io_intr ( ) ;
vnic_intr_coalescing_timer_set ( & enic - > intr [ intr ] ,
2009-12-23 16:27:54 +03:00
INTR_COALESCE_USEC_TO_HW ( tx_coalesce_usecs ) ) ;
break ;
case VNIC_DEV_INTR_MODE_MSI :
if ( tx_coalesce_usecs ! = rx_coalesce_usecs )
return - EINVAL ;
vnic_intr_coalescing_timer_set ( & enic - > intr [ 0 ] ,
INTR_COALESCE_USEC_TO_HW ( tx_coalesce_usecs ) ) ;
break ;
case VNIC_DEV_INTR_MODE_MSIX :
2010-10-20 14:16:59 +04:00
for ( i = 0 ; i < enic - > wq_count ; i + + ) {
intr = enic_msix_wq_intr ( enic , i ) ;
vnic_intr_coalescing_timer_set ( & enic - > intr [ intr ] ,
INTR_COALESCE_USEC_TO_HW ( tx_coalesce_usecs ) ) ;
}
for ( i = 0 ; i < enic - > rq_count ; i + + ) {
intr = enic_msix_rq_intr ( enic , i ) ;
vnic_intr_coalescing_timer_set ( & enic - > intr [ intr ] ,
INTR_COALESCE_USEC_TO_HW ( rx_coalesce_usecs ) ) ;
}
2009-12-23 16:27:54 +03:00
break ;
default :
break ;
}
enic - > tx_coalesce_usecs = tx_coalesce_usecs ;
enic - > rx_coalesce_usecs = rx_coalesce_usecs ;
return 0 ;
}
2009-09-02 12:03:33 +04:00
static const struct ethtool_ops enic_ethtool_ops = {
2008-09-15 20:17:11 +04:00
. get_settings = enic_get_settings ,
. get_drvinfo = enic_get_drvinfo ,
. get_msglevel = enic_get_msglevel ,
. set_msglevel = enic_set_msglevel ,
. get_link = ethtool_op_get_link ,
. get_strings = enic_get_strings ,
2008-09-24 22:23:32 +04:00
. get_sset_count = enic_get_sset_count ,
2008-09-15 20:17:11 +04:00
. get_ethtool_stats = enic_get_ethtool_stats ,
. get_rx_csum = enic_get_rx_csum ,
. set_rx_csum = enic_set_rx_csum ,
. get_tx_csum = ethtool_op_get_tx_csum ,
. set_tx_csum = enic_set_tx_csum ,
. get_sg = ethtool_op_get_sg ,
. set_sg = ethtool_op_set_sg ,
. get_tso = ethtool_op_get_tso ,
. set_tso = enic_set_tso ,
2009-12-23 16:27:54 +03:00
. get_coalesce = enic_get_coalesce ,
. set_coalesce = enic_set_coalesce ,
2008-11-22 08:26:55 +03:00
. get_flags = ethtool_op_get_flags ,
2008-09-15 20:17:11 +04:00
} ;
static void enic_free_wq_buf ( struct vnic_wq * wq , struct vnic_wq_buf * buf )
{
struct enic * enic = vnic_dev_priv ( wq - > vdev ) ;
if ( buf - > sop )
pci_unmap_single ( enic - > pdev , buf - > dma_addr ,
buf - > len , PCI_DMA_TODEVICE ) ;
else
pci_unmap_page ( enic - > pdev , buf - > dma_addr ,
buf - > len , PCI_DMA_TODEVICE ) ;
if ( buf - > os_buf )
dev_kfree_skb_any ( buf - > os_buf ) ;
}
static void enic_wq_free_buf ( struct vnic_wq * wq ,
struct cq_desc * cq_desc , struct vnic_wq_buf * buf , void * opaque )
{
enic_free_wq_buf ( wq , buf ) ;
}
static int enic_wq_service ( struct vnic_dev * vdev , struct cq_desc * cq_desc ,
u8 type , u16 q_number , u16 completed_index , void * opaque )
{
struct enic * enic = vnic_dev_priv ( vdev ) ;
spin_lock ( & enic - > wq_lock [ q_number ] ) ;
vnic_wq_service ( & enic - > wq [ q_number ] , cq_desc ,
completed_index , enic_wq_free_buf ,
opaque ) ;
if ( netif_queue_stopped ( enic - > netdev ) & &
2009-09-03 21:02:03 +04:00
vnic_wq_desc_avail ( & enic - > wq [ q_number ] ) > =
( MAX_SKB_FRAGS + ENIC_DESC_MAX_SPLITS ) )
2008-09-15 20:17:11 +04:00
netif_wake_queue ( enic - > netdev ) ;
spin_unlock ( & enic - > wq_lock [ q_number ] ) ;
return 0 ;
}
static void enic_log_q_error ( struct enic * enic )
{
unsigned int i ;
u32 error_status ;
for ( i = 0 ; i < enic - > wq_count ; i + + ) {
error_status = vnic_wq_error_status ( & enic - > wq [ i ] ) ;
if ( error_status )
2010-06-24 14:50:56 +04:00
netdev_err ( enic - > netdev , " WQ[%d] error_status %d \n " ,
i , error_status ) ;
2008-09-15 20:17:11 +04:00
}
for ( i = 0 ; i < enic - > rq_count ; i + + ) {
error_status = vnic_rq_error_status ( & enic - > rq [ i ] ) ;
if ( error_status )
2010-06-24 14:50:56 +04:00
netdev_err ( enic - > netdev , " RQ[%d] error_status %d \n " ,
i , error_status ) ;
2008-09-15 20:17:11 +04:00
}
}
2010-06-24 14:50:12 +04:00
static void enic_msglvl_check ( struct enic * enic )
2008-09-15 20:17:11 +04:00
{
2010-06-24 14:50:12 +04:00
u32 msg_enable = vnic_dev_msg_lvl ( enic - > vdev ) ;
2008-09-15 20:17:11 +04:00
2010-06-24 14:50:12 +04:00
if ( msg_enable ! = enic - > msg_enable ) {
2010-06-24 14:50:56 +04:00
netdev_info ( enic - > netdev , " msg lvl changed from 0x%x to 0x%x \n " ,
enic - > msg_enable , msg_enable ) ;
2010-06-24 14:50:12 +04:00
enic - > msg_enable = msg_enable ;
2008-09-15 20:17:11 +04:00
}
}
static void enic_mtu_check ( struct enic * enic )
{
u32 mtu = vnic_dev_mtu ( enic - > vdev ) ;
2010-06-24 14:50:56 +04:00
struct net_device * netdev = enic - > netdev ;
2008-09-15 20:17:11 +04:00
2009-09-03 21:02:40 +04:00
if ( mtu & & mtu ! = enic - > port_mtu ) {
2009-12-23 16:27:54 +03:00
enic - > port_mtu = mtu ;
2010-06-24 14:50:56 +04:00
if ( mtu < netdev - > mtu )
netdev_warn ( netdev ,
" interface MTU (%d) set higher "
2008-09-15 20:17:11 +04:00
" than switch port MTU (%d) \n " ,
2010-06-24 14:50:56 +04:00
netdev - > mtu , mtu ) ;
2008-09-15 20:17:11 +04:00
}
}
2010-06-24 14:50:12 +04:00
static void enic_link_check ( struct enic * enic )
2008-09-15 20:17:11 +04:00
{
2010-06-24 14:50:12 +04:00
int link_status = vnic_dev_link_status ( enic - > vdev ) ;
int carrier_ok = netif_carrier_ok ( enic - > netdev ) ;
2008-09-15 20:17:11 +04:00
2010-06-24 14:50:12 +04:00
if ( link_status & & ! carrier_ok ) {
2010-06-24 14:50:56 +04:00
netdev_info ( enic - > netdev , " Link UP \n " ) ;
2010-06-24 14:50:12 +04:00
netif_carrier_on ( enic - > netdev ) ;
} else if ( ! link_status & & carrier_ok ) {
2010-06-24 14:50:56 +04:00
netdev_info ( enic - > netdev , " Link DOWN \n " ) ;
2010-06-24 14:50:12 +04:00
netif_carrier_off ( enic - > netdev ) ;
2008-09-15 20:17:11 +04:00
}
}
static void enic_notify_check ( struct enic * enic )
{
enic_msglvl_check ( enic ) ;
enic_mtu_check ( enic ) ;
enic_link_check ( enic ) ;
}
# define ENIC_TEST_INTR(pba, i) (pba & (1 << i))
static irqreturn_t enic_isr_legacy ( int irq , void * data )
{
struct net_device * netdev = data ;
struct enic * enic = netdev_priv ( netdev ) ;
2010-10-20 14:16:59 +04:00
unsigned int io_intr = enic_legacy_io_intr ( ) ;
unsigned int err_intr = enic_legacy_err_intr ( ) ;
unsigned int notify_intr = enic_legacy_notify_intr ( ) ;
2008-09-15 20:17:11 +04:00
u32 pba ;
2010-10-20 14:16:59 +04:00
vnic_intr_mask ( & enic - > intr [ io_intr ] ) ;
2008-09-15 20:17:11 +04:00
pba = vnic_intr_legacy_pba ( enic - > legacy_pba ) ;
if ( ! pba ) {
2010-10-20 14:16:59 +04:00
vnic_intr_unmask ( & enic - > intr [ io_intr ] ) ;
2008-09-15 20:17:11 +04:00
return IRQ_NONE ; /* not our interrupt */
}
2010-10-20 14:16:59 +04:00
if ( ENIC_TEST_INTR ( pba , notify_intr ) ) {
vnic_intr_return_all_credits ( & enic - > intr [ notify_intr ] ) ;
2008-09-15 20:17:11 +04:00
enic_notify_check ( enic ) ;
2009-02-10 10:23:50 +03:00
}
2008-09-15 20:17:11 +04:00
2010-10-20 14:16:59 +04:00
if ( ENIC_TEST_INTR ( pba , err_intr ) ) {
vnic_intr_return_all_credits ( & enic - > intr [ err_intr ] ) ;
2008-09-15 20:17:11 +04:00
enic_log_q_error ( enic ) ;
/* schedule recovery from WQ/RQ error */
schedule_work ( & enic - > reset ) ;
return IRQ_HANDLED ;
}
2010-10-20 14:16:59 +04:00
if ( ENIC_TEST_INTR ( pba , io_intr ) ) {
if ( napi_schedule_prep ( & enic - > napi [ 0 ] ) )
__napi_schedule ( & enic - > napi [ 0 ] ) ;
2008-09-15 20:17:11 +04:00
} else {
2010-10-20 14:16:59 +04:00
vnic_intr_unmask ( & enic - > intr [ io_intr ] ) ;
2008-09-15 20:17:11 +04:00
}
return IRQ_HANDLED ;
}
static irqreturn_t enic_isr_msi ( int irq , void * data )
{
struct enic * enic = data ;
/* With MSI, there is no sharing of interrupts, so this is
* our interrupt and there is no need to ack it . The device
* is not providing per - vector masking , so the OS will not
* write to PCI config space to mask / unmask the interrupt .
* We ' re using mask_on_assertion for MSI , so the device
* automatically masks the interrupt when the interrupt is
* generated . Later , when exiting polling , the interrupt
* will be unmasked ( see enic_poll ) .
*
* Also , the device uses the same PCIe Traffic Class ( TC )
* for Memory Write data and MSI , so there are no ordering
* issues ; the MSI will always arrive at the Root Complex
* _after_ corresponding Memory Writes ( i . e . descriptor
* writes ) .
*/
2010-10-20 14:16:59 +04:00
napi_schedule ( & enic - > napi [ 0 ] ) ;
2008-09-15 20:17:11 +04:00
return IRQ_HANDLED ;
}
static irqreturn_t enic_isr_msix_rq ( int irq , void * data )
{
2010-10-20 14:16:59 +04:00
struct napi_struct * napi = data ;
2008-09-15 20:17:11 +04:00
/* schedule NAPI polling for RQ cleanup */
2010-10-20 14:16:59 +04:00
napi_schedule ( napi ) ;
2008-09-15 20:17:11 +04:00
return IRQ_HANDLED ;
}
static irqreturn_t enic_isr_msix_wq ( int irq , void * data )
{
struct enic * enic = data ;
2010-10-20 14:16:59 +04:00
unsigned int cq = enic_cq_wq ( enic , 0 ) ;
unsigned int intr = enic_msix_wq_intr ( enic , 0 ) ;
2008-09-15 20:17:11 +04:00
unsigned int wq_work_to_do = - 1 ; /* no limit */
unsigned int wq_work_done ;
2010-10-20 14:16:59 +04:00
wq_work_done = vnic_cq_service ( & enic - > cq [ cq ] ,
2008-09-15 20:17:11 +04:00
wq_work_to_do , enic_wq_service , NULL ) ;
2010-10-20 14:16:59 +04:00
vnic_intr_return_credits ( & enic - > intr [ intr ] ,
2008-09-15 20:17:11 +04:00
wq_work_done ,
1 /* unmask intr */ ,
1 /* reset intr timer */ ) ;
return IRQ_HANDLED ;
}
static irqreturn_t enic_isr_msix_err ( int irq , void * data )
{
struct enic * enic = data ;
2010-10-20 14:16:59 +04:00
unsigned int intr = enic_msix_err_intr ( enic ) ;
2008-09-15 20:17:11 +04:00
2010-10-20 14:16:59 +04:00
vnic_intr_return_all_credits ( & enic - > intr [ intr ] ) ;
2009-02-10 10:23:50 +03:00
2008-09-15 20:17:11 +04:00
enic_log_q_error ( enic ) ;
/* schedule recovery from WQ/RQ error */
schedule_work ( & enic - > reset ) ;
return IRQ_HANDLED ;
}
static irqreturn_t enic_isr_msix_notify ( int irq , void * data )
{
struct enic * enic = data ;
2010-10-20 14:16:59 +04:00
unsigned int intr = enic_msix_notify_intr ( enic ) ;
2008-09-15 20:17:11 +04:00
2010-10-20 14:16:59 +04:00
vnic_intr_return_all_credits ( & enic - > intr [ intr ] ) ;
2008-09-15 20:17:11 +04:00
enic_notify_check ( enic ) ;
return IRQ_HANDLED ;
}
static inline void enic_queue_wq_skb_cont ( struct enic * enic ,
struct vnic_wq * wq , struct sk_buff * skb ,
2010-06-24 14:51:59 +04:00
unsigned int len_left , int loopback )
2008-09-15 20:17:11 +04:00
{
skb_frag_t * frag ;
/* Queue additional data fragments */
for ( frag = skb_shinfo ( skb ) - > frags ; len_left ; frag + + ) {
len_left - = frag - > size ;
enic_queue_wq_desc_cont ( wq , skb ,
pci_map_page ( enic - > pdev , frag - > page ,
frag - > page_offset , frag - > size ,
PCI_DMA_TODEVICE ) ,
frag - > size ,
2010-06-24 14:51:59 +04:00
( len_left = = 0 ) , /* EOP? */
loopback ) ;
2008-09-15 20:17:11 +04:00
}
}
static inline void enic_queue_wq_skb_vlan ( struct enic * enic ,
struct vnic_wq * wq , struct sk_buff * skb ,
2010-06-24 14:51:59 +04:00
int vlan_tag_insert , unsigned int vlan_tag , int loopback )
2008-09-15 20:17:11 +04:00
{
unsigned int head_len = skb_headlen ( skb ) ;
unsigned int len_left = skb - > len - head_len ;
int eop = ( len_left = = 0 ) ;
2009-09-03 21:02:03 +04:00
/* Queue the main skb fragment. The fragments are no larger
* than max MTU ( 9000 ) + ETH_HDR_LEN ( 14 ) bytes , which is less
* than WQ_ENET_MAX_DESC_LEN length . So only one descriptor
* per fragment is queued .
*/
2008-09-15 20:17:11 +04:00
enic_queue_wq_desc ( wq , skb ,
pci_map_single ( enic - > pdev , skb - > data ,
head_len , PCI_DMA_TODEVICE ) ,
head_len ,
vlan_tag_insert , vlan_tag ,
2010-06-24 14:51:59 +04:00
eop , loopback ) ;
2008-09-15 20:17:11 +04:00
if ( ! eop )
2010-06-24 14:51:59 +04:00
enic_queue_wq_skb_cont ( enic , wq , skb , len_left , loopback ) ;
2008-09-15 20:17:11 +04:00
}
static inline void enic_queue_wq_skb_csum_l4 ( struct enic * enic ,
struct vnic_wq * wq , struct sk_buff * skb ,
2010-06-24 14:51:59 +04:00
int vlan_tag_insert , unsigned int vlan_tag , int loopback )
2008-09-15 20:17:11 +04:00
{
unsigned int head_len = skb_headlen ( skb ) ;
unsigned int len_left = skb - > len - head_len ;
unsigned int hdr_len = skb_transport_offset ( skb ) ;
unsigned int csum_offset = hdr_len + skb - > csum_offset ;
int eop = ( len_left = = 0 ) ;
2009-09-03 21:02:03 +04:00
/* Queue the main skb fragment. The fragments are no larger
* than max MTU ( 9000 ) + ETH_HDR_LEN ( 14 ) bytes , which is less
* than WQ_ENET_MAX_DESC_LEN length . So only one descriptor
* per fragment is queued .
*/
2008-09-15 20:17:11 +04:00
enic_queue_wq_desc_csum_l4 ( wq , skb ,
pci_map_single ( enic - > pdev , skb - > data ,
head_len , PCI_DMA_TODEVICE ) ,
head_len ,
csum_offset ,
hdr_len ,
vlan_tag_insert , vlan_tag ,
2010-06-24 14:51:59 +04:00
eop , loopback ) ;
2008-09-15 20:17:11 +04:00
if ( ! eop )
2010-06-24 14:51:59 +04:00
enic_queue_wq_skb_cont ( enic , wq , skb , len_left , loopback ) ;
2008-09-15 20:17:11 +04:00
}
static inline void enic_queue_wq_skb_tso ( struct enic * enic ,
struct vnic_wq * wq , struct sk_buff * skb , unsigned int mss ,
2010-06-24 14:51:59 +04:00
int vlan_tag_insert , unsigned int vlan_tag , int loopback )
2008-09-15 20:17:11 +04:00
{
2009-09-03 21:02:03 +04:00
unsigned int frag_len_left = skb_headlen ( skb ) ;
unsigned int len_left = skb - > len - frag_len_left ;
2008-09-15 20:17:11 +04:00
unsigned int hdr_len = skb_transport_offset ( skb ) + tcp_hdrlen ( skb ) ;
int eop = ( len_left = = 0 ) ;
2009-09-03 21:02:03 +04:00
unsigned int len ;
dma_addr_t dma_addr ;
unsigned int offset = 0 ;
skb_frag_t * frag ;
2008-09-15 20:17:11 +04:00
/* Preload TCP csum field with IP pseudo hdr calculated
* with IP length set to zero . HW will later add in length
* to each TCP segment resulting from the TSO .
*/
2009-02-01 11:45:17 +03:00
if ( skb - > protocol = = cpu_to_be16 ( ETH_P_IP ) ) {
2008-09-15 20:17:11 +04:00
ip_hdr ( skb ) - > check = 0 ;
tcp_hdr ( skb ) - > check = ~ csum_tcpudp_magic ( ip_hdr ( skb ) - > saddr ,
ip_hdr ( skb ) - > daddr , 0 , IPPROTO_TCP , 0 ) ;
2009-02-01 11:45:17 +03:00
} else if ( skb - > protocol = = cpu_to_be16 ( ETH_P_IPV6 ) ) {
2008-09-15 20:17:11 +04:00
tcp_hdr ( skb ) - > check = ~ csum_ipv6_magic ( & ipv6_hdr ( skb ) - > saddr ,
& ipv6_hdr ( skb ) - > daddr , 0 , IPPROTO_TCP , 0 ) ;
}
2009-09-03 21:02:03 +04:00
/* Queue WQ_ENET_MAX_DESC_LEN length descriptors
* for the main skb fragment
*/
while ( frag_len_left ) {
len = min ( frag_len_left , ( unsigned int ) WQ_ENET_MAX_DESC_LEN ) ;
dma_addr = pci_map_single ( enic - > pdev , skb - > data + offset ,
len , PCI_DMA_TODEVICE ) ;
enic_queue_wq_desc_tso ( wq , skb ,
dma_addr ,
len ,
mss , hdr_len ,
vlan_tag_insert , vlan_tag ,
2010-06-24 14:51:59 +04:00
eop & & ( len = = frag_len_left ) , loopback ) ;
2009-09-03 21:02:03 +04:00
frag_len_left - = len ;
offset + = len ;
}
2008-09-15 20:17:11 +04:00
2009-09-03 21:02:03 +04:00
if ( eop )
return ;
/* Queue WQ_ENET_MAX_DESC_LEN length descriptors
* for additional data fragments
*/
for ( frag = skb_shinfo ( skb ) - > frags ; len_left ; frag + + ) {
len_left - = frag - > size ;
frag_len_left = frag - > size ;
offset = frag - > page_offset ;
while ( frag_len_left ) {
len = min ( frag_len_left ,
( unsigned int ) WQ_ENET_MAX_DESC_LEN ) ;
dma_addr = pci_map_page ( enic - > pdev , frag - > page ,
offset , len ,
PCI_DMA_TODEVICE ) ;
enic_queue_wq_desc_cont ( wq , skb ,
dma_addr ,
len ,
( len_left = = 0 ) & &
2010-06-24 14:51:59 +04:00
( len = = frag_len_left ) , /* EOP? */
loopback ) ;
2009-09-03 21:02:03 +04:00
frag_len_left - = len ;
offset + = len ;
}
}
2008-09-15 20:17:11 +04:00
}
static inline void enic_queue_wq_skb ( struct enic * enic ,
struct vnic_wq * wq , struct sk_buff * skb )
{
unsigned int mss = skb_shinfo ( skb ) - > gso_size ;
unsigned int vlan_tag = 0 ;
int vlan_tag_insert = 0 ;
2010-06-24 14:51:59 +04:00
int loopback = 0 ;
2008-09-15 20:17:11 +04:00
2010-10-20 17:56:03 +04:00
if ( vlan_tx_tag_present ( skb ) ) {
2008-09-15 20:17:11 +04:00
/* VLAN tag from trunking driver */
vlan_tag_insert = 1 ;
vlan_tag = vlan_tx_tag_get ( skb ) ;
2010-06-24 14:51:59 +04:00
} else if ( enic - > loop_enable ) {
vlan_tag = enic - > loop_tag ;
loopback = 1 ;
2008-09-15 20:17:11 +04:00
}
if ( mss )
enic_queue_wq_skb_tso ( enic , wq , skb , mss ,
2010-06-24 14:51:59 +04:00
vlan_tag_insert , vlan_tag , loopback ) ;
2008-09-15 20:17:11 +04:00
else if ( skb - > ip_summed = = CHECKSUM_PARTIAL )
enic_queue_wq_skb_csum_l4 ( enic , wq , skb ,
2010-06-24 14:51:59 +04:00
vlan_tag_insert , vlan_tag , loopback ) ;
2008-09-15 20:17:11 +04:00
else
enic_queue_wq_skb_vlan ( enic , wq , skb ,
2010-06-24 14:51:59 +04:00
vlan_tag_insert , vlan_tag , loopback ) ;
2008-09-15 20:17:11 +04:00
}
2009-02-10 10:23:50 +03:00
/* netif_tx_lock held, process context with BHs disabled, or BH */
2009-08-31 23:50:58 +04:00
static netdev_tx_t enic_hard_start_xmit ( struct sk_buff * skb ,
2009-12-23 16:27:59 +03:00
struct net_device * netdev )
2008-09-15 20:17:11 +04:00
{
struct enic * enic = netdev_priv ( netdev ) ;
struct vnic_wq * wq = & enic - > wq [ 0 ] ;
unsigned long flags ;
if ( skb - > len < = 0 ) {
dev_kfree_skb ( skb ) ;
return NETDEV_TX_OK ;
}
/* Non-TSO sends must fit within ENIC_NON_TSO_MAX_DESC descs,
* which is very likely . In the off chance it ' s going to take
* more than * ENIC_NON_TSO_MAX_DESC , linearize the skb .
*/
if ( skb_shinfo ( skb ) - > gso_size = = 0 & &
skb_shinfo ( skb ) - > nr_frags + 1 > ENIC_NON_TSO_MAX_DESC & &
skb_linearize ( skb ) ) {
dev_kfree_skb ( skb ) ;
return NETDEV_TX_OK ;
}
spin_lock_irqsave ( & enic - > wq_lock [ 0 ] , flags ) ;
2009-09-03 21:02:03 +04:00
if ( vnic_wq_desc_avail ( wq ) <
skb_shinfo ( skb ) - > nr_frags + ENIC_DESC_MAX_SPLITS ) {
2008-09-15 20:17:11 +04:00
netif_stop_queue ( netdev ) ;
/* This is a hard error, log it */
2010-06-24 14:50:56 +04:00
netdev_err ( netdev , " BUG! Tx ring full when queue awake! \n " ) ;
2008-09-15 20:17:11 +04:00
spin_unlock_irqrestore ( & enic - > wq_lock [ 0 ] , flags ) ;
return NETDEV_TX_BUSY ;
}
enic_queue_wq_skb ( enic , wq , skb ) ;
2009-09-03 21:02:03 +04:00
if ( vnic_wq_desc_avail ( wq ) < MAX_SKB_FRAGS + ENIC_DESC_MAX_SPLITS )
2008-09-15 20:17:11 +04:00
netif_stop_queue ( netdev ) ;
spin_unlock_irqrestore ( & enic - > wq_lock [ 0 ] , flags ) ;
return NETDEV_TX_OK ;
}
/* dev_base_lock rwlock held, nominally process context */
static struct net_device_stats * enic_get_stats ( struct net_device * netdev )
{
struct enic * enic = netdev_priv ( netdev ) ;
2008-09-24 22:23:32 +04:00
struct net_device_stats * net_stats = & netdev - > stats ;
2008-09-15 20:17:11 +04:00
struct vnic_stats * stats ;
2010-06-24 14:50:12 +04:00
enic_dev_stats_dump ( enic , & stats ) ;
2008-09-15 20:17:11 +04:00
2008-09-24 22:23:32 +04:00
net_stats - > tx_packets = stats - > tx . tx_frames_ok ;
net_stats - > tx_bytes = stats - > tx . tx_bytes_ok ;
net_stats - > tx_errors = stats - > tx . tx_errors ;
net_stats - > tx_dropped = stats - > tx . tx_drops ;
2008-09-15 20:17:11 +04:00
2008-09-24 22:23:32 +04:00
net_stats - > rx_packets = stats - > rx . rx_frames_ok ;
net_stats - > rx_bytes = stats - > rx . rx_bytes_ok ;
net_stats - > rx_errors = stats - > rx . rx_errors ;
net_stats - > multicast = stats - > rx . rx_multicast_frames_ok ;
2009-09-03 21:02:19 +04:00
net_stats - > rx_over_errors = enic - > rq_truncated_pkts ;
2009-02-10 10:24:08 +03:00
net_stats - > rx_crc_errors = enic - > rq_bad_fcs ;
2009-09-03 21:02:19 +04:00
net_stats - > rx_dropped = stats - > rx . rx_no_bufs + stats - > rx . rx_drop ;
2008-09-15 20:17:11 +04:00
2008-09-24 22:23:32 +04:00
return net_stats ;
2008-09-15 20:17:11 +04:00
}
2010-06-24 14:50:00 +04:00
static void enic_reset_multicast_list ( struct enic * enic )
2008-09-15 20:17:11 +04:00
{
enic - > mc_count = 0 ;
2010-06-24 14:50:00 +04:00
enic - > flags = 0 ;
2008-09-15 20:17:11 +04:00
}
static int enic_set_mac_addr ( struct net_device * netdev , char * addr )
{
2010-05-18 09:50:19 +04:00
struct enic * enic = netdev_priv ( netdev ) ;
if ( enic_is_dynamic ( enic ) ) {
if ( ! is_valid_ether_addr ( addr ) & & ! is_zero_ether_addr ( addr ) )
return - EADDRNOTAVAIL ;
} else {
if ( ! is_valid_ether_addr ( addr ) )
return - EADDRNOTAVAIL ;
}
2008-09-15 20:17:11 +04:00
memcpy ( netdev - > dev_addr , addr , netdev - > addr_len ) ;
return 0 ;
}
2010-05-18 09:50:19 +04:00
static int enic_dev_add_station_addr ( struct enic * enic )
{
int err = 0 ;
if ( is_valid_ether_addr ( enic - > netdev - > dev_addr ) ) {
spin_lock ( & enic - > devcmd_lock ) ;
err = vnic_dev_add_addr ( enic - > vdev , enic - > netdev - > dev_addr ) ;
spin_unlock ( & enic - > devcmd_lock ) ;
}
return err ;
}
static int enic_dev_del_station_addr ( struct enic * enic )
{
int err = 0 ;
if ( is_valid_ether_addr ( enic - > netdev - > dev_addr ) ) {
spin_lock ( & enic - > devcmd_lock ) ;
err = vnic_dev_del_addr ( enic - > vdev , enic - > netdev - > dev_addr ) ;
spin_unlock ( & enic - > devcmd_lock ) ;
}
return err ;
}
static int enic_set_mac_address_dynamic ( struct net_device * netdev , void * p )
{
struct enic * enic = netdev_priv ( netdev ) ;
struct sockaddr * saddr = p ;
char * addr = saddr - > sa_data ;
int err ;
if ( netif_running ( enic - > netdev ) ) {
err = enic_dev_del_station_addr ( enic ) ;
if ( err )
return err ;
}
err = enic_set_mac_addr ( netdev , addr ) ;
if ( err )
return err ;
if ( netif_running ( enic - > netdev ) ) {
err = enic_dev_add_station_addr ( enic ) ;
if ( err )
return err ;
}
return err ;
}
static int enic_set_mac_address ( struct net_device * netdev , void * p )
{
2010-08-10 22:54:55 +04:00
struct sockaddr * saddr = p ;
2010-10-20 14:17:04 +04:00
char * addr = saddr - > sa_data ;
struct enic * enic = netdev_priv ( netdev ) ;
int err ;
err = enic_dev_del_station_addr ( enic ) ;
if ( err )
return err ;
err = enic_set_mac_addr ( netdev , addr ) ;
if ( err )
return err ;
2010-08-10 22:54:55 +04:00
2010-10-20 14:17:04 +04:00
return enic_dev_add_station_addr ( enic ) ;
2010-05-18 09:50:19 +04:00
}
2010-06-24 14:50:12 +04:00
static int enic_dev_packet_filter ( struct enic * enic , int directed ,
int multicast , int broadcast , int promisc , int allmulti )
{
int err ;
spin_lock ( & enic - > devcmd_lock ) ;
err = vnic_dev_packet_filter ( enic - > vdev , directed ,
multicast , broadcast , promisc , allmulti ) ;
spin_unlock ( & enic - > devcmd_lock ) ;
return err ;
}
static int enic_dev_add_multicast_addr ( struct enic * enic , u8 * addr )
{
int err ;
spin_lock ( & enic - > devcmd_lock ) ;
err = vnic_dev_add_addr ( enic - > vdev , addr ) ;
spin_unlock ( & enic - > devcmd_lock ) ;
return err ;
}
static int enic_dev_del_multicast_addr ( struct enic * enic , u8 * addr )
{
int err ;
spin_lock ( & enic - > devcmd_lock ) ;
err = vnic_dev_del_addr ( enic - > vdev , addr ) ;
spin_unlock ( & enic - > devcmd_lock ) ;
return err ;
}
2008-09-15 20:17:11 +04:00
/* netif_tx_lock held, BHs disabled */
static void enic_set_multicast_list ( struct net_device * netdev )
{
struct enic * enic = netdev_priv ( netdev ) ;
2010-04-02 01:22:57 +04:00
struct netdev_hw_addr * ha ;
2008-09-15 20:17:11 +04:00
int directed = 1 ;
int multicast = ( netdev - > flags & IFF_MULTICAST ) ? 1 : 0 ;
int broadcast = ( netdev - > flags & IFF_BROADCAST ) ? 1 : 0 ;
int promisc = ( netdev - > flags & IFF_PROMISC ) ? 1 : 0 ;
2010-02-08 07:30:35 +03:00
unsigned int mc_count = netdev_mc_count ( netdev ) ;
2008-09-15 20:17:11 +04:00
int allmulti = ( netdev - > flags & IFF_ALLMULTI ) | |
2010-03-18 19:20:04 +03:00
mc_count > ENIC_MULTICAST_PERFECT_FILTERS ;
2009-12-23 16:27:43 +03:00
unsigned int flags = netdev - > flags | ( allmulti ? IFF_ALLMULTI : 0 ) ;
2008-09-15 20:17:11 +04:00
u8 mc_addr [ ENIC_MULTICAST_PERFECT_FILTERS ] [ ETH_ALEN ] ;
unsigned int i , j ;
if ( mc_count > ENIC_MULTICAST_PERFECT_FILTERS )
mc_count = ENIC_MULTICAST_PERFECT_FILTERS ;
2009-12-23 16:27:43 +03:00
if ( enic - > flags ! = flags ) {
enic - > flags = flags ;
2010-06-24 14:50:12 +04:00
enic_dev_packet_filter ( enic , directed ,
2009-12-23 16:27:43 +03:00
multicast , broadcast , promisc , allmulti ) ;
}
2008-09-15 20:17:11 +04:00
/* Is there an easier way? Trying to minimize to
* calls to add / del multicast addrs . We keep the
* addrs from the last call in enic - > mc_addr and
* look for changes to add / del .
*/
2010-02-22 12:22:26 +03:00
i = 0 ;
2010-04-02 01:22:57 +04:00
netdev_for_each_mc_addr ( ha , netdev ) {
2010-02-22 12:22:26 +03:00
if ( i = = mc_count )
break ;
2010-04-02 01:22:57 +04:00
memcpy ( mc_addr [ i + + ] , ha - > addr , ETH_ALEN ) ;
2008-09-15 20:17:11 +04:00
}
for ( i = 0 ; i < enic - > mc_count ; i + + ) {
for ( j = 0 ; j < mc_count ; j + + )
if ( compare_ether_addr ( enic - > mc_addr [ i ] ,
mc_addr [ j ] ) = = 0 )
break ;
if ( j = = mc_count )
2010-06-24 14:50:12 +04:00
enic_dev_del_multicast_addr ( enic , enic - > mc_addr [ i ] ) ;
2008-09-15 20:17:11 +04:00
}
for ( i = 0 ; i < mc_count ; i + + ) {
for ( j = 0 ; j < enic - > mc_count ; j + + )
if ( compare_ether_addr ( mc_addr [ i ] ,
enic - > mc_addr [ j ] ) = = 0 )
break ;
if ( j = = enic - > mc_count )
2010-06-24 14:50:12 +04:00
enic_dev_add_multicast_addr ( enic , mc_addr [ i ] ) ;
2008-09-15 20:17:11 +04:00
}
/* Save the list to compare against next time
*/
for ( i = 0 ; i < mc_count ; i + + )
memcpy ( enic - > mc_addr [ i ] , mc_addr [ i ] , ETH_ALEN ) ;
enic - > mc_count = mc_count ;
}
/* rtnl lock is held */
static void enic_vlan_rx_register ( struct net_device * netdev ,
struct vlan_group * vlan_group )
{
struct enic * enic = netdev_priv ( netdev ) ;
enic - > vlan_group = vlan_group ;
}
/* rtnl lock is held */
static void enic_vlan_rx_add_vid ( struct net_device * netdev , u16 vid )
{
struct enic * enic = netdev_priv ( netdev ) ;
spin_lock ( & enic - > devcmd_lock ) ;
enic_add_vlan ( enic , vid ) ;
spin_unlock ( & enic - > devcmd_lock ) ;
}
/* rtnl lock is held */
static void enic_vlan_rx_kill_vid ( struct net_device * netdev , u16 vid )
{
struct enic * enic = netdev_priv ( netdev ) ;
spin_lock ( & enic - > devcmd_lock ) ;
enic_del_vlan ( enic , vid ) ;
spin_unlock ( & enic - > devcmd_lock ) ;
}
/* netif_tx_lock held, BHs disabled */
static void enic_tx_timeout ( struct net_device * netdev )
{
struct enic * enic = netdev_priv ( netdev ) ;
schedule_work ( & enic - > reset ) ;
}
2010-05-18 09:50:19 +04:00
static int enic_vnic_dev_deinit ( struct enic * enic )
{
int err ;
spin_lock ( & enic - > devcmd_lock ) ;
err = vnic_dev_deinit ( enic - > vdev ) ;
spin_unlock ( & enic - > devcmd_lock ) ;
return err ;
}
static int enic_dev_init_prov ( struct enic * enic , struct vic_provinfo * vp )
{
int err ;
spin_lock ( & enic - > devcmd_lock ) ;
err = vnic_dev_init_prov ( enic - > vdev ,
( u8 * ) vp , vic_provinfo_size ( vp ) ) ;
spin_unlock ( & enic - > devcmd_lock ) ;
return err ;
}
static int enic_dev_init_done ( struct enic * enic , int * done , int * error )
{
int err ;
spin_lock ( & enic - > devcmd_lock ) ;
err = vnic_dev_init_done ( enic - > vdev , done , error ) ;
spin_unlock ( & enic - > devcmd_lock ) ;
return err ;
}
2010-06-01 12:59:33 +04:00
static int enic_set_port_profile ( struct enic * enic , u8 * mac )
2010-05-18 09:50:19 +04:00
{
struct vic_provinfo * vp ;
u8 oui [ 3 ] = VIC_PROVINFO_CISCO_OUI ;
char uuid_str [ 38 ] ;
int err ;
2010-06-01 12:59:33 +04:00
err = enic_vnic_dev_deinit ( enic ) ;
if ( err )
return err ;
2010-05-18 09:50:19 +04:00
2010-06-01 12:59:33 +04:00
switch ( enic - > pp . request ) {
2010-05-18 09:50:19 +04:00
2010-06-01 12:59:33 +04:00
case PORT_REQUEST_ASSOCIATE :
2010-05-18 09:50:19 +04:00
2010-06-01 12:59:33 +04:00
if ( ! ( enic - > pp . set & ENIC_SET_NAME ) | | ! strlen ( enic - > pp . name ) )
return - EINVAL ;
2010-05-18 09:50:19 +04:00
2010-06-01 12:59:33 +04:00
if ( ! is_valid_ether_addr ( mac ) )
return - EADDRNOTAVAIL ;
2010-05-18 09:50:19 +04:00
2010-06-01 12:59:33 +04:00
vp = vic_provinfo_alloc ( GFP_KERNEL , oui ,
VIC_PROVINFO_LINUX_TYPE ) ;
if ( ! vp )
return - ENOMEM ;
2010-05-18 09:50:19 +04:00
vic_provinfo_add_tlv ( vp ,
2010-06-01 12:59:33 +04:00
VIC_LINUX_PROV_TLV_PORT_PROFILE_NAME_STR ,
strlen ( enic - > pp . name ) + 1 , enic - > pp . name ) ;
2010-05-18 09:50:19 +04:00
2010-06-01 12:59:33 +04:00
vic_provinfo_add_tlv ( vp ,
VIC_LINUX_PROV_TLV_CLIENT_MAC_ADDR ,
ETH_ALEN , mac ) ;
if ( enic - > pp . set & ENIC_SET_INSTANCE ) {
2010-08-03 13:32:24 +04:00
sprintf ( uuid_str , " %pUB " , enic - > pp . instance_uuid ) ;
2010-06-01 12:59:33 +04:00
vic_provinfo_add_tlv ( vp ,
VIC_LINUX_PROV_TLV_CLIENT_UUID_STR ,
sizeof ( uuid_str ) , uuid_str ) ;
}
2010-05-18 09:50:19 +04:00
2010-06-01 12:59:33 +04:00
if ( enic - > pp . set & ENIC_SET_HOST ) {
2010-08-03 13:32:24 +04:00
sprintf ( uuid_str , " %pUB " , enic - > pp . host_uuid ) ;
2010-06-01 12:59:33 +04:00
vic_provinfo_add_tlv ( vp ,
VIC_LINUX_PROV_TLV_HOST_UUID_STR ,
sizeof ( uuid_str ) , uuid_str ) ;
}
2010-05-18 09:50:19 +04:00
2010-06-01 12:59:33 +04:00
err = enic_dev_init_prov ( enic , vp ) ;
vic_provinfo_free ( vp ) ;
if ( err )
return err ;
break ;
2010-05-18 09:50:19 +04:00
2010-06-01 12:59:33 +04:00
case PORT_REQUEST_DISASSOCIATE :
break ;
2010-05-18 09:50:19 +04:00
2010-06-01 12:59:33 +04:00
default :
return - EINVAL ;
}
2010-05-18 09:50:19 +04:00
2010-06-01 12:59:33 +04:00
enic - > pp . set | = ENIC_SET_APPLIED ;
return 0 ;
2010-05-18 09:50:19 +04:00
}
static int enic_set_vf_port ( struct net_device * netdev , int vf ,
struct nlattr * port [ ] )
{
struct enic * enic = netdev_priv ( netdev ) ;
2010-06-01 12:59:33 +04:00
memset ( & enic - > pp , 0 , sizeof ( enic - > pp ) ) ;
if ( port [ IFLA_PORT_REQUEST ] ) {
enic - > pp . set | = ENIC_SET_REQUEST ;
enic - > pp . request = nla_get_u8 ( port [ IFLA_PORT_REQUEST ] ) ;
}
if ( port [ IFLA_PORT_PROFILE ] ) {
enic - > pp . set | = ENIC_SET_NAME ;
memcpy ( enic - > pp . name , nla_data ( port [ IFLA_PORT_PROFILE ] ) ,
PORT_PROFILE_MAX ) ;
}
if ( port [ IFLA_PORT_INSTANCE_UUID ] ) {
enic - > pp . set | = ENIC_SET_INSTANCE ;
memcpy ( enic - > pp . instance_uuid ,
nla_data ( port [ IFLA_PORT_INSTANCE_UUID ] ) , PORT_UUID_MAX ) ;
}
if ( port [ IFLA_PORT_HOST_UUID ] ) {
enic - > pp . set | = ENIC_SET_HOST ;
memcpy ( enic - > pp . host_uuid ,
nla_data ( port [ IFLA_PORT_HOST_UUID ] ) , PORT_UUID_MAX ) ;
}
2010-05-18 09:50:19 +04:00
/* don't support VFs, yet */
if ( vf ! = PORT_SELF_VF )
return - EOPNOTSUPP ;
2010-06-01 12:59:33 +04:00
if ( ! ( enic - > pp . set & ENIC_SET_REQUEST ) )
return - EOPNOTSUPP ;
2010-05-18 09:50:19 +04:00
2010-06-01 12:59:33 +04:00
if ( enic - > pp . request = = PORT_REQUEST_ASSOCIATE ) {
2010-05-18 09:50:19 +04:00
2010-05-22 21:29:58 +04:00
/* If the interface mac addr hasn't been assigned,
* assign a random mac addr before setting port -
* profile .
*/
if ( is_zero_ether_addr ( netdev - > dev_addr ) )
random_ether_addr ( netdev - > dev_addr ) ;
2010-05-18 09:50:19 +04:00
}
2010-06-01 12:59:33 +04:00
return enic_set_port_profile ( enic , netdev - > dev_addr ) ;
2010-05-18 09:50:19 +04:00
}
static int enic_get_vf_port ( struct net_device * netdev , int vf ,
struct sk_buff * skb )
{
struct enic * enic = netdev_priv ( netdev ) ;
int err , error , done ;
u16 response = PORT_PROFILE_RESPONSE_SUCCESS ;
2010-06-01 12:59:33 +04:00
if ( ! ( enic - > pp . set & ENIC_SET_APPLIED ) )
return - ENODATA ;
2010-05-18 09:50:19 +04:00
err = enic_dev_init_done ( enic , & done , & error ) ;
if ( err )
2010-06-01 12:59:33 +04:00
error = err ;
2010-05-18 09:50:19 +04:00
switch ( error ) {
case ERR_SUCCESS :
if ( ! done )
response = PORT_PROFILE_RESPONSE_INPROGRESS ;
break ;
case ERR_EINVAL :
response = PORT_PROFILE_RESPONSE_INVALID ;
break ;
case ERR_EBADSTATE :
response = PORT_PROFILE_RESPONSE_BADSTATE ;
break ;
case ERR_ENOMEM :
response = PORT_PROFILE_RESPONSE_INSUFFICIENT_RESOURCES ;
break ;
default :
response = PORT_PROFILE_RESPONSE_ERROR ;
break ;
}
NLA_PUT_U16 ( skb , IFLA_PORT_REQUEST , enic - > pp . request ) ;
NLA_PUT_U16 ( skb , IFLA_PORT_RESPONSE , response ) ;
2010-06-01 12:59:33 +04:00
if ( enic - > pp . set & ENIC_SET_NAME )
NLA_PUT ( skb , IFLA_PORT_PROFILE , PORT_PROFILE_MAX ,
enic - > pp . name ) ;
if ( enic - > pp . set & ENIC_SET_INSTANCE )
NLA_PUT ( skb , IFLA_PORT_INSTANCE_UUID , PORT_UUID_MAX ,
enic - > pp . instance_uuid ) ;
if ( enic - > pp . set & ENIC_SET_HOST )
NLA_PUT ( skb , IFLA_PORT_HOST_UUID , PORT_UUID_MAX ,
enic - > pp . host_uuid ) ;
2010-05-18 09:50:19 +04:00
return 0 ;
nla_put_failure :
return - EMSGSIZE ;
}
2008-09-15 20:17:11 +04:00
static void enic_free_rq_buf ( struct vnic_rq * rq , struct vnic_rq_buf * buf )
{
struct enic * enic = vnic_dev_priv ( rq - > vdev ) ;
if ( ! buf - > os_buf )
return ;
pci_unmap_single ( enic - > pdev , buf - > dma_addr ,
buf - > len , PCI_DMA_FROMDEVICE ) ;
dev_kfree_skb_any ( buf - > os_buf ) ;
}
static int enic_rq_alloc_buf ( struct vnic_rq * rq )
{
struct enic * enic = vnic_dev_priv ( rq - > vdev ) ;
2009-09-03 21:02:08 +04:00
struct net_device * netdev = enic - > netdev ;
2008-09-15 20:17:11 +04:00
struct sk_buff * skb ;
2010-06-24 14:51:59 +04:00
unsigned int len = netdev - > mtu + VLAN_ETH_HLEN ;
2008-09-15 20:17:11 +04:00
unsigned int os_buf_index = 0 ;
dma_addr_t dma_addr ;
2009-10-13 09:34:20 +04:00
skb = netdev_alloc_skb_ip_align ( netdev , len ) ;
2008-09-15 20:17:11 +04:00
if ( ! skb )
return - ENOMEM ;
dma_addr = pci_map_single ( enic - > pdev , skb - > data ,
len , PCI_DMA_FROMDEVICE ) ;
enic_queue_rq_desc ( rq , skb , os_buf_index ,
dma_addr , len ) ;
return 0 ;
}
2009-09-03 21:01:58 +04:00
static int enic_rq_alloc_buf_a1 ( struct vnic_rq * rq )
{
struct rq_enet_desc * desc = vnic_rq_next_desc ( rq ) ;
if ( vnic_rq_posting_soon ( rq ) ) {
/* SW workaround for A0 HW erratum: if we're just about
* to write posted_index , insert a dummy desc
* of type resvd
*/
rq_enet_desc_enc ( desc , 0 , RQ_ENET_TYPE_RESV2 , 0 ) ;
vnic_rq_post ( rq , 0 , 0 , 0 , 0 ) ;
} else {
return enic_rq_alloc_buf ( rq ) ;
}
return 0 ;
}
2010-06-24 14:50:12 +04:00
static int enic_dev_hw_version ( struct enic * enic ,
enum vnic_dev_hw_version * hw_ver )
{
int err ;
spin_lock ( & enic - > devcmd_lock ) ;
err = vnic_dev_hw_version ( enic - > vdev , hw_ver ) ;
spin_unlock ( & enic - > devcmd_lock ) ;
return err ;
}
2009-09-03 21:01:58 +04:00
static int enic_set_rq_alloc_buf ( struct enic * enic )
{
enum vnic_dev_hw_version hw_ver ;
int err ;
2010-06-24 14:50:12 +04:00
err = enic_dev_hw_version ( enic , & hw_ver ) ;
2009-09-03 21:01:58 +04:00
if ( err )
return err ;
switch ( hw_ver ) {
case VNIC_DEV_HW_VER_A1 :
enic - > rq_alloc_buf = enic_rq_alloc_buf_a1 ;
break ;
case VNIC_DEV_HW_VER_A2 :
case VNIC_DEV_HW_VER_UNKNOWN :
enic - > rq_alloc_buf = enic_rq_alloc_buf ;
break ;
default :
return - ENODEV ;
}
return 0 ;
}
2008-09-15 20:17:11 +04:00
static void enic_rq_indicate_buf ( struct vnic_rq * rq ,
struct cq_desc * cq_desc , struct vnic_rq_buf * buf ,
int skipped , void * opaque )
{
struct enic * enic = vnic_dev_priv ( rq - > vdev ) ;
2008-11-22 08:26:55 +03:00
struct net_device * netdev = enic - > netdev ;
2008-09-15 20:17:11 +04:00
struct sk_buff * skb ;
u8 type , color , eop , sop , ingress_port , vlan_stripped ;
u8 fcoe , fcoe_sof , fcoe_fc_crc_ok , fcoe_enc_error , fcoe_eof ;
u8 tcp_udp_csum_ok , udp , tcp , ipv4_csum_ok ;
u8 ipv6 , ipv4 , ipv4_fragment , fcs_ok , rss_type , csum_not_calc ;
u8 packet_error ;
2010-06-24 14:49:51 +04:00
u16 q_number , completed_index , bytes_written , vlan_tci , checksum ;
2008-09-15 20:17:11 +04:00
u32 rss_hash ;
if ( skipped )
return ;
skb = buf - > os_buf ;
prefetch ( skb - > data - NET_IP_ALIGN ) ;
pci_unmap_single ( enic - > pdev , buf - > dma_addr ,
buf - > len , PCI_DMA_FROMDEVICE ) ;
cq_enet_rq_desc_dec ( ( struct cq_enet_rq_desc * ) cq_desc ,
& type , & color , & q_number , & completed_index ,
& ingress_port , & fcoe , & eop , & sop , & rss_type ,
& csum_not_calc , & rss_hash , & bytes_written ,
2010-06-24 14:49:51 +04:00
& packet_error , & vlan_stripped , & vlan_tci , & checksum ,
2008-09-15 20:17:11 +04:00
& fcoe_sof , & fcoe_fc_crc_ok , & fcoe_enc_error ,
& fcoe_eof , & tcp_udp_csum_ok , & udp , & tcp ,
& ipv4_csum_ok , & ipv6 , & ipv4 , & ipv4_fragment ,
& fcs_ok ) ;
if ( packet_error ) {
2009-09-03 21:02:19 +04:00
if ( ! fcs_ok ) {
if ( bytes_written > 0 )
enic - > rq_bad_fcs + + ;
else if ( bytes_written = = 0 )
enic - > rq_truncated_pkts + + ;
}
2008-09-15 20:17:11 +04:00
dev_kfree_skb_any ( skb ) ;
return ;
}
if ( eop & & bytes_written > 0 ) {
/* Good receive
*/
skb_put ( skb , bytes_written ) ;
2008-11-22 08:26:55 +03:00
skb - > protocol = eth_type_trans ( skb , netdev ) ;
2008-09-15 20:17:11 +04:00
if ( enic - > csum_rx_enabled & & ! csum_not_calc ) {
skb - > csum = htons ( checksum ) ;
skb - > ip_summed = CHECKSUM_COMPLETE ;
}
2008-11-22 08:26:55 +03:00
skb - > dev = netdev ;
2008-09-15 20:17:11 +04:00
2010-06-24 14:49:51 +04:00
if ( enic - > vlan_group & & vlan_stripped & &
( vlan_tci & CQ_ENET_RQ_DESC_VLAN_TCI_VLAN_MASK ) ) {
2008-09-15 20:17:11 +04:00
2010-06-24 14:49:25 +04:00
if ( netdev - > features & NETIF_F_GRO )
2010-10-20 14:16:59 +04:00
vlan_gro_receive ( & enic - > napi [ q_number ] ,
enic - > vlan_group , vlan_tci , skb ) ;
2008-09-15 20:17:11 +04:00
else
vlan_hwaccel_receive_skb ( skb ,
2010-06-24 14:49:51 +04:00
enic - > vlan_group , vlan_tci ) ;
2008-09-15 20:17:11 +04:00
} else {
2010-06-24 14:49:25 +04:00
if ( netdev - > features & NETIF_F_GRO )
2010-10-20 14:16:59 +04:00
napi_gro_receive ( & enic - > napi [ q_number ] , skb ) ;
2008-09-15 20:17:11 +04:00
else
netif_receive_skb ( skb ) ;
}
} else {
/* Buffer overflow
*/
dev_kfree_skb_any ( skb ) ;
}
}
static int enic_rq_service ( struct vnic_dev * vdev , struct cq_desc * cq_desc ,
u8 type , u16 q_number , u16 completed_index , void * opaque )
{
struct enic * enic = vnic_dev_priv ( vdev ) ;
vnic_rq_service ( & enic - > rq [ q_number ] , cq_desc ,
completed_index , VNIC_RQ_RETURN_DESC ,
enic_rq_indicate_buf , opaque ) ;
return 0 ;
}
static int enic_poll ( struct napi_struct * napi , int budget )
{
2010-10-20 14:16:59 +04:00
struct net_device * netdev = napi - > dev ;
struct enic * enic = netdev_priv ( netdev ) ;
unsigned int cq_rq = enic_cq_rq ( enic , 0 ) ;
unsigned int cq_wq = enic_cq_wq ( enic , 0 ) ;
unsigned int intr = enic_legacy_io_intr ( ) ;
2008-09-15 20:17:11 +04:00
unsigned int rq_work_to_do = budget ;
unsigned int wq_work_to_do = - 1 ; /* no limit */
unsigned int work_done , rq_work_done , wq_work_done ;
2009-12-23 16:27:38 +03:00
int err ;
2008-09-15 20:17:11 +04:00
/* Service RQ (first) and WQ
*/
2010-10-20 14:16:59 +04:00
rq_work_done = vnic_cq_service ( & enic - > cq [ cq_rq ] ,
2008-09-15 20:17:11 +04:00
rq_work_to_do , enic_rq_service , NULL ) ;
2010-10-20 14:16:59 +04:00
wq_work_done = vnic_cq_service ( & enic - > cq [ cq_wq ] ,
2008-09-15 20:17:11 +04:00
wq_work_to_do , enic_wq_service , NULL ) ;
/* Accumulate intr event credits for this polling
* cycle . An intr event is the completion of a
* a WQ or RQ packet .
*/
work_done = rq_work_done + wq_work_done ;
if ( work_done > 0 )
2010-10-20 14:16:59 +04:00
vnic_intr_return_credits ( & enic - > intr [ intr ] ,
2008-09-15 20:17:11 +04:00
work_done ,
0 /* don't unmask intr */ ,
0 /* don't reset intr timer */ ) ;
2009-12-23 16:27:38 +03:00
err = vnic_rq_fill ( & enic - > rq [ 0 ] , enic - > rq_alloc_buf ) ;
2008-09-15 20:17:11 +04:00
2009-12-23 16:27:38 +03:00
/* Buffer allocation failed. Stay in polling
* mode so we can try to fill the ring again .
*/
2008-09-15 20:17:11 +04:00
2009-12-23 16:27:38 +03:00
if ( err )
rq_work_done = rq_work_to_do ;
2008-09-15 20:17:11 +04:00
2009-12-23 16:27:38 +03:00
if ( rq_work_done < rq_work_to_do ) {
2008-09-15 20:17:11 +04:00
2009-12-23 16:27:38 +03:00
/* Some work done, but not enough to stay in polling,
2010-06-24 14:49:25 +04:00
* exit polling
2008-09-15 20:17:11 +04:00
*/
2009-01-20 03:43:59 +03:00
napi_complete ( napi ) ;
2010-10-20 14:16:59 +04:00
vnic_intr_unmask ( & enic - > intr [ intr ] ) ;
2008-09-15 20:17:11 +04:00
}
return rq_work_done ;
}
static int enic_poll_msix ( struct napi_struct * napi , int budget )
{
2010-10-20 14:16:59 +04:00
struct net_device * netdev = napi - > dev ;
struct enic * enic = netdev_priv ( netdev ) ;
unsigned int rq = ( napi - & enic - > napi [ 0 ] ) ;
unsigned int cq = enic_cq_rq ( enic , rq ) ;
unsigned int intr = enic_msix_rq_intr ( enic , rq ) ;
2008-09-15 20:17:11 +04:00
unsigned int work_to_do = budget ;
unsigned int work_done ;
2009-12-23 16:27:38 +03:00
int err ;
2008-09-15 20:17:11 +04:00
/* Service RQ
*/
2010-10-20 14:16:59 +04:00
work_done = vnic_cq_service ( & enic - > cq [ cq ] ,
2008-09-15 20:17:11 +04:00
work_to_do , enic_rq_service , NULL ) ;
2009-12-23 16:27:38 +03:00
/* Return intr event credits for this polling
* cycle . An intr event is the completion of a
* RQ packet .
*/
2008-09-15 20:17:11 +04:00
2009-12-23 16:27:38 +03:00
if ( work_done > 0 )
2010-10-20 14:16:59 +04:00
vnic_intr_return_credits ( & enic - > intr [ intr ] ,
2008-09-15 20:17:11 +04:00
work_done ,
0 /* don't unmask intr */ ,
0 /* don't reset intr timer */ ) ;
2010-10-20 14:16:59 +04:00
err = vnic_rq_fill ( & enic - > rq [ rq ] , enic - > rq_alloc_buf ) ;
2009-12-23 16:27:38 +03:00
/* Buffer allocation failed. Stay in polling mode
* so we can try to fill the ring again .
*/
if ( err )
work_done = work_to_do ;
if ( work_done < work_to_do ) {
/* Some work done, but not enough to stay in polling,
2010-06-24 14:49:25 +04:00
* exit polling
2008-09-15 20:17:11 +04:00
*/
2009-01-20 03:43:59 +03:00
napi_complete ( napi ) ;
2010-10-20 14:16:59 +04:00
vnic_intr_unmask ( & enic - > intr [ intr ] ) ;
2008-09-15 20:17:11 +04:00
}
return work_done ;
}
static void enic_notify_timer ( unsigned long data )
{
struct enic * enic = ( struct enic * ) data ;
enic_notify_check ( enic ) ;
2008-09-24 22:23:32 +04:00
mod_timer ( & enic - > notify_timer ,
round_jiffies ( jiffies + ENIC_NOTIFY_TIMER_PERIOD ) ) ;
2008-09-15 20:17:11 +04:00
}
static void enic_free_intr ( struct enic * enic )
{
struct net_device * netdev = enic - > netdev ;
unsigned int i ;
switch ( vnic_dev_get_intr_mode ( enic - > vdev ) ) {
case VNIC_DEV_INTR_MODE_INTX :
free_irq ( enic - > pdev - > irq , netdev ) ;
break ;
2008-09-24 22:23:42 +04:00
case VNIC_DEV_INTR_MODE_MSI :
free_irq ( enic - > pdev - > irq , enic ) ;
break ;
2008-09-15 20:17:11 +04:00
case VNIC_DEV_INTR_MODE_MSIX :
for ( i = 0 ; i < ARRAY_SIZE ( enic - > msix ) ; i + + )
if ( enic - > msix [ i ] . requested )
free_irq ( enic - > msix_entry [ i ] . vector ,
enic - > msix [ i ] . devid ) ;
break ;
default :
break ;
}
}
static int enic_request_intr ( struct enic * enic )
{
struct net_device * netdev = enic - > netdev ;
2010-10-20 14:16:59 +04:00
unsigned int i , intr ;
2008-09-15 20:17:11 +04:00
int err = 0 ;
switch ( vnic_dev_get_intr_mode ( enic - > vdev ) ) {
case VNIC_DEV_INTR_MODE_INTX :
err = request_irq ( enic - > pdev - > irq , enic_isr_legacy ,
IRQF_SHARED , netdev - > name , netdev ) ;
break ;
case VNIC_DEV_INTR_MODE_MSI :
err = request_irq ( enic - > pdev - > irq , enic_isr_msi ,
0 , netdev - > name , enic ) ;
break ;
case VNIC_DEV_INTR_MODE_MSIX :
2010-10-20 14:16:59 +04:00
for ( i = 0 ; i < enic - > rq_count ; i + + ) {
intr = enic_msix_rq_intr ( enic , i ) ;
sprintf ( enic - > msix [ intr ] . devname ,
" %.11s-rx-%d " , netdev - > name , i ) ;
enic - > msix [ intr ] . isr = enic_isr_msix_rq ;
enic - > msix [ intr ] . devid = & enic - > napi [ i ] ;
}
2008-09-15 20:17:11 +04:00
2010-10-20 14:16:59 +04:00
for ( i = 0 ; i < enic - > wq_count ; i + + ) {
intr = enic_msix_wq_intr ( enic , i ) ;
sprintf ( enic - > msix [ intr ] . devname ,
" %.11s-tx-%d " , netdev - > name , i ) ;
enic - > msix [ intr ] . isr = enic_isr_msix_wq ;
enic - > msix [ intr ] . devid = enic ;
}
2008-09-15 20:17:11 +04:00
2010-10-20 14:16:59 +04:00
intr = enic_msix_err_intr ( enic ) ;
sprintf ( enic - > msix [ intr ] . devname ,
2008-09-15 20:17:11 +04:00
" %.11s-err " , netdev - > name ) ;
2010-10-20 14:16:59 +04:00
enic - > msix [ intr ] . isr = enic_isr_msix_err ;
enic - > msix [ intr ] . devid = enic ;
2008-09-15 20:17:11 +04:00
2010-10-20 14:16:59 +04:00
intr = enic_msix_notify_intr ( enic ) ;
sprintf ( enic - > msix [ intr ] . devname ,
2008-09-15 20:17:11 +04:00
" %.11s-notify " , netdev - > name ) ;
2010-10-20 14:16:59 +04:00
enic - > msix [ intr ] . isr = enic_isr_msix_notify ;
enic - > msix [ intr ] . devid = enic ;
for ( i = 0 ; i < ARRAY_SIZE ( enic - > msix ) ; i + + )
enic - > msix [ i ] . requested = 0 ;
2008-09-15 20:17:11 +04:00
2010-10-20 14:16:59 +04:00
for ( i = 0 ; i < enic - > intr_count ; i + + ) {
2008-09-15 20:17:11 +04:00
err = request_irq ( enic - > msix_entry [ i ] . vector ,
enic - > msix [ i ] . isr , 0 ,
enic - > msix [ i ] . devname ,
enic - > msix [ i ] . devid ) ;
if ( err ) {
enic_free_intr ( enic ) ;
break ;
}
enic - > msix [ i ] . requested = 1 ;
}
break ;
default :
break ;
}
return err ;
}
2009-12-23 16:27:30 +03:00
static void enic_synchronize_irqs ( struct enic * enic )
{
unsigned int i ;
switch ( vnic_dev_get_intr_mode ( enic - > vdev ) ) {
case VNIC_DEV_INTR_MODE_INTX :
case VNIC_DEV_INTR_MODE_MSI :
synchronize_irq ( enic - > pdev - > irq ) ;
break ;
case VNIC_DEV_INTR_MODE_MSIX :
for ( i = 0 ; i < enic - > intr_count ; i + + )
synchronize_irq ( enic - > msix_entry [ i ] . vector ) ;
break ;
default :
break ;
}
}
2010-06-24 14:50:12 +04:00
static int enic_dev_notify_set ( struct enic * enic )
2008-09-15 20:17:11 +04:00
{
int err ;
2009-09-03 21:02:14 +04:00
spin_lock ( & enic - > devcmd_lock ) ;
2008-09-15 20:17:11 +04:00
switch ( vnic_dev_get_intr_mode ( enic - > vdev ) ) {
case VNIC_DEV_INTR_MODE_INTX :
2010-10-20 14:16:59 +04:00
err = vnic_dev_notify_set ( enic - > vdev ,
enic_legacy_notify_intr ( ) ) ;
2008-09-15 20:17:11 +04:00
break ;
case VNIC_DEV_INTR_MODE_MSIX :
2010-10-20 14:16:59 +04:00
err = vnic_dev_notify_set ( enic - > vdev ,
enic_msix_notify_intr ( enic ) ) ;
2008-09-15 20:17:11 +04:00
break ;
default :
err = vnic_dev_notify_set ( enic - > vdev , - 1 /* no intr */ ) ;
break ;
}
2009-09-03 21:02:14 +04:00
spin_unlock ( & enic - > devcmd_lock ) ;
2008-09-15 20:17:11 +04:00
return err ;
}
2010-06-24 14:50:12 +04:00
static int enic_dev_notify_unset ( struct enic * enic )
{
int err ;
spin_lock ( & enic - > devcmd_lock ) ;
err = vnic_dev_notify_unset ( enic - > vdev ) ;
spin_unlock ( & enic - > devcmd_lock ) ;
return err ;
}
static int enic_dev_enable ( struct enic * enic )
{
int err ;
spin_lock ( & enic - > devcmd_lock ) ;
2010-10-20 14:17:09 +04:00
err = vnic_dev_enable_wait ( enic - > vdev ) ;
2010-06-24 14:50:12 +04:00
spin_unlock ( & enic - > devcmd_lock ) ;
return err ;
}
static int enic_dev_disable ( struct enic * enic )
{
int err ;
spin_lock ( & enic - > devcmd_lock ) ;
err = vnic_dev_disable ( enic - > vdev ) ;
spin_unlock ( & enic - > devcmd_lock ) ;
return err ;
}
2008-09-15 20:17:11 +04:00
static void enic_notify_timer_start ( struct enic * enic )
{
switch ( vnic_dev_get_intr_mode ( enic - > vdev ) ) {
case VNIC_DEV_INTR_MODE_MSI :
mod_timer ( & enic - > notify_timer , jiffies ) ;
break ;
default :
/* Using intr for notification for INTx/MSI-X */
break ;
} ;
}
/* rtnl lock is held, process context */
static int enic_open ( struct net_device * netdev )
{
struct enic * enic = netdev_priv ( netdev ) ;
unsigned int i ;
int err ;
2008-09-24 22:23:53 +04:00
err = enic_request_intr ( enic ) ;
if ( err ) {
2010-06-24 14:50:56 +04:00
netdev_err ( netdev , " Unable to request irq. \n " ) ;
2008-09-24 22:23:53 +04:00
return err ;
}
2010-06-24 14:50:12 +04:00
err = enic_dev_notify_set ( enic ) ;
2008-09-24 22:23:53 +04:00
if ( err ) {
2010-06-24 14:50:56 +04:00
netdev_err ( netdev ,
" Failed to alloc notify buffer, aborting. \n " ) ;
2008-09-24 22:23:53 +04:00
goto err_out_free_intr ;
}
2008-09-15 20:17:11 +04:00
for ( i = 0 ; i < enic - > rq_count ; i + + ) {
2009-12-23 16:27:38 +03:00
vnic_rq_fill ( & enic - > rq [ i ] , enic - > rq_alloc_buf ) ;
/* Need at least one buffer on ring to get going */
if ( vnic_rq_desc_used ( & enic - > rq [ i ] ) = = 0 ) {
2010-06-24 14:50:56 +04:00
netdev_err ( netdev , " Unable to alloc receive buffers \n " ) ;
2009-12-23 16:27:38 +03:00
err = - ENOMEM ;
2008-09-24 22:23:53 +04:00
goto err_out_notify_unset ;
2008-09-15 20:17:11 +04:00
}
}
for ( i = 0 ; i < enic - > wq_count ; i + + )
vnic_wq_enable ( & enic - > wq [ i ] ) ;
for ( i = 0 ; i < enic - > rq_count ; i + + )
vnic_rq_enable ( & enic - > rq [ i ] ) ;
2010-05-18 09:50:19 +04:00
enic_dev_add_station_addr ( enic ) ;
2008-09-15 20:17:11 +04:00
enic_set_multicast_list ( netdev ) ;
netif_wake_queue ( netdev ) ;
2010-10-20 14:16:59 +04:00
for ( i = 0 ; i < enic - > rq_count ; i + + )
napi_enable ( & enic - > napi [ i ] ) ;
2010-06-24 14:50:12 +04:00
enic_dev_enable ( enic ) ;
2008-09-15 20:17:11 +04:00
for ( i = 0 ; i < enic - > intr_count ; i + + )
vnic_intr_unmask ( & enic - > intr [ i ] ) ;
enic_notify_timer_start ( enic ) ;
return 0 ;
2008-09-24 22:23:53 +04:00
err_out_notify_unset :
2010-06-24 14:50:12 +04:00
enic_dev_notify_unset ( enic ) ;
2008-09-24 22:23:53 +04:00
err_out_free_intr :
enic_free_intr ( enic ) ;
return err ;
2008-09-15 20:17:11 +04:00
}
/* rtnl lock is held, process context */
static int enic_stop ( struct net_device * netdev )
{
struct enic * enic = netdev_priv ( netdev ) ;
unsigned int i ;
int err ;
2010-06-24 14:52:26 +04:00
for ( i = 0 ; i < enic - > intr_count ; i + + ) {
2009-12-23 16:27:30 +03:00
vnic_intr_mask ( & enic - > intr [ i ] ) ;
2010-06-24 14:52:26 +04:00
( void ) vnic_intr_masked ( & enic - > intr [ i ] ) ; /* flush write */
}
2009-12-23 16:27:30 +03:00
enic_synchronize_irqs ( enic ) ;
2008-09-15 20:17:11 +04:00
del_timer_sync ( & enic - > notify_timer ) ;
2010-06-24 14:50:12 +04:00
enic_dev_disable ( enic ) ;
2010-10-20 14:16:59 +04:00
for ( i = 0 ; i < enic - > rq_count ; i + + )
napi_disable ( & enic - > napi [ i ] ) ;
2009-12-23 16:27:30 +03:00
netif_carrier_off ( netdev ) ;
netif_tx_disable ( netdev ) ;
2010-05-18 09:50:19 +04:00
enic_dev_del_station_addr ( enic ) ;
2008-09-15 20:17:11 +04:00
for ( i = 0 ; i < enic - > wq_count ; i + + ) {
err = vnic_wq_disable ( & enic - > wq [ i ] ) ;
if ( err )
return err ;
}
for ( i = 0 ; i < enic - > rq_count ; i + + ) {
err = vnic_rq_disable ( & enic - > rq [ i ] ) ;
if ( err )
return err ;
}
2010-06-24 14:50:12 +04:00
enic_dev_notify_unset ( enic ) ;
2008-09-24 22:23:53 +04:00
enic_free_intr ( enic ) ;
2008-09-15 20:17:11 +04:00
for ( i = 0 ; i < enic - > wq_count ; i + + )
vnic_wq_clean ( & enic - > wq [ i ] , enic_free_wq_buf ) ;
for ( i = 0 ; i < enic - > rq_count ; i + + )
vnic_rq_clean ( & enic - > rq [ i ] , enic_free_rq_buf ) ;
for ( i = 0 ; i < enic - > cq_count ; i + + )
vnic_cq_clean ( & enic - > cq [ i ] ) ;
for ( i = 0 ; i < enic - > intr_count ; i + + )
vnic_intr_clean ( & enic - > intr [ i ] ) ;
return 0 ;
}
static int enic_change_mtu ( struct net_device * netdev , int new_mtu )
{
struct enic * enic = netdev_priv ( netdev ) ;
int running = netif_running ( netdev ) ;
2008-09-24 22:23:32 +04:00
if ( new_mtu < ENIC_MIN_MTU | | new_mtu > ENIC_MAX_MTU )
return - EINVAL ;
2008-09-15 20:17:11 +04:00
if ( running )
enic_stop ( netdev ) ;
netdev - > mtu = new_mtu ;
if ( netdev - > mtu > enic - > port_mtu )
2010-06-24 14:50:56 +04:00
netdev_warn ( netdev ,
" interface MTU (%d) set higher than port MTU (%d) \n " ,
netdev - > mtu , enic - > port_mtu ) ;
2008-09-15 20:17:11 +04:00
if ( running )
enic_open ( netdev ) ;
return 0 ;
}
# ifdef CONFIG_NET_POLL_CONTROLLER
static void enic_poll_controller ( struct net_device * netdev )
{
struct enic * enic = netdev_priv ( netdev ) ;
struct vnic_dev * vdev = enic - > vdev ;
2010-10-20 14:16:59 +04:00
unsigned int i , intr ;
2008-09-15 20:17:11 +04:00
switch ( vnic_dev_get_intr_mode ( vdev ) ) {
case VNIC_DEV_INTR_MODE_MSIX :
2010-10-20 14:16:59 +04:00
for ( i = 0 ; i < enic - > rq_count ; i + + ) {
intr = enic_msix_rq_intr ( enic , i ) ;
enic_isr_msix_rq ( enic - > msix_entry [ intr ] . vector , enic ) ;
}
intr = enic_msix_wq_intr ( enic , i ) ;
enic_isr_msix_wq ( enic - > msix_entry [ intr ] . vector , enic ) ;
2008-09-15 20:17:11 +04:00
break ;
case VNIC_DEV_INTR_MODE_MSI :
enic_isr_msi ( enic - > pdev - > irq , enic ) ;
break ;
case VNIC_DEV_INTR_MODE_INTX :
enic_isr_legacy ( enic - > pdev - > irq , netdev ) ;
break ;
default :
break ;
}
}
# endif
static int enic_dev_wait ( struct vnic_dev * vdev ,
int ( * start ) ( struct vnic_dev * , int ) ,
int ( * finished ) ( struct vnic_dev * , int * ) ,
int arg )
{
unsigned long time ;
int done ;
int err ;
BUG_ON ( in_interrupt ( ) ) ;
err = start ( vdev , arg ) ;
if ( err )
return err ;
/* Wait for func to complete...2 seconds max
*/
time = jiffies + ( HZ * 2 ) ;
do {
err = finished ( vdev , & done ) ;
if ( err )
return err ;
if ( done )
return 0 ;
schedule_timeout_uninterruptible ( HZ / 10 ) ;
} while ( time_after ( time , jiffies ) ) ;
return - ETIMEDOUT ;
}
static int enic_dev_open ( struct enic * enic )
{
int err ;
err = enic_dev_wait ( enic - > vdev , vnic_dev_open ,
vnic_dev_open_done , 0 ) ;
if ( err )
2010-06-24 14:50:56 +04:00
dev_err ( enic_get_dev ( enic ) , " vNIC device open failed, err %d \n " ,
err ) ;
2008-09-15 20:17:11 +04:00
return err ;
}
2010-06-24 14:50:00 +04:00
static int enic_dev_hang_reset ( struct enic * enic )
2008-09-15 20:17:11 +04:00
{
int err ;
2010-06-24 14:50:00 +04:00
err = enic_dev_wait ( enic - > vdev , vnic_dev_hang_reset ,
vnic_dev_hang_reset_done , 0 ) ;
2008-09-15 20:17:11 +04:00
if ( err )
2010-06-24 14:50:56 +04:00
netdev_err ( enic - > netdev , " vNIC hang reset failed, err %d \n " ,
err ) ;
2008-09-15 20:17:11 +04:00
return err ;
}
2010-10-20 14:16:59 +04:00
static int enic_set_rsskey ( struct enic * enic )
{
2010-11-15 11:09:55 +03:00
dma_addr_t rss_key_buf_pa ;
2010-10-20 14:16:59 +04:00
union vnic_rss_key * rss_key_buf_va = NULL ;
union vnic_rss_key rss_key = {
. key [ 0 ] . b = { 85 , 67 , 83 , 97 , 119 , 101 , 115 , 111 , 109 , 101 } ,
. key [ 1 ] . b = { 80 , 65 , 76 , 79 , 117 , 110 , 105 , 113 , 117 , 101 } ,
. key [ 2 ] . b = { 76 , 73 , 78 , 85 , 88 , 114 , 111 , 99 , 107 , 115 } ,
. key [ 3 ] . b = { 69 , 78 , 73 , 67 , 105 , 115 , 99 , 111 , 111 , 108 } ,
} ;
int err ;
rss_key_buf_va = pci_alloc_consistent ( enic - > pdev ,
sizeof ( union vnic_rss_key ) , & rss_key_buf_pa ) ;
if ( ! rss_key_buf_va )
return - ENOMEM ;
memcpy ( rss_key_buf_va , & rss_key , sizeof ( union vnic_rss_key ) ) ;
spin_lock ( & enic - > devcmd_lock ) ;
err = enic_set_rss_key ( enic ,
rss_key_buf_pa ,
sizeof ( union vnic_rss_key ) ) ;
spin_unlock ( & enic - > devcmd_lock ) ;
pci_free_consistent ( enic - > pdev , sizeof ( union vnic_rss_key ) ,
rss_key_buf_va , rss_key_buf_pa ) ;
return err ;
}
static int enic_set_rsscpu ( struct enic * enic , u8 rss_hash_bits )
{
2010-11-15 11:09:55 +03:00
dma_addr_t rss_cpu_buf_pa ;
2010-10-20 14:16:59 +04:00
union vnic_rss_cpu * rss_cpu_buf_va = NULL ;
unsigned int i ;
int err ;
rss_cpu_buf_va = pci_alloc_consistent ( enic - > pdev ,
sizeof ( union vnic_rss_cpu ) , & rss_cpu_buf_pa ) ;
if ( ! rss_cpu_buf_va )
return - ENOMEM ;
for ( i = 0 ; i < ( 1 < < rss_hash_bits ) ; i + + )
( * rss_cpu_buf_va ) . cpu [ i / 4 ] . b [ i % 4 ] = i % enic - > rq_count ;
spin_lock ( & enic - > devcmd_lock ) ;
err = enic_set_rss_cpu ( enic ,
rss_cpu_buf_pa ,
sizeof ( union vnic_rss_cpu ) ) ;
spin_unlock ( & enic - > devcmd_lock ) ;
pci_free_consistent ( enic - > pdev , sizeof ( union vnic_rss_cpu ) ,
rss_cpu_buf_va , rss_cpu_buf_pa ) ;
return err ;
}
static int enic_set_niccfg ( struct enic * enic , u8 rss_default_cpu ,
u8 rss_hash_type , u8 rss_hash_bits , u8 rss_base_cpu , u8 rss_enable )
2009-02-10 10:24:24 +03:00
{
const u8 tso_ipid_split_en = 0 ;
const u8 ig_vlan_strip_en = 1 ;
2010-06-24 14:50:12 +04:00
int err ;
2009-02-10 10:24:24 +03:00
2010-10-20 14:16:59 +04:00
/* Enable VLAN tag stripping.
*/
2009-02-10 10:24:24 +03:00
2010-06-24 14:50:12 +04:00
spin_lock ( & enic - > devcmd_lock ) ;
err = enic_set_nic_cfg ( enic ,
2009-02-10 10:24:24 +03:00
rss_default_cpu , rss_hash_type ,
rss_hash_bits , rss_base_cpu ,
rss_enable , tso_ipid_split_en ,
ig_vlan_strip_en ) ;
2010-06-24 14:50:12 +04:00
spin_unlock ( & enic - > devcmd_lock ) ;
return err ;
}
2010-10-20 14:16:59 +04:00
static int enic_set_rss_nic_cfg ( struct enic * enic )
{
struct device * dev = enic_get_dev ( enic ) ;
const u8 rss_default_cpu = 0 ;
const u8 rss_hash_type = NIC_CFG_RSS_HASH_TYPE_IPV4 |
NIC_CFG_RSS_HASH_TYPE_TCP_IPV4 |
NIC_CFG_RSS_HASH_TYPE_IPV6 |
NIC_CFG_RSS_HASH_TYPE_TCP_IPV6 ;
const u8 rss_hash_bits = 7 ;
const u8 rss_base_cpu = 0 ;
u8 rss_enable = ENIC_SETTING ( enic , RSS ) & & ( enic - > rq_count > 1 ) ;
if ( rss_enable ) {
if ( ! enic_set_rsskey ( enic ) ) {
if ( enic_set_rsscpu ( enic , rss_hash_bits ) ) {
rss_enable = 0 ;
dev_warn ( dev , " RSS disabled, "
" Failed to set RSS cpu indirection table. " ) ;
}
} else {
rss_enable = 0 ;
dev_warn ( dev , " RSS disabled, Failed to set RSS key. \n " ) ;
}
}
return enic_set_niccfg ( enic , rss_default_cpu , rss_hash_type ,
rss_hash_bits , rss_base_cpu , rss_enable ) ;
}
2010-06-24 14:50:12 +04:00
static int enic_dev_hang_notify ( struct enic * enic )
{
int err ;
spin_lock ( & enic - > devcmd_lock ) ;
err = vnic_dev_hang_notify ( enic - > vdev ) ;
spin_unlock ( & enic - > devcmd_lock ) ;
return err ;
2009-02-10 10:24:24 +03:00
}
2010-09-30 17:35:45 +04:00
static int enic_dev_set_ig_vlan_rewrite_mode ( struct enic * enic )
2010-06-24 14:49:51 +04:00
{
int err ;
spin_lock ( & enic - > devcmd_lock ) ;
err = vnic_dev_set_ig_vlan_rewrite_mode ( enic - > vdev ,
IG_VLAN_REWRITE_MODE_PRIORITY_TAG_DEFAULT_VLAN ) ;
spin_unlock ( & enic - > devcmd_lock ) ;
return err ;
}
2008-09-15 20:17:11 +04:00
static void enic_reset ( struct work_struct * work )
{
struct enic * enic = container_of ( work , struct enic , reset ) ;
if ( ! netif_running ( enic - > netdev ) )
return ;
rtnl_lock ( ) ;
2010-06-24 14:50:12 +04:00
enic_dev_hang_notify ( enic ) ;
2008-09-15 20:17:11 +04:00
enic_stop ( enic - > netdev ) ;
2010-06-24 14:50:00 +04:00
enic_dev_hang_reset ( enic ) ;
enic_reset_multicast_list ( enic ) ;
2008-09-15 20:17:11 +04:00
enic_init_vnic_resources ( enic ) ;
2010-10-20 14:16:59 +04:00
enic_set_rss_nic_cfg ( enic ) ;
2010-06-24 14:49:51 +04:00
enic_dev_set_ig_vlan_rewrite_mode ( enic ) ;
2008-09-15 20:17:11 +04:00
enic_open ( enic - > netdev ) ;
rtnl_unlock ( ) ;
}
static int enic_set_intr_mode ( struct enic * enic )
{
2010-10-20 14:16:59 +04:00
unsigned int n = min_t ( unsigned int , enic - > rq_count , ENIC_RQ_MAX ) ;
2009-09-03 21:02:24 +04:00
unsigned int m = 1 ;
2008-09-15 20:17:11 +04:00
unsigned int i ;
/* Set interrupt mode (INTx, MSI, MSI-X) depending
2010-10-20 14:16:59 +04:00
* on system capabilities .
2008-09-15 20:17:11 +04:00
*
* Try MSI - X first
*
* We need n RQs , m WQs , n + m CQs , and n + m + 2 INTRs
* ( the second to last INTR is used for WQ / RQ errors )
* ( the last INTR is used for notifications )
*/
BUG_ON ( ARRAY_SIZE ( enic - > msix_entry ) < n + m + 2 ) ;
for ( i = 0 ; i < n + m + 2 ; i + + )
enic - > msix_entry [ i ] . entry = i ;
2010-10-20 14:16:59 +04:00
/* Use multiple RQs if RSS is enabled
*/
if ( ENIC_SETTING ( enic , RSS ) & &
enic - > config . intr_mode < 1 & &
2008-09-15 20:17:11 +04:00
enic - > rq_count > = n & &
enic - > wq_count > = m & &
enic - > cq_count > = n + m & &
2010-10-20 14:16:59 +04:00
enic - > intr_count > = n + m + 2 ) {
2008-09-15 20:17:11 +04:00
2010-10-20 14:16:59 +04:00
if ( ! pci_enable_msix ( enic - > pdev , enic - > msix_entry , n + m + 2 ) ) {
2008-09-15 20:17:11 +04:00
2010-10-20 14:16:59 +04:00
enic - > rq_count = n ;
enic - > wq_count = m ;
enic - > cq_count = n + m ;
enic - > intr_count = n + m + 2 ;
2008-09-15 20:17:11 +04:00
2010-10-20 14:16:59 +04:00
vnic_dev_set_intr_mode ( enic - > vdev ,
VNIC_DEV_INTR_MODE_MSIX ) ;
return 0 ;
}
}
if ( enic - > config . intr_mode < 1 & &
enic - > rq_count > = 1 & &
enic - > wq_count > = m & &
enic - > cq_count > = 1 + m & &
enic - > intr_count > = 1 + m + 2 ) {
if ( ! pci_enable_msix ( enic - > pdev , enic - > msix_entry , 1 + m + 2 ) ) {
enic - > rq_count = 1 ;
enic - > wq_count = m ;
enic - > cq_count = 1 + m ;
enic - > intr_count = 1 + m + 2 ;
vnic_dev_set_intr_mode ( enic - > vdev ,
VNIC_DEV_INTR_MODE_MSIX ) ;
return 0 ;
}
2008-09-15 20:17:11 +04:00
}
/* Next try MSI
*
* We need 1 RQ , 1 WQ , 2 CQs , and 1 INTR
*/
if ( enic - > config . intr_mode < 2 & &
enic - > rq_count > = 1 & &
enic - > wq_count > = 1 & &
enic - > cq_count > = 2 & &
enic - > intr_count > = 1 & &
! pci_enable_msi ( enic - > pdev ) ) {
enic - > rq_count = 1 ;
enic - > wq_count = 1 ;
enic - > cq_count = 2 ;
enic - > intr_count = 1 ;
vnic_dev_set_intr_mode ( enic - > vdev , VNIC_DEV_INTR_MODE_MSI ) ;
return 0 ;
}
/* Next try INTx
*
* We need 1 RQ , 1 WQ , 2 CQs , and 3 INTRs
* ( the first INTR is used for WQ / RQ )
* ( the second INTR is used for WQ / RQ errors )
* ( the last INTR is used for notifications )
*/
if ( enic - > config . intr_mode < 3 & &
enic - > rq_count > = 1 & &
enic - > wq_count > = 1 & &
enic - > cq_count > = 2 & &
enic - > intr_count > = 3 ) {
enic - > rq_count = 1 ;
enic - > wq_count = 1 ;
enic - > cq_count = 2 ;
enic - > intr_count = 3 ;
vnic_dev_set_intr_mode ( enic - > vdev , VNIC_DEV_INTR_MODE_INTX ) ;
return 0 ;
}
vnic_dev_set_intr_mode ( enic - > vdev , VNIC_DEV_INTR_MODE_UNKNOWN ) ;
return - EINVAL ;
}
static void enic_clear_intr_mode ( struct enic * enic )
{
switch ( vnic_dev_get_intr_mode ( enic - > vdev ) ) {
case VNIC_DEV_INTR_MODE_MSIX :
pci_disable_msix ( enic - > pdev ) ;
break ;
case VNIC_DEV_INTR_MODE_MSI :
pci_disable_msi ( enic - > pdev ) ;
break ;
default :
break ;
}
vnic_dev_set_intr_mode ( enic - > vdev , VNIC_DEV_INTR_MODE_UNKNOWN ) ;
}
2010-05-18 09:50:19 +04:00
static const struct net_device_ops enic_netdev_dynamic_ops = {
. ndo_open = enic_open ,
. ndo_stop = enic_stop ,
. ndo_start_xmit = enic_hard_start_xmit ,
. ndo_get_stats = enic_get_stats ,
. ndo_validate_addr = eth_validate_addr ,
. ndo_set_multicast_list = enic_set_multicast_list ,
. ndo_set_mac_address = enic_set_mac_address_dynamic ,
. ndo_change_mtu = enic_change_mtu ,
. ndo_vlan_rx_register = enic_vlan_rx_register ,
. ndo_vlan_rx_add_vid = enic_vlan_rx_add_vid ,
. ndo_vlan_rx_kill_vid = enic_vlan_rx_kill_vid ,
. ndo_tx_timeout = enic_tx_timeout ,
. ndo_set_vf_port = enic_set_vf_port ,
. ndo_get_vf_port = enic_get_vf_port ,
# ifdef CONFIG_NET_POLL_CONTROLLER
. ndo_poll_controller = enic_poll_controller ,
# endif
} ;
2008-11-20 09:23:26 +03:00
static const struct net_device_ops enic_netdev_ops = {
. ndo_open = enic_open ,
. ndo_stop = enic_stop ,
2008-11-21 07:14:53 +03:00
. ndo_start_xmit = enic_hard_start_xmit ,
2008-11-20 09:23:26 +03:00
. ndo_get_stats = enic_get_stats ,
. ndo_validate_addr = eth_validate_addr ,
2010-05-18 09:50:19 +04:00
. ndo_set_mac_address = enic_set_mac_address ,
2010-06-24 14:50:12 +04:00
. ndo_set_multicast_list = enic_set_multicast_list ,
2008-11-20 09:23:26 +03:00
. ndo_change_mtu = enic_change_mtu ,
. ndo_vlan_rx_register = enic_vlan_rx_register ,
. ndo_vlan_rx_add_vid = enic_vlan_rx_add_vid ,
. ndo_vlan_rx_kill_vid = enic_vlan_rx_kill_vid ,
. ndo_tx_timeout = enic_tx_timeout ,
# ifdef CONFIG_NET_POLL_CONTROLLER
. ndo_poll_controller = enic_poll_controller ,
# endif
} ;
2010-09-30 17:35:45 +04:00
static void enic_dev_deinit ( struct enic * enic )
2009-09-03 21:02:45 +04:00
{
2010-10-20 14:16:59 +04:00
unsigned int i ;
for ( i = 0 ; i < enic - > rq_count ; i + + )
netif_napi_del ( & enic - > napi [ i ] ) ;
2009-09-03 21:02:45 +04:00
enic_free_vnic_resources ( enic ) ;
enic_clear_intr_mode ( enic ) ;
}
2010-09-30 17:35:45 +04:00
static int enic_dev_init ( struct enic * enic )
2009-09-03 21:02:45 +04:00
{
2010-06-24 14:50:56 +04:00
struct device * dev = enic_get_dev ( enic ) ;
2009-09-03 21:02:45 +04:00
struct net_device * netdev = enic - > netdev ;
2010-10-20 14:16:59 +04:00
unsigned int i ;
2009-09-03 21:02:45 +04:00
int err ;
/* Get vNIC configuration
*/
err = enic_get_vnic_config ( enic ) ;
if ( err ) {
2010-06-24 14:50:56 +04:00
dev_err ( dev , " Get vNIC configuration failed, aborting \n " ) ;
2009-09-03 21:02:45 +04:00
return err ;
}
/* Get available resource counts
*/
enic_get_res_counts ( enic ) ;
/* Set interrupt mode based on resource counts and system
* capabilities
*/
err = enic_set_intr_mode ( enic ) ;
if ( err ) {
2010-06-24 14:50:56 +04:00
dev_err ( dev , " Failed to set intr mode based on resource "
" counts and system capabilities, aborting \n " ) ;
2009-09-03 21:02:45 +04:00
return err ;
}
/* Allocate and configure vNIC resources
*/
err = enic_alloc_vnic_resources ( enic ) ;
if ( err ) {
2010-06-24 14:50:56 +04:00
dev_err ( dev , " Failed to alloc vNIC resources, aborting \n " ) ;
2009-09-03 21:02:45 +04:00
goto err_out_free_vnic_resources ;
}
enic_init_vnic_resources ( enic ) ;
err = enic_set_rq_alloc_buf ( enic ) ;
if ( err ) {
2010-06-24 14:50:56 +04:00
dev_err ( dev , " Failed to set RQ buffer allocator, aborting \n " ) ;
2009-09-03 21:02:45 +04:00
goto err_out_free_vnic_resources ;
}
2010-10-20 14:16:59 +04:00
err = enic_set_rss_nic_cfg ( enic ) ;
2009-09-03 21:02:45 +04:00
if ( err ) {
2010-06-24 14:50:56 +04:00
dev_err ( dev , " Failed to config nic, aborting \n " ) ;
2009-09-03 21:02:45 +04:00
goto err_out_free_vnic_resources ;
}
2010-06-24 14:49:51 +04:00
err = enic_dev_set_ig_vlan_rewrite_mode ( enic ) ;
if ( err ) {
2010-10-20 14:17:19 +04:00
dev_err ( dev ,
2010-06-24 14:49:51 +04:00
" Failed to set ingress vlan rewrite mode, aborting. \n " ) ;
goto err_out_free_vnic_resources ;
}
2009-09-03 21:02:45 +04:00
switch ( vnic_dev_get_intr_mode ( enic - > vdev ) ) {
default :
2010-10-20 14:16:59 +04:00
netif_napi_add ( netdev , & enic - > napi [ 0 ] , enic_poll , 64 ) ;
2009-09-03 21:02:45 +04:00
break ;
case VNIC_DEV_INTR_MODE_MSIX :
2010-10-20 14:16:59 +04:00
for ( i = 0 ; i < enic - > rq_count ; i + + )
netif_napi_add ( netdev , & enic - > napi [ i ] ,
enic_poll_msix , 64 ) ;
2009-09-03 21:02:45 +04:00
break ;
}
return 0 ;
err_out_free_vnic_resources :
enic_clear_intr_mode ( enic ) ;
enic_free_vnic_resources ( enic ) ;
return err ;
}
2009-09-03 21:01:53 +04:00
static void enic_iounmap ( struct enic * enic )
{
unsigned int i ;
for ( i = 0 ; i < ARRAY_SIZE ( enic - > bar ) ; i + + )
if ( enic - > bar [ i ] . vaddr )
iounmap ( enic - > bar [ i ] . vaddr ) ;
}
2008-09-15 20:17:11 +04:00
static int __devinit enic_probe ( struct pci_dev * pdev ,
const struct pci_device_id * ent )
{
2010-06-24 14:50:56 +04:00
struct device * dev = & pdev - > dev ;
2008-09-15 20:17:11 +04:00
struct net_device * netdev ;
struct enic * enic ;
int using_dac = 0 ;
unsigned int i ;
int err ;
/* Allocate net device structure and initialize. Private
* instance data is initialized to zero .
*/
netdev = alloc_etherdev ( sizeof ( struct enic ) ) ;
if ( ! netdev ) {
2010-06-24 14:50:56 +04:00
pr_err ( " Etherdev alloc failed, aborting \n " ) ;
2008-09-15 20:17:11 +04:00
return - ENOMEM ;
}
pci_set_drvdata ( pdev , netdev ) ;
SET_NETDEV_DEV ( netdev , & pdev - > dev ) ;
enic = netdev_priv ( netdev ) ;
enic - > netdev = netdev ;
enic - > pdev = pdev ;
/* Setup PCI resources
*/
2010-06-24 14:52:26 +04:00
err = pci_enable_device_mem ( pdev ) ;
2008-09-15 20:17:11 +04:00
if ( err ) {
2010-06-24 14:50:56 +04:00
dev_err ( dev , " Cannot enable PCI device, aborting \n " ) ;
2008-09-15 20:17:11 +04:00
goto err_out_free_netdev ;
}
err = pci_request_regions ( pdev , DRV_NAME ) ;
if ( err ) {
2010-06-24 14:50:56 +04:00
dev_err ( dev , " Cannot request PCI regions, aborting \n " ) ;
2008-09-15 20:17:11 +04:00
goto err_out_disable_device ;
}
pci_set_master ( pdev ) ;
/* Query PCI controller on system for DMA addressing
* limitation for the device . Try 40 - bit first , and
* fail to 32 - bit .
*/
2009-04-07 06:01:14 +04:00
err = pci_set_dma_mask ( pdev , DMA_BIT_MASK ( 40 ) ) ;
2008-09-15 20:17:11 +04:00
if ( err ) {
2009-04-07 06:01:15 +04:00
err = pci_set_dma_mask ( pdev , DMA_BIT_MASK ( 32 ) ) ;
2008-09-15 20:17:11 +04:00
if ( err ) {
2010-06-24 14:50:56 +04:00
dev_err ( dev , " No usable DMA configuration, aborting \n " ) ;
2008-09-15 20:17:11 +04:00
goto err_out_release_regions ;
}
2009-04-07 06:01:15 +04:00
err = pci_set_consistent_dma_mask ( pdev , DMA_BIT_MASK ( 32 ) ) ;
2008-09-15 20:17:11 +04:00
if ( err ) {
2010-06-24 14:50:56 +04:00
dev_err ( dev , " Unable to obtain %u-bit DMA "
" for consistent allocations, aborting \n " , 32 ) ;
2008-09-15 20:17:11 +04:00
goto err_out_release_regions ;
}
} else {
2009-04-07 06:01:14 +04:00
err = pci_set_consistent_dma_mask ( pdev , DMA_BIT_MASK ( 40 ) ) ;
2008-09-15 20:17:11 +04:00
if ( err ) {
2010-06-24 14:50:56 +04:00
dev_err ( dev , " Unable to obtain %u-bit DMA "
" for consistent allocations, aborting \n " , 40 ) ;
2008-09-15 20:17:11 +04:00
goto err_out_release_regions ;
}
using_dac = 1 ;
}
2009-09-03 21:01:53 +04:00
/* Map vNIC resources from BAR0-5
2008-09-15 20:17:11 +04:00
*/
2009-09-03 21:01:53 +04:00
for ( i = 0 ; i < ARRAY_SIZE ( enic - > bar ) ; i + + ) {
if ( ! ( pci_resource_flags ( pdev , i ) & IORESOURCE_MEM ) )
continue ;
enic - > bar [ i ] . len = pci_resource_len ( pdev , i ) ;
enic - > bar [ i ] . vaddr = pci_iomap ( pdev , i , enic - > bar [ i ] . len ) ;
if ( ! enic - > bar [ i ] . vaddr ) {
2010-06-24 14:50:56 +04:00
dev_err ( dev , " Cannot memory-map BAR %d, aborting \n " , i ) ;
2009-09-03 21:01:53 +04:00
err = - ENODEV ;
goto err_out_iounmap ;
}
enic - > bar [ i ] . bus_addr = pci_resource_start ( pdev , i ) ;
2008-09-15 20:17:11 +04:00
}
/* Register vNIC device
*/
2009-09-03 21:01:53 +04:00
enic - > vdev = vnic_dev_register ( NULL , enic , pdev , enic - > bar ,
ARRAY_SIZE ( enic - > bar ) ) ;
2008-09-15 20:17:11 +04:00
if ( ! enic - > vdev ) {
2010-06-24 14:50:56 +04:00
dev_err ( dev , " vNIC registration failed, aborting \n " ) ;
2008-09-15 20:17:11 +04:00
err = - ENODEV ;
goto err_out_iounmap ;
}
/* Issue device open to get device in known state
*/
err = enic_dev_open ( enic ) ;
if ( err ) {
2010-06-24 14:50:56 +04:00
dev_err ( dev , " vNIC dev open failed, aborting \n " ) ;
2008-09-15 20:17:11 +04:00
goto err_out_vnic_unregister ;
}
/* Issue device init to initialize the vnic-to-switch link.
* We ' ll start with carrier off and wait for link UP
* notification later to turn on carrier . We don ' t need
* to wait here for the vnic - to - switch link initialization
* to complete ; link UP notification is the indication that
* the process is complete .
*/
netif_carrier_off ( netdev ) ;
2010-06-24 14:50:56 +04:00
/* Do not call dev_init for a dynamic vnic.
* For a dynamic vnic , init_prov_info will be
* called later by an upper layer .
*/
2010-05-18 09:50:19 +04:00
if ( ! enic_is_dynamic ( enic ) ) {
err = vnic_dev_init ( enic - > vdev , 0 ) ;
if ( err ) {
2010-06-24 14:50:56 +04:00
dev_err ( dev , " vNIC dev init failed, aborting \n " ) ;
2010-05-18 09:50:19 +04:00
goto err_out_dev_close ;
}
2008-09-15 20:17:11 +04:00
}
2010-06-24 14:50:12 +04:00
/* Setup devcmd lock
*/
spin_lock_init ( & enic - > devcmd_lock ) ;
2009-09-03 21:02:45 +04:00
err = enic_dev_init ( enic ) ;
2008-09-15 20:17:11 +04:00
if ( err ) {
2010-06-24 14:50:56 +04:00
dev_err ( dev , " Device initialization failed, aborting \n " ) ;
2008-09-15 20:17:11 +04:00
goto err_out_dev_close ;
}
2010-06-24 14:50:12 +04:00
/* Setup notification timer, HW reset task, and wq locks
2008-09-15 20:17:11 +04:00
*/
init_timer ( & enic - > notify_timer ) ;
enic - > notify_timer . function = enic_notify_timer ;
enic - > notify_timer . data = ( unsigned long ) enic ;
INIT_WORK ( & enic - > reset , enic_reset ) ;
for ( i = 0 ; i < enic - > wq_count ; i + + )
spin_lock_init ( & enic - > wq_lock [ i ] ) ;
/* Register net device
*/
enic - > port_mtu = enic - > config . mtu ;
( void ) enic_change_mtu ( netdev , enic - > port_mtu ) ;
err = enic_set_mac_addr ( netdev , enic - > mac_addr ) ;
if ( err ) {
2010-06-24 14:50:56 +04:00
dev_err ( dev , " Invalid MAC address, aborting \n " ) ;
2009-09-03 21:02:45 +04:00
goto err_out_dev_deinit ;
2008-09-15 20:17:11 +04:00
}
2009-12-23 16:27:54 +03:00
enic - > tx_coalesce_usecs = enic - > config . intr_timer_usec ;
enic - > rx_coalesce_usecs = enic - > tx_coalesce_usecs ;
2010-05-18 09:50:19 +04:00
if ( enic_is_dynamic ( enic ) )
netdev - > netdev_ops = & enic_netdev_dynamic_ops ;
else
netdev - > netdev_ops = & enic_netdev_ops ;
2008-09-15 20:17:11 +04:00
netdev - > watchdog_timeo = 2 * HZ ;
netdev - > ethtool_ops = & enic_ethtool_ops ;
2010-03-18 19:19:54 +03:00
netdev - > features | = NETIF_F_HW_VLAN_TX | NETIF_F_HW_VLAN_RX ;
2010-06-24 14:51:59 +04:00
if ( ENIC_SETTING ( enic , LOOP ) ) {
netdev - > features & = ~ NETIF_F_HW_VLAN_TX ;
enic - > loop_enable = 1 ;
enic - > loop_tag = enic - > config . loop_tag ;
dev_info ( dev , " loopback tag=0x%04x \n " , enic - > loop_tag ) ;
}
2008-09-15 20:17:11 +04:00
if ( ENIC_SETTING ( enic , TXCSUM ) )
netdev - > features | = NETIF_F_SG | NETIF_F_HW_CSUM ;
if ( ENIC_SETTING ( enic , TSO ) )
netdev - > features | = NETIF_F_TSO |
NETIF_F_TSO6 | NETIF_F_TSO_ECN ;
2008-11-22 08:26:55 +03:00
if ( ENIC_SETTING ( enic , LRO ) )
2010-06-24 14:49:25 +04:00
netdev - > features | = NETIF_F_GRO ;
2008-09-15 20:17:11 +04:00
if ( using_dac )
netdev - > features | = NETIF_F_HIGHDMA ;
enic - > csum_rx_enabled = ENIC_SETTING ( enic , RXCSUM ) ;
err = register_netdev ( netdev ) ;
if ( err ) {
2010-06-24 14:50:56 +04:00
dev_err ( dev , " Cannot register net device, aborting \n " ) ;
2009-09-03 21:02:45 +04:00
goto err_out_dev_deinit ;
2008-09-15 20:17:11 +04:00
}
return 0 ;
2009-09-03 21:02:45 +04:00
err_out_dev_deinit :
enic_dev_deinit ( enic ) ;
2008-09-15 20:17:11 +04:00
err_out_dev_close :
vnic_dev_close ( enic - > vdev ) ;
err_out_vnic_unregister :
vnic_dev_unregister ( enic - > vdev ) ;
err_out_iounmap :
enic_iounmap ( enic ) ;
err_out_release_regions :
pci_release_regions ( pdev ) ;
err_out_disable_device :
pci_disable_device ( pdev ) ;
err_out_free_netdev :
pci_set_drvdata ( pdev , NULL ) ;
free_netdev ( netdev ) ;
return err ;
}
static void __devexit enic_remove ( struct pci_dev * pdev )
{
struct net_device * netdev = pci_get_drvdata ( pdev ) ;
if ( netdev ) {
struct enic * enic = netdev_priv ( netdev ) ;
flush_scheduled_work ( ) ;
unregister_netdev ( netdev ) ;
2009-09-03 21:02:45 +04:00
enic_dev_deinit ( enic ) ;
2008-09-15 20:17:11 +04:00
vnic_dev_close ( enic - > vdev ) ;
vnic_dev_unregister ( enic - > vdev ) ;
enic_iounmap ( enic ) ;
pci_release_regions ( pdev ) ;
pci_disable_device ( pdev ) ;
pci_set_drvdata ( pdev , NULL ) ;
free_netdev ( netdev ) ;
}
}
static struct pci_driver enic_driver = {
. name = DRV_NAME ,
. id_table = enic_id_table ,
. probe = enic_probe ,
. remove = __devexit_p ( enic_remove ) ,
} ;
static int __init enic_init_module ( void )
{
2010-06-24 14:50:56 +04:00
pr_info ( " %s, ver %s \n " , DRV_DESCRIPTION , DRV_VERSION ) ;
2008-09-15 20:17:11 +04:00
return pci_register_driver ( & enic_driver ) ;
}
static void __exit enic_cleanup_module ( void )
{
pci_unregister_driver ( & enic_driver ) ;
}
module_init ( enic_init_module ) ;
module_exit ( enic_cleanup_module ) ;