2013-03-02 00:41:11 +00:00
/*
* ASIX AX88179 / 178 A USB 3.0 / 2.0 to Gigabit Ethernet Devices
*
* Copyright ( C ) 2011 - 2013 ASIX
*
* 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 .
*
* 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 .
*
* 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 .
*/
# include <linux/module.h>
# include <linux/etherdevice.h>
# include <linux/mii.h>
# include <linux/usb.h>
# include <linux/crc32.h>
# include <linux/usb/usbnet.h>
# define AX88179_PHY_ID 0x03
# define AX_EEPROM_LEN 0x100
# define AX88179_EEPROM_MAGIC 0x17900b95
# define AX_MCAST_FLTSIZE 8
# define AX_MAX_MCAST 64
# define AX_INT_PPLS_LINK ((u32)BIT(16))
# define AX_RXHDR_L4_TYPE_MASK 0x1c
# define AX_RXHDR_L4_TYPE_UDP 4
# define AX_RXHDR_L4_TYPE_TCP 16
# define AX_RXHDR_L3CSUM_ERR 2
# define AX_RXHDR_L4CSUM_ERR 1
# define AX_RXHDR_CRC_ERR ((u32)BIT(31))
# define AX_RXHDR_DROP_ERR ((u32)BIT(30))
# define AX_ACCESS_MAC 0x01
# define AX_ACCESS_PHY 0x02
# define AX_ACCESS_EEPROM 0x04
# define AX_ACCESS_EFUS 0x05
# define AX_PAUSE_WATERLVL_HIGH 0x54
# define AX_PAUSE_WATERLVL_LOW 0x55
# define PHYSICAL_LINK_STATUS 0x02
# define AX_USB_SS 0x04
# define AX_USB_HS 0x02
# define GENERAL_STATUS 0x03
/* Check AX88179 version. UA1:Bit2 = 0, UA2:Bit2 = 1 */
# define AX_SECLD 0x04
# define AX_SROM_ADDR 0x07
# define AX_SROM_CMD 0x0a
# define EEP_RD 0x04
# define EEP_BUSY 0x10
# define AX_SROM_DATA_LOW 0x08
# define AX_SROM_DATA_HIGH 0x09
# define AX_RX_CTL 0x0b
# define AX_RX_CTL_DROPCRCERR 0x0100
# define AX_RX_CTL_IPE 0x0200
# define AX_RX_CTL_START 0x0080
# define AX_RX_CTL_AP 0x0020
# define AX_RX_CTL_AM 0x0010
# define AX_RX_CTL_AB 0x0008
# define AX_RX_CTL_AMALL 0x0002
# define AX_RX_CTL_PRO 0x0001
# define AX_RX_CTL_STOP 0x0000
# define AX_NODE_ID 0x10
# define AX_MULFLTARY 0x16
# define AX_MEDIUM_STATUS_MODE 0x22
# define AX_MEDIUM_GIGAMODE 0x01
# define AX_MEDIUM_FULL_DUPLEX 0x02
# define AX_MEDIUM_ALWAYS_ONE 0x04
# define AX_MEDIUM_EN_125MHZ 0x08
# define AX_MEDIUM_RXFLOW_CTRLEN 0x10
# define AX_MEDIUM_TXFLOW_CTRLEN 0x20
# define AX_MEDIUM_RECEIVE_EN 0x100
# define AX_MEDIUM_PS 0x200
# define AX_MEDIUM_JUMBO_EN 0x8040
# define AX_MONITOR_MOD 0x24
# define AX_MONITOR_MODE_RWLC 0x02
# define AX_MONITOR_MODE_RWMP 0x04
# define AX_MONITOR_MODE_PMEPOL 0x20
# define AX_MONITOR_MODE_PMETYPE 0x40
# define AX_GPIO_CTRL 0x25
# define AX_GPIO_CTRL_GPIO3EN 0x80
# define AX_GPIO_CTRL_GPIO2EN 0x40
# define AX_GPIO_CTRL_GPIO1EN 0x20
# define AX_PHYPWR_RSTCTL 0x26
# define AX_PHYPWR_RSTCTL_BZ 0x0010
# define AX_PHYPWR_RSTCTL_IPRL 0x0020
# define AX_PHYPWR_RSTCTL_AT 0x1000
# define AX_RX_BULKIN_QCTRL 0x2e
# define AX_CLK_SELECT 0x33
# define AX_CLK_SELECT_BCS 0x01
# define AX_CLK_SELECT_ACS 0x02
# define AX_CLK_SELECT_ULR 0x08
# define AX_RXCOE_CTL 0x34
# define AX_RXCOE_IP 0x01
# define AX_RXCOE_TCP 0x02
# define AX_RXCOE_UDP 0x04
# define AX_RXCOE_TCPV6 0x20
# define AX_RXCOE_UDPV6 0x40
# define AX_TXCOE_CTL 0x35
# define AX_TXCOE_IP 0x01
# define AX_TXCOE_TCP 0x02
# define AX_TXCOE_UDP 0x04
# define AX_TXCOE_TCPV6 0x20
# define AX_TXCOE_UDPV6 0x40
# define AX_LEDCTRL 0x73
# define GMII_PHY_PHYSR 0x11
# define GMII_PHY_PHYSR_SMASK 0xc000
# define GMII_PHY_PHYSR_GIGA 0x8000
# define GMII_PHY_PHYSR_100 0x4000
# define GMII_PHY_PHYSR_FULL 0x2000
# define GMII_PHY_PHYSR_LINK 0x400
# define GMII_LED_ACT 0x1a
# define GMII_LED_ACTIVE_MASK 0xff8f
# define GMII_LED0_ACTIVE BIT(4)
# define GMII_LED1_ACTIVE BIT(5)
# define GMII_LED2_ACTIVE BIT(6)
# define GMII_LED_LINK 0x1c
# define GMII_LED_LINK_MASK 0xf888
# define GMII_LED0_LINK_10 BIT(0)
# define GMII_LED0_LINK_100 BIT(1)
# define GMII_LED0_LINK_1000 BIT(2)
# define GMII_LED1_LINK_10 BIT(4)
# define GMII_LED1_LINK_100 BIT(5)
# define GMII_LED1_LINK_1000 BIT(6)
# define GMII_LED2_LINK_10 BIT(8)
# define GMII_LED2_LINK_100 BIT(9)
# define GMII_LED2_LINK_1000 BIT(10)
# define LED0_ACTIVE BIT(0)
# define LED0_LINK_10 BIT(1)
# define LED0_LINK_100 BIT(2)
# define LED0_LINK_1000 BIT(3)
# define LED0_FD BIT(4)
# define LED0_USB3_MASK 0x001f
# define LED1_ACTIVE BIT(5)
# define LED1_LINK_10 BIT(6)
# define LED1_LINK_100 BIT(7)
# define LED1_LINK_1000 BIT(8)
# define LED1_FD BIT(9)
# define LED1_USB3_MASK 0x03e0
# define LED2_ACTIVE BIT(10)
# define LED2_LINK_1000 BIT(13)
# define LED2_LINK_100 BIT(12)
# define LED2_LINK_10 BIT(11)
# define LED2_FD BIT(14)
# define LED_VALID BIT(15)
# define LED2_USB3_MASK 0x7c00
# define GMII_PHYPAGE 0x1e
# define GMII_PHY_PAGE_SELECT 0x1f
# define GMII_PHY_PGSEL_EXT 0x0007
# define GMII_PHY_PGSEL_PAGE0 0x0000
struct ax88179_data {
u16 rxctl ;
u16 reserved ;
} ;
struct ax88179_int_data {
__le32 intdata1 ;
__le32 intdata2 ;
} ;
static const struct {
unsigned char ctrl , timer_l , timer_h , size , ifg ;
} AX88179_BULKIN_SIZE [ ] = {
{ 7 , 0x4f , 0 , 0x12 , 0xff } ,
{ 7 , 0x20 , 3 , 0x16 , 0xff } ,
{ 7 , 0xae , 7 , 0x18 , 0xff } ,
{ 7 , 0xcc , 0x4c , 0x18 , 8 } ,
} ;
static int __ax88179_read_cmd ( struct usbnet * dev , u8 cmd , u16 value , u16 index ,
u16 size , void * data , int in_pm )
{
int ret ;
int ( * fn ) ( struct usbnet * , u8 , u8 , u16 , u16 , void * , u16 ) ;
BUG_ON ( ! dev ) ;
if ( ! in_pm )
fn = usbnet_read_cmd ;
else
fn = usbnet_read_cmd_nopm ;
ret = fn ( dev , cmd , USB_DIR_IN | USB_TYPE_VENDOR | USB_RECIP_DEVICE ,
value , index , data , size ) ;
if ( unlikely ( ret < 0 ) )
netdev_warn ( dev - > net , " Failed to read reg index 0x%04x: %d \n " ,
index , ret ) ;
return ret ;
}
static int __ax88179_write_cmd ( struct usbnet * dev , u8 cmd , u16 value , u16 index ,
u16 size , void * data , int in_pm )
{
int ret ;
int ( * fn ) ( struct usbnet * , u8 , u8 , u16 , u16 , const void * , u16 ) ;
BUG_ON ( ! dev ) ;
if ( ! in_pm )
fn = usbnet_write_cmd ;
else
fn = usbnet_write_cmd_nopm ;
ret = fn ( dev , cmd , USB_DIR_OUT | USB_TYPE_VENDOR | USB_RECIP_DEVICE ,
value , index , data , size ) ;
if ( unlikely ( ret < 0 ) )
netdev_warn ( dev - > net , " Failed to write reg index 0x%04x: %d \n " ,
index , ret ) ;
return ret ;
}
static void ax88179_write_cmd_async ( struct usbnet * dev , u8 cmd , u16 value ,
u16 index , u16 size , void * data )
{
u16 buf ;
if ( 2 = = size ) {
buf = * ( ( u16 * ) data ) ;
cpu_to_le16s ( & buf ) ;
usbnet_write_cmd_async ( dev , cmd , USB_DIR_OUT | USB_TYPE_VENDOR |
USB_RECIP_DEVICE , value , index , & buf ,
size ) ;
} else {
usbnet_write_cmd_async ( dev , cmd , USB_DIR_OUT | USB_TYPE_VENDOR |
USB_RECIP_DEVICE , value , index , data ,
size ) ;
}
}
static int ax88179_read_cmd_nopm ( struct usbnet * dev , u8 cmd , u16 value ,
u16 index , u16 size , void * data )
{
int ret ;
if ( 2 = = size ) {
u16 buf ;
ret = __ax88179_read_cmd ( dev , cmd , value , index , size , & buf , 1 ) ;
le16_to_cpus ( & buf ) ;
* ( ( u16 * ) data ) = buf ;
} else if ( 4 = = size ) {
u32 buf ;
ret = __ax88179_read_cmd ( dev , cmd , value , index , size , & buf , 1 ) ;
le32_to_cpus ( & buf ) ;
* ( ( u32 * ) data ) = buf ;
} else {
ret = __ax88179_read_cmd ( dev , cmd , value , index , size , data , 1 ) ;
}
return ret ;
}
static int ax88179_write_cmd_nopm ( struct usbnet * dev , u8 cmd , u16 value ,
u16 index , u16 size , void * data )
{
int ret ;
if ( 2 = = size ) {
u16 buf ;
buf = * ( ( u16 * ) data ) ;
cpu_to_le16s ( & buf ) ;
ret = __ax88179_write_cmd ( dev , cmd , value , index ,
size , & buf , 1 ) ;
} else {
ret = __ax88179_write_cmd ( dev , cmd , value , index ,
size , data , 1 ) ;
}
return ret ;
}
static int ax88179_read_cmd ( struct usbnet * dev , u8 cmd , u16 value , u16 index ,
u16 size , void * data )
{
int ret ;
if ( 2 = = size ) {
u16 buf ;
ret = __ax88179_read_cmd ( dev , cmd , value , index , size , & buf , 0 ) ;
le16_to_cpus ( & buf ) ;
* ( ( u16 * ) data ) = buf ;
} else if ( 4 = = size ) {
u32 buf ;
ret = __ax88179_read_cmd ( dev , cmd , value , index , size , & buf , 0 ) ;
le32_to_cpus ( & buf ) ;
* ( ( u32 * ) data ) = buf ;
} else {
ret = __ax88179_read_cmd ( dev , cmd , value , index , size , data , 0 ) ;
}
return ret ;
}
static int ax88179_write_cmd ( struct usbnet * dev , u8 cmd , u16 value , u16 index ,
u16 size , void * data )
{
int ret ;
if ( 2 = = size ) {
u16 buf ;
buf = * ( ( u16 * ) data ) ;
cpu_to_le16s ( & buf ) ;
ret = __ax88179_write_cmd ( dev , cmd , value , index ,
size , & buf , 0 ) ;
} else {
ret = __ax88179_write_cmd ( dev , cmd , value , index ,
size , data , 0 ) ;
}
return ret ;
}
static void ax88179_status ( struct usbnet * dev , struct urb * urb )
{
struct ax88179_int_data * event ;
u32 link ;
if ( urb - > actual_length < 8 )
return ;
event = urb - > transfer_buffer ;
le32_to_cpus ( ( void * ) & event - > intdata1 ) ;
link = ( ( ( __force u32 ) event - > intdata1 ) & AX_INT_PPLS_LINK ) > > 16 ;
if ( netif_carrier_ok ( dev - > net ) ! = link ) {
2013-04-11 04:40:35 +00:00
usbnet_link_change ( dev , link , 1 ) ;
2013-03-02 00:41:11 +00:00
netdev_info ( dev - > net , " ax88179 - Link status is: %d \n " , link ) ;
}
}
static int ax88179_mdio_read ( struct net_device * netdev , int phy_id , int loc )
{
struct usbnet * dev = netdev_priv ( netdev ) ;
u16 res ;
ax88179_read_cmd ( dev , AX_ACCESS_PHY , phy_id , ( __u16 ) loc , 2 , & res ) ;
return res ;
}
static void ax88179_mdio_write ( struct net_device * netdev , int phy_id , int loc ,
int val )
{
struct usbnet * dev = netdev_priv ( netdev ) ;
u16 res = ( u16 ) val ;
ax88179_write_cmd ( dev , AX_ACCESS_PHY , phy_id , ( __u16 ) loc , 2 , & res ) ;
}
static int ax88179_suspend ( struct usb_interface * intf , pm_message_t message )
{
struct usbnet * dev = usb_get_intfdata ( intf ) ;
u16 tmp16 ;
u8 tmp8 ;
usbnet_suspend ( intf , message ) ;
/* Disable RX path */
ax88179_read_cmd_nopm ( dev , AX_ACCESS_MAC , AX_MEDIUM_STATUS_MODE ,
2 , 2 , & tmp16 ) ;
tmp16 & = ~ AX_MEDIUM_RECEIVE_EN ;
ax88179_write_cmd_nopm ( dev , AX_ACCESS_MAC , AX_MEDIUM_STATUS_MODE ,
2 , 2 , & tmp16 ) ;
/* Force bulk-in zero length */
ax88179_read_cmd_nopm ( dev , AX_ACCESS_MAC , AX_PHYPWR_RSTCTL ,
2 , 2 , & tmp16 ) ;
tmp16 | = AX_PHYPWR_RSTCTL_BZ | AX_PHYPWR_RSTCTL_IPRL ;
ax88179_write_cmd_nopm ( dev , AX_ACCESS_MAC , AX_PHYPWR_RSTCTL ,
2 , 2 , & tmp16 ) ;
/* change clock */
tmp8 = 0 ;
ax88179_write_cmd_nopm ( dev , AX_ACCESS_MAC , AX_CLK_SELECT , 1 , 1 , & tmp8 ) ;
/* Configure RX control register => stop operation */
tmp16 = AX_RX_CTL_STOP ;
ax88179_write_cmd_nopm ( dev , AX_ACCESS_MAC , AX_RX_CTL , 2 , 2 , & tmp16 ) ;
return 0 ;
}
/* This function is used to enable the autodetach function. */
/* This function is determined by offset 0x43 of EEPROM */
static int ax88179_auto_detach ( struct usbnet * dev , int in_pm )
{
u16 tmp16 ;
u8 tmp8 ;
int ( * fnr ) ( struct usbnet * , u8 , u16 , u16 , u16 , void * ) ;
int ( * fnw ) ( struct usbnet * , u8 , u16 , u16 , u16 , void * ) ;
if ( ! in_pm ) {
fnr = ax88179_read_cmd ;
fnw = ax88179_write_cmd ;
} else {
fnr = ax88179_read_cmd_nopm ;
fnw = ax88179_write_cmd_nopm ;
}
if ( fnr ( dev , AX_ACCESS_EEPROM , 0x43 , 1 , 2 , & tmp16 ) < 0 )
return 0 ;
if ( ( tmp16 = = 0xFFFF ) | | ( ! ( tmp16 & 0x0100 ) ) )
return 0 ;
/* Enable Auto Detach bit */
tmp8 = 0 ;
fnr ( dev , AX_ACCESS_MAC , AX_CLK_SELECT , 1 , 1 , & tmp8 ) ;
tmp8 | = AX_CLK_SELECT_ULR ;
fnw ( dev , AX_ACCESS_MAC , AX_CLK_SELECT , 1 , 1 , & tmp8 ) ;
fnr ( dev , AX_ACCESS_MAC , AX_PHYPWR_RSTCTL , 2 , 2 , & tmp16 ) ;
tmp16 | = AX_PHYPWR_RSTCTL_AT ;
fnw ( dev , AX_ACCESS_MAC , AX_PHYPWR_RSTCTL , 2 , 2 , & tmp16 ) ;
return 0 ;
}
static int ax88179_resume ( struct usb_interface * intf )
{
struct usbnet * dev = usb_get_intfdata ( intf ) ;
u16 tmp16 ;
u8 tmp8 ;
2013-04-11 04:40:35 +00:00
usbnet_link_change ( dev , 0 , 0 ) ;
2013-03-02 00:41:11 +00:00
/* Power up ethernet PHY */
tmp16 = 0 ;
ax88179_write_cmd_nopm ( dev , AX_ACCESS_MAC , AX_PHYPWR_RSTCTL ,
2 , 2 , & tmp16 ) ;
udelay ( 1000 ) ;
tmp16 = AX_PHYPWR_RSTCTL_IPRL ;
ax88179_write_cmd_nopm ( dev , AX_ACCESS_MAC , AX_PHYPWR_RSTCTL ,
2 , 2 , & tmp16 ) ;
msleep ( 200 ) ;
/* Ethernet PHY Auto Detach*/
ax88179_auto_detach ( dev , 1 ) ;
/* Enable clock */
ax88179_read_cmd_nopm ( dev , AX_ACCESS_MAC , AX_CLK_SELECT , 1 , 1 , & tmp8 ) ;
tmp8 | = AX_CLK_SELECT_ACS | AX_CLK_SELECT_BCS ;
ax88179_write_cmd_nopm ( dev , AX_ACCESS_MAC , AX_CLK_SELECT , 1 , 1 , & tmp8 ) ;
msleep ( 100 ) ;
/* Configure RX control register => start operation */
tmp16 = AX_RX_CTL_DROPCRCERR | AX_RX_CTL_IPE | AX_RX_CTL_START |
AX_RX_CTL_AP | AX_RX_CTL_AMALL | AX_RX_CTL_AB ;
ax88179_write_cmd_nopm ( dev , AX_ACCESS_MAC , AX_RX_CTL , 2 , 2 , & tmp16 ) ;
return usbnet_resume ( intf ) ;
}
static void
ax88179_get_wol ( struct net_device * net , struct ethtool_wolinfo * wolinfo )
{
struct usbnet * dev = netdev_priv ( net ) ;
u8 opt ;
if ( ax88179_read_cmd ( dev , AX_ACCESS_MAC , AX_MONITOR_MOD ,
1 , 1 , & opt ) < 0 ) {
wolinfo - > supported = 0 ;
wolinfo - > wolopts = 0 ;
return ;
}
wolinfo - > supported = WAKE_PHY | WAKE_MAGIC ;
wolinfo - > wolopts = 0 ;
if ( opt & AX_MONITOR_MODE_RWLC )
wolinfo - > wolopts | = WAKE_PHY ;
if ( opt & AX_MONITOR_MODE_RWMP )
wolinfo - > wolopts | = WAKE_MAGIC ;
}
static int
ax88179_set_wol ( struct net_device * net , struct ethtool_wolinfo * wolinfo )
{
struct usbnet * dev = netdev_priv ( net ) ;
u8 opt = 0 ;
if ( wolinfo - > wolopts & WAKE_PHY )
opt | = AX_MONITOR_MODE_RWLC ;
if ( wolinfo - > wolopts & WAKE_MAGIC )
opt | = AX_MONITOR_MODE_RWMP ;
if ( ax88179_write_cmd ( dev , AX_ACCESS_MAC , AX_MONITOR_MOD ,
1 , 1 , & opt ) < 0 )
return - EINVAL ;
return 0 ;
}
static int ax88179_get_eeprom_len ( struct net_device * net )
{
return AX_EEPROM_LEN ;
}
static int
ax88179_get_eeprom ( struct net_device * net , struct ethtool_eeprom * eeprom ,
u8 * data )
{
struct usbnet * dev = netdev_priv ( net ) ;
u16 * eeprom_buff ;
int first_word , last_word ;
int i , ret ;
if ( eeprom - > len = = 0 )
return - EINVAL ;
eeprom - > magic = AX88179_EEPROM_MAGIC ;
first_word = eeprom - > offset > > 1 ;
last_word = ( eeprom - > offset + eeprom - > len - 1 ) > > 1 ;
eeprom_buff = kmalloc ( sizeof ( u16 ) * ( last_word - first_word + 1 ) ,
GFP_KERNEL ) ;
if ( ! eeprom_buff )
return - ENOMEM ;
/* ax88179/178A returns 2 bytes from eeprom on read */
for ( i = first_word ; i < = last_word ; i + + ) {
ret = __ax88179_read_cmd ( dev , AX_ACCESS_EEPROM , i , 1 , 2 ,
& eeprom_buff [ i - first_word ] ,
0 ) ;
if ( ret < 0 ) {
kfree ( eeprom_buff ) ;
return - EIO ;
}
}
memcpy ( data , ( u8 * ) eeprom_buff + ( eeprom - > offset & 1 ) , eeprom - > len ) ;
kfree ( eeprom_buff ) ;
return 0 ;
}
static int ax88179_get_settings ( struct net_device * net , struct ethtool_cmd * cmd )
{
struct usbnet * dev = netdev_priv ( net ) ;
return mii_ethtool_gset ( & dev - > mii , cmd ) ;
}
static int ax88179_set_settings ( struct net_device * net , struct ethtool_cmd * cmd )
{
struct usbnet * dev = netdev_priv ( net ) ;
return mii_ethtool_sset ( & dev - > mii , cmd ) ;
}
static int ax88179_ioctl ( struct net_device * net , struct ifreq * rq , int cmd )
{
struct usbnet * dev = netdev_priv ( net ) ;
return generic_mii_ioctl ( & dev - > mii , if_mii ( rq ) , cmd , NULL ) ;
}
static const struct ethtool_ops ax88179_ethtool_ops = {
. get_link = ethtool_op_get_link ,
. get_msglevel = usbnet_get_msglevel ,
. set_msglevel = usbnet_set_msglevel ,
. get_wol = ax88179_get_wol ,
. set_wol = ax88179_set_wol ,
. get_eeprom_len = ax88179_get_eeprom_len ,
. get_eeprom = ax88179_get_eeprom ,
. get_settings = ax88179_get_settings ,
. set_settings = ax88179_set_settings ,
. nway_reset = usbnet_nway_reset ,
} ;
static void ax88179_set_multicast ( struct net_device * net )
{
struct usbnet * dev = netdev_priv ( net ) ;
struct ax88179_data * data = ( struct ax88179_data * ) dev - > data ;
u8 * m_filter = ( ( u8 * ) dev - > data ) + 12 ;
data - > rxctl = ( AX_RX_CTL_START | AX_RX_CTL_AB | AX_RX_CTL_IPE ) ;
if ( net - > flags & IFF_PROMISC ) {
data - > rxctl | = AX_RX_CTL_PRO ;
} else if ( net - > flags & IFF_ALLMULTI | |
netdev_mc_count ( net ) > AX_MAX_MCAST ) {
data - > rxctl | = AX_RX_CTL_AMALL ;
} else if ( netdev_mc_empty ( net ) ) {
/* just broadcast and directed */
} else {
/* We use the 20 byte dev->data for our 8 byte filter buffer
* to avoid allocating memory that is tricky to free later
*/
u32 crc_bits ;
struct netdev_hw_addr * ha ;
memset ( m_filter , 0 , AX_MCAST_FLTSIZE ) ;
netdev_for_each_mc_addr ( ha , net ) {
crc_bits = ether_crc ( ETH_ALEN , ha - > addr ) > > 26 ;
* ( m_filter + ( crc_bits > > 3 ) ) | = ( 1 < < ( crc_bits & 7 ) ) ;
}
ax88179_write_cmd_async ( dev , AX_ACCESS_MAC , AX_MULFLTARY ,
AX_MCAST_FLTSIZE , AX_MCAST_FLTSIZE ,
m_filter ) ;
data - > rxctl | = AX_RX_CTL_AM ;
}
ax88179_write_cmd_async ( dev , AX_ACCESS_MAC , AX_RX_CTL ,
2 , 2 , & data - > rxctl ) ;
}
static int
ax88179_set_features ( struct net_device * net , netdev_features_t features )
{
u8 tmp ;
struct usbnet * dev = netdev_priv ( net ) ;
netdev_features_t changed = net - > features ^ features ;
if ( changed & NETIF_F_IP_CSUM ) {
ax88179_read_cmd ( dev , AX_ACCESS_MAC , AX_TXCOE_CTL , 1 , 1 , & tmp ) ;
tmp ^ = AX_TXCOE_TCP | AX_TXCOE_UDP ;
ax88179_write_cmd ( dev , AX_ACCESS_MAC , AX_TXCOE_CTL , 1 , 1 , & tmp ) ;
}
if ( changed & NETIF_F_IPV6_CSUM ) {
ax88179_read_cmd ( dev , AX_ACCESS_MAC , AX_TXCOE_CTL , 1 , 1 , & tmp ) ;
tmp ^ = AX_TXCOE_TCPV6 | AX_TXCOE_UDPV6 ;
ax88179_write_cmd ( dev , AX_ACCESS_MAC , AX_TXCOE_CTL , 1 , 1 , & tmp ) ;
}
if ( changed & NETIF_F_RXCSUM ) {
ax88179_read_cmd ( dev , AX_ACCESS_MAC , AX_RXCOE_CTL , 1 , 1 , & tmp ) ;
tmp ^ = AX_RXCOE_IP | AX_RXCOE_TCP | AX_RXCOE_UDP |
AX_RXCOE_TCPV6 | AX_RXCOE_UDPV6 ;
ax88179_write_cmd ( dev , AX_ACCESS_MAC , AX_RXCOE_CTL , 1 , 1 , & tmp ) ;
}
return 0 ;
}
static int ax88179_change_mtu ( struct net_device * net , int new_mtu )
{
struct usbnet * dev = netdev_priv ( net ) ;
u16 tmp16 ;
if ( new_mtu < = 0 | | new_mtu > 4088 )
return - EINVAL ;
net - > mtu = new_mtu ;
dev - > hard_mtu = net - > mtu + net - > hard_header_len ;
if ( net - > mtu > 1500 ) {
ax88179_read_cmd ( dev , AX_ACCESS_MAC , AX_MEDIUM_STATUS_MODE ,
2 , 2 , & tmp16 ) ;
tmp16 | = AX_MEDIUM_JUMBO_EN ;
ax88179_write_cmd ( dev , AX_ACCESS_MAC , AX_MEDIUM_STATUS_MODE ,
2 , 2 , & tmp16 ) ;
} else {
ax88179_read_cmd ( dev , AX_ACCESS_MAC , AX_MEDIUM_STATUS_MODE ,
2 , 2 , & tmp16 ) ;
tmp16 & = ~ AX_MEDIUM_JUMBO_EN ;
ax88179_write_cmd ( dev , AX_ACCESS_MAC , AX_MEDIUM_STATUS_MODE ,
2 , 2 , & tmp16 ) ;
}
return 0 ;
}
static int ax88179_set_mac_addr ( struct net_device * net , void * p )
{
struct usbnet * dev = netdev_priv ( net ) ;
struct sockaddr * addr = p ;
if ( netif_running ( net ) )
return - EBUSY ;
if ( ! is_valid_ether_addr ( addr - > sa_data ) )
return - EADDRNOTAVAIL ;
memcpy ( net - > dev_addr , addr - > sa_data , ETH_ALEN ) ;
/* Set the MAC address */
return ax88179_write_cmd ( dev , AX_ACCESS_MAC , AX_NODE_ID , ETH_ALEN ,
ETH_ALEN , net - > dev_addr ) ;
}
static const struct net_device_ops ax88179_netdev_ops = {
. ndo_open = usbnet_open ,
. ndo_stop = usbnet_stop ,
. ndo_start_xmit = usbnet_start_xmit ,
. ndo_tx_timeout = usbnet_tx_timeout ,
. ndo_change_mtu = ax88179_change_mtu ,
. ndo_set_mac_address = ax88179_set_mac_addr ,
. ndo_validate_addr = eth_validate_addr ,
. ndo_do_ioctl = ax88179_ioctl ,
. ndo_set_rx_mode = ax88179_set_multicast ,
. ndo_set_features = ax88179_set_features ,
} ;
static int ax88179_check_eeprom ( struct usbnet * dev )
{
u8 i , buf , eeprom [ 20 ] ;
u16 csum , delay = HZ / 10 ;
unsigned long jtimeout ;
/* Read EEPROM content */
for ( i = 0 ; i < 6 ; i + + ) {
buf = i ;
if ( ax88179_write_cmd ( dev , AX_ACCESS_MAC , AX_SROM_ADDR ,
1 , 1 , & buf ) < 0 )
return - EINVAL ;
buf = EEP_RD ;
if ( ax88179_write_cmd ( dev , AX_ACCESS_MAC , AX_SROM_CMD ,
1 , 1 , & buf ) < 0 )
return - EINVAL ;
jtimeout = jiffies + delay ;
do {
ax88179_read_cmd ( dev , AX_ACCESS_MAC , AX_SROM_CMD ,
1 , 1 , & buf ) ;
if ( time_after ( jiffies , jtimeout ) )
return - EINVAL ;
} while ( buf & EEP_BUSY ) ;
__ax88179_read_cmd ( dev , AX_ACCESS_MAC , AX_SROM_DATA_LOW ,
2 , 2 , & eeprom [ i * 2 ] , 0 ) ;
if ( ( i = = 0 ) & & ( eeprom [ 0 ] = = 0xFF ) )
return - EINVAL ;
}
csum = eeprom [ 6 ] + eeprom [ 7 ] + eeprom [ 8 ] + eeprom [ 9 ] ;
csum = ( csum > > 8 ) + ( csum & 0xff ) ;
if ( ( csum + eeprom [ 10 ] ) ! = 0xff )
return - EINVAL ;
return 0 ;
}
static int ax88179_check_efuse ( struct usbnet * dev , u16 * ledmode )
{
u8 i ;
u8 efuse [ 64 ] ;
u16 csum = 0 ;
if ( ax88179_read_cmd ( dev , AX_ACCESS_EFUS , 0 , 64 , 64 , efuse ) < 0 )
return - EINVAL ;
if ( * efuse = = 0xFF )
return - EINVAL ;
for ( i = 0 ; i < 64 ; i + + )
csum = csum + efuse [ i ] ;
while ( csum > 255 )
csum = ( csum & 0x00FF ) + ( ( csum > > 8 ) & 0x00FF ) ;
if ( csum ! = 0xFF )
return - EINVAL ;
* ledmode = ( efuse [ 51 ] < < 8 ) | efuse [ 52 ] ;
return 0 ;
}
static int ax88179_convert_old_led ( struct usbnet * dev , u16 * ledvalue )
{
u16 led ;
/* Loaded the old eFuse LED Mode */
if ( ax88179_read_cmd ( dev , AX_ACCESS_EEPROM , 0x3C , 1 , 2 , & led ) < 0 )
return - EINVAL ;
led > > = 8 ;
switch ( led ) {
case 0xFF :
led = LED0_ACTIVE | LED1_LINK_10 | LED1_LINK_100 |
LED1_LINK_1000 | LED2_ACTIVE | LED2_LINK_10 |
LED2_LINK_100 | LED2_LINK_1000 | LED_VALID ;
break ;
case 0xFE :
led = LED0_ACTIVE | LED1_LINK_1000 | LED2_LINK_100 | LED_VALID ;
break ;
case 0xFD :
led = LED0_ACTIVE | LED1_LINK_1000 | LED2_LINK_100 |
LED2_LINK_10 | LED_VALID ;
break ;
case 0xFC :
led = LED0_ACTIVE | LED1_ACTIVE | LED1_LINK_1000 | LED2_ACTIVE |
LED2_LINK_100 | LED2_LINK_10 | LED_VALID ;
break ;
default :
led = LED0_ACTIVE | LED1_LINK_10 | LED1_LINK_100 |
LED1_LINK_1000 | LED2_ACTIVE | LED2_LINK_10 |
LED2_LINK_100 | LED2_LINK_1000 | LED_VALID ;
break ;
}
* ledvalue = led ;
return 0 ;
}
static int ax88179_led_setting ( struct usbnet * dev )
{
u8 ledfd , value = 0 ;
u16 tmp , ledact , ledlink , ledvalue = 0 , delay = HZ / 10 ;
unsigned long jtimeout ;
/* Check AX88179 version. UA1 or UA2*/
ax88179_read_cmd ( dev , AX_ACCESS_MAC , GENERAL_STATUS , 1 , 1 , & value ) ;
if ( ! ( value & AX_SECLD ) ) { /* UA1 */
value = AX_GPIO_CTRL_GPIO3EN | AX_GPIO_CTRL_GPIO2EN |
AX_GPIO_CTRL_GPIO1EN ;
if ( ax88179_write_cmd ( dev , AX_ACCESS_MAC , AX_GPIO_CTRL ,
1 , 1 , & value ) < 0 )
return - EINVAL ;
}
/* Check EEPROM */
if ( ! ax88179_check_eeprom ( dev ) ) {
value = 0x42 ;
if ( ax88179_write_cmd ( dev , AX_ACCESS_MAC , AX_SROM_ADDR ,
1 , 1 , & value ) < 0 )
return - EINVAL ;
value = EEP_RD ;
if ( ax88179_write_cmd ( dev , AX_ACCESS_MAC , AX_SROM_CMD ,
1 , 1 , & value ) < 0 )
return - EINVAL ;
jtimeout = jiffies + delay ;
do {
ax88179_read_cmd ( dev , AX_ACCESS_MAC , AX_SROM_CMD ,
1 , 1 , & value ) ;
if ( time_after ( jiffies , jtimeout ) )
return - EINVAL ;
} while ( value & EEP_BUSY ) ;
ax88179_read_cmd ( dev , AX_ACCESS_MAC , AX_SROM_DATA_HIGH ,
1 , 1 , & value ) ;
ledvalue = ( value < < 8 ) ;
ax88179_read_cmd ( dev , AX_ACCESS_MAC , AX_SROM_DATA_LOW ,
1 , 1 , & value ) ;
ledvalue | = value ;
/* load internal ROM for defaule setting */
if ( ( ledvalue = = 0xFFFF ) | | ( ( ledvalue & LED_VALID ) = = 0 ) )
ax88179_convert_old_led ( dev , & ledvalue ) ;
} else if ( ! ax88179_check_efuse ( dev , & ledvalue ) ) {
if ( ( ledvalue = = 0xFFFF ) | | ( ( ledvalue & LED_VALID ) = = 0 ) )
ax88179_convert_old_led ( dev , & ledvalue ) ;
} else {
ax88179_convert_old_led ( dev , & ledvalue ) ;
}
tmp = GMII_PHY_PGSEL_EXT ;
ax88179_write_cmd ( dev , AX_ACCESS_PHY , AX88179_PHY_ID ,
GMII_PHY_PAGE_SELECT , 2 , & tmp ) ;
tmp = 0x2c ;
ax88179_write_cmd ( dev , AX_ACCESS_PHY , AX88179_PHY_ID ,
GMII_PHYPAGE , 2 , & tmp ) ;
ax88179_read_cmd ( dev , AX_ACCESS_PHY , AX88179_PHY_ID ,
GMII_LED_ACT , 2 , & ledact ) ;
ax88179_read_cmd ( dev , AX_ACCESS_PHY , AX88179_PHY_ID ,
GMII_LED_LINK , 2 , & ledlink ) ;
ledact & = GMII_LED_ACTIVE_MASK ;
ledlink & = GMII_LED_LINK_MASK ;
if ( ledvalue & LED0_ACTIVE )
ledact | = GMII_LED0_ACTIVE ;
if ( ledvalue & LED1_ACTIVE )
ledact | = GMII_LED1_ACTIVE ;
if ( ledvalue & LED2_ACTIVE )
ledact | = GMII_LED2_ACTIVE ;
if ( ledvalue & LED0_LINK_10 )
ledlink | = GMII_LED0_LINK_10 ;
if ( ledvalue & LED1_LINK_10 )
ledlink | = GMII_LED1_LINK_10 ;
if ( ledvalue & LED2_LINK_10 )
ledlink | = GMII_LED2_LINK_10 ;
if ( ledvalue & LED0_LINK_100 )
ledlink | = GMII_LED0_LINK_100 ;
if ( ledvalue & LED1_LINK_100 )
ledlink | = GMII_LED1_LINK_100 ;
if ( ledvalue & LED2_LINK_100 )
ledlink | = GMII_LED2_LINK_100 ;
if ( ledvalue & LED0_LINK_1000 )
ledlink | = GMII_LED0_LINK_1000 ;
if ( ledvalue & LED1_LINK_1000 )
ledlink | = GMII_LED1_LINK_1000 ;
if ( ledvalue & LED2_LINK_1000 )
ledlink | = GMII_LED2_LINK_1000 ;
tmp = ledact ;
ax88179_write_cmd ( dev , AX_ACCESS_PHY , AX88179_PHY_ID ,
GMII_LED_ACT , 2 , & tmp ) ;
tmp = ledlink ;
ax88179_write_cmd ( dev , AX_ACCESS_PHY , AX88179_PHY_ID ,
GMII_LED_LINK , 2 , & tmp ) ;
tmp = GMII_PHY_PGSEL_PAGE0 ;
ax88179_write_cmd ( dev , AX_ACCESS_PHY , AX88179_PHY_ID ,
GMII_PHY_PAGE_SELECT , 2 , & tmp ) ;
/* LED full duplex setting */
ledfd = 0 ;
if ( ledvalue & LED0_FD )
ledfd | = 0x01 ;
else if ( ( ledvalue & LED0_USB3_MASK ) = = 0 )
ledfd | = 0x02 ;
if ( ledvalue & LED1_FD )
ledfd | = 0x04 ;
else if ( ( ledvalue & LED1_USB3_MASK ) = = 0 )
ledfd | = 0x08 ;
if ( ledvalue & LED2_FD )
ledfd | = 0x10 ;
else if ( ( ledvalue & LED2_USB3_MASK ) = = 0 )
ledfd | = 0x20 ;
ax88179_write_cmd ( dev , AX_ACCESS_MAC , AX_LEDCTRL , 1 , 1 , & ledfd ) ;
return 0 ;
}
static int ax88179_bind ( struct usbnet * dev , struct usb_interface * intf )
{
u8 buf [ 5 ] ;
u16 * tmp16 ;
u8 * tmp ;
struct ax88179_data * ax179_data = ( struct ax88179_data * ) dev - > data ;
usbnet_get_endpoints ( dev , intf ) ;
tmp16 = ( u16 * ) buf ;
tmp = ( u8 * ) buf ;
memset ( ax179_data , 0 , sizeof ( * ax179_data ) ) ;
/* Power up ethernet PHY */
* tmp16 = 0 ;
ax88179_write_cmd ( dev , AX_ACCESS_MAC , AX_PHYPWR_RSTCTL , 2 , 2 , tmp16 ) ;
* tmp16 = AX_PHYPWR_RSTCTL_IPRL ;
ax88179_write_cmd ( dev , AX_ACCESS_MAC , AX_PHYPWR_RSTCTL , 2 , 2 , tmp16 ) ;
msleep ( 200 ) ;
* tmp = AX_CLK_SELECT_ACS | AX_CLK_SELECT_BCS ;
ax88179_write_cmd ( dev , AX_ACCESS_MAC , AX_CLK_SELECT , 1 , 1 , tmp ) ;
msleep ( 100 ) ;
ax88179_read_cmd ( dev , AX_ACCESS_MAC , AX_NODE_ID , ETH_ALEN ,
ETH_ALEN , dev - > net - > dev_addr ) ;
memcpy ( dev - > net - > perm_addr , dev - > net - > dev_addr , ETH_ALEN ) ;
/* RX bulk configuration */
memcpy ( tmp , & AX88179_BULKIN_SIZE [ 0 ] , 5 ) ;
ax88179_write_cmd ( dev , AX_ACCESS_MAC , AX_RX_BULKIN_QCTRL , 5 , 5 , tmp ) ;
dev - > rx_urb_size = 1024 * 20 ;
* tmp = 0x34 ;
ax88179_write_cmd ( dev , AX_ACCESS_MAC , AX_PAUSE_WATERLVL_LOW , 1 , 1 , tmp ) ;
* tmp = 0x52 ;
ax88179_write_cmd ( dev , AX_ACCESS_MAC , AX_PAUSE_WATERLVL_HIGH ,
1 , 1 , tmp ) ;
dev - > net - > netdev_ops = & ax88179_netdev_ops ;
dev - > net - > ethtool_ops = & ax88179_ethtool_ops ;
dev - > net - > needed_headroom = 8 ;
/* Initialize MII structure */
dev - > mii . dev = dev - > net ;
dev - > mii . mdio_read = ax88179_mdio_read ;
dev - > mii . mdio_write = ax88179_mdio_write ;
dev - > mii . phy_id_mask = 0xff ;
dev - > mii . reg_num_mask = 0xff ;
dev - > mii . phy_id = 0x03 ;
dev - > mii . supports_gmii = 1 ;
dev - > net - > features | = NETIF_F_IP_CSUM | NETIF_F_IPV6_CSUM |
NETIF_F_RXCSUM | NETIF_F_SG | NETIF_F_TSO ;
dev - > net - > hw_features | = NETIF_F_IP_CSUM | NETIF_F_IPV6_CSUM |
NETIF_F_RXCSUM | NETIF_F_SG | NETIF_F_TSO ;
/* Enable checksum offload */
* tmp = AX_RXCOE_IP | AX_RXCOE_TCP | AX_RXCOE_UDP |
AX_RXCOE_TCPV6 | AX_RXCOE_UDPV6 ;
ax88179_write_cmd ( dev , AX_ACCESS_MAC , AX_RXCOE_CTL , 1 , 1 , tmp ) ;
* tmp = AX_TXCOE_IP | AX_TXCOE_TCP | AX_TXCOE_UDP |
AX_TXCOE_TCPV6 | AX_TXCOE_UDPV6 ;
ax88179_write_cmd ( dev , AX_ACCESS_MAC , AX_TXCOE_CTL , 1 , 1 , tmp ) ;
/* Configure RX control register => start operation */
* tmp16 = AX_RX_CTL_DROPCRCERR | AX_RX_CTL_IPE | AX_RX_CTL_START |
AX_RX_CTL_AP | AX_RX_CTL_AMALL | AX_RX_CTL_AB ;
ax88179_write_cmd ( dev , AX_ACCESS_MAC , AX_RX_CTL , 2 , 2 , tmp16 ) ;
* tmp = AX_MONITOR_MODE_PMETYPE | AX_MONITOR_MODE_PMEPOL |
AX_MONITOR_MODE_RWMP ;
ax88179_write_cmd ( dev , AX_ACCESS_MAC , AX_MONITOR_MOD , 1 , 1 , tmp ) ;
/* Configure default medium type => giga */
* tmp16 = AX_MEDIUM_RECEIVE_EN | AX_MEDIUM_TXFLOW_CTRLEN |
AX_MEDIUM_RXFLOW_CTRLEN | AX_MEDIUM_ALWAYS_ONE |
AX_MEDIUM_FULL_DUPLEX | AX_MEDIUM_GIGAMODE ;
ax88179_write_cmd ( dev , AX_ACCESS_MAC , AX_MEDIUM_STATUS_MODE ,
2 , 2 , tmp16 ) ;
ax88179_led_setting ( dev ) ;
/* Restart autoneg */
mii_nway_restart ( & dev - > mii ) ;
2013-04-11 04:40:35 +00:00
usbnet_link_change ( dev , 0 , 0 ) ;
2013-03-02 00:41:11 +00:00
return 0 ;
}
static void ax88179_unbind ( struct usbnet * dev , struct usb_interface * intf )
{
u16 tmp16 ;
/* Configure RX control register => stop operation */
tmp16 = AX_RX_CTL_STOP ;
ax88179_write_cmd ( dev , AX_ACCESS_MAC , AX_RX_CTL , 2 , 2 , & tmp16 ) ;
tmp16 = 0 ;
ax88179_write_cmd ( dev , AX_ACCESS_MAC , AX_CLK_SELECT , 1 , 1 , & tmp16 ) ;
/* Power down ethernet PHY */
tmp16 = 0 ;
ax88179_write_cmd ( dev , AX_ACCESS_MAC , AX_PHYPWR_RSTCTL , 2 , 2 , & tmp16 ) ;
}
static void
ax88179_rx_checksum ( struct sk_buff * skb , u32 * pkt_hdr )
{
skb - > ip_summed = CHECKSUM_NONE ;
/* checksum error bit is set */
if ( ( * pkt_hdr & AX_RXHDR_L3CSUM_ERR ) | |
( * pkt_hdr & AX_RXHDR_L4CSUM_ERR ) )
return ;
/* It must be a TCP or UDP packet with a valid checksum */
if ( ( ( * pkt_hdr & AX_RXHDR_L4_TYPE_MASK ) = = AX_RXHDR_L4_TYPE_TCP ) | |
( ( * pkt_hdr & AX_RXHDR_L4_TYPE_MASK ) = = AX_RXHDR_L4_TYPE_UDP ) )
skb - > ip_summed = CHECKSUM_UNNECESSARY ;
}
static int ax88179_rx_fixup ( struct usbnet * dev , struct sk_buff * skb )
{
struct sk_buff * ax_skb ;
int pkt_cnt ;
u32 rx_hdr ;
u16 hdr_off ;
u32 * pkt_hdr ;
skb_trim ( skb , skb - > len - 4 ) ;
memcpy ( & rx_hdr , skb_tail_pointer ( skb ) , 4 ) ;
le32_to_cpus ( & rx_hdr ) ;
pkt_cnt = ( u16 ) rx_hdr ;
hdr_off = ( u16 ) ( rx_hdr > > 16 ) ;
pkt_hdr = ( u32 * ) ( skb - > data + hdr_off ) ;
while ( pkt_cnt - - ) {
u16 pkt_len ;
le32_to_cpus ( pkt_hdr ) ;
pkt_len = ( * pkt_hdr > > 16 ) & 0x1fff ;
/* Check CRC or runt packet */
if ( ( * pkt_hdr & AX_RXHDR_CRC_ERR ) | |
( * pkt_hdr & AX_RXHDR_DROP_ERR ) ) {
skb_pull ( skb , ( pkt_len + 7 ) & 0xFFF8 ) ;
pkt_hdr + + ;
continue ;
}
if ( pkt_cnt = = 0 ) {
/* Skip IP alignment psudo header */
skb_pull ( skb , 2 ) ;
skb - > len = pkt_len ;
skb_set_tail_pointer ( skb , pkt_len ) ;
skb - > truesize = pkt_len + sizeof ( struct sk_buff ) ;
ax88179_rx_checksum ( skb , pkt_hdr ) ;
return 1 ;
}
ax_skb = skb_clone ( skb , GFP_ATOMIC ) ;
if ( ax_skb ) {
ax_skb - > len = pkt_len ;
ax_skb - > data = skb - > data + 2 ;
skb_set_tail_pointer ( ax_skb , pkt_len ) ;
ax_skb - > truesize = pkt_len + sizeof ( struct sk_buff ) ;
ax88179_rx_checksum ( ax_skb , pkt_hdr ) ;
usbnet_skb_return ( dev , ax_skb ) ;
} else {
return 0 ;
}
skb_pull ( skb , ( pkt_len + 7 ) & 0xFFF8 ) ;
pkt_hdr + + ;
}
return 1 ;
}
static struct sk_buff *
ax88179_tx_fixup ( struct usbnet * dev , struct sk_buff * skb , gfp_t flags )
{
u32 tx_hdr1 , tx_hdr2 ;
int frame_size = dev - > maxpacket ;
int mss = skb_shinfo ( skb ) - > gso_size ;
int headroom ;
int tailroom ;
tx_hdr1 = skb - > len ;
tx_hdr2 = mss ;
if ( ( ( skb - > len + 8 ) % frame_size ) = = 0 )
tx_hdr2 | = 0x80008000 ; /* Enable padding */
skb_linearize ( skb ) ;
headroom = skb_headroom ( skb ) ;
tailroom = skb_tailroom ( skb ) ;
if ( ! skb_header_cloned ( skb ) & &
! skb_cloned ( skb ) & &
( headroom + tailroom ) > = 8 ) {
if ( headroom < 8 ) {
skb - > data = memmove ( skb - > head + 8 , skb - > data , skb - > len ) ;
skb_set_tail_pointer ( skb , skb - > len ) ;
}
} else {
struct sk_buff * skb2 ;
skb2 = skb_copy_expand ( skb , 8 , 0 , flags ) ;
dev_kfree_skb_any ( skb ) ;
skb = skb2 ;
if ( ! skb )
return NULL ;
}
skb_push ( skb , 4 ) ;
cpu_to_le32s ( & tx_hdr2 ) ;
skb_copy_to_linear_data ( skb , & tx_hdr2 , 4 ) ;
skb_push ( skb , 4 ) ;
cpu_to_le32s ( & tx_hdr1 ) ;
skb_copy_to_linear_data ( skb , & tx_hdr1 , 4 ) ;
return skb ;
}
static int ax88179_link_reset ( struct usbnet * dev )
{
struct ax88179_data * ax179_data = ( struct ax88179_data * ) dev - > data ;
u8 tmp [ 5 ] , link_sts ;
u16 mode , tmp16 , delay = HZ / 10 ;
u32 tmp32 = 0x40000000 ;
unsigned long jtimeout ;
jtimeout = jiffies + delay ;
while ( tmp32 & 0x40000000 ) {
mode = 0 ;
ax88179_write_cmd ( dev , AX_ACCESS_MAC , AX_RX_CTL , 2 , 2 , & mode ) ;
ax88179_write_cmd ( dev , AX_ACCESS_MAC , AX_RX_CTL , 2 , 2 ,
& ax179_data - > rxctl ) ;
/*link up, check the usb device control TX FIFO full or empty*/
ax88179_read_cmd ( dev , 0x81 , 0x8c , 0 , 4 , & tmp32 ) ;
if ( time_after ( jiffies , jtimeout ) )
return 0 ;
}
mode = AX_MEDIUM_RECEIVE_EN | AX_MEDIUM_TXFLOW_CTRLEN |
AX_MEDIUM_RXFLOW_CTRLEN | AX_MEDIUM_ALWAYS_ONE ;
ax88179_read_cmd ( dev , AX_ACCESS_MAC , PHYSICAL_LINK_STATUS ,
1 , 1 , & link_sts ) ;
ax88179_read_cmd ( dev , AX_ACCESS_PHY , AX88179_PHY_ID ,
GMII_PHY_PHYSR , 2 , & tmp16 ) ;
if ( ! ( tmp16 & GMII_PHY_PHYSR_LINK ) ) {
return 0 ;
} else if ( GMII_PHY_PHYSR_GIGA = = ( tmp16 & GMII_PHY_PHYSR_SMASK ) ) {
mode | = AX_MEDIUM_GIGAMODE | AX_MEDIUM_EN_125MHZ ;
if ( dev - > net - > mtu > 1500 )
mode | = AX_MEDIUM_JUMBO_EN ;
if ( link_sts & AX_USB_SS )
memcpy ( tmp , & AX88179_BULKIN_SIZE [ 0 ] , 5 ) ;
else if ( link_sts & AX_USB_HS )
memcpy ( tmp , & AX88179_BULKIN_SIZE [ 1 ] , 5 ) ;
else
memcpy ( tmp , & AX88179_BULKIN_SIZE [ 3 ] , 5 ) ;
} else if ( GMII_PHY_PHYSR_100 = = ( tmp16 & GMII_PHY_PHYSR_SMASK ) ) {
mode | = AX_MEDIUM_PS ;
if ( link_sts & ( AX_USB_SS | AX_USB_HS ) )
memcpy ( tmp , & AX88179_BULKIN_SIZE [ 2 ] , 5 ) ;
else
memcpy ( tmp , & AX88179_BULKIN_SIZE [ 3 ] , 5 ) ;
} else {
memcpy ( tmp , & AX88179_BULKIN_SIZE [ 3 ] , 5 ) ;
}
/* RX bulk configuration */
ax88179_write_cmd ( dev , AX_ACCESS_MAC , AX_RX_BULKIN_QCTRL , 5 , 5 , tmp ) ;
dev - > rx_urb_size = ( 1024 * ( tmp [ 3 ] + 2 ) ) ;
if ( tmp16 & GMII_PHY_PHYSR_FULL )
mode | = AX_MEDIUM_FULL_DUPLEX ;
ax88179_write_cmd ( dev , AX_ACCESS_MAC , AX_MEDIUM_STATUS_MODE ,
2 , 2 , & mode ) ;
netif_carrier_on ( dev - > net ) ;
return 0 ;
}
static int ax88179_reset ( struct usbnet * dev )
{
u8 buf [ 5 ] ;
u16 * tmp16 ;
u8 * tmp ;
tmp16 = ( u16 * ) buf ;
tmp = ( u8 * ) buf ;
/* Power up ethernet PHY */
* tmp16 = 0 ;
ax88179_write_cmd ( dev , AX_ACCESS_MAC , AX_PHYPWR_RSTCTL , 2 , 2 , tmp16 ) ;
* tmp16 = AX_PHYPWR_RSTCTL_IPRL ;
ax88179_write_cmd ( dev , AX_ACCESS_MAC , AX_PHYPWR_RSTCTL , 2 , 2 , tmp16 ) ;
msleep ( 200 ) ;
* tmp = AX_CLK_SELECT_ACS | AX_CLK_SELECT_BCS ;
ax88179_write_cmd ( dev , AX_ACCESS_MAC , AX_CLK_SELECT , 1 , 1 , tmp ) ;
msleep ( 100 ) ;
/* Ethernet PHY Auto Detach*/
ax88179_auto_detach ( dev , 0 ) ;
ax88179_read_cmd ( dev , AX_ACCESS_MAC , AX_NODE_ID , ETH_ALEN , ETH_ALEN ,
dev - > net - > dev_addr ) ;
memcpy ( dev - > net - > perm_addr , dev - > net - > dev_addr , ETH_ALEN ) ;
/* RX bulk configuration */
memcpy ( tmp , & AX88179_BULKIN_SIZE [ 0 ] , 5 ) ;
ax88179_write_cmd ( dev , AX_ACCESS_MAC , AX_RX_BULKIN_QCTRL , 5 , 5 , tmp ) ;
dev - > rx_urb_size = 1024 * 20 ;
* tmp = 0x34 ;
ax88179_write_cmd ( dev , AX_ACCESS_MAC , AX_PAUSE_WATERLVL_LOW , 1 , 1 , tmp ) ;
* tmp = 0x52 ;
ax88179_write_cmd ( dev , AX_ACCESS_MAC , AX_PAUSE_WATERLVL_HIGH ,
1 , 1 , tmp ) ;
dev - > net - > features | = NETIF_F_IP_CSUM | NETIF_F_IPV6_CSUM |
NETIF_F_RXCSUM | NETIF_F_SG | NETIF_F_TSO ;
dev - > net - > hw_features | = NETIF_F_IP_CSUM | NETIF_F_IPV6_CSUM |
NETIF_F_RXCSUM | NETIF_F_SG | NETIF_F_TSO ;
/* Enable checksum offload */
* tmp = AX_RXCOE_IP | AX_RXCOE_TCP | AX_RXCOE_UDP |
AX_RXCOE_TCPV6 | AX_RXCOE_UDPV6 ;
ax88179_write_cmd ( dev , AX_ACCESS_MAC , AX_RXCOE_CTL , 1 , 1 , tmp ) ;
* tmp = AX_TXCOE_IP | AX_TXCOE_TCP | AX_TXCOE_UDP |
AX_TXCOE_TCPV6 | AX_TXCOE_UDPV6 ;
ax88179_write_cmd ( dev , AX_ACCESS_MAC , AX_TXCOE_CTL , 1 , 1 , tmp ) ;
/* Configure RX control register => start operation */
* tmp16 = AX_RX_CTL_DROPCRCERR | AX_RX_CTL_IPE | AX_RX_CTL_START |
AX_RX_CTL_AP | AX_RX_CTL_AMALL | AX_RX_CTL_AB ;
ax88179_write_cmd ( dev , AX_ACCESS_MAC , AX_RX_CTL , 2 , 2 , tmp16 ) ;
* tmp = AX_MONITOR_MODE_PMETYPE | AX_MONITOR_MODE_PMEPOL |
AX_MONITOR_MODE_RWMP ;
ax88179_write_cmd ( dev , AX_ACCESS_MAC , AX_MONITOR_MOD , 1 , 1 , tmp ) ;
/* Configure default medium type => giga */
* tmp16 = AX_MEDIUM_RECEIVE_EN | AX_MEDIUM_TXFLOW_CTRLEN |
AX_MEDIUM_RXFLOW_CTRLEN | AX_MEDIUM_ALWAYS_ONE |
AX_MEDIUM_FULL_DUPLEX | AX_MEDIUM_GIGAMODE ;
ax88179_write_cmd ( dev , AX_ACCESS_MAC , AX_MEDIUM_STATUS_MODE ,
2 , 2 , tmp16 ) ;
ax88179_led_setting ( dev ) ;
/* Restart autoneg */
mii_nway_restart ( & dev - > mii ) ;
2013-04-11 04:40:35 +00:00
usbnet_link_change ( dev , 0 , 0 ) ;
2013-03-02 00:41:11 +00:00
return 0 ;
}
static int ax88179_stop ( struct usbnet * dev )
{
u16 tmp16 ;
ax88179_read_cmd ( dev , AX_ACCESS_MAC , AX_MEDIUM_STATUS_MODE ,
2 , 2 , & tmp16 ) ;
tmp16 & = ~ AX_MEDIUM_RECEIVE_EN ;
ax88179_write_cmd ( dev , AX_ACCESS_MAC , AX_MEDIUM_STATUS_MODE ,
2 , 2 , & tmp16 ) ;
return 0 ;
}
static const struct driver_info ax88179_info = {
. description = " ASIX AX88179 USB 3.0 Gigibit Ethernet " ,
. bind = ax88179_bind ,
. unbind = ax88179_unbind ,
. status = ax88179_status ,
. link_reset = ax88179_link_reset ,
. reset = ax88179_reset ,
. stop = ax88179_stop ,
. flags = FLAG_ETHER | FLAG_FRAMING_AX ,
. rx_fixup = ax88179_rx_fixup ,
. tx_fixup = ax88179_tx_fixup ,
} ;
static const struct driver_info ax88178a_info = {
. description = " ASIX AX88178A USB 2.0 Gigibit Ethernet " ,
. bind = ax88179_bind ,
. unbind = ax88179_unbind ,
. status = ax88179_status ,
. link_reset = ax88179_link_reset ,
. reset = ax88179_reset ,
. stop = ax88179_stop ,
. flags = FLAG_ETHER | FLAG_FRAMING_AX ,
. rx_fixup = ax88179_rx_fixup ,
. tx_fixup = ax88179_tx_fixup ,
} ;
static const struct driver_info sitecom_info = {
. description = " Sitecom USB 3.0 to Gigabit Adapter " ,
. bind = ax88179_bind ,
. unbind = ax88179_unbind ,
. status = ax88179_status ,
. link_reset = ax88179_link_reset ,
. reset = ax88179_reset ,
. stop = ax88179_stop ,
. flags = FLAG_ETHER | FLAG_FRAMING_AX ,
. rx_fixup = ax88179_rx_fixup ,
. tx_fixup = ax88179_tx_fixup ,
} ;
static const struct usb_device_id products [ ] = {
{
/* ASIX AX88179 10/100/1000 */
USB_DEVICE ( 0x0b95 , 0x1790 ) ,
. driver_info = ( unsigned long ) & ax88179_info ,
} , {
/* ASIX AX88178A 10/100/1000 */
USB_DEVICE ( 0x0b95 , 0x178a ) ,
. driver_info = ( unsigned long ) & ax88178a_info ,
} , {
/* Sitecom USB 3.0 to Gigabit Adapter */
USB_DEVICE ( 0x0df6 , 0x0072 ) ,
. driver_info = ( unsigned long ) & sitecom_info ,
} ,
{ } ,
} ;
MODULE_DEVICE_TABLE ( usb , products ) ;
static struct usb_driver ax88179_178a_driver = {
. name = " ax88179_178a " ,
. id_table = products ,
. probe = usbnet_probe ,
. suspend = ax88179_suspend ,
. resume = ax88179_resume ,
. disconnect = usbnet_disconnect ,
. supports_autosuspend = 1 ,
. disable_hub_initiated_lpm = 1 ,
} ;
module_usb_driver ( ax88179_178a_driver ) ;
MODULE_DESCRIPTION ( " ASIX AX88179/178A based USB 3.0/2.0 Gigabit Ethernet Devices " ) ;
MODULE_LICENSE ( " GPL " ) ;