2006-10-21 23:33:03 +04:00
/*
2009-02-25 18:57:56 +03:00
* Copyright ( C ) 2003 - 2009 NetXen , Inc .
2006-10-21 23:33:03 +04:00
* All rights reserved .
2006-12-04 20:18:00 +03:00
*
2006-10-21 23:33:03 +04:00
* This program is free software ; you can redistribute it and / or
* modify it under the terms of the GNU General Public License
* as published by the Free Software Foundation ; either version 2
* of the License , or ( at your option ) any later version .
2006-12-04 20:18:00 +03:00
*
2006-10-21 23:33:03 +04:00
* This program is distributed in the hope that it will be useful , but
* WITHOUT ANY WARRANTY ; without even the implied warranty of
* MERCHANTABILITY or FITNESS FOR A PARTICULAR PURPOSE . See the
* GNU General Public License for more details .
2006-12-04 20:18:00 +03:00
*
2006-10-21 23:33:03 +04:00
* You should have received a copy of the GNU General Public License
* along with this program ; if not , write to the Free Software
* Foundation , Inc . , 59 Temple Place - Suite 330 , Boston ,
* MA 02111 - 1307 , USA .
2006-12-04 20:18:00 +03:00
*
2006-10-21 23:33:03 +04:00
* The full GNU General Public License is included in this distribution
* in the file called LICENSE .
2006-12-04 20:18:00 +03:00
*
2006-10-21 23:33:03 +04:00
* Contact Information :
* info @ netxen . com
2009-02-25 18:57:56 +03:00
* NetXen Inc ,
* 18922 Forge Drive
* Cupertino , CA 95014 - 0701
2006-10-21 23:33:03 +04:00
*
*/
# include <linux/netdevice.h>
# include <linux/delay.h>
# include "netxen_nic.h"
# include "netxen_nic_hw.h"
# include "netxen_nic_phan_reg.h"
struct crb_addr_pair {
2007-02-28 16:16:40 +03:00
u32 addr ;
u32 data ;
2006-10-21 23:33:03 +04:00
} ;
# define NETXEN_MAX_CRB_XFORM 60
static unsigned int crb_addr_xform [ NETXEN_MAX_CRB_XFORM ] ;
2007-02-28 16:16:40 +03:00
# define NETXEN_ADDR_ERROR (0xffffffff)
2006-10-21 23:33:03 +04:00
# define crb_addr_transform(name) \
crb_addr_xform [ NETXEN_HW_PX_MAP_CRB_ # # name ] = \
NETXEN_HW_CRB_HUB_AGT_ADR_ # # name < < 20
2006-11-29 20:00:10 +03:00
# define NETXEN_NIC_XDMA_RESET 0x8000ff
2007-11-05 20:07:31 +03:00
static void netxen_post_rx_buffers_nodb ( struct netxen_adapter * adapter ,
uint32_t ctx , uint32_t ringid ) ;
2006-10-21 23:33:03 +04:00
static void crb_addr_transform_setup ( void )
{
crb_addr_transform ( XDMA ) ;
crb_addr_transform ( TIMR ) ;
crb_addr_transform ( SRE ) ;
crb_addr_transform ( SQN3 ) ;
crb_addr_transform ( SQN2 ) ;
crb_addr_transform ( SQN1 ) ;
crb_addr_transform ( SQN0 ) ;
crb_addr_transform ( SQS3 ) ;
crb_addr_transform ( SQS2 ) ;
crb_addr_transform ( SQS1 ) ;
crb_addr_transform ( SQS0 ) ;
crb_addr_transform ( RPMX7 ) ;
crb_addr_transform ( RPMX6 ) ;
crb_addr_transform ( RPMX5 ) ;
crb_addr_transform ( RPMX4 ) ;
crb_addr_transform ( RPMX3 ) ;
crb_addr_transform ( RPMX2 ) ;
crb_addr_transform ( RPMX1 ) ;
crb_addr_transform ( RPMX0 ) ;
crb_addr_transform ( ROMUSB ) ;
crb_addr_transform ( SN ) ;
crb_addr_transform ( QMN ) ;
crb_addr_transform ( QMS ) ;
crb_addr_transform ( PGNI ) ;
crb_addr_transform ( PGND ) ;
crb_addr_transform ( PGN3 ) ;
crb_addr_transform ( PGN2 ) ;
crb_addr_transform ( PGN1 ) ;
crb_addr_transform ( PGN0 ) ;
crb_addr_transform ( PGSI ) ;
crb_addr_transform ( PGSD ) ;
crb_addr_transform ( PGS3 ) ;
crb_addr_transform ( PGS2 ) ;
crb_addr_transform ( PGS1 ) ;
crb_addr_transform ( PGS0 ) ;
crb_addr_transform ( PS ) ;
crb_addr_transform ( PH ) ;
crb_addr_transform ( NIU ) ;
crb_addr_transform ( I2Q ) ;
crb_addr_transform ( EG ) ;
crb_addr_transform ( MN ) ;
crb_addr_transform ( MS ) ;
crb_addr_transform ( CAS2 ) ;
crb_addr_transform ( CAS1 ) ;
crb_addr_transform ( CAS0 ) ;
crb_addr_transform ( CAM ) ;
crb_addr_transform ( C2C1 ) ;
crb_addr_transform ( C2C0 ) ;
2007-02-05 18:35:26 +03:00
crb_addr_transform ( SMB ) ;
2008-07-22 06:44:02 +04:00
crb_addr_transform ( OCM0 ) ;
crb_addr_transform ( I2C0 ) ;
2006-10-21 23:33:03 +04:00
}
int netxen_init_firmware ( struct netxen_adapter * adapter )
{
u32 state = 0 , loops = 0 , err = 0 ;
/* Window 1 call */
2008-07-22 06:44:03 +04:00
state = adapter - > pci_read_normalize ( adapter , CRB_CMDPEG_STATE ) ;
2006-10-21 23:33:03 +04:00
if ( state = = PHAN_INITIALIZE_ACK )
return 0 ;
while ( state ! = PHAN_INITIALIZE_COMPLETE & & loops < 2000 ) {
2008-07-22 06:44:04 +04:00
msleep ( 1 ) ;
2006-10-21 23:33:03 +04:00
/* Window 1 call */
2008-07-22 06:44:03 +04:00
state = adapter - > pci_read_normalize ( adapter , CRB_CMDPEG_STATE ) ;
2006-10-21 23:33:03 +04:00
loops + + ;
}
if ( loops > = 2000 ) {
printk ( KERN_ERR " Cmd Peg initialization not complete:%x. \n " ,
state ) ;
err = - EIO ;
return err ;
}
/* Window 1 call */
2008-07-22 06:44:03 +04:00
adapter - > pci_write_normalize ( adapter ,
CRB_NIC_CAPABILITIES_HOST , INTR_SCHEME_PERPORT ) ;
adapter - > pci_write_normalize ( adapter ,
CRB_NIC_MSI_MODE_HOST , MSI_MODE_MULTIFUNC ) ;
adapter - > pci_write_normalize ( adapter ,
CRB_MPORT_MODE , MPORT_MULTI_FUNCTION_MODE ) ;
adapter - > pci_write_normalize ( adapter ,
CRB_CMDPEG_STATE , PHAN_INITIALIZE_ACK ) ;
2006-10-21 23:33:03 +04:00
return err ;
}
2008-07-22 06:44:04 +04:00
void netxen_release_rx_buffers ( struct netxen_adapter * adapter )
2006-10-21 23:33:03 +04:00
{
2008-07-22 06:44:04 +04:00
struct netxen_recv_context * recv_ctx ;
2008-07-22 06:44:06 +04:00
struct nx_host_rds_ring * rds_ring ;
2008-07-22 06:44:04 +04:00
struct netxen_rx_buffer * rx_buf ;
int i , ctxid , ring ;
2006-10-21 23:33:03 +04:00
for ( ctxid = 0 ; ctxid < MAX_RCV_CTX ; + + ctxid ) {
2008-07-22 06:44:04 +04:00
recv_ctx = & adapter - > recv_ctx [ ctxid ] ;
2008-07-22 06:44:06 +04:00
for ( ring = 0 ; ring < adapter - > max_rds_rings ; ring + + ) {
rds_ring = & recv_ctx - > rds_rings [ ring ] ;
for ( i = 0 ; i < rds_ring - > max_rx_desc_count ; + + i ) {
rx_buf = & ( rds_ring - > rx_buf_arr [ i ] ) ;
2008-07-22 06:44:04 +04:00
if ( rx_buf - > state = = NETXEN_BUFFER_FREE )
continue ;
pci_unmap_single ( adapter - > pdev ,
rx_buf - > dma ,
2008-07-22 06:44:06 +04:00
rds_ring - > dma_size ,
2008-07-22 06:44:04 +04:00
PCI_DMA_FROMDEVICE ) ;
if ( rx_buf - > skb ! = NULL )
dev_kfree_skb_any ( rx_buf - > skb ) ;
}
}
}
}
void netxen_release_tx_buffers ( struct netxen_adapter * adapter )
{
struct netxen_cmd_buffer * cmd_buf ;
struct netxen_skb_frag * buffrag ;
int i , j ;
cmd_buf = adapter - > cmd_buf_arr ;
for ( i = 0 ; i < adapter - > max_tx_desc_count ; i + + ) {
buffrag = cmd_buf - > frag_array ;
if ( buffrag - > dma ) {
pci_unmap_single ( adapter - > pdev , buffrag - > dma ,
buffrag - > length , PCI_DMA_TODEVICE ) ;
buffrag - > dma = 0ULL ;
}
for ( j = 0 ; j < cmd_buf - > frag_count ; j + + ) {
buffrag + + ;
if ( buffrag - > dma ) {
pci_unmap_page ( adapter - > pdev , buffrag - > dma ,
buffrag - > length ,
PCI_DMA_TODEVICE ) ;
buffrag - > dma = 0ULL ;
}
}
/* Free the skb we received in netxen_nic_xmit_frame */
if ( cmd_buf - > skb ) {
dev_kfree_skb_any ( cmd_buf - > skb ) ;
cmd_buf - > skb = NULL ;
}
cmd_buf + + ;
}
}
void netxen_free_sw_resources ( struct netxen_adapter * adapter )
{
struct netxen_recv_context * recv_ctx ;
2008-07-22 06:44:06 +04:00
struct nx_host_rds_ring * rds_ring ;
2008-07-22 06:44:04 +04:00
int ctx , ring ;
for ( ctx = 0 ; ctx < MAX_RCV_CTX ; ctx + + ) {
recv_ctx = & adapter - > recv_ctx [ ctx ] ;
2008-07-22 06:44:06 +04:00
for ( ring = 0 ; ring < adapter - > max_rds_rings ; ring + + ) {
rds_ring = & recv_ctx - > rds_rings [ ring ] ;
if ( rds_ring - > rx_buf_arr ) {
vfree ( rds_ring - > rx_buf_arr ) ;
rds_ring - > rx_buf_arr = NULL ;
2008-07-22 06:44:04 +04:00
}
}
}
if ( adapter - > cmd_buf_arr )
vfree ( adapter - > cmd_buf_arr ) ;
return ;
}
int netxen_alloc_sw_resources ( struct netxen_adapter * adapter )
{
struct netxen_recv_context * recv_ctx ;
2008-07-22 06:44:06 +04:00
struct nx_host_rds_ring * rds_ring ;
2008-07-22 06:44:04 +04:00
struct netxen_rx_buffer * rx_buf ;
int ctx , ring , i , num_rx_bufs ;
struct netxen_cmd_buffer * cmd_buf_arr ;
struct net_device * netdev = adapter - > netdev ;
cmd_buf_arr = ( struct netxen_cmd_buffer * ) vmalloc ( TX_RINGSIZE ) ;
if ( cmd_buf_arr = = NULL ) {
printk ( KERN_ERR " %s: Failed to allocate cmd buffer ring \n " ,
netdev - > name ) ;
return - ENOMEM ;
}
memset ( cmd_buf_arr , 0 , TX_RINGSIZE ) ;
adapter - > cmd_buf_arr = cmd_buf_arr ;
for ( ctx = 0 ; ctx < MAX_RCV_CTX ; ctx + + ) {
recv_ctx = & adapter - > recv_ctx [ ctx ] ;
2008-07-22 06:44:06 +04:00
for ( ring = 0 ; ring < adapter - > max_rds_rings ; ring + + ) {
rds_ring = & recv_ctx - > rds_rings [ ring ] ;
2008-07-22 06:44:04 +04:00
switch ( RCV_DESC_TYPE ( ring ) ) {
case RCV_DESC_NORMAL :
2008-07-22 06:44:06 +04:00
rds_ring - > max_rx_desc_count =
2008-07-22 06:44:04 +04:00
adapter - > max_rx_desc_count ;
2008-07-22 06:44:06 +04:00
rds_ring - > flags = RCV_DESC_NORMAL ;
2008-07-22 06:44:08 +04:00
if ( adapter - > ahw . cut_through ) {
rds_ring - > dma_size =
NX_CT_DEFAULT_RX_BUF_LEN ;
rds_ring - > skb_size =
NX_CT_DEFAULT_RX_BUF_LEN ;
} else {
rds_ring - > dma_size = RX_DMA_MAP_LEN ;
rds_ring - > skb_size =
MAX_RX_BUFFER_LENGTH ;
}
2008-07-22 06:44:04 +04:00
break ;
case RCV_DESC_JUMBO :
2008-07-22 06:44:06 +04:00
rds_ring - > max_rx_desc_count =
2008-07-22 06:44:04 +04:00
adapter - > max_jumbo_rx_desc_count ;
2008-07-22 06:44:06 +04:00
rds_ring - > flags = RCV_DESC_JUMBO ;
2008-07-22 06:44:08 +04:00
if ( NX_IS_REVISION_P3 ( adapter - > ahw . revision_id ) )
rds_ring - > dma_size =
NX_P3_RX_JUMBO_BUF_MAX_LEN ;
else
rds_ring - > dma_size =
NX_P2_RX_JUMBO_BUF_MAX_LEN ;
2008-07-22 06:44:06 +04:00
rds_ring - > skb_size =
2008-07-22 06:44:08 +04:00
rds_ring - > dma_size + NET_IP_ALIGN ;
2008-07-22 06:44:04 +04:00
break ;
case RCV_RING_LRO :
2008-07-22 06:44:06 +04:00
rds_ring - > max_rx_desc_count =
2008-07-22 06:44:04 +04:00
adapter - > max_lro_rx_desc_count ;
2008-07-22 06:44:06 +04:00
rds_ring - > flags = RCV_DESC_LRO ;
rds_ring - > dma_size = RX_LRO_DMA_MAP_LEN ;
rds_ring - > skb_size = MAX_RX_LRO_BUFFER_LENGTH ;
2008-07-22 06:44:04 +04:00
break ;
}
2008-07-22 06:44:06 +04:00
rds_ring - > rx_buf_arr = ( struct netxen_rx_buffer * )
2008-07-22 06:44:04 +04:00
vmalloc ( RCV_BUFFSIZE ) ;
2008-07-22 06:44:06 +04:00
if ( rds_ring - > rx_buf_arr = = NULL ) {
2008-07-22 06:44:04 +04:00
printk ( KERN_ERR " %s: Failed to allocate "
" rx buffer ring %d \n " ,
netdev - > name , ring ) ;
/* free whatever was already allocated */
goto err_out ;
}
2008-07-22 06:44:06 +04:00
memset ( rds_ring - > rx_buf_arr , 0 , RCV_BUFFSIZE ) ;
2008-07-22 06:44:08 +04:00
INIT_LIST_HEAD ( & rds_ring - > free_list ) ;
2006-10-21 23:33:03 +04:00
/*
* Now go through all of them , set reference handles
* and put them in the queues .
*/
2008-07-22 06:44:06 +04:00
num_rx_bufs = rds_ring - > max_rx_desc_count ;
rx_buf = rds_ring - > rx_buf_arr ;
2006-10-21 23:33:03 +04:00
for ( i = 0 ; i < num_rx_bufs ; i + + ) {
2008-07-22 06:44:08 +04:00
list_add_tail ( & rx_buf - > list ,
& rds_ring - > free_list ) ;
2006-10-21 23:33:03 +04:00
rx_buf - > ref_handle = i ;
rx_buf - > state = NETXEN_BUFFER_FREE ;
rx_buf + + ;
}
}
}
2008-07-22 06:44:04 +04:00
return 0 ;
err_out :
netxen_free_sw_resources ( adapter ) ;
return - ENOMEM ;
2006-10-21 23:33:03 +04:00
}
void netxen_initialize_adapter_ops ( struct netxen_adapter * adapter )
{
switch ( adapter - > ahw . board_type ) {
case NETXEN_NIC_GBE :
2006-12-04 20:18:00 +03:00
adapter - > enable_phy_interrupts =
2006-10-21 23:33:03 +04:00
netxen_niu_gbe_enable_phy_interrupts ;
2006-12-04 20:18:00 +03:00
adapter - > disable_phy_interrupts =
2006-10-21 23:33:03 +04:00
netxen_niu_gbe_disable_phy_interrupts ;
2006-12-04 20:18:00 +03:00
adapter - > macaddr_set = netxen_niu_macaddr_set ;
adapter - > set_mtu = netxen_nic_set_mtu_gb ;
adapter - > set_promisc = netxen_niu_set_promiscuous_mode ;
adapter - > phy_read = netxen_niu_gbe_phy_read ;
adapter - > phy_write = netxen_niu_gbe_phy_write ;
2008-07-22 06:44:07 +04:00
adapter - > init_port = netxen_niu_gbe_init_port ;
2006-12-04 20:18:00 +03:00
adapter - > stop_port = netxen_niu_disable_gbe_port ;
2006-10-21 23:33:03 +04:00
break ;
case NETXEN_NIC_XGBE :
2006-12-04 20:18:00 +03:00
adapter - > enable_phy_interrupts =
2006-10-21 23:33:03 +04:00
netxen_niu_xgbe_enable_phy_interrupts ;
2006-12-04 20:18:00 +03:00
adapter - > disable_phy_interrupts =
2006-10-21 23:33:03 +04:00
netxen_niu_xgbe_disable_phy_interrupts ;
2006-12-04 20:18:00 +03:00
adapter - > macaddr_set = netxen_niu_xg_macaddr_set ;
adapter - > set_mtu = netxen_nic_set_mtu_xgb ;
adapter - > init_port = netxen_niu_xg_init_port ;
adapter - > set_promisc = netxen_niu_xg_set_promiscuous_mode ;
adapter - > stop_port = netxen_niu_disable_xg_port ;
2006-10-21 23:33:03 +04:00
break ;
default :
break ;
}
2008-08-01 14:14:59 +04:00
if ( NX_IS_REVISION_P3 ( adapter - > ahw . revision_id ) ) {
adapter - > set_mtu = nx_fw_cmd_set_mtu ;
adapter - > set_promisc = netxen_p3_nic_set_promisc ;
}
2006-10-21 23:33:03 +04:00
}
/*
* netxen_decode_crb_addr ( 0 - utility to translate from internal Phantom CRB
* address to external PCI CRB address .
*/
2007-11-05 20:07:31 +03:00
static u32 netxen_decode_crb_addr ( u32 addr )
2006-10-21 23:33:03 +04:00
{
int i ;
2007-02-28 16:16:40 +03:00
u32 base_addr , offset , pci_base ;
2006-10-21 23:33:03 +04:00
crb_addr_transform_setup ( ) ;
pci_base = NETXEN_ADDR_ERROR ;
base_addr = addr & 0xfff00000 ;
offset = addr & 0x000fffff ;
for ( i = 0 ; i < NETXEN_MAX_CRB_XFORM ; i + + ) {
if ( crb_addr_xform [ i ] = = base_addr ) {
pci_base = i < < 20 ;
break ;
}
}
if ( pci_base = = NETXEN_ADDR_ERROR )
return pci_base ;
else
return ( pci_base + offset ) ;
}
2007-04-20 18:53:05 +04:00
static long rom_max_timeout = 100 ;
static long rom_lock_timeout = 10000 ;
2006-10-21 23:33:03 +04:00
2007-11-05 20:07:31 +03:00
static int rom_lock ( struct netxen_adapter * adapter )
2006-10-21 23:33:03 +04:00
{
int iter ;
u32 done = 0 ;
int timeout = 0 ;
while ( ! done ) {
/* acquire semaphore2 from PCI HW block */
netxen_nic_read_w0 ( adapter , NETXEN_PCIE_REG ( PCIE_SEM2_LOCK ) ,
& done ) ;
if ( done = = 1 )
break ;
if ( timeout > = rom_lock_timeout )
return - EIO ;
timeout + + ;
/*
* Yield CPU
*/
if ( ! in_atomic ( ) )
schedule ( ) ;
else {
for ( iter = 0 ; iter < 20 ; iter + + )
cpu_relax ( ) ; /*This a nop instr on i386 */
}
}
netxen_nic_reg_write ( adapter , NETXEN_ROM_LOCK_ID , ROM_LOCK_DRIVER ) ;
return 0 ;
}
2007-11-05 20:07:31 +03:00
static int netxen_wait_rom_done ( struct netxen_adapter * adapter )
2006-10-21 23:33:03 +04:00
{
long timeout = 0 ;
long done = 0 ;
2009-01-15 07:49:00 +03:00
cond_resched ( ) ;
2006-10-21 23:33:03 +04:00
while ( done = = 0 ) {
done = netxen_nic_reg_read ( adapter , NETXEN_ROMUSB_GLB_STATUS ) ;
done & = 2 ;
timeout + + ;
if ( timeout > = rom_max_timeout ) {
printk ( " Timeout reached waiting for rom done " ) ;
return - EIO ;
}
}
return 0 ;
}
2007-11-05 20:07:31 +03:00
static void netxen_rom_unlock ( struct netxen_adapter * adapter )
2006-11-29 20:00:10 +03:00
{
u32 val ;
/* release semaphore2 */
netxen_nic_read_w0 ( adapter , NETXEN_PCIE_REG ( PCIE_SEM2_UNLOCK ) , & val ) ;
}
2007-11-05 20:07:31 +03:00
static int do_rom_fast_read ( struct netxen_adapter * adapter ,
int addr , int * valp )
2006-10-21 23:33:03 +04:00
{
netxen_nic_reg_write ( adapter , NETXEN_ROMUSB_ROM_ADDRESS , addr ) ;
netxen_nic_reg_write ( adapter , NETXEN_ROMUSB_ROM_DUMMY_BYTE_CNT , 0 ) ;
2009-01-15 07:49:00 +03:00
netxen_nic_reg_write ( adapter , NETXEN_ROMUSB_ROM_ABYTE_CNT , 3 ) ;
2006-10-21 23:33:03 +04:00
netxen_nic_reg_write ( adapter , NETXEN_ROMUSB_ROM_INSTR_OPCODE , 0xb ) ;
if ( netxen_wait_rom_done ( adapter ) ) {
printk ( " Error waiting for rom done \n " ) ;
return - EIO ;
}
/* reset abyte_cnt and dummy_byte_cnt */
netxen_nic_reg_write ( adapter , NETXEN_ROMUSB_ROM_ABYTE_CNT , 0 ) ;
2009-01-15 07:49:00 +03:00
udelay ( 10 ) ;
2006-10-21 23:33:03 +04:00
netxen_nic_reg_write ( adapter , NETXEN_ROMUSB_ROM_DUMMY_BYTE_CNT , 0 ) ;
* valp = netxen_nic_reg_read ( adapter , NETXEN_ROMUSB_ROM_RDATA ) ;
return 0 ;
}
2007-11-05 20:07:31 +03:00
static int do_rom_fast_read_words ( struct netxen_adapter * adapter , int addr ,
u8 * bytes , size_t size )
2007-02-05 18:40:49 +03:00
{
int addridx ;
int ret = 0 ;
for ( addridx = addr ; addridx < ( addr + size ) ; addridx + = 4 ) {
2007-12-22 22:44:00 +03:00
int v ;
ret = do_rom_fast_read ( adapter , addridx , & v ) ;
2007-02-05 18:40:49 +03:00
if ( ret ! = 0 )
break ;
2007-12-22 22:44:00 +03:00
* ( __le32 * ) bytes = cpu_to_le32 ( v ) ;
2007-02-05 18:40:49 +03:00
bytes + = 4 ;
}
return ret ;
}
int
2007-11-24 05:23:36 +03:00
netxen_rom_fast_read_words ( struct netxen_adapter * adapter , int addr ,
2007-02-05 18:40:49 +03:00
u8 * bytes , size_t size )
{
int ret ;
ret = rom_lock ( adapter ) ;
if ( ret < 0 )
return ret ;
ret = do_rom_fast_read_words ( adapter , addr , bytes , size ) ;
netxen_rom_unlock ( adapter ) ;
return ret ;
}
2006-10-21 23:33:03 +04:00
int netxen_rom_fast_read ( struct netxen_adapter * adapter , int addr , int * valp )
{
int ret ;
if ( rom_lock ( adapter ) ! = 0 )
return - EIO ;
ret = do_rom_fast_read ( adapter , addr , valp ) ;
2006-11-29 20:00:10 +03:00
netxen_rom_unlock ( adapter ) ;
return ret ;
}
2006-10-21 23:33:03 +04:00
# define NETXEN_BOARDTYPE 0x4008
# define NETXEN_BOARDNUM 0x400c
# define NETXEN_CHIPNUM 0x4010
int netxen_pinit_from_rom ( struct netxen_adapter * adapter , int verbose )
{
2008-06-16 09:59:45 +04:00
int addr , val ;
2009-01-15 07:49:00 +03:00
int i , n , init_delay = 0 ;
2006-10-21 23:33:03 +04:00
struct crb_addr_pair * buf ;
2009-01-15 07:49:00 +03:00
unsigned offset ;
2007-02-28 16:16:40 +03:00
u32 off ;
2006-10-21 23:33:03 +04:00
/* resetall */
2009-01-15 07:49:00 +03:00
rom_lock ( adapter ) ;
2006-10-21 23:33:03 +04:00
netxen_crb_writelit_adapter ( adapter , NETXEN_ROMUSB_GLB_SW_RESET ,
2008-07-22 06:44:04 +04:00
0xffffffff ) ;
2009-01-15 07:49:00 +03:00
netxen_rom_unlock ( adapter ) ;
2006-10-21 23:33:03 +04:00
if ( verbose ) {
if ( netxen_rom_fast_read ( adapter , NETXEN_BOARDTYPE , & val ) = = 0 )
printk ( " P2 ROM board type: 0x%08x \n " , val ) ;
else
printk ( " Could not read board type \n " ) ;
if ( netxen_rom_fast_read ( adapter , NETXEN_BOARDNUM , & val ) = = 0 )
printk ( " P2 ROM board num: 0x%08x \n " , val ) ;
else
printk ( " Could not read board number \n " ) ;
if ( netxen_rom_fast_read ( adapter , NETXEN_CHIPNUM , & val ) = = 0 )
printk ( " P2 ROM chip num: 0x%08x \n " , val ) ;
else
printk ( " Could not read chip number \n " ) ;
}
2008-07-22 06:44:04 +04:00
if ( NX_IS_REVISION_P3 ( adapter - > ahw . revision_id ) ) {
if ( netxen_rom_fast_read ( adapter , 0 , & n ) ! = 0 | |
2009-01-15 07:49:00 +03:00
( n ! = 0xcafecafe ) | |
2008-07-22 06:44:04 +04:00
netxen_rom_fast_read ( adapter , 4 , & n ) ! = 0 ) {
printk ( KERN_ERR " %s: ERROR Reading crb_init area: "
" n: %08x \n " , netxen_nic_driver_name , n ) ;
2006-10-21 23:33:03 +04:00
return - EIO ;
}
2008-07-22 06:44:04 +04:00
offset = n & 0xffffU ;
n = ( n > > 16 ) & 0xffffU ;
} else {
if ( netxen_rom_fast_read ( adapter , 0 , & n ) ! = 0 | |
! ( n & 0x80000000 ) ) {
printk ( KERN_ERR " %s: ERROR Reading crb_init area: "
" n: %08x \n " , netxen_nic_driver_name , n ) ;
return - EIO ;
2006-10-21 23:33:03 +04:00
}
2008-07-22 06:44:04 +04:00
offset = 1 ;
n & = ~ 0x80000000 ;
}
if ( n < 1024 ) {
if ( verbose )
printk ( KERN_DEBUG " %s: %d CRB init values found "
" in ROM. \n " , netxen_nic_driver_name , n ) ;
} else {
printk ( KERN_ERR " %s:n=0x%x Error! NetXen card flash not "
" initialized. \n " , __func__ , n ) ;
return - EIO ;
}
2006-10-21 23:33:03 +04:00
2008-07-22 06:44:04 +04:00
buf = kcalloc ( n , sizeof ( struct crb_addr_pair ) , GFP_KERNEL ) ;
if ( buf = = NULL ) {
printk ( " %s: netxen_pinit_from_rom: Unable to calloc memory. \n " ,
netxen_nic_driver_name ) ;
return - ENOMEM ;
}
for ( i = 0 ; i < n ; i + + ) {
if ( netxen_rom_fast_read ( adapter , 8 * i + 4 * offset , & val ) ! = 0 | |
2009-01-29 11:55:56 +03:00
netxen_rom_fast_read ( adapter , 8 * i + 4 * offset + 4 , & addr ) ! = 0 ) {
kfree ( buf ) ;
2008-07-22 06:44:04 +04:00
return - EIO ;
2009-01-29 11:55:56 +03:00
}
2008-07-22 06:44:04 +04:00
buf [ i ] . addr = addr ;
buf [ i ] . data = val ;
if ( verbose )
printk ( KERN_DEBUG " %s: PCI: 0x%08x == 0x%08x \n " ,
netxen_nic_driver_name ,
( u32 ) netxen_decode_crb_addr ( addr ) , val ) ;
}
for ( i = 0 ; i < n ; i + + ) {
off = netxen_decode_crb_addr ( buf [ i ] . addr ) ;
if ( off = = NETXEN_ADDR_ERROR ) {
printk ( KERN_ERR " CRB init value out of range %x \n " ,
2007-02-05 18:35:26 +03:00
buf [ i ] . addr ) ;
2008-07-22 06:44:04 +04:00
continue ;
}
off + = NETXEN_PCI_CRBSPACE ;
/* skipping cold reboot MAGIC */
if ( off = = NETXEN_CAM_RAM ( 0x1fc ) )
continue ;
if ( NX_IS_REVISION_P3 ( adapter - > ahw . revision_id ) ) {
/* do not reset PCI */
if ( off = = ( ROMUSB_GLB + 0xbc ) )
2007-02-05 18:35:26 +03:00
continue ;
2009-01-15 07:49:00 +03:00
if ( off = = ( ROMUSB_GLB + 0xa8 ) )
continue ;
if ( off = = ( ROMUSB_GLB + 0xc8 ) ) /* core clock */
continue ;
if ( off = = ( ROMUSB_GLB + 0x24 ) ) /* MN clock */
continue ;
if ( off = = ( ROMUSB_GLB + 0x1c ) ) /* MS clock */
continue ;
2008-07-22 06:44:04 +04:00
if ( off = = ( NETXEN_CRB_PEG_NET_1 + 0x18 ) )
buf [ i ] . data = 0x1020 ;
/* skip the function enable register */
if ( off = = NETXEN_PCIE_REG ( PCIE_SETUP_FUNCTION ) )
2006-10-21 23:33:03 +04:00
continue ;
2008-07-22 06:44:04 +04:00
if ( off = = NETXEN_PCIE_REG ( PCIE_SETUP_FUNCTION2 ) )
continue ;
if ( ( off & 0x0ff00000 ) = = NETXEN_CRB_SMB )
continue ;
}
2006-10-21 23:33:03 +04:00
2008-07-22 06:44:04 +04:00
if ( off = = NETXEN_ADDR_ERROR ) {
printk ( KERN_ERR " %s: Err: Unknown addr: 0x%08x \n " ,
netxen_nic_driver_name , buf [ i ] . addr ) ;
continue ;
}
2009-01-15 07:49:00 +03:00
init_delay = 1 ;
2008-07-22 06:44:04 +04:00
/* After writing this register, HW needs time for CRB */
/* to quiet down (else crb_window returns 0xffffffff) */
if ( off = = NETXEN_ROMUSB_GLB_SW_RESET ) {
2009-01-15 07:49:00 +03:00
init_delay = 1000 ;
2008-07-22 06:44:04 +04:00
if ( NX_IS_REVISION_P2 ( adapter - > ahw . revision_id ) ) {
2006-10-21 23:33:03 +04:00
/* hold xdma in reset also */
2006-11-29 20:00:10 +03:00
buf [ i ] . data = NETXEN_NIC_XDMA_RESET ;
2009-01-15 07:49:00 +03:00
buf [ i ] . data = 0x8000ff ;
2006-10-21 23:33:03 +04:00
}
2008-07-22 06:44:04 +04:00
}
2006-10-21 23:33:03 +04:00
2008-07-22 06:44:04 +04:00
adapter - > hw_write_wx ( adapter , off , & buf [ i ] . data , 4 ) ;
2006-10-21 23:33:03 +04:00
2009-01-15 07:49:00 +03:00
msleep ( init_delay ) ;
2008-07-22 06:44:04 +04:00
}
kfree ( buf ) ;
2006-10-21 23:33:03 +04:00
2008-07-22 06:44:04 +04:00
/* disable_peg_cache_all */
2006-10-21 23:33:03 +04:00
2008-07-22 06:44:04 +04:00
/* unreset_net_cache */
if ( NX_IS_REVISION_P2 ( adapter - > ahw . revision_id ) ) {
adapter - > hw_read_wx ( adapter ,
NETXEN_ROMUSB_GLB_SW_RESET , & val , 4 ) ;
2006-10-21 23:33:03 +04:00
netxen_crb_writelit_adapter ( adapter ,
2008-07-22 06:44:04 +04:00
NETXEN_ROMUSB_GLB_SW_RESET , ( val & 0xffffff0f ) ) ;
2006-10-21 23:33:03 +04:00
}
2008-07-22 06:44:04 +04:00
/* p2dn replyCount */
netxen_crb_writelit_adapter ( adapter , NETXEN_CRB_PEG_NET_D + 0xec , 0x1e ) ;
/* disable_peg_cache 0 */
netxen_crb_writelit_adapter ( adapter , NETXEN_CRB_PEG_NET_D + 0x4c , 8 ) ;
/* disable_peg_cache 1 */
netxen_crb_writelit_adapter ( adapter , NETXEN_CRB_PEG_NET_I + 0x4c , 8 ) ;
/* peg_clr_all */
/* peg_clr 0 */
netxen_crb_writelit_adapter ( adapter , NETXEN_CRB_PEG_NET_0 + 0x8 , 0 ) ;
netxen_crb_writelit_adapter ( adapter , NETXEN_CRB_PEG_NET_0 + 0xc , 0 ) ;
/* peg_clr 1 */
netxen_crb_writelit_adapter ( adapter , NETXEN_CRB_PEG_NET_1 + 0x8 , 0 ) ;
netxen_crb_writelit_adapter ( adapter , NETXEN_CRB_PEG_NET_1 + 0xc , 0 ) ;
/* peg_clr 2 */
netxen_crb_writelit_adapter ( adapter , NETXEN_CRB_PEG_NET_2 + 0x8 , 0 ) ;
netxen_crb_writelit_adapter ( adapter , NETXEN_CRB_PEG_NET_2 + 0xc , 0 ) ;
/* peg_clr 3 */
netxen_crb_writelit_adapter ( adapter , NETXEN_CRB_PEG_NET_3 + 0x8 , 0 ) ;
netxen_crb_writelit_adapter ( adapter , NETXEN_CRB_PEG_NET_3 + 0xc , 0 ) ;
2006-10-21 23:33:03 +04:00
return 0 ;
}
2006-12-04 20:23:25 +03:00
int netxen_initialize_adapter_offload ( struct netxen_adapter * adapter )
{
uint64_t addr ;
uint32_t hi ;
uint32_t lo ;
adapter - > dummy_dma . addr =
2008-07-22 06:44:00 +04:00
pci_alloc_consistent ( adapter - > pdev ,
2006-12-04 20:23:25 +03:00
NETXEN_HOST_DUMMY_DMA_SIZE ,
& adapter - > dummy_dma . phys_addr ) ;
if ( adapter - > dummy_dma . addr = = NULL ) {
printk ( " %s: ERROR: Could not allocate dummy DMA memory \n " ,
2008-07-22 06:44:04 +04:00
__func__ ) ;
2006-12-04 20:23:25 +03:00
return - ENOMEM ;
}
addr = ( uint64_t ) adapter - > dummy_dma . phys_addr ;
hi = ( addr > > 32 ) & 0xffffffff ;
lo = addr & 0xffffffff ;
2008-07-22 06:44:03 +04:00
adapter - > pci_write_normalize ( adapter , CRB_HOST_DUMMY_BUF_ADDR_HI , hi ) ;
adapter - > pci_write_normalize ( adapter , CRB_HOST_DUMMY_BUF_ADDR_LO , lo ) ;
2006-12-04 20:23:25 +03:00
2008-07-22 06:44:04 +04:00
if ( NX_IS_REVISION_P3 ( adapter - > ahw . revision_id ) ) {
uint32_t temp = 0 ;
adapter - > hw_write_wx ( adapter , CRB_HOST_DUMMY_BUF , & temp , 4 ) ;
}
2006-12-04 20:23:25 +03:00
return 0 ;
}
void netxen_free_adapter_offload ( struct netxen_adapter * adapter )
{
2008-08-08 11:08:42 +04:00
int i = 100 ;
if ( ! adapter - > dummy_dma . addr )
return ;
2008-06-16 09:59:46 +04:00
2008-08-08 11:08:42 +04:00
if ( NX_IS_REVISION_P2 ( adapter - > ahw . revision_id ) ) {
2008-06-16 09:59:46 +04:00
do {
if ( dma_watchdog_shutdown_request ( adapter ) = = 1 )
break ;
msleep ( 50 ) ;
if ( dma_watchdog_shutdown_poll_result ( adapter ) = = 1 )
break ;
} while ( - - i ) ;
2008-08-08 11:08:42 +04:00
}
2008-06-16 09:59:46 +04:00
2008-08-08 11:08:42 +04:00
if ( i ) {
pci_free_consistent ( adapter - > pdev ,
NETXEN_HOST_DUMMY_DMA_SIZE ,
adapter - > dummy_dma . addr ,
adapter - > dummy_dma . phys_addr ) ;
adapter - > dummy_dma . addr = NULL ;
} else {
printk ( KERN_ERR " %s: dma_watchdog_shutdown failed \n " ,
adapter - > netdev - > name ) ;
2006-12-04 20:23:25 +03:00
}
}
2007-07-02 08:07:57 +04:00
int netxen_phantom_init ( struct netxen_adapter * adapter , int pegtune_val )
2006-10-21 23:33:03 +04:00
{
u32 val = 0 ;
2008-07-22 06:44:04 +04:00
int retries = 60 ;
2006-10-21 23:33:03 +04:00
2006-11-29 20:00:10 +03:00
if ( ! pegtune_val ) {
2007-07-02 08:07:57 +04:00
do {
2008-07-22 06:44:03 +04:00
val = adapter - > pci_read_normalize ( adapter ,
CRB_CMDPEG_STATE ) ;
2007-07-02 08:07:57 +04:00
if ( val = = PHAN_INITIALIZE_COMPLETE | |
val = = PHAN_INITIALIZE_ACK )
return 0 ;
2008-07-22 06:44:04 +04:00
msleep ( 500 ) ;
2007-07-02 08:07:57 +04:00
} while ( - - retries ) ;
2008-07-22 06:44:04 +04:00
2007-07-02 08:07:57 +04:00
if ( ! retries ) {
2008-07-22 06:44:04 +04:00
pegtune_val = adapter - > pci_read_normalize ( adapter ,
NETXEN_ROMUSB_GLB_PEGTUNE_DONE ) ;
2007-07-02 08:07:57 +04:00
printk ( KERN_WARNING " netxen_phantom_init: init failed, "
" pegtune_val=%x \n " , pegtune_val ) ;
return - 1 ;
2006-10-21 23:33:03 +04:00
}
}
2007-07-02 08:07:57 +04:00
return 0 ;
2006-10-21 23:33:03 +04:00
}
2008-07-22 06:44:04 +04:00
int netxen_receive_peg_ready ( struct netxen_adapter * adapter )
{
u32 val = 0 ;
int retries = 2000 ;
do {
val = adapter - > pci_read_normalize ( adapter , CRB_RCVPEG_STATE ) ;
if ( val = = PHAN_PEG_RCV_INITIALIZED )
return 0 ;
msleep ( 10 ) ;
} while ( - - retries ) ;
if ( ! retries ) {
printk ( KERN_ERR " Receive Peg initialization not "
" complete, state: 0x%x. \n " , val ) ;
return - EIO ;
}
return 0 ;
}
2008-07-22 06:44:08 +04:00
static struct sk_buff * netxen_process_rxbuf ( struct netxen_adapter * adapter ,
struct nx_host_rds_ring * rds_ring , u16 index , u16 cksum )
{
struct netxen_rx_buffer * buffer ;
struct sk_buff * skb ;
buffer = & rds_ring - > rx_buf_arr [ index ] ;
pci_unmap_single ( adapter - > pdev , buffer - > dma , rds_ring - > dma_size ,
PCI_DMA_FROMDEVICE ) ;
skb = buffer - > skb ;
if ( ! skb )
goto no_skb ;
if ( likely ( adapter - > rx_csum & & cksum = = STATUS_CKSUM_OK ) ) {
adapter - > stats . csummed + + ;
skb - > ip_summed = CHECKSUM_UNNECESSARY ;
} else
skb - > ip_summed = CHECKSUM_NONE ;
skb - > dev = adapter - > netdev ;
buffer - > skb = NULL ;
no_skb :
buffer - > state = NETXEN_BUFFER_FREE ;
buffer - > lro_current_frags = 0 ;
buffer - > lro_expected_frags = 0 ;
list_add_tail ( & buffer - > list , & rds_ring - > free_list ) ;
return skb ;
}
2007-11-05 20:07:31 +03:00
static void netxen_process_rcv ( struct netxen_adapter * adapter , int ctxid ,
2009-03-09 11:50:52 +03:00
struct status_desc * desc )
2006-10-21 23:33:03 +04:00
{
2007-04-20 18:52:37 +04:00
struct net_device * netdev = adapter - > netdev ;
2007-12-31 21:08:57 +03:00
u64 sts_data = le64_to_cpu ( desc - > status_desc_data ) ;
int index = netxen_get_sts_refhandle ( sts_data ) ;
2006-10-21 23:33:03 +04:00
struct netxen_recv_context * recv_ctx = & ( adapter - > recv_ctx [ ctxid ] ) ;
struct netxen_rx_buffer * buffer ;
struct sk_buff * skb ;
2007-12-31 21:08:57 +03:00
u32 length = netxen_get_sts_totallength ( sts_data ) ;
2006-10-21 23:33:03 +04:00
u32 desc_ctx ;
2008-07-22 06:44:08 +04:00
u16 pkt_offset = 0 , cksum ;
2008-07-22 06:44:06 +04:00
struct nx_host_rds_ring * rds_ring ;
2006-10-21 23:33:03 +04:00
2007-12-31 21:08:57 +03:00
desc_ctx = netxen_get_sts_type ( sts_data ) ;
2006-10-21 23:33:03 +04:00
if ( unlikely ( desc_ctx > = NUM_RCV_DESC_RINGS ) ) {
return ;
}
2008-07-22 06:44:06 +04:00
rds_ring = & recv_ctx - > rds_rings [ desc_ctx ] ;
if ( unlikely ( index > rds_ring - > max_rx_desc_count ) ) {
2006-12-04 20:23:25 +03:00
return ;
}
2008-07-22 06:44:06 +04:00
buffer = & rds_ring - > rx_buf_arr [ index ] ;
2006-12-04 20:23:25 +03:00
if ( desc_ctx = = RCV_DESC_LRO_CTXID ) {
buffer - > lro_current_frags + + ;
if ( netxen_get_sts_desc_lro_last_frag ( desc ) ) {
buffer - > lro_expected_frags =
netxen_get_sts_desc_lro_cnt ( desc ) ;
buffer - > lro_length = length ;
}
if ( buffer - > lro_current_frags ! = buffer - > lro_expected_frags ) {
return ;
}
}
2006-10-21 23:33:03 +04:00
2008-07-22 06:44:08 +04:00
cksum = netxen_get_sts_status ( sts_data ) ;
2006-10-21 23:33:03 +04:00
2008-07-22 06:44:08 +04:00
skb = netxen_process_rxbuf ( adapter , rds_ring , index , cksum ) ;
if ( ! skb )
return ;
2007-09-03 09:03:35 +04:00
2006-12-04 20:23:25 +03:00
if ( desc_ctx = = RCV_DESC_LRO_CTXID ) {
/* True length was only available on the last pkt */
skb_put ( skb , buffer - > lro_length ) ;
} else {
2008-07-22 06:44:08 +04:00
if ( length > rds_ring - > skb_size )
skb_put ( skb , rds_ring - > skb_size ) ;
else
skb_put ( skb , length ) ;
pkt_offset = netxen_get_sts_pkt_offset ( sts_data ) ;
if ( pkt_offset )
skb_pull ( skb , pkt_offset ) ;
2006-12-04 20:23:25 +03:00
}
2006-10-21 23:33:03 +04:00
skb - > protocol = eth_type_trans ( skb , netdev ) ;
2009-03-09 11:50:52 +03:00
netif_receive_skb ( skb ) ;
2008-07-22 06:44:08 +04:00
2009-03-09 11:50:52 +03:00
adapter - > stats . no_rcv + + ;
adapter - > stats . rxbytes + = length ;
2006-10-21 23:33:03 +04:00
}
/* Process Receive status ring */
u32 netxen_process_rcv_ring ( struct netxen_adapter * adapter , int ctxid , int max )
{
struct netxen_recv_context * recv_ctx = & ( adapter - > recv_ctx [ ctxid ] ) ;
struct status_desc * desc_head = recv_ctx - > rcv_status_desc_head ;
2009-03-09 11:50:52 +03:00
struct status_desc * desc ;
2006-10-21 23:33:03 +04:00
u32 consumer = recv_ctx - > status_rx_consumer ;
int count = 0 , ring ;
2008-07-22 06:44:08 +04:00
u64 sts_data ;
u16 opcode ;
2006-10-21 23:33:03 +04:00
while ( count < max ) {
desc = & desc_head [ consumer ] ;
2009-03-09 11:50:52 +03:00
sts_data = le64_to_cpu ( desc - > status_desc_data ) ;
if ( ! ( sts_data & STATUS_OWNER_HOST ) )
2006-10-21 23:33:03 +04:00
break ;
2008-07-22 06:44:08 +04:00
opcode = netxen_get_sts_opcode ( sts_data ) ;
2009-03-09 11:50:52 +03:00
netxen_process_rcv ( adapter , ctxid , desc ) ;
2008-07-22 06:44:08 +04:00
2009-03-09 11:50:52 +03:00
desc - > status_desc_data = cpu_to_le64 ( STATUS_OWNER_PHANTOM ) ;
2008-07-22 06:44:08 +04:00
consumer = get_next_index ( consumer ,
adapter - > max_rx_desc_count ) ;
2006-10-21 23:33:03 +04:00
count + + ;
}
2009-03-09 11:50:52 +03:00
2008-07-22 06:44:06 +04:00
for ( ring = 0 ; ring < adapter - > max_rds_rings ; ring + + )
2008-03-18 05:59:49 +03:00
netxen_post_rx_buffers_nodb ( adapter , ctxid , ring ) ;
2006-10-21 23:33:03 +04:00
if ( count ) {
recv_ctx - > status_rx_consumer = consumer ;
2008-07-22 06:44:03 +04:00
adapter - > pci_write_normalize ( adapter ,
recv_ctx - > crb_sts_consumer , consumer ) ;
2006-10-21 23:33:03 +04:00
}
return count ;
}
/* Process Command status ring */
2008-03-18 05:59:49 +03:00
int netxen_process_cmd_ring ( struct netxen_adapter * adapter )
2006-10-21 23:33:03 +04:00
{
2008-03-18 05:59:50 +03:00
u32 last_consumer , consumer ;
int count = 0 , i ;
2006-10-21 23:33:03 +04:00
struct netxen_cmd_buffer * buffer ;
2008-03-18 05:59:50 +03:00
struct pci_dev * pdev = adapter - > pdev ;
struct net_device * netdev = adapter - > netdev ;
2006-10-21 23:33:03 +04:00
struct netxen_skb_frag * frag ;
2008-03-18 05:59:50 +03:00
int done = 0 ;
2006-10-21 23:33:03 +04:00
last_consumer = adapter - > last_cmd_consumer ;
2009-03-09 11:50:53 +03:00
barrier ( ) ; /* cmd_consumer can change underneath */
2007-02-09 16:45:18 +03:00
consumer = le32_to_cpu ( * ( adapter - > cmd_consumer ) ) ;
2006-10-21 23:33:03 +04:00
2008-03-18 05:59:50 +03:00
while ( last_consumer ! = consumer ) {
2006-10-21 23:33:03 +04:00
buffer = & adapter - > cmd_buf_arr [ last_consumer ] ;
2007-12-26 21:23:58 +03:00
if ( buffer - > skb ) {
frag = & buffer - > frag_array [ 0 ] ;
2006-10-21 23:33:03 +04:00
pci_unmap_single ( pdev , frag - > dma , frag - > length ,
PCI_DMA_TODEVICE ) ;
2007-07-02 08:07:57 +04:00
frag - > dma = 0ULL ;
2006-10-21 23:33:03 +04:00
for ( i = 1 ; i < buffer - > frag_count ; i + + ) {
frag + + ; /* Get the next frag */
pci_unmap_page ( pdev , frag - > dma , frag - > length ,
PCI_DMA_TODEVICE ) ;
2007-07-02 08:07:57 +04:00
frag - > dma = 0ULL ;
2006-10-21 23:33:03 +04:00
}
2008-03-18 05:59:50 +03:00
adapter - > stats . xmitfinished + + ;
2007-12-26 21:23:58 +03:00
dev_kfree_skb_any ( buffer - > skb ) ;
buffer - > skb = NULL ;
2006-10-21 23:33:03 +04:00
}
last_consumer = get_next_index ( last_consumer ,
adapter - > max_tx_desc_count ) ;
2008-03-18 05:59:50 +03:00
if ( + + count > = MAX_STATUS_HANDLE )
break ;
2006-10-21 23:33:03 +04:00
}
2008-03-18 05:59:50 +03:00
if ( count ) {
2006-10-21 23:33:03 +04:00
adapter - > last_cmd_consumer = last_consumer ;
2008-03-18 05:59:50 +03:00
smp_mb ( ) ;
if ( netif_queue_stopped ( netdev ) & & netif_running ( netdev ) ) {
netif_tx_lock ( netdev ) ;
netif_wake_queue ( netdev ) ;
smp_mb ( ) ;
netif_tx_unlock ( netdev ) ;
2006-10-21 23:33:03 +04:00
}
}
2006-12-04 20:23:25 +03:00
/*
* If everything is freed up to consumer then check if the ring is full
* If the ring is full then check if more needs to be freed and
* schedule the call back again .
*
* This happens when there are 2 CPUs . One could be freeing and the
* other filling it . If the ring is full when we get out of here and
* the card has already interrupted the host then the host can miss the
* interrupt .
*
* There is still a possible race condition and the host could miss an
* interrupt . The card has to take care of this .
*/
2009-03-09 11:50:53 +03:00
barrier ( ) ; /* cmd_consumer can change underneath */
2008-03-18 05:59:50 +03:00
consumer = le32_to_cpu ( * ( adapter - > cmd_consumer ) ) ;
done = ( last_consumer = = consumer ) ;
2006-10-21 23:33:03 +04:00
2006-12-04 20:23:25 +03:00
return ( done ) ;
2006-10-21 23:33:03 +04:00
}
/*
* netxen_post_rx_buffers puts buffer in the Phantom memory
*/
void netxen_post_rx_buffers ( struct netxen_adapter * adapter , u32 ctx , u32 ringid )
{
2008-07-22 06:44:00 +04:00
struct pci_dev * pdev = adapter - > pdev ;
2006-10-21 23:33:03 +04:00
struct sk_buff * skb ;
struct netxen_recv_context * recv_ctx = & ( adapter - > recv_ctx [ ctx ] ) ;
2008-07-22 06:44:06 +04:00
struct nx_host_rds_ring * rds_ring = NULL ;
2006-12-04 20:23:25 +03:00
uint producer ;
2006-10-21 23:33:03 +04:00
struct rcv_desc * pdesc ;
struct netxen_rx_buffer * buffer ;
int count = 0 ;
2006-12-04 20:23:25 +03:00
netxen_ctx_msg msg = 0 ;
dma_addr_t dma ;
2008-07-22 06:44:08 +04:00
struct list_head * head ;
2006-10-21 23:33:03 +04:00
2008-07-22 06:44:06 +04:00
rds_ring = & recv_ctx - > rds_rings [ ringid ] ;
2006-10-21 23:33:03 +04:00
2008-07-22 06:44:06 +04:00
producer = rds_ring - > producer ;
2008-07-22 06:44:08 +04:00
head = & rds_ring - > free_list ;
2006-10-21 23:33:03 +04:00
/* We can start writing rx descriptors into the phantom memory. */
2008-07-22 06:44:08 +04:00
while ( ! list_empty ( head ) ) {
2008-07-22 06:44:06 +04:00
skb = dev_alloc_skb ( rds_ring - > skb_size ) ;
2006-10-21 23:33:03 +04:00
if ( unlikely ( ! skb ) ) {
break ;
}
2006-12-04 20:23:25 +03:00
2009-01-15 07:50:00 +03:00
if ( ! adapter - > ahw . cut_through )
skb_reserve ( skb , 2 ) ;
dma = pci_map_single ( pdev , skb - > data ,
rds_ring - > dma_size , PCI_DMA_FROMDEVICE ) ;
if ( pci_dma_mapping_error ( pdev , dma ) ) {
dev_kfree_skb_any ( skb ) ;
break ;
}
count + + ;
2008-07-22 06:44:08 +04:00
buffer = list_entry ( head - > next , struct netxen_rx_buffer , list ) ;
list_del ( & buffer - > list ) ;
2006-12-04 20:23:25 +03:00
buffer - > skb = skb ;
buffer - > state = NETXEN_BUFFER_BUSY ;
buffer - > dma = dma ;
2009-01-15 07:50:00 +03:00
2006-12-04 20:23:25 +03:00
/* make a rcv descriptor */
2009-01-15 07:50:00 +03:00
pdesc = & rds_ring - > desc_head [ producer ] ;
pdesc - > addr_buffer = cpu_to_le64 ( dma ) ;
2006-12-18 16:53:59 +03:00
pdesc - > reference_handle = cpu_to_le16 ( buffer - > ref_handle ) ;
2008-07-22 06:44:06 +04:00
pdesc - > buffer_length = cpu_to_le32 ( rds_ring - > dma_size ) ;
2009-01-15 07:50:00 +03:00
producer = get_next_index ( producer , rds_ring - > max_rx_desc_count ) ;
2006-12-04 20:23:25 +03:00
}
/* if we did allocate buffers, then write the count to Phantom */
if ( count ) {
2008-07-22 06:44:06 +04:00
rds_ring - > producer = producer ;
2006-12-04 20:23:25 +03:00
/* Window = 1 */
2008-07-22 06:44:03 +04:00
adapter - > pci_write_normalize ( adapter ,
2008-07-22 06:44:06 +04:00
rds_ring - > crb_rcv_producer ,
( producer - 1 ) & ( rds_ring - > max_rx_desc_count - 1 ) ) ;
if ( adapter - > fw_major < 4 ) {
2006-12-04 20:23:25 +03:00
/*
* Write a doorbell msg to tell phanmon of change in
* receive ring producer
2008-07-22 06:44:06 +04:00
* Only for firmware version < 4.0 .0
2006-12-04 20:23:25 +03:00
*/
netxen_set_msg_peg_id ( msg , NETXEN_RCV_PEG_DB_ID ) ;
netxen_set_msg_privid ( msg ) ;
netxen_set_msg_count ( msg ,
( ( producer -
2008-07-22 06:44:06 +04:00
1 ) & ( rds_ring - >
2006-12-04 20:23:25 +03:00
max_rx_desc_count - 1 ) ) ) ;
2007-04-20 18:52:37 +04:00
netxen_set_msg_ctxid ( msg , adapter - > portnum ) ;
2006-12-04 20:23:25 +03:00
netxen_set_msg_opcode ( msg , NETXEN_RCV_PRODUCER ( ringid ) ) ;
writel ( msg ,
DB_NORMALIZE ( adapter ,
NETXEN_RCV_PRODUCER_OFFSET ) ) ;
2008-07-22 06:44:06 +04:00
}
2006-12-04 20:23:25 +03:00
}
}
2007-11-05 20:07:31 +03:00
static void netxen_post_rx_buffers_nodb ( struct netxen_adapter * adapter ,
uint32_t ctx , uint32_t ringid )
2006-12-04 20:23:25 +03:00
{
2008-07-22 06:44:00 +04:00
struct pci_dev * pdev = adapter - > pdev ;
2006-12-04 20:23:25 +03:00
struct sk_buff * skb ;
struct netxen_recv_context * recv_ctx = & ( adapter - > recv_ctx [ ctx ] ) ;
2008-07-22 06:44:06 +04:00
struct nx_host_rds_ring * rds_ring = NULL ;
2006-12-04 20:23:25 +03:00
u32 producer ;
struct rcv_desc * pdesc ;
struct netxen_rx_buffer * buffer ;
int count = 0 ;
2008-07-22 06:44:08 +04:00
struct list_head * head ;
2009-01-15 07:50:00 +03:00
dma_addr_t dma ;
2006-12-04 20:23:25 +03:00
2008-07-22 06:44:06 +04:00
rds_ring = & recv_ctx - > rds_rings [ ringid ] ;
2006-12-04 20:23:25 +03:00
2008-07-22 06:44:06 +04:00
producer = rds_ring - > producer ;
2008-07-22 06:44:08 +04:00
head = & rds_ring - > free_list ;
2006-12-04 20:23:25 +03:00
/* We can start writing rx descriptors into the phantom memory. */
2008-07-22 06:44:08 +04:00
while ( ! list_empty ( head ) ) {
2008-07-22 06:44:06 +04:00
skb = dev_alloc_skb ( rds_ring - > skb_size ) ;
2006-12-04 20:23:25 +03:00
if ( unlikely ( ! skb ) ) {
break ;
}
2008-07-22 06:44:08 +04:00
2009-01-15 07:50:00 +03:00
if ( ! adapter - > ahw . cut_through )
skb_reserve ( skb , 2 ) ;
dma = pci_map_single ( pdev , skb - > data ,
rds_ring - > dma_size , PCI_DMA_FROMDEVICE ) ;
if ( pci_dma_mapping_error ( pdev , dma ) ) {
dev_kfree_skb_any ( skb ) ;
break ;
}
count + + ;
2008-07-22 06:44:08 +04:00
buffer = list_entry ( head - > next , struct netxen_rx_buffer , list ) ;
list_del ( & buffer - > list ) ;
2006-10-21 23:33:03 +04:00
buffer - > skb = skb ;
buffer - > state = NETXEN_BUFFER_BUSY ;
2009-01-15 07:50:00 +03:00
buffer - > dma = dma ;
2006-12-04 20:23:25 +03:00
2006-10-21 23:33:03 +04:00
/* make a rcv descriptor */
2009-01-15 07:50:00 +03:00
pdesc = & rds_ring - > desc_head [ producer ] ;
2006-12-18 16:53:59 +03:00
pdesc - > reference_handle = cpu_to_le16 ( buffer - > ref_handle ) ;
2008-07-22 06:44:06 +04:00
pdesc - > buffer_length = cpu_to_le32 ( rds_ring - > dma_size ) ;
2006-10-21 23:33:03 +04:00
pdesc - > addr_buffer = cpu_to_le64 ( buffer - > dma ) ;
2009-01-15 07:50:00 +03:00
producer = get_next_index ( producer , rds_ring - > max_rx_desc_count ) ;
2006-10-21 23:33:03 +04:00
}
/* if we did allocate buffers, then write the count to Phantom */
if ( count ) {
2008-07-22 06:44:06 +04:00
rds_ring - > producer = producer ;
2006-10-21 23:33:03 +04:00
/* Window = 1 */
2008-07-22 06:44:03 +04:00
adapter - > pci_write_normalize ( adapter ,
2008-07-22 06:44:06 +04:00
rds_ring - > crb_rcv_producer ,
( producer - 1 ) & ( rds_ring - > max_rx_desc_count - 1 ) ) ;
2006-10-21 23:33:03 +04:00
wmb ( ) ;
}
}
void netxen_nic_clear_stats ( struct netxen_adapter * adapter )
{
memset ( & adapter - > stats , 0 , sizeof ( adapter - > stats ) ) ;
2007-04-20 18:52:37 +04:00
return ;
2006-10-21 23:33:03 +04:00
}