diff options
Diffstat (limited to 'drivers/net/usb')
| -rw-r--r-- | drivers/net/usb/Kconfig | 20 | ||||
| -rw-r--r-- | drivers/net/usb/Makefile | 1 | ||||
| -rw-r--r-- | drivers/net/usb/asix_devices.c | 31 | ||||
| -rw-r--r-- | drivers/net/usb/ax88179_178a.c | 1448 | ||||
| -rw-r--r-- | drivers/net/usb/cdc_mbim.c | 11 | ||||
| -rw-r--r-- | drivers/net/usb/cdc_ncm.c | 57 | ||||
| -rw-r--r-- | drivers/net/usb/qmi_wwan.c | 49 | ||||
| -rw-r--r-- | drivers/net/usb/smsc75xx.c | 12 | 
8 files changed, 1564 insertions, 65 deletions
diff --git a/drivers/net/usb/Kconfig b/drivers/net/usb/Kconfig index da92ed3797a..7c769d8e25a 100644 --- a/drivers/net/usb/Kconfig +++ b/drivers/net/usb/Kconfig @@ -156,6 +156,24 @@ config USB_NET_AX8817X  	  This driver creates an interface named "ethX", where X depends on  	  what other networking devices you have in use. +config USB_NET_AX88179_178A +	tristate "ASIX AX88179/178A USB 3.0/2.0 to Gigabit Ethernet" +	depends on USB_USBNET +	select CRC32 +	select PHYLIB +	default y +	help +	  This option adds support for ASIX AX88179 based USB 3.0/2.0 +	  to Gigabit Ethernet adapters. + +	  This driver should work with at least the following devices: +	    * ASIX AX88179 +	    * ASIX AX88178A +	    * Sitcomm LN-032 + +	  This driver creates an interface named "ethX", where X depends on +	  what other networking devices you have in use. +  config USB_NET_CDCETHER  	tristate "CDC Ethernet support (smart devices such as cable modems)"  	depends on USB_USBNET @@ -250,7 +268,7 @@ config USB_NET_SMSC75XX  	select CRC16  	select CRC32  	help -	  This option adds support for SMSC LAN95XX based USB 2.0 +	  This option adds support for SMSC LAN75XX based USB 2.0  	  Gigabit Ethernet adapters.  config USB_NET_SMSC95XX diff --git a/drivers/net/usb/Makefile b/drivers/net/usb/Makefile index 478691326f3..119b06c9aa1 100644 --- a/drivers/net/usb/Makefile +++ b/drivers/net/usb/Makefile @@ -9,6 +9,7 @@ obj-$(CONFIG_USB_RTL8150)	+= rtl8150.o  obj-$(CONFIG_USB_HSO)		+= hso.o  obj-$(CONFIG_USB_NET_AX8817X)	+= asix.o  asix-y := asix_devices.o asix_common.o ax88172a.o +obj-$(CONFIG_USB_NET_AX88179_178A)      += ax88179_178a.o  obj-$(CONFIG_USB_NET_CDCETHER)	+= cdc_ether.o  obj-$(CONFIG_USB_NET_CDC_EEM)	+= cdc_eem.o  obj-$(CONFIG_USB_NET_DM9601)	+= dm9601.o diff --git a/drivers/net/usb/asix_devices.c b/drivers/net/usb/asix_devices.c index 2205dbc8d32..70975346909 100644 --- a/drivers/net/usb/asix_devices.c +++ b/drivers/net/usb/asix_devices.c @@ -924,6 +924,29 @@ static const struct driver_info ax88178_info = {  	.tx_fixup = asix_tx_fixup,  }; +/* + * USBLINK 20F9 "USB 2.0 LAN" USB ethernet adapter, typically found in + * no-name packaging. + * USB device strings are: + *   1: Manufacturer: USBLINK + *   2: Product: HG20F9 USB2.0 + *   3: Serial: 000003 + * Appears to be compatible with Asix 88772B. + */ +static const struct driver_info hg20f9_info = { +	.description = "HG20F9 USB 2.0 Ethernet", +	.bind = ax88772_bind, +	.unbind = ax88772_unbind, +	.status = asix_status, +	.link_reset = ax88772_link_reset, +	.reset = ax88772_reset, +	.flags = FLAG_ETHER | FLAG_FRAMING_AX | FLAG_LINK_INTR | +	         FLAG_MULTI_PACKET, +	.rx_fixup = asix_rx_fixup_common, +	.tx_fixup = asix_tx_fixup, +	.data = FLAG_EEPROM_MAC, +}; +  extern const struct driver_info ax88172a_info;  static const struct usb_device_id	products [] = { @@ -1063,6 +1086,14 @@ static const struct usb_device_id	products [] = {  	/* ASIX 88172a demo board */  	USB_DEVICE(0x0b95, 0x172a),  	.driver_info = (unsigned long) &ax88172a_info, +}, { +	/* +	 * USBLINK HG20F9 "USB 2.0 LAN" +	 * Appears to have gazumped Linksys's manufacturer ID but +	 * doesn't (yet) conflict with any known Linksys product. +	 */ +	USB_DEVICE(0x066b, 0x20f9), +	.driver_info = (unsigned long) &hg20f9_info,  },  	{ },		// END  }; diff --git a/drivers/net/usb/ax88179_178a.c b/drivers/net/usb/ax88179_178a.c new file mode 100644 index 00000000000..71c27d8d214 --- /dev/null +++ b/drivers/net/usb/ax88179_178a.c @@ -0,0 +1,1448 @@ +/* + * ASIX AX88179/178A 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) { +		if (link) +			usbnet_defer_kevent(dev, EVENT_LINK_RESET); +		else +			netif_carrier_off(dev->net); + +		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; + +	netif_carrier_off(dev->net); + +	/* 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); + +	netif_carrier_off(dev->net); + +	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); + +	netif_carrier_off(dev->net); + +	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"); diff --git a/drivers/net/usb/cdc_mbim.c b/drivers/net/usb/cdc_mbim.c index 248d2dc765a..16c84299729 100644 --- a/drivers/net/usb/cdc_mbim.c +++ b/drivers/net/usb/cdc_mbim.c @@ -68,18 +68,9 @@ static int cdc_mbim_bind(struct usbnet *dev, struct usb_interface *intf)  	struct cdc_ncm_ctx *ctx;  	struct usb_driver *subdriver = ERR_PTR(-ENODEV);  	int ret = -ENODEV; -	u8 data_altsetting = CDC_NCM_DATA_ALTSETTING_NCM; +	u8 data_altsetting = cdc_ncm_select_altsetting(dev, intf);  	struct cdc_mbim_state *info = (void *)&dev->data; -	/* see if interface supports MBIM alternate setting */ -	if (intf->num_altsetting == 2) { -		if (!cdc_ncm_comm_intf_is_mbim(intf->cur_altsetting)) -			usb_set_interface(dev->udev, -					  intf->cur_altsetting->desc.bInterfaceNumber, -					  CDC_NCM_COMM_ALTSETTING_MBIM); -		data_altsetting = CDC_NCM_DATA_ALTSETTING_MBIM; -	} -  	/* Probably NCM, defer for cdc_ncm_bind */  	if (!cdc_ncm_comm_intf_is_mbim(intf->cur_altsetting))  		goto err; diff --git a/drivers/net/usb/cdc_ncm.c b/drivers/net/usb/cdc_ncm.c index 4a8c25a2229..4709fa3497c 100644 --- a/drivers/net/usb/cdc_ncm.c +++ b/drivers/net/usb/cdc_ncm.c @@ -55,6 +55,14 @@  #define	DRIVER_VERSION				"14-Mar-2012" +#if IS_ENABLED(CONFIG_USB_NET_CDC_MBIM) +static bool prefer_mbim = true; +#else +static bool prefer_mbim; +#endif +module_param(prefer_mbim, bool, S_IRUGO | S_IWUSR); +MODULE_PARM_DESC(prefer_mbim, "Prefer MBIM setting on dual NCM/MBIM functions"); +  static void cdc_ncm_txpath_bh(unsigned long param);  static void cdc_ncm_tx_timeout_start(struct cdc_ncm_ctx *ctx);  static enum hrtimer_restart cdc_ncm_tx_timer_cb(struct hrtimer *hr_timer); @@ -550,9 +558,12 @@ void cdc_ncm_unbind(struct usbnet *dev, struct usb_interface *intf)  }  EXPORT_SYMBOL_GPL(cdc_ncm_unbind); -static int cdc_ncm_bind(struct usbnet *dev, struct usb_interface *intf) +/* Select the MBIM altsetting iff it is preferred and available, + * returning the number of the corresponding data interface altsetting + */ +u8 cdc_ncm_select_altsetting(struct usbnet *dev, struct usb_interface *intf)  { -	int ret; +	struct usb_host_interface *alt;  	/* The MBIM spec defines a NCM compatible default altsetting,  	 * which we may have matched: @@ -568,23 +579,27 @@ static int cdc_ncm_bind(struct usbnet *dev, struct usb_interface *intf)  	 *   endpoint descriptors, shall be constructed according to  	 *   the rules given in section 6 (USB Device Model) of this  	 *   specification." -	 * -	 * Do not bind to such interfaces, allowing cdc_mbim to handle -	 * them  	 */ -#if IS_ENABLED(CONFIG_USB_NET_CDC_MBIM) -	if ((intf->num_altsetting == 2) && -	    !usb_set_interface(dev->udev, -			       intf->cur_altsetting->desc.bInterfaceNumber, -			       CDC_NCM_COMM_ALTSETTING_MBIM)) { -		if (cdc_ncm_comm_intf_is_mbim(intf->cur_altsetting)) -			return -ENODEV; -		else -			usb_set_interface(dev->udev, -					  intf->cur_altsetting->desc.bInterfaceNumber, -					  CDC_NCM_COMM_ALTSETTING_NCM); +	if (prefer_mbim && intf->num_altsetting == 2) { +		alt = usb_altnum_to_altsetting(intf, CDC_NCM_COMM_ALTSETTING_MBIM); +		if (alt && cdc_ncm_comm_intf_is_mbim(alt) && +		    !usb_set_interface(dev->udev, +				       intf->cur_altsetting->desc.bInterfaceNumber, +				       CDC_NCM_COMM_ALTSETTING_MBIM)) +			return CDC_NCM_DATA_ALTSETTING_MBIM;  	} -#endif +	return CDC_NCM_DATA_ALTSETTING_NCM; +} +EXPORT_SYMBOL_GPL(cdc_ncm_select_altsetting); + +static int cdc_ncm_bind(struct usbnet *dev, struct usb_interface *intf) +{ +	int ret; + +	/* MBIM backwards compatible function? */ +	cdc_ncm_select_altsetting(dev, intf); +	if (cdc_ncm_comm_intf_is_mbim(intf->cur_altsetting)) +		return -ENODEV;  	/* NCM data altsetting is always 1 */  	ret = cdc_ncm_bind_common(dev, intf, 1); @@ -1213,6 +1228,14 @@ static const struct usb_device_id cdc_devs[] = {  	  .driver_info = (unsigned long) &wwan_info,  	}, +	/* tag Huawei devices as wwan */ +	{ USB_VENDOR_AND_INTERFACE_INFO(0x12d1, +					USB_CLASS_COMM, +					USB_CDC_SUBCLASS_NCM, +					USB_CDC_PROTO_NONE), +	  .driver_info = (unsigned long)&wwan_info, +	}, +  	/* Huawei NCM devices disguised as vendor specific */  	{ USB_VENDOR_AND_INTERFACE_INFO(0x12d1, 0xff, 0x02, 0x16),  	  .driver_info = (unsigned long)&wwan_info, diff --git a/drivers/net/usb/qmi_wwan.c b/drivers/net/usb/qmi_wwan.c index efb5c7c33a2..968d5d50751 100644 --- a/drivers/net/usb/qmi_wwan.c +++ b/drivers/net/usb/qmi_wwan.c @@ -139,16 +139,9 @@ static int qmi_wwan_bind(struct usbnet *dev, struct usb_interface *intf)  	BUILD_BUG_ON((sizeof(((struct usbnet *)0)->data) < sizeof(struct qmi_wwan_state))); -	/* control and data is shared? */ -	if (intf->cur_altsetting->desc.bNumEndpoints == 3) { -		info->control = intf; -		info->data = intf; -		goto shared; -	} - -	/* else require a single interrupt status endpoint on control intf */ -	if (intf->cur_altsetting->desc.bNumEndpoints != 1) -		goto err; +	/* set up initial state */ +	info->control = intf; +	info->data = intf;  	/* and a number of CDC descriptors */  	while (len > 3) { @@ -207,25 +200,14 @@ next_desc:  		buf += h->bLength;  	} -	/* did we find all the required ones? */ -	if (!(found & (1 << USB_CDC_HEADER_TYPE)) || -	    !(found & (1 << USB_CDC_UNION_TYPE))) { -		dev_err(&intf->dev, "CDC functional descriptors missing\n"); -		goto err; -	} - -	/* verify CDC Union */ -	if (desc->bInterfaceNumber != cdc_union->bMasterInterface0) { -		dev_err(&intf->dev, "bogus CDC Union: master=%u\n", cdc_union->bMasterInterface0); -		goto err; -	} - -	/* need to save these for unbind */ -	info->control = intf; -	info->data = usb_ifnum_to_if(dev->udev,	cdc_union->bSlaveInterface0); -	if (!info->data) { -		dev_err(&intf->dev, "bogus CDC Union: slave=%u\n", cdc_union->bSlaveInterface0); -		goto err; +	/* Use separate control and data interfaces if we found a CDC Union */ +	if (cdc_union) { +		info->data = usb_ifnum_to_if(dev->udev, cdc_union->bSlaveInterface0); +		if (desc->bInterfaceNumber != cdc_union->bMasterInterface0 || !info->data) { +			dev_err(&intf->dev, "bogus CDC Union: master=%u, slave=%u\n", +				cdc_union->bMasterInterface0, cdc_union->bSlaveInterface0); +			goto err; +		}  	}  	/* errors aren't fatal - we can live with the dynamic address */ @@ -235,11 +217,12 @@ next_desc:  	}  	/* claim data interface and set it up */ -	status = usb_driver_claim_interface(driver, info->data, dev); -	if (status < 0) -		goto err; +	if (info->control != info->data) { +		status = usb_driver_claim_interface(driver, info->data, dev); +		if (status < 0) +			goto err; +	} -shared:  	status = qmi_wwan_register_subdriver(dev);  	if (status < 0 && info->control != info->data) {  		usb_set_intfdata(info->data, NULL); diff --git a/drivers/net/usb/smsc75xx.c b/drivers/net/usb/smsc75xx.c index 9abe51710f2..1a15ec14c38 100644 --- a/drivers/net/usb/smsc75xx.c +++ b/drivers/net/usb/smsc75xx.c @@ -914,8 +914,12 @@ static int smsc75xx_set_rx_max_frame_length(struct usbnet *dev, int size)  static int smsc75xx_change_mtu(struct net_device *netdev, int new_mtu)  {  	struct usbnet *dev = netdev_priv(netdev); +	int ret; + +	if (new_mtu > MAX_SINGLE_PACKET_SIZE) +		return -EINVAL; -	int ret = smsc75xx_set_rx_max_frame_length(dev, new_mtu); +	ret = smsc75xx_set_rx_max_frame_length(dev, new_mtu + ETH_HLEN);  	if (ret < 0) {  		netdev_warn(dev->net, "Failed to set mac rx frame length\n");  		return ret; @@ -1324,7 +1328,7 @@ static int smsc75xx_reset(struct usbnet *dev)  	netif_dbg(dev, ifup, dev->net, "FCT_TX_CTL set to 0x%08x\n", buf); -	ret = smsc75xx_set_rx_max_frame_length(dev, 1514); +	ret = smsc75xx_set_rx_max_frame_length(dev, dev->net->mtu + ETH_HLEN);  	if (ret < 0) {  		netdev_warn(dev->net, "Failed to set max rx frame length\n");  		return ret; @@ -2134,8 +2138,8 @@ static int smsc75xx_rx_fixup(struct usbnet *dev, struct sk_buff *skb)  			else if (rx_cmd_a & (RX_CMD_A_LONG | RX_CMD_A_RUNT))  				dev->net->stats.rx_frame_errors++;  		} else { -			/* ETH_FRAME_LEN + 4(CRC) + 2(COE) + 4(Vlan) */ -			if (unlikely(size > (ETH_FRAME_LEN + 12))) { +			/* MAX_SINGLE_PACKET_SIZE + 4(CRC) + 2(COE) + 4(Vlan) */ +			if (unlikely(size > (MAX_SINGLE_PACKET_SIZE + ETH_HLEN + 12))) {  				netif_dbg(dev, rx_err, dev->net,  					  "size err rx_cmd_a=0x%08x\n",  					  rx_cmd_a);  |