2008-04-27 15:55:59 +04:00
/****************************************************************************
* Driver for Solarflare Solarstorm network controllers and boards
* Copyright 2005 - 2006 Fen Systems Ltd .
2011-02-25 03:01:34 +03:00
* Copyright 2006 - 2010 Solarflare Communications Inc .
2008-04-27 15:55:59 +04:00
*
* This program is free software ; you can redistribute it and / or modify it
* under the terms of the GNU General Public License version 2 as published
* by the Free Software Foundation , incorporated herein by reference .
*/
# ifndef EFX_EFX_H
# define EFX_EFX_H
# include "net_driver.h"
2010-09-20 12:43:07 +04:00
# include "filter.h"
2008-04-27 15:55:59 +04:00
2009-10-23 12:32:33 +04:00
/* Solarstorm controllers use BAR 0 for I/O space and BAR 2(&3) for memory */
# define EFX_MEM_BAR 2
2008-04-27 15:55:59 +04:00
/* TX */
2009-11-23 19:07:30 +03:00
extern int efx_probe_tx_queue ( struct efx_tx_queue * tx_queue ) ;
extern void efx_remove_tx_queue ( struct efx_tx_queue * tx_queue ) ;
extern void efx_init_tx_queue ( struct efx_tx_queue * tx_queue ) ;
2011-01-12 21:39:40 +03:00
extern void efx_init_tx_queue_core_txq ( struct efx_tx_queue * tx_queue ) ;
2009-11-23 19:07:30 +03:00
extern void efx_fini_tx_queue ( struct efx_tx_queue * tx_queue ) ;
extern netdev_tx_t
efx_hard_start_xmit ( struct sk_buff * skb , struct net_device * net_dev ) ;
extern netdev_tx_t
efx_enqueue_skb ( struct efx_tx_queue * tx_queue , struct sk_buff * skb ) ;
2009-10-23 12:31:54 +04:00
extern void efx_xmit_done ( struct efx_tx_queue * tx_queue , unsigned int index ) ;
2011-01-11 00:18:20 +03:00
extern int efx_setup_tc ( struct net_device * net_dev , u8 num_tc ) ;
2012-07-30 19:57:44 +04:00
extern unsigned int efx_tx_max_skb_descs ( struct efx_nic * efx ) ;
2008-04-27 15:55:59 +04:00
/* RX */
2013-02-13 14:54:41 +04:00
extern void efx_rx_config_page_split ( struct efx_nic * efx ) ;
2009-11-23 19:07:30 +03:00
extern int efx_probe_rx_queue ( struct efx_rx_queue * rx_queue ) ;
extern void efx_remove_rx_queue ( struct efx_rx_queue * rx_queue ) ;
extern void efx_init_rx_queue ( struct efx_rx_queue * rx_queue ) ;
extern void efx_fini_rx_queue ( struct efx_rx_queue * rx_queue ) ;
extern void efx_fast_push_rx_descriptors ( struct efx_rx_queue * rx_queue ) ;
2010-06-01 15:19:39 +04:00
extern void efx_rx_slow_fill ( unsigned long context ) ;
2013-01-30 03:33:15 +04:00
extern void __efx_rx_packet ( struct efx_channel * channel ) ;
extern void efx_rx_packet ( struct efx_rx_queue * rx_queue ,
unsigned int index , unsigned int n_frags ,
2011-08-26 21:05:11 +04:00
unsigned int len , u16 flags ) ;
2013-01-30 03:33:14 +04:00
static inline void efx_rx_flush_packet ( struct efx_channel * channel )
{
2013-01-30 03:33:15 +04:00
if ( channel - > rx_pkt_n_frags )
__efx_rx_packet ( channel ) ;
2013-01-30 03:33:14 +04:00
}
2010-06-01 15:19:39 +04:00
extern void efx_schedule_slow_fill ( struct efx_rx_queue * rx_queue ) ;
2010-09-10 10:42:22 +04:00
# define EFX_MAX_DMAQ_SIZE 4096UL
# define EFX_DEFAULT_DMAQ_SIZE 1024UL
# define EFX_MIN_DMAQ_SIZE 512UL
# define EFX_MAX_EVQ_SIZE 16384UL
# define EFX_MIN_EVQ_SIZE 512UL
2008-04-27 15:55:59 +04:00
2012-07-30 19:57:44 +04:00
/* Maximum number of TCP segments we support for soft-TSO */
# define EFX_TSO_MAX_SEGS 100
/* The smallest [rt]xq_entries that the driver supports. RX minimum
* is a bit arbitrary . For TX , we must have space for at least 2
* TSO skbs .
*/
# define EFX_RXQ_MIN_ENT 128U
# define EFX_TXQ_MIN_ENT(efx) (2 * efx_tx_max_skb_descs(efx))
2010-09-10 10:42:33 +04:00
2010-09-20 12:43:07 +04:00
/* Filters */
2012-11-08 05:46:53 +04:00
/**
* efx_filter_insert_filter - add or replace a filter
* @ efx : NIC in which to insert the filter
* @ spec : Specification for the filter
* @ replace_equal : Flag for whether the specified filter may replace an
* existing filter with equal priority
*
* On success , return the filter ID .
* On failure , return a negative error code .
*
2013-01-16 02:00:07 +04:00
* If existing filters have equal match values to the new filter spec ,
* then the new filter might replace them or the function might fail ,
* as follows .
*
* 1. If the existing filters have lower priority , or @ replace_equal
* is set and they have equal priority , replace them .
*
* 2. If the existing filters have higher priority , return - % EPERM .
*
* 3. If ! efx_filter_is_mc_recipient ( @ spec ) , or the NIC does not
* support delivery to multiple recipients , return - % EEXIST .
*
* This implies that filters for multiple multicast recipients must
* all be inserted with the same priority and @ replace_equal = % false .
2012-11-08 05:46:53 +04:00
*/
static inline s32 efx_filter_insert_filter ( struct efx_nic * efx ,
struct efx_filter_spec * spec ,
bool replace_equal )
{
return efx - > type - > filter_insert ( efx , spec , replace_equal ) ;
}
/**
* efx_filter_remove_id_safe - remove a filter by ID , carefully
* @ efx : NIC from which to remove the filter
* @ priority : Priority of filter , as passed to @ efx_filter_insert_filter
* @ filter_id : ID of filter , as returned by @ efx_filter_insert_filter
*
* This function will range - check @ filter_id , so it is safe to call
* with a value passed from userland .
*/
static inline int efx_filter_remove_id_safe ( struct efx_nic * efx ,
enum efx_filter_priority priority ,
u32 filter_id )
{
return efx - > type - > filter_remove_safe ( efx , priority , filter_id ) ;
}
/**
* efx_filter_get_filter_safe - retrieve a filter by ID , carefully
* @ efx : NIC from which to remove the filter
* @ priority : Priority of filter , as passed to @ efx_filter_insert_filter
* @ filter_id : ID of filter , as returned by @ efx_filter_insert_filter
* @ spec : Buffer in which to store filter specification
*
* This function will range - check @ filter_id , so it is safe to call
* with a value passed from userland .
*/
static inline int
efx_filter_get_filter_safe ( struct efx_nic * efx ,
enum efx_filter_priority priority ,
u32 filter_id , struct efx_filter_spec * spec )
{
return efx - > type - > filter_get_safe ( efx , priority , filter_id , spec ) ;
}
/**
* efx_farch_filter_clear_rx - remove RX filters by priority
* @ efx : NIC from which to remove the filters
* @ priority : Maximum priority to remove
*/
static inline void efx_filter_clear_rx ( struct efx_nic * efx ,
enum efx_filter_priority priority )
{
return efx - > type - > filter_clear_rx ( efx , priority ) ;
}
static inline u32 efx_filter_count_rx_used ( struct efx_nic * efx ,
enum efx_filter_priority priority )
{
return efx - > type - > filter_count_rx_used ( efx , priority ) ;
}
static inline u32 efx_filter_get_rx_id_limit ( struct efx_nic * efx )
{
return efx - > type - > filter_get_rx_id_limit ( efx ) ;
}
static inline s32 efx_filter_get_rx_ids ( struct efx_nic * efx ,
enum efx_filter_priority priority ,
u32 * buf , u32 size )
{
return efx - > type - > filter_get_rx_ids ( efx , priority , buf , size ) ;
}
2011-01-05 03:50:41 +03:00
# ifdef CONFIG_RFS_ACCEL
extern int efx_filter_rfs ( struct net_device * net_dev , const struct sk_buff * skb ,
u16 rxq_index , u32 flow_id ) ;
extern bool __efx_filter_rfs_expire ( struct efx_nic * efx , unsigned quota ) ;
static inline void efx_filter_rfs_expire ( struct efx_channel * channel )
{
if ( channel - > rfs_filters_added > = 60 & &
__efx_filter_rfs_expire ( channel - > efx , 100 ) )
channel - > rfs_filters_added - = 60 ;
}
# define efx_filter_rfs_enabled() 1
# else
static inline void efx_filter_rfs_expire ( struct efx_channel * channel ) { }
# define efx_filter_rfs_enabled() 0
# endif
2013-01-16 02:00:07 +04:00
extern bool efx_filter_is_mc_recipient ( const struct efx_filter_spec * spec ) ;
2010-09-20 12:43:07 +04:00
2008-04-27 15:55:59 +04:00
/* Channels */
2012-02-14 03:45:02 +04:00
extern int efx_channel_dummy_op_int ( struct efx_channel * channel ) ;
2012-07-18 12:52:11 +04:00
extern void efx_channel_dummy_op_void ( struct efx_channel * channel ) ;
2010-09-10 10:42:33 +04:00
extern int
efx_realloc_channels ( struct efx_nic * efx , u32 rxq_entries , u32 txq_entries ) ;
2008-04-27 15:55:59 +04:00
/* Ports */
2009-11-29 06:42:41 +03:00
extern int efx_reconfigure_port ( struct efx_nic * efx ) ;
extern int __efx_reconfigure_port ( struct efx_nic * efx ) ;
2008-09-01 15:49:02 +04:00
2009-11-23 19:07:30 +03:00
/* Ethtool support */
extern const struct ethtool_ops efx_ethtool_ops ;
2008-09-01 15:49:02 +04:00
/* Reset handling */
2009-11-29 06:43:15 +03:00
extern int efx_reset ( struct efx_nic * efx , enum reset_type method ) ;
2009-11-29 06:42:41 +03:00
extern void efx_reset_down ( struct efx_nic * efx , enum reset_type method ) ;
extern int efx_reset_up ( struct efx_nic * efx , enum reset_type method , bool ok ) ;
2013-03-21 20:41:43 +04:00
extern int efx_try_recovery ( struct efx_nic * efx ) ;
2008-04-27 15:55:59 +04:00
/* Global */
extern void efx_schedule_reset ( struct efx_nic * efx , enum reset_type type ) ;
2011-09-05 11:43:04 +04:00
extern int efx_init_irq_moderation ( struct efx_nic * efx , unsigned int tx_usecs ,
unsigned int rx_usecs , bool rx_adaptive ,
bool rx_may_override_tx ) ;
2011-09-05 11:42:25 +04:00
extern void efx_get_irq_moderation ( struct efx_nic * efx , unsigned int * tx_usecs ,
unsigned int * rx_usecs , bool * rx_adaptive ) ;
2008-04-27 15:55:59 +04:00
/* Dummy PHY ops for PHY drivers */
extern int efx_port_dummy_op_int ( struct efx_nic * efx ) ;
extern void efx_port_dummy_op_void ( struct efx_nic * efx ) ;
2010-10-18 09:27:31 +04:00
2008-04-27 15:55:59 +04:00
2008-11-04 23:34:28 +03:00
/* MTD */
# ifdef CONFIG_SFC_MTD
2012-11-28 08:38:14 +04:00
extern int efx_mtd_add ( struct efx_nic * efx , struct efx_mtd_partition * parts ,
size_t n_parts , size_t sizeof_part ) ;
static inline int efx_mtd_probe ( struct efx_nic * efx )
{
return efx - > type - > mtd_probe ( efx ) ;
}
2008-11-04 23:34:28 +03:00
extern void efx_mtd_rename ( struct efx_nic * efx ) ;
extern void efx_mtd_remove ( struct efx_nic * efx ) ;
# else
static inline int efx_mtd_probe ( struct efx_nic * efx ) { return 0 ; }
static inline void efx_mtd_rename ( struct efx_nic * efx ) { }
static inline void efx_mtd_remove ( struct efx_nic * efx ) { }
# endif
2008-04-27 15:55:59 +04:00
static inline void efx_schedule_channel ( struct efx_channel * channel )
{
2010-06-23 15:30:07 +04:00
netif_vdbg ( channel - > efx , intr , channel - > efx - > net_dev ,
" channel %d scheduling NAPI poll on CPU%d \n " ,
channel - > channel , raw_smp_processor_id ( ) ) ;
2008-04-27 15:55:59 +04:00
2009-01-20 03:43:59 +03:00
napi_schedule ( & channel - > napi_str ) ;
2008-04-27 15:55:59 +04:00
}
2012-01-06 00:14:10 +04:00
static inline void efx_schedule_channel_irq ( struct efx_channel * channel )
{
2012-02-29 03:40:21 +04:00
channel - > event_test_cpu = raw_smp_processor_id ( ) ;
2012-01-06 00:14:10 +04:00
efx_schedule_channel ( channel ) ;
}
2009-11-28 08:34:05 +03:00
extern void efx_link_status_changed ( struct efx_nic * efx ) ;
2009-11-29 06:42:41 +03:00
extern void efx_link_set_advertising ( struct efx_nic * efx , u32 ) ;
2011-05-18 01:53:22 +04:00
extern void efx_link_set_wanted_fc ( struct efx_nic * efx , u8 ) ;
2009-11-28 08:34:05 +03:00
2012-10-17 16:21:23 +04:00
static inline void efx_device_detach_sync ( struct efx_nic * efx )
{
struct net_device * dev = efx - > net_dev ;
/* Lock/freeze all TX queues so that we can be sure the
* TX scheduler is stopped when we ' re done and before
* netif_device_present ( ) becomes false .
*/
2013-03-05 05:03:47 +04:00
netif_tx_lock_bh ( dev ) ;
2012-10-17 16:21:23 +04:00
netif_device_detach ( dev ) ;
2013-03-05 05:03:47 +04:00
netif_tx_unlock_bh ( dev ) ;
2012-10-17 16:21:23 +04:00
}
2008-04-27 15:55:59 +04:00
# endif /* EFX_EFX_H */