2018-11-26 12:32:59 +03:00
// SPDX-License-Identifier: GPL-2.0-or-later
/* Aquantia Corp. Aquantia AQtion USB to 5GbE Controller
* Copyright ( C ) 2003 - 2005 David Hollis < dhollis @ davehollis . com >
* Copyright ( C ) 2005 Phil Chang < pchang23 @ sbcglobal . net >
* Copyright ( C ) 2002 - 2003 TiVo Inc .
* Copyright ( C ) 2017 - 2018 ASIX
* Copyright ( C ) 2018 Aquantia Corp .
*/
# include <linux/module.h>
# include <linux/netdevice.h>
# include <linux/mii.h>
# include <linux/usb.h>
2018-11-26 12:33:14 +03:00
# include <linux/if_vlan.h>
2018-11-26 12:32:59 +03:00
# include <linux/usb/cdc.h>
# include <linux/usb/usbnet.h>
2018-11-26 12:33:04 +03:00
# include "aqc111.h"
static int aqc111_read_cmd_nopm ( struct usbnet * dev , u8 cmd , u16 value ,
u16 index , u16 size , void * data )
{
int ret ;
ret = usbnet_read_cmd_nopm ( 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(0x%x) reg index 0x%04x: %d \n " ,
cmd , index , ret ) ;
return ret ;
}
static int aqc111_read_cmd ( struct usbnet * dev , u8 cmd , u16 value ,
u16 index , u16 size , void * data )
{
int ret ;
ret = usbnet_read_cmd ( 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(0x%x) reg index 0x%04x: %d \n " ,
cmd , index , ret ) ;
return ret ;
}
2018-11-26 12:33:07 +03:00
static int aqc111_read16_cmd ( struct usbnet * dev , u8 cmd , u16 value ,
u16 index , u16 * data )
{
int ret = 0 ;
ret = aqc111_read_cmd ( dev , cmd , value , index , sizeof ( * data ) , data ) ;
le16_to_cpus ( data ) ;
return ret ;
}
2018-11-26 12:33:04 +03:00
static int __aqc111_write_cmd ( struct usbnet * dev , u8 cmd , u8 reqtype ,
u16 value , u16 index , u16 size , const void * data )
{
int err = - ENOMEM ;
void * buf = NULL ;
netdev_dbg ( dev - > net ,
" %s cmd=%#x reqtype=%#x value=%#x index=%#x size=%d \n " ,
__func__ , cmd , reqtype , value , index , size ) ;
if ( data ) {
buf = kmemdup ( data , size , GFP_KERNEL ) ;
if ( ! buf )
goto out ;
}
err = usb_control_msg ( dev - > udev , usb_sndctrlpipe ( dev - > udev , 0 ) ,
cmd , reqtype , value , index , buf , size ,
( cmd = = AQ_PHY_POWER ) ? AQ_USB_PHY_SET_TIMEOUT :
AQ_USB_SET_TIMEOUT ) ;
if ( unlikely ( err < 0 ) )
netdev_warn ( dev - > net ,
" Failed to write(0x%x) reg index 0x%04x: %d \n " ,
cmd , index , err ) ;
kfree ( buf ) ;
out :
return err ;
}
static int aqc111_write_cmd_nopm ( struct usbnet * dev , u8 cmd , u16 value ,
u16 index , u16 size , void * data )
{
int ret ;
ret = __aqc111_write_cmd ( dev , cmd , USB_DIR_OUT | USB_TYPE_VENDOR |
USB_RECIP_DEVICE , value , index , size , data ) ;
return ret ;
}
static int aqc111_write_cmd ( struct usbnet * dev , u8 cmd , u16 value ,
u16 index , u16 size , void * data )
{
int ret ;
if ( usb_autopm_get_interface ( dev - > intf ) < 0 )
return - ENODEV ;
ret = __aqc111_write_cmd ( dev , cmd , USB_DIR_OUT | USB_TYPE_VENDOR |
USB_RECIP_DEVICE , value , index , size , data ) ;
usb_autopm_put_interface ( dev - > intf ) ;
return ret ;
}
2018-11-26 12:33:07 +03:00
static int aqc111_write16_cmd_nopm ( struct usbnet * dev , u8 cmd , u16 value ,
u16 index , u16 * data )
{
u16 tmp = * data ;
cpu_to_le16s ( & tmp ) ;
return aqc111_write_cmd_nopm ( dev , cmd , value , index , sizeof ( tmp ) , & tmp ) ;
}
static int aqc111_write16_cmd ( struct usbnet * dev , u8 cmd , u16 value ,
u16 index , u16 * data )
{
u16 tmp = * data ;
cpu_to_le16s ( & tmp ) ;
return aqc111_write_cmd ( dev , cmd , value , index , sizeof ( tmp ) , & tmp ) ;
}
2018-11-26 12:33:09 +03:00
static int aqc111_write32_cmd_nopm ( struct usbnet * dev , u8 cmd , u16 value ,
u16 index , u32 * data )
{
u32 tmp = * data ;
cpu_to_le32s ( & tmp ) ;
return aqc111_write_cmd_nopm ( dev , cmd , value , index , sizeof ( tmp ) , & tmp ) ;
}
static int aqc111_write32_cmd ( struct usbnet * dev , u8 cmd , u16 value ,
u16 index , u32 * data )
{
u32 tmp = * data ;
cpu_to_le32s ( & tmp ) ;
return aqc111_write_cmd ( dev , cmd , value , index , sizeof ( tmp ) , & tmp ) ;
}
2018-11-26 12:33:12 +03:00
static void aqc111_set_phy_speed ( struct usbnet * dev , u8 autoneg , u16 speed )
{
struct aqc111_data * aqc111_data = dev - > driver_priv ;
aqc111_data - > phy_cfg & = ~ AQ_ADV_MASK ;
aqc111_data - > phy_cfg | = AQ_PAUSE ;
aqc111_data - > phy_cfg | = AQ_ASYM_PAUSE ;
aqc111_data - > phy_cfg | = AQ_DOWNSHIFT ;
aqc111_data - > phy_cfg & = ~ AQ_DSH_RETRIES_MASK ;
aqc111_data - > phy_cfg | = ( 3 < < AQ_DSH_RETRIES_SHIFT ) &
AQ_DSH_RETRIES_MASK ;
if ( autoneg = = AUTONEG_ENABLE ) {
switch ( speed ) {
case SPEED_5000 :
aqc111_data - > phy_cfg | = AQ_ADV_5G ;
/* fall-through */
case SPEED_2500 :
aqc111_data - > phy_cfg | = AQ_ADV_2G5 ;
/* fall-through */
case SPEED_1000 :
aqc111_data - > phy_cfg | = AQ_ADV_1G ;
/* fall-through */
case SPEED_100 :
aqc111_data - > phy_cfg | = AQ_ADV_100M ;
/* fall-through */
}
} else {
switch ( speed ) {
case SPEED_5000 :
aqc111_data - > phy_cfg | = AQ_ADV_5G ;
break ;
case SPEED_2500 :
aqc111_data - > phy_cfg | = AQ_ADV_2G5 ;
break ;
case SPEED_1000 :
aqc111_data - > phy_cfg | = AQ_ADV_1G ;
break ;
case SPEED_100 :
aqc111_data - > phy_cfg | = AQ_ADV_100M ;
break ;
}
}
aqc111_write32_cmd ( dev , AQ_PHY_OPS , 0 , 0 , & aqc111_data - > phy_cfg ) ;
}
2018-11-26 12:33:14 +03:00
static int aqc111_set_mac_addr ( struct net_device * net , void * p )
{
struct usbnet * dev = netdev_priv ( net ) ;
int ret = 0 ;
ret = eth_mac_addr ( net , p ) ;
if ( ret < 0 )
return ret ;
/* Set the MAC address */
return aqc111_write_cmd ( dev , AQ_ACCESS_MAC , SFR_NODE_ID , ETH_ALEN ,
ETH_ALEN , net - > dev_addr ) ;
}
2018-11-26 12:33:02 +03:00
static const struct net_device_ops aqc111_netdev_ops = {
. ndo_open = usbnet_open ,
. ndo_stop = usbnet_stop ,
2018-11-26 12:33:16 +03:00
. ndo_start_xmit = usbnet_start_xmit ,
. ndo_tx_timeout = usbnet_tx_timeout ,
. ndo_get_stats64 = usbnet_get_stats64 ,
2018-11-26 12:33:14 +03:00
. ndo_set_mac_address = aqc111_set_mac_addr ,
. ndo_validate_addr = eth_validate_addr ,
2018-11-26 12:33:02 +03:00
} ;
2018-11-26 12:33:14 +03:00
static int aqc111_read_perm_mac ( struct usbnet * dev )
{
u8 buf [ ETH_ALEN ] ;
int ret ;
ret = aqc111_read_cmd ( dev , AQ_FLASH_PARAMETERS , 0 , 0 , ETH_ALEN , buf ) ;
if ( ret < 0 )
goto out ;
ether_addr_copy ( dev - > net - > perm_addr , buf ) ;
return 0 ;
out :
return ret ;
}
2018-11-26 12:33:09 +03:00
static void aqc111_read_fw_version ( struct usbnet * dev ,
struct aqc111_data * aqc111_data )
{
aqc111_read_cmd ( dev , AQ_ACCESS_MAC , AQ_FW_VER_MAJOR ,
1 , 1 , & aqc111_data - > fw_ver . major ) ;
aqc111_read_cmd ( dev , AQ_ACCESS_MAC , AQ_FW_VER_MINOR ,
1 , 1 , & aqc111_data - > fw_ver . minor ) ;
aqc111_read_cmd ( dev , AQ_ACCESS_MAC , AQ_FW_VER_REV ,
1 , 1 , & aqc111_data - > fw_ver . rev ) ;
if ( aqc111_data - > fw_ver . major & 0x80 )
aqc111_data - > fw_ver . major & = ~ 0x80 ;
}
2018-11-26 12:33:02 +03:00
static int aqc111_bind ( struct usbnet * dev , struct usb_interface * intf )
{
struct usb_device * udev = interface_to_usbdev ( intf ) ;
2018-11-26 12:33:12 +03:00
enum usb_device_speed usb_speed = udev - > speed ;
2018-11-26 12:33:09 +03:00
struct aqc111_data * aqc111_data ;
2018-11-26 12:33:02 +03:00
int ret ;
/* Check if vendor configuration */
if ( udev - > actconfig - > desc . bConfigurationValue ! = 1 ) {
usb_driver_set_configuration ( udev , 1 ) ;
return - ENODEV ;
}
usb_reset_configuration ( dev - > udev ) ;
ret = usbnet_get_endpoints ( dev , intf ) ;
if ( ret < 0 ) {
netdev_dbg ( dev - > net , " usbnet_get_endpoints failed " ) ;
return ret ;
}
2018-11-26 12:33:09 +03:00
aqc111_data = kzalloc ( sizeof ( * aqc111_data ) , GFP_KERNEL ) ;
if ( ! aqc111_data )
return - ENOMEM ;
/* store aqc111_data pointer in device data field */
dev - > driver_priv = aqc111_data ;
2018-11-26 12:33:14 +03:00
/* Init the MAC address */
ret = aqc111_read_perm_mac ( dev ) ;
if ( ret )
goto out ;
ether_addr_copy ( dev - > net - > dev_addr , dev - > net - > perm_addr ) ;
2018-11-26 12:33:16 +03:00
2018-11-26 12:33:19 +03:00
/* Set Rx urb size */
dev - > rx_urb_size = URB_SIZE ;
2018-11-26 12:33:16 +03:00
/* Set TX needed headroom & tailroom */
dev - > net - > needed_headroom + = sizeof ( u64 ) ;
dev - > net - > needed_tailroom + = sizeof ( u64 ) ;
2018-11-26 12:33:02 +03:00
dev - > net - > netdev_ops = & aqc111_netdev_ops ;
2018-11-26 12:33:16 +03:00
if ( usb_device_no_sg_constraint ( dev - > udev ) )
dev - > can_dma_sg = 1 ;
dev - > net - > hw_features | = AQ_SUPPORT_HW_FEATURE ;
dev - > net - > features | = AQ_SUPPORT_FEATURE ;
2018-11-26 12:33:09 +03:00
aqc111_read_fw_version ( dev , aqc111_data ) ;
2018-11-26 12:33:12 +03:00
aqc111_data - > autoneg = AUTONEG_ENABLE ;
aqc111_data - > advertised_speed = ( usb_speed = = USB_SPEED_SUPER ) ?
SPEED_5000 : SPEED_1000 ;
2018-11-26 12:33:09 +03:00
2018-11-26 12:33:02 +03:00
return 0 ;
2018-11-26 12:33:14 +03:00
out :
kfree ( aqc111_data ) ;
return ret ;
2018-11-26 12:33:02 +03:00
}
static void aqc111_unbind ( struct usbnet * dev , struct usb_interface * intf )
{
2018-11-26 12:33:09 +03:00
struct aqc111_data * aqc111_data = dev - > driver_priv ;
2018-11-26 12:33:07 +03:00
u16 reg16 ;
/* Force bz */
reg16 = SFR_PHYPWR_RSTCTL_BZ ;
aqc111_write16_cmd_nopm ( dev , AQ_ACCESS_MAC , SFR_PHYPWR_RSTCTL ,
2 , & reg16 ) ;
reg16 = 0 ;
aqc111_write16_cmd_nopm ( dev , AQ_ACCESS_MAC , SFR_PHYPWR_RSTCTL ,
2 , & reg16 ) ;
2018-11-26 12:33:09 +03:00
/* Power down ethernet PHY */
2018-11-26 12:33:12 +03:00
aqc111_data - > phy_cfg & = ~ AQ_ADV_MASK ;
2018-11-26 12:33:09 +03:00
aqc111_data - > phy_cfg | = AQ_LOW_POWER ;
aqc111_data - > phy_cfg & = ~ AQ_PHY_POWER_EN ;
aqc111_write32_cmd_nopm ( dev , AQ_PHY_OPS , 0 , 0 ,
& aqc111_data - > phy_cfg ) ;
kfree ( aqc111_data ) ;
2018-11-26 12:33:07 +03:00
}
2018-11-26 12:33:12 +03:00
static void aqc111_status ( struct usbnet * dev , struct urb * urb )
{
struct aqc111_data * aqc111_data = dev - > driver_priv ;
u64 * event_data = NULL ;
int link = 0 ;
if ( urb - > actual_length < sizeof ( * event_data ) )
return ;
event_data = urb - > transfer_buffer ;
le64_to_cpus ( event_data ) ;
if ( * event_data & AQ_LS_MASK )
link = 1 ;
else
link = 0 ;
aqc111_data - > link_speed = ( * event_data & AQ_SPEED_MASK ) > >
AQ_SPEED_SHIFT ;
aqc111_data - > link = link ;
if ( netif_carrier_ok ( dev - > net ) ! = link )
usbnet_defer_kevent ( dev , EVENT_LINK_RESET ) ;
}
static void aqc111_configure_rx ( struct usbnet * dev ,
struct aqc111_data * aqc111_data )
{
enum usb_device_speed usb_speed = dev - > udev - > speed ;
u16 link_speed = 0 , usb_host = 0 ;
u8 buf [ 5 ] = { 0 } ;
u8 queue_num = 0 ;
u16 reg16 = 0 ;
u8 reg8 = 0 ;
buf [ 0 ] = 0x00 ;
buf [ 1 ] = 0xF8 ;
buf [ 2 ] = 0x07 ;
switch ( aqc111_data - > link_speed ) {
case AQ_INT_SPEED_5G :
link_speed = 5000 ;
reg8 = 0x05 ;
reg16 = 0x001F ;
break ;
case AQ_INT_SPEED_2_5G :
link_speed = 2500 ;
reg16 = 0x003F ;
break ;
case AQ_INT_SPEED_1G :
link_speed = 1000 ;
reg16 = 0x009F ;
break ;
case AQ_INT_SPEED_100M :
link_speed = 100 ;
queue_num = 1 ;
reg16 = 0x063F ;
buf [ 1 ] = 0xFB ;
buf [ 2 ] = 0x4 ;
break ;
}
aqc111_write_cmd ( dev , AQ_ACCESS_MAC , SFR_INTER_PACKET_GAP_0 ,
1 , 1 , & reg8 ) ;
aqc111_write_cmd ( dev , AQ_ACCESS_MAC , SFR_TX_PAUSE_RESEND_T , 3 , 3 , buf ) ;
switch ( usb_speed ) {
case USB_SPEED_SUPER :
usb_host = 3 ;
break ;
case USB_SPEED_HIGH :
usb_host = 2 ;
break ;
case USB_SPEED_FULL :
case USB_SPEED_LOW :
usb_host = 1 ;
queue_num = 0 ;
break ;
default :
usb_host = 0 ;
break ;
}
memcpy ( buf , & AQC111_BULKIN_SIZE [ queue_num ] , 5 ) ;
/* RX bulk configuration */
aqc111_write_cmd ( dev , AQ_ACCESS_MAC , SFR_RX_BULKIN_QCTRL , 5 , 5 , buf ) ;
/* Set high low water level */
reg16 = 0x0810 ;
aqc111_write16_cmd ( dev , AQ_ACCESS_MAC , SFR_PAUSE_WATERLVL_LOW ,
2 , & reg16 ) ;
netdev_info ( dev - > net , " Link Speed %d, USB %d " , link_speed , usb_host ) ;
}
static int aqc111_link_reset ( struct usbnet * dev )
{
struct aqc111_data * aqc111_data = dev - > driver_priv ;
u16 reg16 = 0 ;
u8 reg8 = 0 ;
if ( aqc111_data - > link = = 1 ) { /* Link up */
aqc111_configure_rx ( dev , aqc111_data ) ;
/* Vlan Tag Filter */
reg8 = SFR_VLAN_CONTROL_VSO ;
aqc111_write_cmd ( dev , AQ_ACCESS_MAC , SFR_VLAN_ID_CONTROL ,
1 , 1 , & reg8 ) ;
reg8 = 0x0 ;
aqc111_write_cmd ( dev , AQ_ACCESS_MAC , SFR_BMRX_DMA_CONTROL ,
1 , 1 , & reg8 ) ;
aqc111_write_cmd ( dev , AQ_ACCESS_MAC , SFR_BMTX_DMA_CONTROL ,
1 , 1 , & reg8 ) ;
aqc111_write_cmd ( dev , AQ_ACCESS_MAC , SFR_ARC_CTRL , 1 , 1 , & reg8 ) ;
reg16 = SFR_RX_CTL_IPE | SFR_RX_CTL_AB ;
aqc111_write16_cmd ( dev , AQ_ACCESS_MAC , SFR_RX_CTL , 2 , & reg16 ) ;
reg8 = SFR_RX_PATH_READY ;
aqc111_write_cmd ( dev , AQ_ACCESS_MAC , SFR_ETH_MAC_PATH ,
1 , 1 , & reg8 ) ;
reg8 = SFR_BULK_OUT_EFF_EN ;
aqc111_write_cmd ( dev , AQ_ACCESS_MAC , SFR_BULK_OUT_CTRL ,
1 , 1 , & reg8 ) ;
reg16 = 0 ;
aqc111_write16_cmd ( dev , AQ_ACCESS_MAC , SFR_MEDIUM_STATUS_MODE ,
2 , & reg16 ) ;
reg16 = SFR_MEDIUM_XGMIIMODE | SFR_MEDIUM_FULL_DUPLEX ;
aqc111_write16_cmd ( dev , AQ_ACCESS_MAC , SFR_MEDIUM_STATUS_MODE ,
2 , & reg16 ) ;
aqc111_read16_cmd ( dev , AQ_ACCESS_MAC , SFR_MEDIUM_STATUS_MODE ,
2 , & reg16 ) ;
reg16 | = SFR_MEDIUM_RECEIVE_EN | SFR_MEDIUM_RXFLOW_CTRLEN |
SFR_MEDIUM_TXFLOW_CTRLEN ;
aqc111_write16_cmd ( dev , AQ_ACCESS_MAC , SFR_MEDIUM_STATUS_MODE ,
2 , & reg16 ) ;
reg16 = SFR_RX_CTL_IPE | SFR_RX_CTL_AB | SFR_RX_CTL_START ;
aqc111_write16_cmd ( dev , AQ_ACCESS_MAC , SFR_RX_CTL , 2 , & reg16 ) ;
netif_carrier_on ( dev - > net ) ;
} else {
aqc111_read16_cmd ( dev , AQ_ACCESS_MAC , SFR_MEDIUM_STATUS_MODE ,
2 , & reg16 ) ;
reg16 & = ~ SFR_MEDIUM_RECEIVE_EN ;
aqc111_write16_cmd ( dev , AQ_ACCESS_MAC , SFR_MEDIUM_STATUS_MODE ,
2 , & reg16 ) ;
aqc111_read16_cmd ( dev , AQ_ACCESS_MAC , SFR_RX_CTL , 2 , & reg16 ) ;
reg16 & = ~ SFR_RX_CTL_START ;
aqc111_write16_cmd ( dev , AQ_ACCESS_MAC , SFR_RX_CTL , 2 , & reg16 ) ;
reg8 = SFR_BULK_OUT_FLUSH_EN | SFR_BULK_OUT_EFF_EN ;
aqc111_write_cmd ( dev , AQ_ACCESS_MAC , SFR_BULK_OUT_CTRL ,
1 , 1 , & reg8 ) ;
reg8 = SFR_BULK_OUT_EFF_EN ;
aqc111_write_cmd ( dev , AQ_ACCESS_MAC , SFR_BULK_OUT_CTRL ,
1 , 1 , & reg8 ) ;
netif_carrier_off ( dev - > net ) ;
}
return 0 ;
}
2018-11-26 12:33:07 +03:00
static int aqc111_reset ( struct usbnet * dev )
{
2018-11-26 12:33:09 +03:00
struct aqc111_data * aqc111_data = dev - > driver_priv ;
2018-11-26 12:33:07 +03:00
u8 reg8 = 0 ;
2018-11-26 12:33:19 +03:00
dev - > rx_urb_size = URB_SIZE ;
2018-11-26 12:33:16 +03:00
if ( usb_device_no_sg_constraint ( dev - > udev ) )
dev - > can_dma_sg = 1 ;
dev - > net - > hw_features | = AQ_SUPPORT_HW_FEATURE ;
dev - > net - > features | = AQ_SUPPORT_FEATURE ;
2018-11-26 12:33:09 +03:00
/* Power up ethernet PHY */
aqc111_data - > phy_cfg = AQ_PHY_POWER_EN ;
aqc111_write32_cmd ( dev , AQ_PHY_OPS , 0 , 0 ,
& aqc111_data - > phy_cfg ) ;
2018-11-26 12:33:14 +03:00
/* Set the MAC address */
aqc111_write_cmd ( dev , AQ_ACCESS_MAC , SFR_NODE_ID , ETH_ALEN ,
ETH_ALEN , dev - > net - > dev_addr ) ;
2018-11-26 12:33:07 +03:00
reg8 = 0xFF ;
aqc111_write_cmd ( dev , AQ_ACCESS_MAC , SFR_BM_INT_MASK , 1 , 1 , & reg8 ) ;
reg8 = 0x0 ;
aqc111_write_cmd ( dev , AQ_ACCESS_MAC , SFR_SWP_CTRL , 1 , 1 , & reg8 ) ;
aqc111_read_cmd ( dev , AQ_ACCESS_MAC , SFR_MONITOR_MODE , 1 , 1 , & reg8 ) ;
reg8 & = ~ ( SFR_MONITOR_MODE_EPHYRW | SFR_MONITOR_MODE_RWLC |
SFR_MONITOR_MODE_RWMP | SFR_MONITOR_MODE_RWWF |
SFR_MONITOR_MODE_RW_FLAG ) ;
aqc111_write_cmd ( dev , AQ_ACCESS_MAC , SFR_MONITOR_MODE , 1 , 1 , & reg8 ) ;
2018-11-26 12:33:12 +03:00
netif_carrier_off ( dev - > net ) ;
/* Phy advertise */
aqc111_set_phy_speed ( dev , aqc111_data - > autoneg ,
aqc111_data - > advertised_speed ) ;
2018-11-26 12:33:07 +03:00
return 0 ;
}
static int aqc111_stop ( struct usbnet * dev )
{
2018-11-26 12:33:09 +03:00
struct aqc111_data * aqc111_data = dev - > driver_priv ;
2018-11-26 12:33:07 +03:00
u16 reg16 = 0 ;
aqc111_read16_cmd ( dev , AQ_ACCESS_MAC , SFR_MEDIUM_STATUS_MODE ,
2 , & reg16 ) ;
reg16 & = ~ SFR_MEDIUM_RECEIVE_EN ;
aqc111_write16_cmd ( dev , AQ_ACCESS_MAC , SFR_MEDIUM_STATUS_MODE ,
2 , & reg16 ) ;
reg16 = 0 ;
aqc111_write16_cmd ( dev , AQ_ACCESS_MAC , SFR_RX_CTL , 2 , & reg16 ) ;
2018-11-26 12:33:09 +03:00
/* Put PHY to low power*/
aqc111_data - > phy_cfg | = AQ_LOW_POWER ;
aqc111_write32_cmd ( dev , AQ_PHY_OPS , 0 , 0 ,
& aqc111_data - > phy_cfg ) ;
2018-11-26 12:33:12 +03:00
netif_carrier_off ( dev - > net ) ;
2018-11-26 12:33:07 +03:00
return 0 ;
2018-11-26 12:33:02 +03:00
}
2018-11-26 12:33:19 +03:00
static int aqc111_rx_fixup ( struct usbnet * dev , struct sk_buff * skb )
{
struct sk_buff * new_skb = NULL ;
u32 pkt_total_offset = 0 ;
u64 * pkt_desc_ptr = NULL ;
u32 start_of_descs = 0 ;
u32 desc_offset = 0 ; /*RX Header Offset*/
u16 pkt_count = 0 ;
u64 desc_hdr = 0 ;
u32 skb_len = 0 ;
if ( ! skb )
goto err ;
if ( skb - > len = = 0 )
goto err ;
skb_len = skb - > len ;
/* RX Descriptor Header */
skb_trim ( skb , skb - > len - sizeof ( desc_hdr ) ) ;
desc_hdr = le64_to_cpup ( ( u64 * ) skb_tail_pointer ( skb ) ) ;
/* Check these packets */
desc_offset = ( desc_hdr & AQ_RX_DH_DESC_OFFSET_MASK ) > >
AQ_RX_DH_DESC_OFFSET_SHIFT ;
pkt_count = desc_hdr & AQ_RX_DH_PKT_CNT_MASK ;
start_of_descs = skb_len - ( ( pkt_count + 1 ) * sizeof ( desc_hdr ) ) ;
/* self check descs position */
if ( start_of_descs ! = desc_offset )
goto err ;
/* self check desc_offset from header*/
if ( desc_offset > = skb_len )
goto err ;
if ( pkt_count = = 0 )
goto err ;
/* Get the first RX packet descriptor */
pkt_desc_ptr = ( u64 * ) ( skb - > data + desc_offset ) ;
while ( pkt_count - - ) {
u64 pkt_desc = le64_to_cpup ( pkt_desc_ptr ) ;
u32 pkt_len_with_padd = 0 ;
u32 pkt_len = 0 ;
pkt_len = ( u32 ) ( ( pkt_desc & AQ_RX_PD_LEN_MASK ) > >
AQ_RX_PD_LEN_SHIFT ) ;
pkt_len_with_padd = ( ( pkt_len + 7 ) & 0x7FFF8 ) ;
pkt_total_offset + = pkt_len_with_padd ;
if ( pkt_total_offset > desc_offset | |
( pkt_count = = 0 & & pkt_total_offset ! = desc_offset ) ) {
goto err ;
}
if ( pkt_desc & AQ_RX_PD_DROP | |
! ( pkt_desc & AQ_RX_PD_RX_OK ) | |
pkt_len > ( dev - > hard_mtu + AQ_RX_HW_PAD ) ) {
skb_pull ( skb , pkt_len_with_padd ) ;
/* Next RX Packet Descriptor */
pkt_desc_ptr + + ;
continue ;
}
/* Clone SKB */
new_skb = skb_clone ( skb , GFP_ATOMIC ) ;
if ( ! new_skb )
goto err ;
new_skb - > len = pkt_len ;
skb_pull ( new_skb , AQ_RX_HW_PAD ) ;
skb_set_tail_pointer ( new_skb , new_skb - > len ) ;
new_skb - > truesize = SKB_TRUESIZE ( new_skb - > len ) ;
usbnet_skb_return ( dev , new_skb ) ;
if ( pkt_count = = 0 )
break ;
skb_pull ( skb , pkt_len_with_padd ) ;
/* Next RX Packet Header */
pkt_desc_ptr + + ;
new_skb = NULL ;
}
return 1 ;
err :
return 0 ;
}
2018-11-26 12:33:16 +03:00
static struct sk_buff * aqc111_tx_fixup ( struct usbnet * dev , struct sk_buff * skb ,
gfp_t flags )
{
int frame_size = dev - > maxpacket ;
struct sk_buff * new_skb = NULL ;
u64 * tx_desc_ptr = NULL ;
int padding_size = 0 ;
int headroom = 0 ;
int tailroom = 0 ;
u64 tx_desc = 0 ;
/*Length of actual data*/
tx_desc | = skb - > len & AQ_TX_DESC_LEN_MASK ;
headroom = ( skb - > len + sizeof ( tx_desc ) ) % 8 ;
if ( headroom ! = 0 )
padding_size = 8 - headroom ;
if ( ( ( skb - > len + sizeof ( tx_desc ) + padding_size ) % frame_size ) = = 0 ) {
padding_size + = 8 ;
tx_desc | = AQ_TX_DESC_DROP_PADD ;
}
if ( ! dev - > can_dma_sg & & ( dev - > net - > features & NETIF_F_SG ) & &
skb_linearize ( skb ) )
return NULL ;
headroom = skb_headroom ( skb ) ;
tailroom = skb_tailroom ( skb ) ;
if ( ! ( headroom > = sizeof ( tx_desc ) & & tailroom > = padding_size ) ) {
new_skb = skb_copy_expand ( skb , sizeof ( tx_desc ) ,
padding_size , flags ) ;
dev_kfree_skb_any ( skb ) ;
skb = new_skb ;
if ( ! skb )
return NULL ;
}
if ( padding_size ! = 0 )
skb_put_zero ( skb , padding_size ) ;
/* Copy TX header */
tx_desc_ptr = skb_push ( skb , sizeof ( tx_desc ) ) ;
* tx_desc_ptr = cpu_to_le64 ( tx_desc ) ;
usbnet_set_skb_tx_stats ( skb , 1 , 0 ) ;
return skb ;
}
2018-11-26 12:32:59 +03:00
static const struct driver_info aqc111_info = {
. description = " Aquantia AQtion USB to 5GbE Controller " ,
2018-11-26 12:33:02 +03:00
. bind = aqc111_bind ,
. unbind = aqc111_unbind ,
2018-11-26 12:33:12 +03:00
. status = aqc111_status ,
. link_reset = aqc111_link_reset ,
2018-11-26 12:33:07 +03:00
. reset = aqc111_reset ,
. stop = aqc111_stop ,
2018-11-26 12:33:16 +03:00
. flags = FLAG_ETHER | FLAG_FRAMING_AX |
FLAG_AVOID_UNLINK_URBS | FLAG_MULTI_PACKET ,
2018-11-26 12:33:19 +03:00
. rx_fixup = aqc111_rx_fixup ,
2018-11-26 12:33:16 +03:00
. tx_fixup = aqc111_tx_fixup ,
2018-11-26 12:32:59 +03:00
} ;
# define AQC111_USB_ETH_DEV(vid, pid, table) \
USB_DEVICE_INTERFACE_CLASS ( ( vid ) , ( pid ) , USB_CLASS_VENDOR_SPEC ) , \
. driver_info = ( unsigned long ) & ( table ) \
} , \
{ \
USB_DEVICE_AND_INTERFACE_INFO ( ( vid ) , ( pid ) , \
USB_CLASS_COMM , \
USB_CDC_SUBCLASS_ETHERNET , \
USB_CDC_PROTO_NONE ) , \
. driver_info = ( unsigned long ) & ( table ) ,
static const struct usb_device_id products [ ] = {
{ AQC111_USB_ETH_DEV ( 0x2eca , 0xc101 , aqc111_info ) } ,
{ } , /* END */
} ;
MODULE_DEVICE_TABLE ( usb , products ) ;
static struct usb_driver aq_driver = {
. name = " aqc111 " ,
. id_table = products ,
. probe = usbnet_probe ,
. disconnect = usbnet_disconnect ,
} ;
module_usb_driver ( aq_driver ) ;
MODULE_DESCRIPTION ( " Aquantia AQtion USB to 5/2.5GbE Controllers " ) ;
MODULE_LICENSE ( " GPL " ) ;