2005-04-16 15:20:36 -07:00
/*
* drivers / net / gianfar_ethtool . c
*
* Gianfar Ethernet Driver
* Ethtool support for Gianfar Enet
* Based on e1000 ethtool support
*
* Author : Andy Fleming
2005-11-13 16:06:30 -08:00
* Maintainer : Kumar Gala
2005-04-16 15:20:36 -07:00
*
* Copyright ( c ) 2003 , 2004 Freescale Semiconductor , Inc .
*
2006-09-13 13:24:59 -04:00
* This software may be used and distributed according to
* the terms of the GNU Public License , Version 2 , incorporated herein
2005-04-16 15:20:36 -07:00
* by reference .
*/
# include <linux/kernel.h>
# include <linux/string.h>
# include <linux/errno.h>
# include <linux/slab.h>
# include <linux/interrupt.h>
# include <linux/init.h>
# include <linux/delay.h>
# include <linux/netdevice.h>
# include <linux/etherdevice.h>
# include <linux/skbuff.h>
# include <linux/spinlock.h>
# include <linux/mm.h>
# include <asm/io.h>
# include <asm/irq.h>
# include <asm/uaccess.h>
# include <linux/module.h>
# include <linux/crc32.h>
# include <asm/types.h>
# include <linux/ethtool.h>
2005-09-23 22:54:21 -04:00
# include <linux/mii.h>
# include <linux/phy.h>
2005-04-16 15:20:36 -07:00
# include "gianfar.h"
2005-06-20 10:54:21 -05:00
extern void gfar_start ( struct net_device * dev ) ;
extern int gfar_clean_rx_ring ( struct net_device * dev , int rx_work_limit ) ;
2005-04-16 15:20:36 -07:00
2005-09-23 22:54:21 -04:00
# define GFAR_MAX_COAL_USECS 0xffff
# define GFAR_MAX_COAL_FRAMES 0xff
2005-06-20 10:54:21 -05:00
static void gfar_fill_stats ( struct net_device * dev , struct ethtool_stats * dummy ,
2005-04-16 15:20:36 -07:00
u64 * buf ) ;
2005-06-20 10:54:21 -05:00
static void gfar_gstrings ( struct net_device * dev , u32 stringset , u8 * buf ) ;
static int gfar_gcoalesce ( struct net_device * dev , struct ethtool_coalesce * cvals ) ;
static int gfar_scoalesce ( struct net_device * dev , struct ethtool_coalesce * cvals ) ;
static void gfar_gringparam ( struct net_device * dev , struct ethtool_ringparam * rvals ) ;
static int gfar_sringparam ( struct net_device * dev , struct ethtool_ringparam * rvals ) ;
static void gfar_gdrvinfo ( struct net_device * dev , struct ethtool_drvinfo * drvinfo ) ;
2005-04-16 15:20:36 -07:00
static char stat_gstrings [ ] [ ETH_GSTRING_LEN ] = {
" rx-dropped-by-kernel " ,
" rx-large-frame-errors " ,
" rx-short-frame-errors " ,
" rx-non-octet-errors " ,
" rx-crc-errors " ,
" rx-overrun-errors " ,
" rx-busy-errors " ,
" rx-babbling-errors " ,
" rx-truncated-frames " ,
" ethernet-bus-error " ,
" tx-babbling-errors " ,
" tx-underrun-errors " ,
" rx-skb-missing-errors " ,
" tx-timeout-errors " ,
" tx-rx-64-frames " ,
" tx-rx-65-127-frames " ,
" tx-rx-128-255-frames " ,
" tx-rx-256-511-frames " ,
" tx-rx-512-1023-frames " ,
" tx-rx-1024-1518-frames " ,
" tx-rx-1519-1522-good-vlan " ,
" rx-bytes " ,
" rx-packets " ,
" rx-fcs-errors " ,
" receive-multicast-packet " ,
" receive-broadcast-packet " ,
" rx-control-frame-packets " ,
" rx-pause-frame-packets " ,
" rx-unknown-op-code " ,
" rx-alignment-error " ,
" rx-frame-length-error " ,
" rx-code-error " ,
" rx-carrier-sense-error " ,
" rx-undersize-packets " ,
" rx-oversize-packets " ,
" rx-fragmented-frames " ,
" rx-jabber-frames " ,
" rx-dropped-frames " ,
" tx-byte-counter " ,
" tx-packets " ,
" tx-multicast-packets " ,
" tx-broadcast-packets " ,
" tx-pause-control-frames " ,
" tx-deferral-packets " ,
" tx-excessive-deferral-packets " ,
" tx-single-collision-packets " ,
" tx-multiple-collision-packets " ,
" tx-late-collision-packets " ,
" tx-excessive-collision-packets " ,
" tx-total-collision " ,
" reserved " ,
" tx-dropped-frames " ,
" tx-jabber-frames " ,
" tx-fcs-errors " ,
" tx-control-frames " ,
" tx-oversize-frames " ,
" tx-undersize-frames " ,
" tx-fragmented-frames " ,
} ;
2005-06-20 10:54:21 -05:00
/* Fill in a buffer with the strings which correspond to the
* stats */
static void gfar_gstrings ( struct net_device * dev , u32 stringset , u8 * buf )
{
struct gfar_private * priv = netdev_priv ( dev ) ;
2005-11-11 12:38:59 -06:00
2008-12-16 15:29:15 -08:00
if ( priv - > device_flags & FSL_GIANFAR_DEV_HAS_RMON )
2005-06-20 10:54:21 -05:00
memcpy ( buf , stat_gstrings , GFAR_STATS_LEN * ETH_GSTRING_LEN ) ;
else
memcpy ( buf , stat_gstrings ,
GFAR_EXTRA_STATS_LEN * ETH_GSTRING_LEN ) ;
}
2005-04-16 15:20:36 -07:00
/* Fill in an array of 64-bit statistics from various sources.
* This array will be appended to the end of the ethtool_stats
* structure , and returned to user space
*/
2005-06-20 10:54:21 -05:00
static void gfar_fill_stats ( struct net_device * dev , struct ethtool_stats * dummy , u64 * buf )
2005-04-16 15:20:36 -07:00
{
int i ;
struct gfar_private * priv = netdev_priv ( dev ) ;
u64 * extra = ( u64 * ) & priv - > extra_stats ;
2008-12-16 15:29:15 -08:00
if ( priv - > device_flags & FSL_GIANFAR_DEV_HAS_RMON ) {
2006-02-01 15:18:03 -06:00
u32 __iomem * rmon = ( u32 __iomem * ) & priv - > regs - > rmon ;
2005-06-20 10:54:21 -05:00
struct gfar_stats * stats = ( struct gfar_stats * ) buf ;
2005-04-16 15:20:36 -07:00
2005-06-20 10:54:21 -05:00
for ( i = 0 ; i < GFAR_RMON_LEN ; i + + )
2006-02-01 15:18:03 -06:00
stats - > rmon [ i ] = ( u64 ) gfar_read ( & rmon [ i ] ) ;
2005-04-16 15:20:36 -07:00
2005-06-20 10:54:21 -05:00
for ( i = 0 ; i < GFAR_EXTRA_STATS_LEN ; i + + )
stats - > extra [ i ] = extra [ i ] ;
} else
for ( i = 0 ; i < GFAR_EXTRA_STATS_LEN ; i + + )
buf [ i ] = extra [ i ] ;
2005-04-16 15:20:36 -07:00
}
2007-10-03 18:07:32 -07:00
static int gfar_sset_count ( struct net_device * dev , int sset )
2005-04-16 15:20:36 -07:00
{
struct gfar_private * priv = netdev_priv ( dev ) ;
2007-10-03 18:07:32 -07:00
switch ( sset ) {
case ETH_SS_STATS :
2008-12-16 15:29:15 -08:00
if ( priv - > device_flags & FSL_GIANFAR_DEV_HAS_RMON )
2007-10-03 18:07:32 -07:00
return GFAR_STATS_LEN ;
else
return GFAR_EXTRA_STATS_LEN ;
default :
return - EOPNOTSUPP ;
}
2005-04-16 15:20:36 -07:00
}
/* Fills in the drvinfo structure with some basic info */
2005-06-20 10:54:21 -05:00
static void gfar_gdrvinfo ( struct net_device * dev , struct
2005-04-16 15:20:36 -07:00
ethtool_drvinfo * drvinfo )
{
strncpy ( drvinfo - > driver , DRV_NAME , GFAR_INFOSTR_LEN ) ;
strncpy ( drvinfo - > version , gfar_driver_version , GFAR_INFOSTR_LEN ) ;
strncpy ( drvinfo - > fw_version , " N/A " , GFAR_INFOSTR_LEN ) ;
strncpy ( drvinfo - > bus_info , " N/A " , GFAR_INFOSTR_LEN ) ;
drvinfo - > regdump_len = 0 ;
drvinfo - > eedump_len = 0 ;
}
2005-09-23 22:54:21 -04:00
static int gfar_ssettings ( struct net_device * dev , struct ethtool_cmd * cmd )
{
struct gfar_private * priv = netdev_priv ( dev ) ;
struct phy_device * phydev = priv - > phydev ;
if ( NULL = = phydev )
return - ENODEV ;
return phy_ethtool_sset ( phydev , cmd ) ;
}
2005-04-16 15:20:36 -07:00
/* Return the current settings in the ethtool_cmd structure */
2005-06-20 10:54:21 -05:00
static int gfar_gsettings ( struct net_device * dev , struct ethtool_cmd * cmd )
2005-04-16 15:20:36 -07:00
{
struct gfar_private * priv = netdev_priv ( dev ) ;
2005-09-23 22:54:21 -04:00
struct phy_device * phydev = priv - > phydev ;
if ( NULL = = phydev )
return - ENODEV ;
2006-09-13 13:24:59 -04:00
2008-12-16 15:29:52 -08:00
cmd - > maxtxpkt = get_icft_value ( priv - > txic ) ;
cmd - > maxrxpkt = get_icft_value ( priv - > rxic ) ;
2005-04-16 15:20:36 -07:00
2005-09-23 22:54:21 -04:00
return phy_ethtool_gset ( phydev , cmd ) ;
2005-04-16 15:20:36 -07:00
}
/* Return the length of the register structure */
2005-06-20 10:54:21 -05:00
static int gfar_reglen ( struct net_device * dev )
2005-04-16 15:20:36 -07:00
{
return sizeof ( struct gfar ) ;
}
/* Return a dump of the GFAR register space */
2005-06-20 10:54:21 -05:00
static void gfar_get_regs ( struct net_device * dev , struct ethtool_regs * regs , void * regbuf )
2005-04-16 15:20:36 -07:00
{
int i ;
struct gfar_private * priv = netdev_priv ( dev ) ;
2006-02-01 15:18:03 -06:00
u32 __iomem * theregs = ( u32 __iomem * ) priv - > regs ;
2005-04-16 15:20:36 -07:00
u32 * buf = ( u32 * ) regbuf ;
for ( i = 0 ; i < sizeof ( struct gfar ) / sizeof ( u32 ) ; i + + )
2006-02-01 15:18:03 -06:00
buf [ i ] = gfar_read ( & theregs [ i ] ) ;
2005-04-16 15:20:36 -07:00
}
/* Convert microseconds to ethernet clock ticks, which changes
* depending on what speed the controller is running at */
static unsigned int gfar_usecs2ticks ( struct gfar_private * priv , unsigned int usecs )
{
unsigned int count ;
/* The timer is different, depending on the interface speed */
2005-09-23 22:54:21 -04:00
switch ( priv - > phydev - > speed ) {
case SPEED_1000 :
2005-04-16 15:20:36 -07:00
count = GFAR_GBIT_TIME ;
break ;
2005-09-23 22:54:21 -04:00
case SPEED_100 :
2005-04-16 15:20:36 -07:00
count = GFAR_100_TIME ;
break ;
2005-09-23 22:54:21 -04:00
case SPEED_10 :
2005-04-16 15:20:36 -07:00
default :
count = GFAR_10_TIME ;
break ;
}
/* Make sure we return a number greater than 0
* if usecs > 0 */
return ( ( usecs * 1000 + count - 1 ) / count ) ;
}
/* Convert ethernet clock ticks to microseconds */
static unsigned int gfar_ticks2usecs ( struct gfar_private * priv , unsigned int ticks )
{
unsigned int count ;
/* The timer is different, depending on the interface speed */
2005-09-23 22:54:21 -04:00
switch ( priv - > phydev - > speed ) {
case SPEED_1000 :
2005-04-16 15:20:36 -07:00
count = GFAR_GBIT_TIME ;
break ;
2005-09-23 22:54:21 -04:00
case SPEED_100 :
2005-04-16 15:20:36 -07:00
count = GFAR_100_TIME ;
break ;
2005-09-23 22:54:21 -04:00
case SPEED_10 :
2005-04-16 15:20:36 -07:00
default :
count = GFAR_10_TIME ;
break ;
}
/* Make sure we return a number greater than 0 */
/* if ticks is > 0 */
return ( ( ticks * count ) / 1000 ) ;
}
/* Get the coalescing parameters, and put them in the cvals
* structure . */
2005-06-20 10:54:21 -05:00
static int gfar_gcoalesce ( struct net_device * dev , struct ethtool_coalesce * cvals )
2005-04-16 15:20:36 -07:00
{
struct gfar_private * priv = netdev_priv ( dev ) ;
2008-12-16 15:29:52 -08:00
unsigned long rxtime ;
unsigned long rxcount ;
unsigned long txtime ;
unsigned long txcount ;
2006-09-13 13:24:59 -04:00
2008-12-16 15:29:15 -08:00
if ( ! ( priv - > device_flags & FSL_GIANFAR_DEV_HAS_COALESCE ) )
2005-06-20 10:54:21 -05:00
return - EOPNOTSUPP ;
2005-04-16 15:20:36 -07:00
2005-09-23 22:54:21 -04:00
if ( NULL = = priv - > phydev )
return - ENODEV ;
2008-12-16 15:29:52 -08:00
rxtime = get_ictt_value ( priv - > rxic ) ;
rxcount = get_icft_value ( priv - > rxic ) ;
txtime = get_ictt_value ( priv - > txic ) ;
2009-08-18 11:18:35 -07:00
txcount = get_icft_value ( priv - > txic ) ;
2008-12-16 15:29:52 -08:00
cvals - > rx_coalesce_usecs = gfar_ticks2usecs ( priv , rxtime ) ;
cvals - > rx_max_coalesced_frames = rxcount ;
2005-04-16 15:20:36 -07:00
2008-12-16 15:29:52 -08:00
cvals - > tx_coalesce_usecs = gfar_ticks2usecs ( priv , txtime ) ;
cvals - > tx_max_coalesced_frames = txcount ;
2005-04-16 15:20:36 -07:00
cvals - > use_adaptive_rx_coalesce = 0 ;
cvals - > use_adaptive_tx_coalesce = 0 ;
cvals - > pkt_rate_low = 0 ;
cvals - > rx_coalesce_usecs_low = 0 ;
cvals - > rx_max_coalesced_frames_low = 0 ;
cvals - > tx_coalesce_usecs_low = 0 ;
cvals - > tx_max_coalesced_frames_low = 0 ;
/* When the packet rate is below pkt_rate_high but above
* pkt_rate_low ( both measured in packets per second ) the
* normal { rx , tx } _ * coalescing parameters are used .
*/
/* When the packet rate is (measured in packets per second)
* is above pkt_rate_high , the { rx , tx } _ * _high parameters are
* used .
*/
cvals - > pkt_rate_high = 0 ;
cvals - > rx_coalesce_usecs_high = 0 ;
cvals - > rx_max_coalesced_frames_high = 0 ;
cvals - > tx_coalesce_usecs_high = 0 ;
cvals - > tx_max_coalesced_frames_high = 0 ;
/* How often to do adaptive coalescing packet rate sampling,
* measured in seconds . Must not be zero .
*/
cvals - > rate_sample_interval = 0 ;
return 0 ;
}
/* Change the coalescing values.
* Both cvals - > * _usecs and cvals - > * _frames have to be > 0
* in order for coalescing to be active
*/
2005-06-20 10:54:21 -05:00
static int gfar_scoalesce ( struct net_device * dev , struct ethtool_coalesce * cvals )
2005-04-16 15:20:36 -07:00
{
struct gfar_private * priv = netdev_priv ( dev ) ;
2008-12-16 15:29:15 -08:00
if ( ! ( priv - > device_flags & FSL_GIANFAR_DEV_HAS_COALESCE ) )
2005-06-20 10:54:21 -05:00
return - EOPNOTSUPP ;
2005-04-16 15:20:36 -07:00
/* Set up rx coalescing */
if ( ( cvals - > rx_coalesce_usecs = = 0 ) | |
( cvals - > rx_max_coalesced_frames = = 0 ) )
priv - > rxcoalescing = 0 ;
else
priv - > rxcoalescing = 1 ;
2005-09-23 22:54:21 -04:00
if ( NULL = = priv - > phydev )
return - ENODEV ;
/* Check the bounds of the values */
if ( cvals - > rx_coalesce_usecs > GFAR_MAX_COAL_USECS ) {
pr_info ( " Coalescing is limited to %d microseconds \n " ,
GFAR_MAX_COAL_USECS ) ;
return - EINVAL ;
}
if ( cvals - > rx_max_coalesced_frames > GFAR_MAX_COAL_FRAMES ) {
pr_info ( " Coalescing is limited to %d frames \n " ,
GFAR_MAX_COAL_FRAMES ) ;
return - EINVAL ;
}
2009-07-30 14:20:42 -07:00
priv - > rxic = mk_ic_value ( cvals - > rx_max_coalesced_frames ,
gfar_usecs2ticks ( priv , cvals - > rx_coalesce_usecs ) ) ;
2005-04-16 15:20:36 -07:00
/* Set up tx coalescing */
if ( ( cvals - > tx_coalesce_usecs = = 0 ) | |
( cvals - > tx_max_coalesced_frames = = 0 ) )
priv - > txcoalescing = 0 ;
else
priv - > txcoalescing = 1 ;
2005-09-23 22:54:21 -04:00
/* Check the bounds of the values */
if ( cvals - > tx_coalesce_usecs > GFAR_MAX_COAL_USECS ) {
pr_info ( " Coalescing is limited to %d microseconds \n " ,
GFAR_MAX_COAL_USECS ) ;
return - EINVAL ;
}
if ( cvals - > tx_max_coalesced_frames > GFAR_MAX_COAL_FRAMES ) {
pr_info ( " Coalescing is limited to %d frames \n " ,
GFAR_MAX_COAL_FRAMES ) ;
return - EINVAL ;
}
2009-07-30 14:20:42 -07:00
priv - > txic = mk_ic_value ( cvals - > tx_max_coalesced_frames ,
gfar_usecs2ticks ( priv , cvals - > tx_coalesce_usecs ) ) ;
2005-04-16 15:20:36 -07:00
2008-12-16 15:29:52 -08:00
gfar_write ( & priv - > regs - > rxic , 0 ) ;
2005-04-16 15:20:36 -07:00
if ( priv - > rxcoalescing )
2008-12-16 15:29:52 -08:00
gfar_write ( & priv - > regs - > rxic , priv - > rxic ) ;
2005-04-16 15:20:36 -07:00
2008-12-16 15:29:52 -08:00
gfar_write ( & priv - > regs - > txic , 0 ) ;
2005-04-16 15:20:36 -07:00
if ( priv - > txcoalescing )
2008-12-16 15:29:52 -08:00
gfar_write ( & priv - > regs - > txic , priv - > txic ) ;
2005-04-16 15:20:36 -07:00
return 0 ;
}
/* Fills in rvals with the current ring parameters. Currently,
* rx , rx_mini , and rx_jumbo rings are the same size , as mini and
* jumbo are ignored by the driver */
2005-06-20 10:54:21 -05:00
static void gfar_gringparam ( struct net_device * dev , struct ethtool_ringparam * rvals )
2005-04-16 15:20:36 -07:00
{
struct gfar_private * priv = netdev_priv ( dev ) ;
rvals - > rx_max_pending = GFAR_RX_MAX_RING_SIZE ;
rvals - > rx_mini_max_pending = GFAR_RX_MAX_RING_SIZE ;
rvals - > rx_jumbo_max_pending = GFAR_RX_MAX_RING_SIZE ;
rvals - > tx_max_pending = GFAR_TX_MAX_RING_SIZE ;
/* Values changeable by the user. The valid values are
* in the range 1 to the " *_max_pending " counterpart above .
*/
rvals - > rx_pending = priv - > rx_ring_size ;
rvals - > rx_mini_pending = priv - > rx_ring_size ;
rvals - > rx_jumbo_pending = priv - > rx_ring_size ;
rvals - > tx_pending = priv - > tx_ring_size ;
}
/* Change the current ring parameters, stopping the controller if
* necessary so that we don ' t mess things up while we ' re in
* motion . We wait for the ring to be clean before reallocating
* the rings . */
2005-06-20 10:54:21 -05:00
static int gfar_sringparam ( struct net_device * dev , struct ethtool_ringparam * rvals )
2005-04-16 15:20:36 -07:00
{
struct gfar_private * priv = netdev_priv ( dev ) ;
int err = 0 ;
if ( rvals - > rx_pending > GFAR_RX_MAX_RING_SIZE )
return - EINVAL ;
if ( ! is_power_of_2 ( rvals - > rx_pending ) ) {
printk ( " %s: Ring sizes must be a power of 2 \n " ,
dev - > name ) ;
return - EINVAL ;
}
if ( rvals - > tx_pending > GFAR_TX_MAX_RING_SIZE )
return - EINVAL ;
if ( ! is_power_of_2 ( rvals - > tx_pending ) ) {
printk ( " %s: Ring sizes must be a power of 2 \n " ,
dev - > name ) ;
return - EINVAL ;
}
2005-06-20 10:54:21 -05:00
if ( dev - > flags & IFF_UP ) {
unsigned long flags ;
2005-04-16 15:20:36 -07:00
2005-06-20 10:54:21 -05:00
/* Halt TX and RX, and process the frames which
* have already been received */
2006-04-20 16:44:29 -05:00
spin_lock_irqsave ( & priv - > txlock , flags ) ;
spin_lock ( & priv - > rxlock ) ;
2005-06-20 10:54:21 -05:00
gfar_halt ( dev ) ;
2006-04-20 16:44:29 -05:00
spin_unlock ( & priv - > rxlock ) ;
spin_unlock_irqrestore ( & priv - > txlock , flags ) ;
2005-04-16 15:20:36 -07:00
2008-12-16 15:30:20 -08:00
gfar_clean_rx_ring ( dev , priv - > rx_ring_size ) ;
2005-06-20 10:54:21 -05:00
/* Now we take down the rings to rebuild them */
stop_gfar ( dev ) ;
}
2005-04-16 15:20:36 -07:00
2005-06-20 10:54:21 -05:00
/* Change the size */
priv - > rx_ring_size = rvals - > rx_pending ;
priv - > tx_ring_size = rvals - > tx_pending ;
2008-12-17 16:51:04 -08:00
priv - > num_txbdfree = priv - > tx_ring_size ;
2005-04-16 15:20:36 -07:00
2005-06-20 10:54:21 -05:00
/* Rebuild the rings with the new size */
2008-12-16 15:30:20 -08:00
if ( dev - > flags & IFF_UP ) {
2005-06-20 10:54:21 -05:00
err = startup_gfar ( dev ) ;
2008-12-16 15:30:20 -08:00
netif_wake_queue ( dev ) ;
}
2005-06-20 10:54:21 -05:00
return err ;
}
2005-04-16 15:20:36 -07:00
2005-06-20 10:54:21 -05:00
static int gfar_set_rx_csum ( struct net_device * dev , uint32_t data )
{
struct gfar_private * priv = netdev_priv ( dev ) ;
2008-07-11 18:04:45 -05:00
unsigned long flags ;
2005-06-20 10:54:21 -05:00
int err = 0 ;
2005-04-16 15:20:36 -07:00
2008-12-16 15:29:15 -08:00
if ( ! ( priv - > device_flags & FSL_GIANFAR_DEV_HAS_CSUM ) )
2005-06-20 10:54:21 -05:00
return - EOPNOTSUPP ;
if ( dev - > flags & IFF_UP ) {
/* Halt TX and RX, and process the frames which
* have already been received */
2006-04-20 16:44:29 -05:00
spin_lock_irqsave ( & priv - > txlock , flags ) ;
spin_lock ( & priv - > rxlock ) ;
2005-06-20 10:54:21 -05:00
gfar_halt ( dev ) ;
2006-04-20 16:44:29 -05:00
spin_unlock ( & priv - > rxlock ) ;
spin_unlock_irqrestore ( & priv - > txlock , flags ) ;
2005-06-20 10:54:21 -05:00
2008-12-16 15:30:20 -08:00
gfar_clean_rx_ring ( dev , priv - > rx_ring_size ) ;
2005-06-20 10:54:21 -05:00
/* Now we take down the rings to rebuild them */
2005-04-16 15:20:36 -07:00
stop_gfar ( dev ) ;
}
2008-07-11 18:04:45 -05:00
spin_lock_irqsave ( & priv - > bflock , flags ) ;
2005-06-20 10:54:21 -05:00
priv - > rx_csum_enable = data ;
2008-07-11 18:04:45 -05:00
spin_unlock_irqrestore ( & priv - > bflock , flags ) ;
2005-04-16 15:20:36 -07:00
2008-12-16 15:30:20 -08:00
if ( dev - > flags & IFF_UP ) {
2005-04-16 15:20:36 -07:00
err = startup_gfar ( dev ) ;
2008-12-16 15:30:20 -08:00
netif_wake_queue ( dev ) ;
}
2005-04-16 15:20:36 -07:00
return err ;
}
2005-06-20 10:54:21 -05:00
static uint32_t gfar_get_rx_csum ( struct net_device * dev )
{
struct gfar_private * priv = netdev_priv ( dev ) ;
2008-12-16 15:29:15 -08:00
if ( ! ( priv - > device_flags & FSL_GIANFAR_DEV_HAS_CSUM ) )
2005-06-20 10:54:21 -05:00
return 0 ;
return priv - > rx_csum_enable ;
}
static int gfar_set_tx_csum ( struct net_device * dev , uint32_t data )
{
struct gfar_private * priv = netdev_priv ( dev ) ;
2008-12-16 15:29:15 -08:00
if ( ! ( priv - > device_flags & FSL_GIANFAR_DEV_HAS_CSUM ) )
2005-06-20 10:54:21 -05:00
return - EOPNOTSUPP ;
2008-12-16 15:30:20 -08:00
netif_tx_lock_bh ( dev ) ;
2005-06-20 10:54:21 -05:00
if ( data )
dev - > features | = NETIF_F_IP_CSUM ;
else
dev - > features & = ~ NETIF_F_IP_CSUM ;
2008-12-16 15:30:20 -08:00
netif_tx_unlock_bh ( dev ) ;
2005-06-20 10:54:21 -05:00
return 0 ;
}
static uint32_t gfar_get_tx_csum ( struct net_device * dev )
{
struct gfar_private * priv = netdev_priv ( dev ) ;
2008-12-16 15:29:15 -08:00
if ( ! ( priv - > device_flags & FSL_GIANFAR_DEV_HAS_CSUM ) )
2005-06-20 10:54:21 -05:00
return 0 ;
return ( dev - > features & NETIF_F_IP_CSUM ) ! = 0 ;
}
static uint32_t gfar_get_msglevel ( struct net_device * dev )
2006-09-13 13:24:59 -04:00
{
2005-06-20 10:54:21 -05:00
struct gfar_private * priv = netdev_priv ( dev ) ;
return priv - > msg_enable ;
2006-09-13 13:24:59 -04:00
}
2005-06-20 10:54:21 -05:00
static void gfar_set_msglevel ( struct net_device * dev , uint32_t data )
2006-09-13 13:24:59 -04:00
{
2005-06-20 10:54:21 -05:00
struct gfar_private * priv = netdev_priv ( dev ) ;
priv - > msg_enable = data ;
}
2008-07-11 18:04:45 -05:00
# ifdef CONFIG_PM
static void gfar_get_wol ( struct net_device * dev , struct ethtool_wolinfo * wol )
{
struct gfar_private * priv = netdev_priv ( dev ) ;
2008-12-16 15:29:15 -08:00
if ( priv - > device_flags & FSL_GIANFAR_DEV_HAS_MAGIC_PACKET ) {
2008-07-11 18:04:45 -05:00
wol - > supported = WAKE_MAGIC ;
wol - > wolopts = priv - > wol_en ? WAKE_MAGIC : 0 ;
} else {
wol - > supported = wol - > wolopts = 0 ;
}
}
static int gfar_set_wol ( struct net_device * dev , struct ethtool_wolinfo * wol )
{
struct gfar_private * priv = netdev_priv ( dev ) ;
unsigned long flags ;
2008-12-16 15:29:15 -08:00
if ( ! ( priv - > device_flags & FSL_GIANFAR_DEV_HAS_MAGIC_PACKET ) & &
2008-07-11 18:04:45 -05:00
wol - > wolopts ! = 0 )
return - EINVAL ;
if ( wol - > wolopts & ~ WAKE_MAGIC )
return - EINVAL ;
spin_lock_irqsave ( & priv - > bflock , flags ) ;
priv - > wol_en = wol - > wolopts & WAKE_MAGIC ? 1 : 0 ;
2009-02-01 00:52:34 -08:00
device_set_wakeup_enable ( & dev - > dev , priv - > wol_en ) ;
2008-07-11 18:04:45 -05:00
spin_unlock_irqrestore ( & priv - > bflock , flags ) ;
return 0 ;
}
# endif
2005-06-20 10:54:21 -05:00
2006-09-13 14:30:00 -04:00
const struct ethtool_ops gfar_ethtool_ops = {
2005-04-16 15:20:36 -07:00
. get_settings = gfar_gsettings ,
2005-09-23 22:54:21 -04:00
. set_settings = gfar_ssettings ,
2005-04-16 15:20:36 -07:00
. get_drvinfo = gfar_gdrvinfo ,
. get_regs_len = gfar_reglen ,
. get_regs = gfar_get_regs ,
. get_link = ethtool_op_get_link ,
. get_coalesce = gfar_gcoalesce ,
. set_coalesce = gfar_scoalesce ,
. get_ringparam = gfar_gringparam ,
. set_ringparam = gfar_sringparam ,
. get_strings = gfar_gstrings ,
2007-10-03 18:07:32 -07:00
. get_sset_count = gfar_sset_count ,
2005-04-16 15:20:36 -07:00
. get_ethtool_stats = gfar_fill_stats ,
2005-06-20 10:54:21 -05:00
. get_rx_csum = gfar_get_rx_csum ,
. get_tx_csum = gfar_get_tx_csum ,
. set_rx_csum = gfar_set_rx_csum ,
. set_tx_csum = gfar_set_tx_csum ,
2008-12-17 16:51:04 -08:00
. set_sg = ethtool_op_set_sg ,
2005-06-20 10:54:21 -05:00
. get_msglevel = gfar_get_msglevel ,
. set_msglevel = gfar_set_msglevel ,
2008-07-11 18:04:45 -05:00
# ifdef CONFIG_PM
. get_wol = gfar_get_wol ,
. set_wol = gfar_set_wol ,
# endif
2005-04-16 15:20:36 -07:00
} ;