xref: /openbmc/linux/drivers/net/usb/lan78xx.c (revision ecc23d0a422a3118fcf6e4f0a46e17a6c2047b02)
16be665a5SStefan Wahren // SPDX-License-Identifier: GPL-2.0+
255d7de9dSWoojung.Huh@microchip.com /*
355d7de9dSWoojung.Huh@microchip.com  * Copyright (C) 2015 Microchip Technology
455d7de9dSWoojung.Huh@microchip.com  */
555d7de9dSWoojung.Huh@microchip.com #include <linux/module.h>
655d7de9dSWoojung.Huh@microchip.com #include <linux/netdevice.h>
755d7de9dSWoojung.Huh@microchip.com #include <linux/etherdevice.h>
855d7de9dSWoojung.Huh@microchip.com #include <linux/ethtool.h>
955d7de9dSWoojung.Huh@microchip.com #include <linux/usb.h>
1055d7de9dSWoojung.Huh@microchip.com #include <linux/crc32.h>
1155d7de9dSWoojung.Huh@microchip.com #include <linux/signal.h>
1255d7de9dSWoojung.Huh@microchip.com #include <linux/slab.h>
1355d7de9dSWoojung.Huh@microchip.com #include <linux/if_vlan.h>
1455d7de9dSWoojung.Huh@microchip.com #include <linux/uaccess.h>
153c1bcc86SAndrew Lunn #include <linux/linkmode.h>
1655d7de9dSWoojung.Huh@microchip.com #include <linux/list.h>
1755d7de9dSWoojung.Huh@microchip.com #include <linux/ip.h>
1855d7de9dSWoojung.Huh@microchip.com #include <linux/ipv6.h>
1955d7de9dSWoojung.Huh@microchip.com #include <linux/mdio.h>
20c6e970a0SAndrew Lunn #include <linux/phy.h>
2155d7de9dSWoojung.Huh@microchip.com #include <net/ip6_checksum.h>
22ce896476SJames Hughes #include <net/vxlan.h>
23cc89c323SWoojung Huh #include <linux/interrupt.h>
24cc89c323SWoojung Huh #include <linux/irqdomain.h>
25cc89c323SWoojung Huh #include <linux/irq.h>
26cc89c323SWoojung Huh #include <linux/irqchip/chained_irq.h>
27bdfba55eSWoojung.Huh@microchip.com #include <linux/microchipphy.h>
2889b36fb5SRaghuram Chary J #include <linux/phy_fixed.h>
291827b067SPhil Elwell #include <linux/of_mdio.h>
30760db29bSPhil Elwell #include <linux/of_net.h>
3155d7de9dSWoojung.Huh@microchip.com #include "lan78xx.h"
3255d7de9dSWoojung.Huh@microchip.com 
3355d7de9dSWoojung.Huh@microchip.com #define DRIVER_AUTHOR	"WOOJUNG HUH <woojung.huh@microchip.com>"
3455d7de9dSWoojung.Huh@microchip.com #define DRIVER_DESC	"LAN78XX USB 3.0 Gigabit Ethernet Devices"
3555d7de9dSWoojung.Huh@microchip.com #define DRIVER_NAME	"lan78xx"
3655d7de9dSWoojung.Huh@microchip.com 
3755d7de9dSWoojung.Huh@microchip.com #define TX_TIMEOUT_JIFFIES		(5 * HZ)
3855d7de9dSWoojung.Huh@microchip.com #define THROTTLE_JIFFIES		(HZ / 8)
3955d7de9dSWoojung.Huh@microchip.com #define UNLINK_TIMEOUT_MS		3
4055d7de9dSWoojung.Huh@microchip.com 
4155d7de9dSWoojung.Huh@microchip.com #define RX_MAX_QUEUE_MEMORY		(60 * 1518)
4255d7de9dSWoojung.Huh@microchip.com 
4355d7de9dSWoojung.Huh@microchip.com #define SS_USB_PKT_SIZE			(1024)
4455d7de9dSWoojung.Huh@microchip.com #define HS_USB_PKT_SIZE			(512)
4555d7de9dSWoojung.Huh@microchip.com #define FS_USB_PKT_SIZE			(64)
4655d7de9dSWoojung.Huh@microchip.com 
4755d7de9dSWoojung.Huh@microchip.com #define MAX_RX_FIFO_SIZE		(12 * 1024)
4855d7de9dSWoojung.Huh@microchip.com #define MAX_TX_FIFO_SIZE		(12 * 1024)
49dc35f854SJohn Efstathiades 
50dc35f854SJohn Efstathiades #define FLOW_THRESHOLD(n)		((((n) + 511) / 512) & 0x7F)
51dc35f854SJohn Efstathiades #define FLOW_CTRL_THRESHOLD(on, off)	((FLOW_THRESHOLD(on)  << 0) | \
52dc35f854SJohn Efstathiades 					 (FLOW_THRESHOLD(off) << 8))
53dc35f854SJohn Efstathiades 
54dc35f854SJohn Efstathiades /* Flow control turned on when Rx FIFO level rises above this level (bytes) */
55dc35f854SJohn Efstathiades #define FLOW_ON_SS			9216
56dc35f854SJohn Efstathiades #define FLOW_ON_HS			8704
57dc35f854SJohn Efstathiades 
58dc35f854SJohn Efstathiades /* Flow control turned off when Rx FIFO level falls below this level (bytes) */
59dc35f854SJohn Efstathiades #define FLOW_OFF_SS			4096
60dc35f854SJohn Efstathiades #define FLOW_OFF_HS			1024
61dc35f854SJohn Efstathiades 
6255d7de9dSWoojung.Huh@microchip.com #define DEFAULT_BURST_CAP_SIZE		(MAX_TX_FIFO_SIZE)
6355d7de9dSWoojung.Huh@microchip.com #define DEFAULT_BULK_IN_DELAY		(0x0800)
6455d7de9dSWoojung.Huh@microchip.com #define MAX_SINGLE_PACKET_SIZE		(9000)
6555d7de9dSWoojung.Huh@microchip.com #define DEFAULT_TX_CSUM_ENABLE		(true)
6655d7de9dSWoojung.Huh@microchip.com #define DEFAULT_RX_CSUM_ENABLE		(true)
6755d7de9dSWoojung.Huh@microchip.com #define DEFAULT_TSO_CSUM_ENABLE		(true)
6855d7de9dSWoojung.Huh@microchip.com #define DEFAULT_VLAN_FILTER_ENABLE	(true)
69ec21ecf0SDave Stevenson #define DEFAULT_VLAN_RX_OFFLOAD		(true)
70d383216aSJohn Efstathiades #define TX_ALIGNMENT			(4)
7155d7de9dSWoojung.Huh@microchip.com #define RXW_PADDING			2
7255d7de9dSWoojung.Huh@microchip.com 
7355d7de9dSWoojung.Huh@microchip.com #define LAN78XX_USB_VENDOR_ID		(0x0424)
7455d7de9dSWoojung.Huh@microchip.com #define LAN7800_USB_PRODUCT_ID		(0x7800)
7555d7de9dSWoojung.Huh@microchip.com #define LAN7850_USB_PRODUCT_ID		(0x7850)
7602dc1f3dSWoojung Huh #define LAN7801_USB_PRODUCT_ID		(0x7801)
7755d7de9dSWoojung.Huh@microchip.com #define LAN78XX_EEPROM_MAGIC		(0x78A5)
7855d7de9dSWoojung.Huh@microchip.com #define LAN78XX_OTP_MAGIC		(0x78F3)
79ef8a0f6eSGreg Jesionowski #define AT29M2AF_USB_VENDOR_ID		(0x07C9)
80ef8a0f6eSGreg Jesionowski #define AT29M2AF_USB_PRODUCT_ID	(0x0012)
8155d7de9dSWoojung.Huh@microchip.com 
8255d7de9dSWoojung.Huh@microchip.com #define	MII_READ			1
8355d7de9dSWoojung.Huh@microchip.com #define	MII_WRITE			0
8455d7de9dSWoojung.Huh@microchip.com 
8555d7de9dSWoojung.Huh@microchip.com #define EEPROM_INDICATOR		(0xA5)
8655d7de9dSWoojung.Huh@microchip.com #define EEPROM_MAC_OFFSET		(0x01)
8755d7de9dSWoojung.Huh@microchip.com #define MAX_EEPROM_SIZE			512
8855d7de9dSWoojung.Huh@microchip.com #define OTP_INDICATOR_1			(0xF3)
8955d7de9dSWoojung.Huh@microchip.com #define OTP_INDICATOR_2			(0xF7)
9055d7de9dSWoojung.Huh@microchip.com 
9155d7de9dSWoojung.Huh@microchip.com #define WAKE_ALL			(WAKE_PHY | WAKE_UCAST | \
9255d7de9dSWoojung.Huh@microchip.com 					 WAKE_MCAST | WAKE_BCAST | \
9355d7de9dSWoojung.Huh@microchip.com 					 WAKE_ARP | WAKE_MAGIC)
9455d7de9dSWoojung.Huh@microchip.com 
95d383216aSJohn Efstathiades #define TX_URB_NUM			10
96d383216aSJohn Efstathiades #define TX_SS_URB_NUM			TX_URB_NUM
97d383216aSJohn Efstathiades #define TX_HS_URB_NUM			TX_URB_NUM
98d383216aSJohn Efstathiades #define TX_FS_URB_NUM			TX_URB_NUM
99d383216aSJohn Efstathiades 
100d383216aSJohn Efstathiades /* A single URB buffer must be large enough to hold a complete jumbo packet
101d383216aSJohn Efstathiades  */
102d383216aSJohn Efstathiades #define TX_SS_URB_SIZE			(32 * 1024)
103d383216aSJohn Efstathiades #define TX_HS_URB_SIZE			(16 * 1024)
104d383216aSJohn Efstathiades #define TX_FS_URB_SIZE			(10 * 1024)
105d383216aSJohn Efstathiades 
106c450a8ebSJohn Efstathiades #define RX_SS_URB_NUM			30
107c450a8ebSJohn Efstathiades #define RX_HS_URB_NUM			10
108c450a8ebSJohn Efstathiades #define RX_FS_URB_NUM			10
109c450a8ebSJohn Efstathiades #define RX_SS_URB_SIZE			TX_SS_URB_SIZE
110c450a8ebSJohn Efstathiades #define RX_HS_URB_SIZE			TX_HS_URB_SIZE
111c450a8ebSJohn Efstathiades #define RX_FS_URB_SIZE			TX_FS_URB_SIZE
112c450a8ebSJohn Efstathiades 
113c450a8ebSJohn Efstathiades #define SS_BURST_CAP_SIZE		RX_SS_URB_SIZE
114c450a8ebSJohn Efstathiades #define SS_BULK_IN_DELAY		0x2000
115c450a8ebSJohn Efstathiades #define HS_BURST_CAP_SIZE		RX_HS_URB_SIZE
116c450a8ebSJohn Efstathiades #define HS_BULK_IN_DELAY		0x2000
117c450a8ebSJohn Efstathiades #define FS_BURST_CAP_SIZE		RX_FS_URB_SIZE
118c450a8ebSJohn Efstathiades #define FS_BULK_IN_DELAY		0x2000
119c450a8ebSJohn Efstathiades 
120d383216aSJohn Efstathiades #define TX_CMD_LEN			8
121d383216aSJohn Efstathiades #define TX_SKB_MIN_LEN			(TX_CMD_LEN + ETH_HLEN)
122d383216aSJohn Efstathiades #define LAN78XX_TSO_SIZE(dev)		((dev)->tx_urb_size - TX_SKB_MIN_LEN)
123d383216aSJohn Efstathiades 
1240dd87266SJohn Efstathiades #define RX_CMD_LEN			10
1250dd87266SJohn Efstathiades #define RX_SKB_MIN_LEN			(RX_CMD_LEN + ETH_HLEN)
1260dd87266SJohn Efstathiades #define RX_MAX_FRAME_LEN(mtu)		((mtu) + ETH_HLEN + VLAN_HLEN)
1270dd87266SJohn Efstathiades 
12855d7de9dSWoojung.Huh@microchip.com /* USB related defines */
12955d7de9dSWoojung.Huh@microchip.com #define BULK_IN_PIPE			1
13055d7de9dSWoojung.Huh@microchip.com #define BULK_OUT_PIPE			2
13155d7de9dSWoojung.Huh@microchip.com 
13255d7de9dSWoojung.Huh@microchip.com /* default autosuspend delay (mSec)*/
13355d7de9dSWoojung.Huh@microchip.com #define DEFAULT_AUTOSUSPEND_DELAY	(10 * 1000)
13455d7de9dSWoojung.Huh@microchip.com 
13520ff5565SWoojung Huh /* statistic update interval (mSec) */
13620ff5565SWoojung Huh #define STAT_UPDATE_TIMER		(1 * 1000)
13720ff5565SWoojung Huh 
138e1210fe6SJohn Efstathiades /* time to wait for MAC or FCT to stop (jiffies) */
139e1210fe6SJohn Efstathiades #define HW_DISABLE_TIMEOUT		(HZ / 10)
140e1210fe6SJohn Efstathiades 
141e1210fe6SJohn Efstathiades /* time to wait between polling MAC or FCT state (ms) */
142e1210fe6SJohn Efstathiades #define HW_DISABLE_DELAY_MS		1
143e1210fe6SJohn Efstathiades 
144cc89c323SWoojung Huh /* defines interrupts from interrupt EP */
145cc89c323SWoojung Huh #define MAX_INT_EP			(32)
146cc89c323SWoojung Huh #define INT_EP_INTEP			(31)
147cc89c323SWoojung Huh #define INT_EP_OTP_WR_DONE		(28)
148cc89c323SWoojung Huh #define INT_EP_EEE_TX_LPI_START		(26)
149cc89c323SWoojung Huh #define INT_EP_EEE_TX_LPI_STOP		(25)
150cc89c323SWoojung Huh #define INT_EP_EEE_RX_LPI		(24)
151cc89c323SWoojung Huh #define INT_EP_MAC_RESET_TIMEOUT	(23)
152cc89c323SWoojung Huh #define INT_EP_RDFO			(22)
153cc89c323SWoojung Huh #define INT_EP_TXE			(21)
154cc89c323SWoojung Huh #define INT_EP_USB_STATUS		(20)
155cc89c323SWoojung Huh #define INT_EP_TX_DIS			(19)
156cc89c323SWoojung Huh #define INT_EP_RX_DIS			(18)
157cc89c323SWoojung Huh #define INT_EP_PHY			(17)
158cc89c323SWoojung Huh #define INT_EP_DP			(16)
159cc89c323SWoojung Huh #define INT_EP_MAC_ERR			(15)
160cc89c323SWoojung Huh #define INT_EP_TDFU			(14)
161cc89c323SWoojung Huh #define INT_EP_TDFO			(13)
162cc89c323SWoojung Huh #define INT_EP_UTX			(12)
163cc89c323SWoojung Huh #define INT_EP_GPIO_11			(11)
164cc89c323SWoojung Huh #define INT_EP_GPIO_10			(10)
165cc89c323SWoojung Huh #define INT_EP_GPIO_9			(9)
166cc89c323SWoojung Huh #define INT_EP_GPIO_8			(8)
167cc89c323SWoojung Huh #define INT_EP_GPIO_7			(7)
168cc89c323SWoojung Huh #define INT_EP_GPIO_6			(6)
169cc89c323SWoojung Huh #define INT_EP_GPIO_5			(5)
170cc89c323SWoojung Huh #define INT_EP_GPIO_4			(4)
171cc89c323SWoojung Huh #define INT_EP_GPIO_3			(3)
172cc89c323SWoojung Huh #define INT_EP_GPIO_2			(2)
173cc89c323SWoojung Huh #define INT_EP_GPIO_1			(1)
174cc89c323SWoojung Huh #define INT_EP_GPIO_0			(0)
175cc89c323SWoojung Huh 
17655d7de9dSWoojung.Huh@microchip.com static const char lan78xx_gstrings[][ETH_GSTRING_LEN] = {
17755d7de9dSWoojung.Huh@microchip.com 	"RX FCS Errors",
17855d7de9dSWoojung.Huh@microchip.com 	"RX Alignment Errors",
17955d7de9dSWoojung.Huh@microchip.com 	"Rx Fragment Errors",
18055d7de9dSWoojung.Huh@microchip.com 	"RX Jabber Errors",
18155d7de9dSWoojung.Huh@microchip.com 	"RX Undersize Frame Errors",
18255d7de9dSWoojung.Huh@microchip.com 	"RX Oversize Frame Errors",
18355d7de9dSWoojung.Huh@microchip.com 	"RX Dropped Frames",
18455d7de9dSWoojung.Huh@microchip.com 	"RX Unicast Byte Count",
18555d7de9dSWoojung.Huh@microchip.com 	"RX Broadcast Byte Count",
18655d7de9dSWoojung.Huh@microchip.com 	"RX Multicast Byte Count",
18755d7de9dSWoojung.Huh@microchip.com 	"RX Unicast Frames",
18855d7de9dSWoojung.Huh@microchip.com 	"RX Broadcast Frames",
18955d7de9dSWoojung.Huh@microchip.com 	"RX Multicast Frames",
19055d7de9dSWoojung.Huh@microchip.com 	"RX Pause Frames",
19155d7de9dSWoojung.Huh@microchip.com 	"RX 64 Byte Frames",
19255d7de9dSWoojung.Huh@microchip.com 	"RX 65 - 127 Byte Frames",
19355d7de9dSWoojung.Huh@microchip.com 	"RX 128 - 255 Byte Frames",
19455d7de9dSWoojung.Huh@microchip.com 	"RX 256 - 511 Bytes Frames",
19555d7de9dSWoojung.Huh@microchip.com 	"RX 512 - 1023 Byte Frames",
19655d7de9dSWoojung.Huh@microchip.com 	"RX 1024 - 1518 Byte Frames",
19755d7de9dSWoojung.Huh@microchip.com 	"RX Greater 1518 Byte Frames",
19855d7de9dSWoojung.Huh@microchip.com 	"EEE RX LPI Transitions",
19955d7de9dSWoojung.Huh@microchip.com 	"EEE RX LPI Time",
20055d7de9dSWoojung.Huh@microchip.com 	"TX FCS Errors",
20155d7de9dSWoojung.Huh@microchip.com 	"TX Excess Deferral Errors",
20255d7de9dSWoojung.Huh@microchip.com 	"TX Carrier Errors",
20355d7de9dSWoojung.Huh@microchip.com 	"TX Bad Byte Count",
20455d7de9dSWoojung.Huh@microchip.com 	"TX Single Collisions",
20555d7de9dSWoojung.Huh@microchip.com 	"TX Multiple Collisions",
20655d7de9dSWoojung.Huh@microchip.com 	"TX Excessive Collision",
20755d7de9dSWoojung.Huh@microchip.com 	"TX Late Collisions",
20855d7de9dSWoojung.Huh@microchip.com 	"TX Unicast Byte Count",
20955d7de9dSWoojung.Huh@microchip.com 	"TX Broadcast Byte Count",
21055d7de9dSWoojung.Huh@microchip.com 	"TX Multicast Byte Count",
21155d7de9dSWoojung.Huh@microchip.com 	"TX Unicast Frames",
21255d7de9dSWoojung.Huh@microchip.com 	"TX Broadcast Frames",
21355d7de9dSWoojung.Huh@microchip.com 	"TX Multicast Frames",
21455d7de9dSWoojung.Huh@microchip.com 	"TX Pause Frames",
21555d7de9dSWoojung.Huh@microchip.com 	"TX 64 Byte Frames",
21655d7de9dSWoojung.Huh@microchip.com 	"TX 65 - 127 Byte Frames",
21755d7de9dSWoojung.Huh@microchip.com 	"TX 128 - 255 Byte Frames",
21855d7de9dSWoojung.Huh@microchip.com 	"TX 256 - 511 Bytes Frames",
21955d7de9dSWoojung.Huh@microchip.com 	"TX 512 - 1023 Byte Frames",
22055d7de9dSWoojung.Huh@microchip.com 	"TX 1024 - 1518 Byte Frames",
22155d7de9dSWoojung.Huh@microchip.com 	"TX Greater 1518 Byte Frames",
22255d7de9dSWoojung.Huh@microchip.com 	"EEE TX LPI Transitions",
22355d7de9dSWoojung.Huh@microchip.com 	"EEE TX LPI Time",
22455d7de9dSWoojung.Huh@microchip.com };
22555d7de9dSWoojung.Huh@microchip.com 
22655d7de9dSWoojung.Huh@microchip.com struct lan78xx_statstage {
22755d7de9dSWoojung.Huh@microchip.com 	u32 rx_fcs_errors;
22855d7de9dSWoojung.Huh@microchip.com 	u32 rx_alignment_errors;
22955d7de9dSWoojung.Huh@microchip.com 	u32 rx_fragment_errors;
23055d7de9dSWoojung.Huh@microchip.com 	u32 rx_jabber_errors;
23155d7de9dSWoojung.Huh@microchip.com 	u32 rx_undersize_frame_errors;
23255d7de9dSWoojung.Huh@microchip.com 	u32 rx_oversize_frame_errors;
23355d7de9dSWoojung.Huh@microchip.com 	u32 rx_dropped_frames;
23455d7de9dSWoojung.Huh@microchip.com 	u32 rx_unicast_byte_count;
23555d7de9dSWoojung.Huh@microchip.com 	u32 rx_broadcast_byte_count;
23655d7de9dSWoojung.Huh@microchip.com 	u32 rx_multicast_byte_count;
23755d7de9dSWoojung.Huh@microchip.com 	u32 rx_unicast_frames;
23855d7de9dSWoojung.Huh@microchip.com 	u32 rx_broadcast_frames;
23955d7de9dSWoojung.Huh@microchip.com 	u32 rx_multicast_frames;
24055d7de9dSWoojung.Huh@microchip.com 	u32 rx_pause_frames;
24155d7de9dSWoojung.Huh@microchip.com 	u32 rx_64_byte_frames;
24255d7de9dSWoojung.Huh@microchip.com 	u32 rx_65_127_byte_frames;
24355d7de9dSWoojung.Huh@microchip.com 	u32 rx_128_255_byte_frames;
24455d7de9dSWoojung.Huh@microchip.com 	u32 rx_256_511_bytes_frames;
24555d7de9dSWoojung.Huh@microchip.com 	u32 rx_512_1023_byte_frames;
24655d7de9dSWoojung.Huh@microchip.com 	u32 rx_1024_1518_byte_frames;
24755d7de9dSWoojung.Huh@microchip.com 	u32 rx_greater_1518_byte_frames;
24855d7de9dSWoojung.Huh@microchip.com 	u32 eee_rx_lpi_transitions;
24955d7de9dSWoojung.Huh@microchip.com 	u32 eee_rx_lpi_time;
25055d7de9dSWoojung.Huh@microchip.com 	u32 tx_fcs_errors;
25155d7de9dSWoojung.Huh@microchip.com 	u32 tx_excess_deferral_errors;
25255d7de9dSWoojung.Huh@microchip.com 	u32 tx_carrier_errors;
25355d7de9dSWoojung.Huh@microchip.com 	u32 tx_bad_byte_count;
25455d7de9dSWoojung.Huh@microchip.com 	u32 tx_single_collisions;
25555d7de9dSWoojung.Huh@microchip.com 	u32 tx_multiple_collisions;
25655d7de9dSWoojung.Huh@microchip.com 	u32 tx_excessive_collision;
25755d7de9dSWoojung.Huh@microchip.com 	u32 tx_late_collisions;
25855d7de9dSWoojung.Huh@microchip.com 	u32 tx_unicast_byte_count;
25955d7de9dSWoojung.Huh@microchip.com 	u32 tx_broadcast_byte_count;
26055d7de9dSWoojung.Huh@microchip.com 	u32 tx_multicast_byte_count;
26155d7de9dSWoojung.Huh@microchip.com 	u32 tx_unicast_frames;
26255d7de9dSWoojung.Huh@microchip.com 	u32 tx_broadcast_frames;
26355d7de9dSWoojung.Huh@microchip.com 	u32 tx_multicast_frames;
26455d7de9dSWoojung.Huh@microchip.com 	u32 tx_pause_frames;
26555d7de9dSWoojung.Huh@microchip.com 	u32 tx_64_byte_frames;
26655d7de9dSWoojung.Huh@microchip.com 	u32 tx_65_127_byte_frames;
26755d7de9dSWoojung.Huh@microchip.com 	u32 tx_128_255_byte_frames;
26855d7de9dSWoojung.Huh@microchip.com 	u32 tx_256_511_bytes_frames;
26955d7de9dSWoojung.Huh@microchip.com 	u32 tx_512_1023_byte_frames;
27055d7de9dSWoojung.Huh@microchip.com 	u32 tx_1024_1518_byte_frames;
27155d7de9dSWoojung.Huh@microchip.com 	u32 tx_greater_1518_byte_frames;
27255d7de9dSWoojung.Huh@microchip.com 	u32 eee_tx_lpi_transitions;
27355d7de9dSWoojung.Huh@microchip.com 	u32 eee_tx_lpi_time;
27455d7de9dSWoojung.Huh@microchip.com };
27555d7de9dSWoojung.Huh@microchip.com 
27620ff5565SWoojung Huh struct lan78xx_statstage64 {
27720ff5565SWoojung Huh 	u64 rx_fcs_errors;
27820ff5565SWoojung Huh 	u64 rx_alignment_errors;
27920ff5565SWoojung Huh 	u64 rx_fragment_errors;
28020ff5565SWoojung Huh 	u64 rx_jabber_errors;
28120ff5565SWoojung Huh 	u64 rx_undersize_frame_errors;
28220ff5565SWoojung Huh 	u64 rx_oversize_frame_errors;
28320ff5565SWoojung Huh 	u64 rx_dropped_frames;
28420ff5565SWoojung Huh 	u64 rx_unicast_byte_count;
28520ff5565SWoojung Huh 	u64 rx_broadcast_byte_count;
28620ff5565SWoojung Huh 	u64 rx_multicast_byte_count;
28720ff5565SWoojung Huh 	u64 rx_unicast_frames;
28820ff5565SWoojung Huh 	u64 rx_broadcast_frames;
28920ff5565SWoojung Huh 	u64 rx_multicast_frames;
29020ff5565SWoojung Huh 	u64 rx_pause_frames;
29120ff5565SWoojung Huh 	u64 rx_64_byte_frames;
29220ff5565SWoojung Huh 	u64 rx_65_127_byte_frames;
29320ff5565SWoojung Huh 	u64 rx_128_255_byte_frames;
29420ff5565SWoojung Huh 	u64 rx_256_511_bytes_frames;
29520ff5565SWoojung Huh 	u64 rx_512_1023_byte_frames;
29620ff5565SWoojung Huh 	u64 rx_1024_1518_byte_frames;
29720ff5565SWoojung Huh 	u64 rx_greater_1518_byte_frames;
29820ff5565SWoojung Huh 	u64 eee_rx_lpi_transitions;
29920ff5565SWoojung Huh 	u64 eee_rx_lpi_time;
30020ff5565SWoojung Huh 	u64 tx_fcs_errors;
30120ff5565SWoojung Huh 	u64 tx_excess_deferral_errors;
30220ff5565SWoojung Huh 	u64 tx_carrier_errors;
30320ff5565SWoojung Huh 	u64 tx_bad_byte_count;
30420ff5565SWoojung Huh 	u64 tx_single_collisions;
30520ff5565SWoojung Huh 	u64 tx_multiple_collisions;
30620ff5565SWoojung Huh 	u64 tx_excessive_collision;
30720ff5565SWoojung Huh 	u64 tx_late_collisions;
30820ff5565SWoojung Huh 	u64 tx_unicast_byte_count;
30920ff5565SWoojung Huh 	u64 tx_broadcast_byte_count;
31020ff5565SWoojung Huh 	u64 tx_multicast_byte_count;
31120ff5565SWoojung Huh 	u64 tx_unicast_frames;
31220ff5565SWoojung Huh 	u64 tx_broadcast_frames;
31320ff5565SWoojung Huh 	u64 tx_multicast_frames;
31420ff5565SWoojung Huh 	u64 tx_pause_frames;
31520ff5565SWoojung Huh 	u64 tx_64_byte_frames;
31620ff5565SWoojung Huh 	u64 tx_65_127_byte_frames;
31720ff5565SWoojung Huh 	u64 tx_128_255_byte_frames;
31820ff5565SWoojung Huh 	u64 tx_256_511_bytes_frames;
31920ff5565SWoojung Huh 	u64 tx_512_1023_byte_frames;
32020ff5565SWoojung Huh 	u64 tx_1024_1518_byte_frames;
32120ff5565SWoojung Huh 	u64 tx_greater_1518_byte_frames;
32220ff5565SWoojung Huh 	u64 eee_tx_lpi_transitions;
32320ff5565SWoojung Huh 	u64 eee_tx_lpi_time;
32420ff5565SWoojung Huh };
32520ff5565SWoojung Huh 
32649621865SRaghuram Chary J static u32 lan78xx_regs[] = {
32749621865SRaghuram Chary J 	ID_REV,
32849621865SRaghuram Chary J 	INT_STS,
32949621865SRaghuram Chary J 	HW_CFG,
33049621865SRaghuram Chary J 	PMT_CTL,
33149621865SRaghuram Chary J 	E2P_CMD,
33249621865SRaghuram Chary J 	E2P_DATA,
33349621865SRaghuram Chary J 	USB_STATUS,
33449621865SRaghuram Chary J 	VLAN_TYPE,
33549621865SRaghuram Chary J 	MAC_CR,
33649621865SRaghuram Chary J 	MAC_RX,
33749621865SRaghuram Chary J 	MAC_TX,
33849621865SRaghuram Chary J 	FLOW,
33949621865SRaghuram Chary J 	ERR_STS,
34049621865SRaghuram Chary J 	MII_ACC,
34149621865SRaghuram Chary J 	MII_DATA,
34249621865SRaghuram Chary J 	EEE_TX_LPI_REQ_DLY,
34349621865SRaghuram Chary J 	EEE_TW_TX_SYS,
34449621865SRaghuram Chary J 	EEE_TX_LPI_REM_DLY,
34549621865SRaghuram Chary J 	WUCSR
34649621865SRaghuram Chary J };
34749621865SRaghuram Chary J 
34849621865SRaghuram Chary J #define PHY_REG_SIZE (32 * sizeof(u32))
34949621865SRaghuram Chary J 
35055d7de9dSWoojung.Huh@microchip.com struct lan78xx_net;
35155d7de9dSWoojung.Huh@microchip.com 
35255d7de9dSWoojung.Huh@microchip.com struct lan78xx_priv {
35355d7de9dSWoojung.Huh@microchip.com 	struct lan78xx_net *dev;
35455d7de9dSWoojung.Huh@microchip.com 	u32 rfe_ctl;
355f62c4f38SZheng Yongjun 	u32 mchash_table[DP_SEL_VHF_HASH_LEN]; /* multicast hash table */
35655d7de9dSWoojung.Huh@microchip.com 	u32 pfilter_table[NUM_OF_MAF][2]; /* perfect filter table */
35755d7de9dSWoojung.Huh@microchip.com 	u32 vlan_table[DP_SEL_VHF_VLAN_LEN];
35855d7de9dSWoojung.Huh@microchip.com 	struct mutex dataport_mutex; /* for dataport access */
35955d7de9dSWoojung.Huh@microchip.com 	spinlock_t rfe_ctl_lock; /* for rfe register access */
36055d7de9dSWoojung.Huh@microchip.com 	struct work_struct set_multicast;
36155d7de9dSWoojung.Huh@microchip.com 	struct work_struct set_vlan;
36255d7de9dSWoojung.Huh@microchip.com 	u32 wol;
36355d7de9dSWoojung.Huh@microchip.com };
36455d7de9dSWoojung.Huh@microchip.com 
36555d7de9dSWoojung.Huh@microchip.com enum skb_state {
36655d7de9dSWoojung.Huh@microchip.com 	illegal = 0,
36755d7de9dSWoojung.Huh@microchip.com 	tx_start,
36855d7de9dSWoojung.Huh@microchip.com 	tx_done,
36955d7de9dSWoojung.Huh@microchip.com 	rx_start,
37055d7de9dSWoojung.Huh@microchip.com 	rx_done,
37155d7de9dSWoojung.Huh@microchip.com 	rx_cleanup,
37255d7de9dSWoojung.Huh@microchip.com 	unlink_start
37355d7de9dSWoojung.Huh@microchip.com };
37455d7de9dSWoojung.Huh@microchip.com 
37555d7de9dSWoojung.Huh@microchip.com struct skb_data {		/* skb->cb is one of these */
37655d7de9dSWoojung.Huh@microchip.com 	struct urb *urb;
37755d7de9dSWoojung.Huh@microchip.com 	struct lan78xx_net *dev;
37855d7de9dSWoojung.Huh@microchip.com 	enum skb_state state;
37955d7de9dSWoojung.Huh@microchip.com 	size_t length;
38074d79a2eSWoojung Huh 	int num_of_packet;
38155d7de9dSWoojung.Huh@microchip.com };
38255d7de9dSWoojung.Huh@microchip.com 
38355d7de9dSWoojung.Huh@microchip.com struct usb_context {
38455d7de9dSWoojung.Huh@microchip.com 	struct usb_ctrlrequest req;
38555d7de9dSWoojung.Huh@microchip.com 	struct lan78xx_net *dev;
38655d7de9dSWoojung.Huh@microchip.com };
38755d7de9dSWoojung.Huh@microchip.com 
38855d7de9dSWoojung.Huh@microchip.com #define EVENT_TX_HALT			0
38955d7de9dSWoojung.Huh@microchip.com #define EVENT_RX_HALT			1
39055d7de9dSWoojung.Huh@microchip.com #define EVENT_RX_MEMORY			2
39155d7de9dSWoojung.Huh@microchip.com #define EVENT_STS_SPLIT			3
39255d7de9dSWoojung.Huh@microchip.com #define EVENT_LINK_RESET		4
39355d7de9dSWoojung.Huh@microchip.com #define EVENT_RX_PAUSED			5
39455d7de9dSWoojung.Huh@microchip.com #define EVENT_DEV_WAKING		6
39555d7de9dSWoojung.Huh@microchip.com #define EVENT_DEV_ASLEEP		7
39655d7de9dSWoojung.Huh@microchip.com #define EVENT_DEV_OPEN			8
39720ff5565SWoojung Huh #define EVENT_STAT_UPDATE		9
39877dfff5bSJohn Efstathiades #define EVENT_DEV_DISCONNECT		10
39920ff5565SWoojung Huh 
40020ff5565SWoojung Huh struct statstage {
40120ff5565SWoojung Huh 	struct mutex			access_lock;	/* for stats access */
40220ff5565SWoojung Huh 	struct lan78xx_statstage	saved;
40320ff5565SWoojung Huh 	struct lan78xx_statstage	rollover_count;
40420ff5565SWoojung Huh 	struct lan78xx_statstage	rollover_max;
40520ff5565SWoojung Huh 	struct lan78xx_statstage64	curr_stat;
40620ff5565SWoojung Huh };
40755d7de9dSWoojung.Huh@microchip.com 
408cc89c323SWoojung Huh struct irq_domain_data {
409cc89c323SWoojung Huh 	struct irq_domain	*irqdomain;
410cc89c323SWoojung Huh 	unsigned int		phyirq;
411cc89c323SWoojung Huh 	struct irq_chip		*irqchip;
412cc89c323SWoojung Huh 	irq_flow_handler_t	irq_handler;
413cc89c323SWoojung Huh 	u32			irqenable;
414cc89c323SWoojung Huh 	struct mutex		irq_lock;		/* for irq bus access */
415cc89c323SWoojung Huh };
416cc89c323SWoojung Huh 
41755d7de9dSWoojung.Huh@microchip.com struct lan78xx_net {
41855d7de9dSWoojung.Huh@microchip.com 	struct net_device	*net;
41955d7de9dSWoojung.Huh@microchip.com 	struct usb_device	*udev;
42055d7de9dSWoojung.Huh@microchip.com 	struct usb_interface	*intf;
42155d7de9dSWoojung.Huh@microchip.com 	void			*driver_priv;
42255d7de9dSWoojung.Huh@microchip.com 
423d383216aSJohn Efstathiades 	unsigned int		tx_pend_data_len;
424d383216aSJohn Efstathiades 	size_t			n_tx_urbs;
425c450a8ebSJohn Efstathiades 	size_t			n_rx_urbs;
426d383216aSJohn Efstathiades 	size_t			tx_urb_size;
427c450a8ebSJohn Efstathiades 	size_t			rx_urb_size;
428d383216aSJohn Efstathiades 
429c450a8ebSJohn Efstathiades 	struct sk_buff_head	rxq_free;
43055d7de9dSWoojung.Huh@microchip.com 	struct sk_buff_head	rxq;
431c450a8ebSJohn Efstathiades 	struct sk_buff_head	rxq_done;
432ec4c7e12SJohn Efstathiades 	struct sk_buff_head	rxq_overflow;
433d383216aSJohn Efstathiades 	struct sk_buff_head	txq_free;
434d383216aSJohn Efstathiades 	struct sk_buff_head	txq;
43555d7de9dSWoojung.Huh@microchip.com 	struct sk_buff_head	txq_pend;
43655d7de9dSWoojung.Huh@microchip.com 
437ec4c7e12SJohn Efstathiades 	struct napi_struct	napi;
438ec4c7e12SJohn Efstathiades 
43955d7de9dSWoojung.Huh@microchip.com 	struct delayed_work	wq;
44055d7de9dSWoojung.Huh@microchip.com 
44155d7de9dSWoojung.Huh@microchip.com 	int			msg_enable;
44255d7de9dSWoojung.Huh@microchip.com 
44355d7de9dSWoojung.Huh@microchip.com 	struct urb		*urb_intr;
44455d7de9dSWoojung.Huh@microchip.com 	struct usb_anchor	deferred;
44555d7de9dSWoojung.Huh@microchip.com 
4465f4cc6e2SJohn Efstathiades 	struct mutex		dev_mutex; /* serialise open/stop wrt suspend/resume */
44755d7de9dSWoojung.Huh@microchip.com 	struct mutex		phy_mutex; /* for phy access */
4489ceec7d3SJohn Efstathiades 	unsigned int		pipe_in, pipe_out, pipe_intr;
44955d7de9dSWoojung.Huh@microchip.com 
450c450a8ebSJohn Efstathiades 	unsigned int		bulk_in_delay;
451c450a8ebSJohn Efstathiades 	unsigned int		burst_cap;
45255d7de9dSWoojung.Huh@microchip.com 
45355d7de9dSWoojung.Huh@microchip.com 	unsigned long		flags;
45455d7de9dSWoojung.Huh@microchip.com 
45555d7de9dSWoojung.Huh@microchip.com 	wait_queue_head_t	*wait;
45655d7de9dSWoojung.Huh@microchip.com 	unsigned char		suspend_count;
45755d7de9dSWoojung.Huh@microchip.com 
4589ceec7d3SJohn Efstathiades 	unsigned int		maxpacket;
45920ff5565SWoojung Huh 	struct timer_list	stat_monitor;
46055d7de9dSWoojung.Huh@microchip.com 
46155d7de9dSWoojung.Huh@microchip.com 	unsigned long		data[5];
46255d7de9dSWoojung.Huh@microchip.com 
46355d7de9dSWoojung.Huh@microchip.com 	int			link_on;
46455d7de9dSWoojung.Huh@microchip.com 	u8			mdix_ctrl;
465ce85e13aSWoojung.Huh@microchip.com 
46687177ba6SWoojung.Huh@microchip.com 	u32			chipid;
46787177ba6SWoojung.Huh@microchip.com 	u32			chiprev;
468ce85e13aSWoojung.Huh@microchip.com 	struct mii_bus		*mdiobus;
46902dc1f3dSWoojung Huh 	phy_interface_t		interface;
470349e0c5eSWoojung.Huh@microchip.com 
471349e0c5eSWoojung.Huh@microchip.com 	int			fc_autoneg;
472349e0c5eSWoojung.Huh@microchip.com 	u8			fc_request_control;
47320ff5565SWoojung Huh 
47420ff5565SWoojung Huh 	int			delta;
47520ff5565SWoojung Huh 	struct statstage	stats;
476cc89c323SWoojung Huh 
477cc89c323SWoojung Huh 	struct irq_domain_data	domain_data;
47855d7de9dSWoojung.Huh@microchip.com };
47955d7de9dSWoojung.Huh@microchip.com 
48002dc1f3dSWoojung Huh /* define external phy id */
48102dc1f3dSWoojung Huh #define	PHY_LAN8835			(0x0007C130)
48202dc1f3dSWoojung Huh #define	PHY_KSZ9031RNX			(0x00221620)
48302dc1f3dSWoojung Huh 
48455d7de9dSWoojung.Huh@microchip.com /* use ethtool to change the level for any given device */
48555d7de9dSWoojung.Huh@microchip.com static int msg_level = -1;
48655d7de9dSWoojung.Huh@microchip.com module_param(msg_level, int, 0);
48755d7de9dSWoojung.Huh@microchip.com MODULE_PARM_DESC(msg_level, "Override default message level");
48855d7de9dSWoojung.Huh@microchip.com 
lan78xx_get_buf(struct sk_buff_head * buf_pool)489d383216aSJohn Efstathiades static struct sk_buff *lan78xx_get_buf(struct sk_buff_head *buf_pool)
490d383216aSJohn Efstathiades {
491d383216aSJohn Efstathiades 	if (skb_queue_empty(buf_pool))
492d383216aSJohn Efstathiades 		return NULL;
493d383216aSJohn Efstathiades 
494d383216aSJohn Efstathiades 	return skb_dequeue(buf_pool);
495d383216aSJohn Efstathiades }
496d383216aSJohn Efstathiades 
lan78xx_release_buf(struct sk_buff_head * buf_pool,struct sk_buff * buf)497d383216aSJohn Efstathiades static void lan78xx_release_buf(struct sk_buff_head *buf_pool,
498d383216aSJohn Efstathiades 				struct sk_buff *buf)
499d383216aSJohn Efstathiades {
500d383216aSJohn Efstathiades 	buf->data = buf->head;
501d383216aSJohn Efstathiades 	skb_reset_tail_pointer(buf);
502d383216aSJohn Efstathiades 
503d383216aSJohn Efstathiades 	buf->len = 0;
504d383216aSJohn Efstathiades 	buf->data_len = 0;
505d383216aSJohn Efstathiades 
506d383216aSJohn Efstathiades 	skb_queue_tail(buf_pool, buf);
507d383216aSJohn Efstathiades }
508d383216aSJohn Efstathiades 
lan78xx_free_buf_pool(struct sk_buff_head * buf_pool)509d383216aSJohn Efstathiades static void lan78xx_free_buf_pool(struct sk_buff_head *buf_pool)
510d383216aSJohn Efstathiades {
511d383216aSJohn Efstathiades 	struct skb_data *entry;
512d383216aSJohn Efstathiades 	struct sk_buff *buf;
513d383216aSJohn Efstathiades 
514d383216aSJohn Efstathiades 	while (!skb_queue_empty(buf_pool)) {
515d383216aSJohn Efstathiades 		buf = skb_dequeue(buf_pool);
516d383216aSJohn Efstathiades 		if (buf) {
517d383216aSJohn Efstathiades 			entry = (struct skb_data *)buf->cb;
518d383216aSJohn Efstathiades 			usb_free_urb(entry->urb);
519d383216aSJohn Efstathiades 			dev_kfree_skb_any(buf);
520d383216aSJohn Efstathiades 		}
521d383216aSJohn Efstathiades 	}
522d383216aSJohn Efstathiades }
523d383216aSJohn Efstathiades 
lan78xx_alloc_buf_pool(struct sk_buff_head * buf_pool,size_t n_urbs,size_t urb_size,struct lan78xx_net * dev)524d383216aSJohn Efstathiades static int lan78xx_alloc_buf_pool(struct sk_buff_head *buf_pool,
525d383216aSJohn Efstathiades 				  size_t n_urbs, size_t urb_size,
526d383216aSJohn Efstathiades 				  struct lan78xx_net *dev)
527d383216aSJohn Efstathiades {
528d383216aSJohn Efstathiades 	struct skb_data *entry;
529d383216aSJohn Efstathiades 	struct sk_buff *buf;
530d383216aSJohn Efstathiades 	struct urb *urb;
531d383216aSJohn Efstathiades 	int i;
532d383216aSJohn Efstathiades 
533d383216aSJohn Efstathiades 	skb_queue_head_init(buf_pool);
534d383216aSJohn Efstathiades 
535d383216aSJohn Efstathiades 	for (i = 0; i < n_urbs; i++) {
536d383216aSJohn Efstathiades 		buf = alloc_skb(urb_size, GFP_ATOMIC);
537d383216aSJohn Efstathiades 		if (!buf)
538d383216aSJohn Efstathiades 			goto error;
539d383216aSJohn Efstathiades 
540d383216aSJohn Efstathiades 		if (skb_linearize(buf) != 0) {
541d383216aSJohn Efstathiades 			dev_kfree_skb_any(buf);
542d383216aSJohn Efstathiades 			goto error;
543d383216aSJohn Efstathiades 		}
544d383216aSJohn Efstathiades 
545d383216aSJohn Efstathiades 		urb = usb_alloc_urb(0, GFP_ATOMIC);
546d383216aSJohn Efstathiades 		if (!urb) {
547d383216aSJohn Efstathiades 			dev_kfree_skb_any(buf);
548d383216aSJohn Efstathiades 			goto error;
549d383216aSJohn Efstathiades 		}
550d383216aSJohn Efstathiades 
551d383216aSJohn Efstathiades 		entry = (struct skb_data *)buf->cb;
552d383216aSJohn Efstathiades 		entry->urb = urb;
553d383216aSJohn Efstathiades 		entry->dev = dev;
554d383216aSJohn Efstathiades 		entry->length = 0;
555d383216aSJohn Efstathiades 		entry->num_of_packet = 0;
556d383216aSJohn Efstathiades 
557d383216aSJohn Efstathiades 		skb_queue_tail(buf_pool, buf);
558d383216aSJohn Efstathiades 	}
559d383216aSJohn Efstathiades 
560d383216aSJohn Efstathiades 	return 0;
561d383216aSJohn Efstathiades 
562d383216aSJohn Efstathiades error:
563d383216aSJohn Efstathiades 	lan78xx_free_buf_pool(buf_pool);
564d383216aSJohn Efstathiades 
565d383216aSJohn Efstathiades 	return -ENOMEM;
566d383216aSJohn Efstathiades }
567d383216aSJohn Efstathiades 
lan78xx_get_rx_buf(struct lan78xx_net * dev)568c450a8ebSJohn Efstathiades static struct sk_buff *lan78xx_get_rx_buf(struct lan78xx_net *dev)
569c450a8ebSJohn Efstathiades {
570c450a8ebSJohn Efstathiades 	return lan78xx_get_buf(&dev->rxq_free);
571c450a8ebSJohn Efstathiades }
572c450a8ebSJohn Efstathiades 
lan78xx_release_rx_buf(struct lan78xx_net * dev,struct sk_buff * rx_buf)573c450a8ebSJohn Efstathiades static void lan78xx_release_rx_buf(struct lan78xx_net *dev,
574c450a8ebSJohn Efstathiades 				   struct sk_buff *rx_buf)
575c450a8ebSJohn Efstathiades {
576c450a8ebSJohn Efstathiades 	lan78xx_release_buf(&dev->rxq_free, rx_buf);
577c450a8ebSJohn Efstathiades }
578c450a8ebSJohn Efstathiades 
lan78xx_free_rx_resources(struct lan78xx_net * dev)579c450a8ebSJohn Efstathiades static void lan78xx_free_rx_resources(struct lan78xx_net *dev)
580c450a8ebSJohn Efstathiades {
581c450a8ebSJohn Efstathiades 	lan78xx_free_buf_pool(&dev->rxq_free);
582c450a8ebSJohn Efstathiades }
583c450a8ebSJohn Efstathiades 
lan78xx_alloc_rx_resources(struct lan78xx_net * dev)584c450a8ebSJohn Efstathiades static int lan78xx_alloc_rx_resources(struct lan78xx_net *dev)
585c450a8ebSJohn Efstathiades {
586c450a8ebSJohn Efstathiades 	return lan78xx_alloc_buf_pool(&dev->rxq_free,
587c450a8ebSJohn Efstathiades 				      dev->n_rx_urbs, dev->rx_urb_size, dev);
588c450a8ebSJohn Efstathiades }
589c450a8ebSJohn Efstathiades 
lan78xx_get_tx_buf(struct lan78xx_net * dev)590d383216aSJohn Efstathiades static struct sk_buff *lan78xx_get_tx_buf(struct lan78xx_net *dev)
591d383216aSJohn Efstathiades {
592d383216aSJohn Efstathiades 	return lan78xx_get_buf(&dev->txq_free);
593d383216aSJohn Efstathiades }
594d383216aSJohn Efstathiades 
lan78xx_release_tx_buf(struct lan78xx_net * dev,struct sk_buff * tx_buf)595d383216aSJohn Efstathiades static void lan78xx_release_tx_buf(struct lan78xx_net *dev,
596d383216aSJohn Efstathiades 				   struct sk_buff *tx_buf)
597d383216aSJohn Efstathiades {
598d383216aSJohn Efstathiades 	lan78xx_release_buf(&dev->txq_free, tx_buf);
599d383216aSJohn Efstathiades }
600d383216aSJohn Efstathiades 
lan78xx_free_tx_resources(struct lan78xx_net * dev)601d383216aSJohn Efstathiades static void lan78xx_free_tx_resources(struct lan78xx_net *dev)
602d383216aSJohn Efstathiades {
603d383216aSJohn Efstathiades 	lan78xx_free_buf_pool(&dev->txq_free);
604d383216aSJohn Efstathiades }
605d383216aSJohn Efstathiades 
lan78xx_alloc_tx_resources(struct lan78xx_net * dev)606d383216aSJohn Efstathiades static int lan78xx_alloc_tx_resources(struct lan78xx_net *dev)
607d383216aSJohn Efstathiades {
608d383216aSJohn Efstathiades 	return lan78xx_alloc_buf_pool(&dev->txq_free,
609d383216aSJohn Efstathiades 				      dev->n_tx_urbs, dev->tx_urb_size, dev);
610d383216aSJohn Efstathiades }
611d383216aSJohn Efstathiades 
lan78xx_read_reg(struct lan78xx_net * dev,u32 index,u32 * data)61255d7de9dSWoojung.Huh@microchip.com static int lan78xx_read_reg(struct lan78xx_net *dev, u32 index, u32 *data)
61355d7de9dSWoojung.Huh@microchip.com {
61477dfff5bSJohn Efstathiades 	u32 *buf;
61555d7de9dSWoojung.Huh@microchip.com 	int ret;
61655d7de9dSWoojung.Huh@microchip.com 
61777dfff5bSJohn Efstathiades 	if (test_bit(EVENT_DEV_DISCONNECT, &dev->flags))
61877dfff5bSJohn Efstathiades 		return -ENODEV;
61977dfff5bSJohn Efstathiades 
62077dfff5bSJohn Efstathiades 	buf = kmalloc(sizeof(u32), GFP_KERNEL);
62155d7de9dSWoojung.Huh@microchip.com 	if (!buf)
62255d7de9dSWoojung.Huh@microchip.com 		return -ENOMEM;
62355d7de9dSWoojung.Huh@microchip.com 
62455d7de9dSWoojung.Huh@microchip.com 	ret = usb_control_msg(dev->udev, usb_rcvctrlpipe(dev->udev, 0),
62555d7de9dSWoojung.Huh@microchip.com 			      USB_VENDOR_REQUEST_READ_REGISTER,
62655d7de9dSWoojung.Huh@microchip.com 			      USB_DIR_IN | USB_TYPE_VENDOR | USB_RECIP_DEVICE,
62755d7de9dSWoojung.Huh@microchip.com 			      0, index, buf, 4, USB_CTRL_GET_TIMEOUT);
62855d7de9dSWoojung.Huh@microchip.com 	if (likely(ret >= 0)) {
62955d7de9dSWoojung.Huh@microchip.com 		le32_to_cpus(buf);
63055d7de9dSWoojung.Huh@microchip.com 		*data = *buf;
631df0d6f7aSJohn Efstathiades 	} else if (net_ratelimit()) {
63255d7de9dSWoojung.Huh@microchip.com 		netdev_warn(dev->net,
63355d7de9dSWoojung.Huh@microchip.com 			    "Failed to read register index 0x%08x. ret = %d",
63455d7de9dSWoojung.Huh@microchip.com 			    index, ret);
63555d7de9dSWoojung.Huh@microchip.com 	}
63655d7de9dSWoojung.Huh@microchip.com 
63755d7de9dSWoojung.Huh@microchip.com 	kfree(buf);
63855d7de9dSWoojung.Huh@microchip.com 
63955d7de9dSWoojung.Huh@microchip.com 	return ret;
64055d7de9dSWoojung.Huh@microchip.com }
64155d7de9dSWoojung.Huh@microchip.com 
lan78xx_write_reg(struct lan78xx_net * dev,u32 index,u32 data)64255d7de9dSWoojung.Huh@microchip.com static int lan78xx_write_reg(struct lan78xx_net *dev, u32 index, u32 data)
64355d7de9dSWoojung.Huh@microchip.com {
64477dfff5bSJohn Efstathiades 	u32 *buf;
64555d7de9dSWoojung.Huh@microchip.com 	int ret;
64655d7de9dSWoojung.Huh@microchip.com 
64777dfff5bSJohn Efstathiades 	if (test_bit(EVENT_DEV_DISCONNECT, &dev->flags))
64877dfff5bSJohn Efstathiades 		return -ENODEV;
64977dfff5bSJohn Efstathiades 
65077dfff5bSJohn Efstathiades 	buf = kmalloc(sizeof(u32), GFP_KERNEL);
65155d7de9dSWoojung.Huh@microchip.com 	if (!buf)
65255d7de9dSWoojung.Huh@microchip.com 		return -ENOMEM;
65355d7de9dSWoojung.Huh@microchip.com 
65455d7de9dSWoojung.Huh@microchip.com 	*buf = data;
65555d7de9dSWoojung.Huh@microchip.com 	cpu_to_le32s(buf);
65655d7de9dSWoojung.Huh@microchip.com 
65755d7de9dSWoojung.Huh@microchip.com 	ret = usb_control_msg(dev->udev, usb_sndctrlpipe(dev->udev, 0),
65855d7de9dSWoojung.Huh@microchip.com 			      USB_VENDOR_REQUEST_WRITE_REGISTER,
65955d7de9dSWoojung.Huh@microchip.com 			      USB_DIR_OUT | USB_TYPE_VENDOR | USB_RECIP_DEVICE,
66055d7de9dSWoojung.Huh@microchip.com 			      0, index, buf, 4, USB_CTRL_SET_TIMEOUT);
661df0d6f7aSJohn Efstathiades 	if (unlikely(ret < 0) &&
662df0d6f7aSJohn Efstathiades 	    net_ratelimit()) {
66355d7de9dSWoojung.Huh@microchip.com 		netdev_warn(dev->net,
66455d7de9dSWoojung.Huh@microchip.com 			    "Failed to write register index 0x%08x. ret = %d",
66555d7de9dSWoojung.Huh@microchip.com 			    index, ret);
66655d7de9dSWoojung.Huh@microchip.com 	}
66755d7de9dSWoojung.Huh@microchip.com 
66855d7de9dSWoojung.Huh@microchip.com 	kfree(buf);
66955d7de9dSWoojung.Huh@microchip.com 
67055d7de9dSWoojung.Huh@microchip.com 	return ret;
67155d7de9dSWoojung.Huh@microchip.com }
67255d7de9dSWoojung.Huh@microchip.com 
lan78xx_update_reg(struct lan78xx_net * dev,u32 reg,u32 mask,u32 data)673e1210fe6SJohn Efstathiades static int lan78xx_update_reg(struct lan78xx_net *dev, u32 reg, u32 mask,
674e1210fe6SJohn Efstathiades 			      u32 data)
675e1210fe6SJohn Efstathiades {
676e1210fe6SJohn Efstathiades 	int ret;
677e1210fe6SJohn Efstathiades 	u32 buf;
678e1210fe6SJohn Efstathiades 
679e1210fe6SJohn Efstathiades 	ret = lan78xx_read_reg(dev, reg, &buf);
680e1210fe6SJohn Efstathiades 	if (ret < 0)
681e1210fe6SJohn Efstathiades 		return ret;
682e1210fe6SJohn Efstathiades 
683e1210fe6SJohn Efstathiades 	buf &= ~mask;
684e1210fe6SJohn Efstathiades 	buf |= (mask & data);
685e1210fe6SJohn Efstathiades 
686e1210fe6SJohn Efstathiades 	ret = lan78xx_write_reg(dev, reg, buf);
687e1210fe6SJohn Efstathiades 	if (ret < 0)
688e1210fe6SJohn Efstathiades 		return ret;
689e1210fe6SJohn Efstathiades 
690e1210fe6SJohn Efstathiades 	return 0;
691e1210fe6SJohn Efstathiades }
692e1210fe6SJohn Efstathiades 
lan78xx_read_stats(struct lan78xx_net * dev,struct lan78xx_statstage * data)69355d7de9dSWoojung.Huh@microchip.com static int lan78xx_read_stats(struct lan78xx_net *dev,
69455d7de9dSWoojung.Huh@microchip.com 			      struct lan78xx_statstage *data)
69555d7de9dSWoojung.Huh@microchip.com {
69655d7de9dSWoojung.Huh@microchip.com 	int ret = 0;
69755d7de9dSWoojung.Huh@microchip.com 	int i;
69855d7de9dSWoojung.Huh@microchip.com 	struct lan78xx_statstage *stats;
69955d7de9dSWoojung.Huh@microchip.com 	u32 *src;
70055d7de9dSWoojung.Huh@microchip.com 	u32 *dst;
70155d7de9dSWoojung.Huh@microchip.com 
70255d7de9dSWoojung.Huh@microchip.com 	stats = kmalloc(sizeof(*stats), GFP_KERNEL);
70355d7de9dSWoojung.Huh@microchip.com 	if (!stats)
70455d7de9dSWoojung.Huh@microchip.com 		return -ENOMEM;
70555d7de9dSWoojung.Huh@microchip.com 
70655d7de9dSWoojung.Huh@microchip.com 	ret = usb_control_msg(dev->udev,
70755d7de9dSWoojung.Huh@microchip.com 			      usb_rcvctrlpipe(dev->udev, 0),
70855d7de9dSWoojung.Huh@microchip.com 			      USB_VENDOR_REQUEST_GET_STATS,
70955d7de9dSWoojung.Huh@microchip.com 			      USB_DIR_IN | USB_TYPE_VENDOR | USB_RECIP_DEVICE,
71055d7de9dSWoojung.Huh@microchip.com 			      0,
71155d7de9dSWoojung.Huh@microchip.com 			      0,
71255d7de9dSWoojung.Huh@microchip.com 			      (void *)stats,
71355d7de9dSWoojung.Huh@microchip.com 			      sizeof(*stats),
71455d7de9dSWoojung.Huh@microchip.com 			      USB_CTRL_SET_TIMEOUT);
71555d7de9dSWoojung.Huh@microchip.com 	if (likely(ret >= 0)) {
71655d7de9dSWoojung.Huh@microchip.com 		src = (u32 *)stats;
71755d7de9dSWoojung.Huh@microchip.com 		dst = (u32 *)data;
71855d7de9dSWoojung.Huh@microchip.com 		for (i = 0; i < sizeof(*stats) / sizeof(u32); i++) {
71955d7de9dSWoojung.Huh@microchip.com 			le32_to_cpus(&src[i]);
72055d7de9dSWoojung.Huh@microchip.com 			dst[i] = src[i];
72155d7de9dSWoojung.Huh@microchip.com 		}
72255d7de9dSWoojung.Huh@microchip.com 	} else {
72355d7de9dSWoojung.Huh@microchip.com 		netdev_warn(dev->net,
724858ce8caSCristian Birsan 			    "Failed to read stat ret = %d", ret);
72555d7de9dSWoojung.Huh@microchip.com 	}
72655d7de9dSWoojung.Huh@microchip.com 
72755d7de9dSWoojung.Huh@microchip.com 	kfree(stats);
72855d7de9dSWoojung.Huh@microchip.com 
72955d7de9dSWoojung.Huh@microchip.com 	return ret;
73055d7de9dSWoojung.Huh@microchip.com }
73155d7de9dSWoojung.Huh@microchip.com 
7329ceec7d3SJohn Efstathiades #define check_counter_rollover(struct1, dev_stats, member)		\
7339ceec7d3SJohn Efstathiades 	do {								\
7349ceec7d3SJohn Efstathiades 		if ((struct1)->member < (dev_stats).saved.member)	\
7359ceec7d3SJohn Efstathiades 			(dev_stats).rollover_count.member++;		\
7369ceec7d3SJohn Efstathiades 	} while (0)
73720ff5565SWoojung Huh 
lan78xx_check_stat_rollover(struct lan78xx_net * dev,struct lan78xx_statstage * stats)73820ff5565SWoojung Huh static void lan78xx_check_stat_rollover(struct lan78xx_net *dev,
73920ff5565SWoojung Huh 					struct lan78xx_statstage *stats)
74020ff5565SWoojung Huh {
74120ff5565SWoojung Huh 	check_counter_rollover(stats, dev->stats, rx_fcs_errors);
74220ff5565SWoojung Huh 	check_counter_rollover(stats, dev->stats, rx_alignment_errors);
74320ff5565SWoojung Huh 	check_counter_rollover(stats, dev->stats, rx_fragment_errors);
74420ff5565SWoojung Huh 	check_counter_rollover(stats, dev->stats, rx_jabber_errors);
74520ff5565SWoojung Huh 	check_counter_rollover(stats, dev->stats, rx_undersize_frame_errors);
74620ff5565SWoojung Huh 	check_counter_rollover(stats, dev->stats, rx_oversize_frame_errors);
74720ff5565SWoojung Huh 	check_counter_rollover(stats, dev->stats, rx_dropped_frames);
74820ff5565SWoojung Huh 	check_counter_rollover(stats, dev->stats, rx_unicast_byte_count);
74920ff5565SWoojung Huh 	check_counter_rollover(stats, dev->stats, rx_broadcast_byte_count);
75020ff5565SWoojung Huh 	check_counter_rollover(stats, dev->stats, rx_multicast_byte_count);
75120ff5565SWoojung Huh 	check_counter_rollover(stats, dev->stats, rx_unicast_frames);
75220ff5565SWoojung Huh 	check_counter_rollover(stats, dev->stats, rx_broadcast_frames);
75320ff5565SWoojung Huh 	check_counter_rollover(stats, dev->stats, rx_multicast_frames);
75420ff5565SWoojung Huh 	check_counter_rollover(stats, dev->stats, rx_pause_frames);
75520ff5565SWoojung Huh 	check_counter_rollover(stats, dev->stats, rx_64_byte_frames);
75620ff5565SWoojung Huh 	check_counter_rollover(stats, dev->stats, rx_65_127_byte_frames);
75720ff5565SWoojung Huh 	check_counter_rollover(stats, dev->stats, rx_128_255_byte_frames);
75820ff5565SWoojung Huh 	check_counter_rollover(stats, dev->stats, rx_256_511_bytes_frames);
75920ff5565SWoojung Huh 	check_counter_rollover(stats, dev->stats, rx_512_1023_byte_frames);
76020ff5565SWoojung Huh 	check_counter_rollover(stats, dev->stats, rx_1024_1518_byte_frames);
76120ff5565SWoojung Huh 	check_counter_rollover(stats, dev->stats, rx_greater_1518_byte_frames);
76220ff5565SWoojung Huh 	check_counter_rollover(stats, dev->stats, eee_rx_lpi_transitions);
76320ff5565SWoojung Huh 	check_counter_rollover(stats, dev->stats, eee_rx_lpi_time);
76420ff5565SWoojung Huh 	check_counter_rollover(stats, dev->stats, tx_fcs_errors);
76520ff5565SWoojung Huh 	check_counter_rollover(stats, dev->stats, tx_excess_deferral_errors);
76620ff5565SWoojung Huh 	check_counter_rollover(stats, dev->stats, tx_carrier_errors);
76720ff5565SWoojung Huh 	check_counter_rollover(stats, dev->stats, tx_bad_byte_count);
76820ff5565SWoojung Huh 	check_counter_rollover(stats, dev->stats, tx_single_collisions);
76920ff5565SWoojung Huh 	check_counter_rollover(stats, dev->stats, tx_multiple_collisions);
77020ff5565SWoojung Huh 	check_counter_rollover(stats, dev->stats, tx_excessive_collision);
77120ff5565SWoojung Huh 	check_counter_rollover(stats, dev->stats, tx_late_collisions);
77220ff5565SWoojung Huh 	check_counter_rollover(stats, dev->stats, tx_unicast_byte_count);
77320ff5565SWoojung Huh 	check_counter_rollover(stats, dev->stats, tx_broadcast_byte_count);
77420ff5565SWoojung Huh 	check_counter_rollover(stats, dev->stats, tx_multicast_byte_count);
77520ff5565SWoojung Huh 	check_counter_rollover(stats, dev->stats, tx_unicast_frames);
77620ff5565SWoojung Huh 	check_counter_rollover(stats, dev->stats, tx_broadcast_frames);
77720ff5565SWoojung Huh 	check_counter_rollover(stats, dev->stats, tx_multicast_frames);
77820ff5565SWoojung Huh 	check_counter_rollover(stats, dev->stats, tx_pause_frames);
77920ff5565SWoojung Huh 	check_counter_rollover(stats, dev->stats, tx_64_byte_frames);
78020ff5565SWoojung Huh 	check_counter_rollover(stats, dev->stats, tx_65_127_byte_frames);
78120ff5565SWoojung Huh 	check_counter_rollover(stats, dev->stats, tx_128_255_byte_frames);
78220ff5565SWoojung Huh 	check_counter_rollover(stats, dev->stats, tx_256_511_bytes_frames);
78320ff5565SWoojung Huh 	check_counter_rollover(stats, dev->stats, tx_512_1023_byte_frames);
78420ff5565SWoojung Huh 	check_counter_rollover(stats, dev->stats, tx_1024_1518_byte_frames);
78520ff5565SWoojung Huh 	check_counter_rollover(stats, dev->stats, tx_greater_1518_byte_frames);
78620ff5565SWoojung Huh 	check_counter_rollover(stats, dev->stats, eee_tx_lpi_transitions);
78720ff5565SWoojung Huh 	check_counter_rollover(stats, dev->stats, eee_tx_lpi_time);
78820ff5565SWoojung Huh 
78920ff5565SWoojung Huh 	memcpy(&dev->stats.saved, stats, sizeof(struct lan78xx_statstage));
79020ff5565SWoojung Huh }
79120ff5565SWoojung Huh 
lan78xx_update_stats(struct lan78xx_net * dev)79220ff5565SWoojung Huh static void lan78xx_update_stats(struct lan78xx_net *dev)
79320ff5565SWoojung Huh {
79420ff5565SWoojung Huh 	u32 *p, *count, *max;
79520ff5565SWoojung Huh 	u64 *data;
79620ff5565SWoojung Huh 	int i;
79720ff5565SWoojung Huh 	struct lan78xx_statstage lan78xx_stats;
79820ff5565SWoojung Huh 
79920ff5565SWoojung Huh 	if (usb_autopm_get_interface(dev->intf) < 0)
80020ff5565SWoojung Huh 		return;
80120ff5565SWoojung Huh 
80220ff5565SWoojung Huh 	p = (u32 *)&lan78xx_stats;
80320ff5565SWoojung Huh 	count = (u32 *)&dev->stats.rollover_count;
80420ff5565SWoojung Huh 	max = (u32 *)&dev->stats.rollover_max;
80520ff5565SWoojung Huh 	data = (u64 *)&dev->stats.curr_stat;
80620ff5565SWoojung Huh 
80720ff5565SWoojung Huh 	mutex_lock(&dev->stats.access_lock);
80820ff5565SWoojung Huh 
80920ff5565SWoojung Huh 	if (lan78xx_read_stats(dev, &lan78xx_stats) > 0)
81020ff5565SWoojung Huh 		lan78xx_check_stat_rollover(dev, &lan78xx_stats);
81120ff5565SWoojung Huh 
81220ff5565SWoojung Huh 	for (i = 0; i < (sizeof(lan78xx_stats) / (sizeof(u32))); i++)
81320ff5565SWoojung Huh 		data[i] = (u64)p[i] + ((u64)count[i] * ((u64)max[i] + 1));
81420ff5565SWoojung Huh 
81520ff5565SWoojung Huh 	mutex_unlock(&dev->stats.access_lock);
81620ff5565SWoojung Huh 
81720ff5565SWoojung Huh 	usb_autopm_put_interface(dev->intf);
81820ff5565SWoojung Huh }
81920ff5565SWoojung Huh 
82055d7de9dSWoojung.Huh@microchip.com /* Loop until the read is completed with timeout called with phy_mutex held */
lan78xx_phy_wait_not_busy(struct lan78xx_net * dev)82155d7de9dSWoojung.Huh@microchip.com static int lan78xx_phy_wait_not_busy(struct lan78xx_net *dev)
82255d7de9dSWoojung.Huh@microchip.com {
82355d7de9dSWoojung.Huh@microchip.com 	unsigned long start_time = jiffies;
82455d7de9dSWoojung.Huh@microchip.com 	u32 val;
82555d7de9dSWoojung.Huh@microchip.com 	int ret;
82655d7de9dSWoojung.Huh@microchip.com 
82755d7de9dSWoojung.Huh@microchip.com 	do {
82855d7de9dSWoojung.Huh@microchip.com 		ret = lan78xx_read_reg(dev, MII_ACC, &val);
82955d7de9dSWoojung.Huh@microchip.com 		if (unlikely(ret < 0))
83055d7de9dSWoojung.Huh@microchip.com 			return -EIO;
83155d7de9dSWoojung.Huh@microchip.com 
83255d7de9dSWoojung.Huh@microchip.com 		if (!(val & MII_ACC_MII_BUSY_))
83355d7de9dSWoojung.Huh@microchip.com 			return 0;
83455d7de9dSWoojung.Huh@microchip.com 	} while (!time_after(jiffies, start_time + HZ));
83555d7de9dSWoojung.Huh@microchip.com 
83655d7de9dSWoojung.Huh@microchip.com 	return -EIO;
83755d7de9dSWoojung.Huh@microchip.com }
83855d7de9dSWoojung.Huh@microchip.com 
mii_access(int id,int index,int read)83955d7de9dSWoojung.Huh@microchip.com static inline u32 mii_access(int id, int index, int read)
84055d7de9dSWoojung.Huh@microchip.com {
84155d7de9dSWoojung.Huh@microchip.com 	u32 ret;
84255d7de9dSWoojung.Huh@microchip.com 
84355d7de9dSWoojung.Huh@microchip.com 	ret = ((u32)id << MII_ACC_PHY_ADDR_SHIFT_) & MII_ACC_PHY_ADDR_MASK_;
84455d7de9dSWoojung.Huh@microchip.com 	ret |= ((u32)index << MII_ACC_MIIRINDA_SHIFT_) & MII_ACC_MIIRINDA_MASK_;
84555d7de9dSWoojung.Huh@microchip.com 	if (read)
84655d7de9dSWoojung.Huh@microchip.com 		ret |= MII_ACC_MII_READ_;
84755d7de9dSWoojung.Huh@microchip.com 	else
84855d7de9dSWoojung.Huh@microchip.com 		ret |= MII_ACC_MII_WRITE_;
84955d7de9dSWoojung.Huh@microchip.com 	ret |= MII_ACC_MII_BUSY_;
85055d7de9dSWoojung.Huh@microchip.com 
85155d7de9dSWoojung.Huh@microchip.com 	return ret;
85255d7de9dSWoojung.Huh@microchip.com }
85355d7de9dSWoojung.Huh@microchip.com 
lan78xx_wait_eeprom(struct lan78xx_net * dev)85455d7de9dSWoojung.Huh@microchip.com static int lan78xx_wait_eeprom(struct lan78xx_net *dev)
85555d7de9dSWoojung.Huh@microchip.com {
85655d7de9dSWoojung.Huh@microchip.com 	unsigned long start_time = jiffies;
85755d7de9dSWoojung.Huh@microchip.com 	u32 val;
85855d7de9dSWoojung.Huh@microchip.com 	int ret;
85955d7de9dSWoojung.Huh@microchip.com 
86055d7de9dSWoojung.Huh@microchip.com 	do {
86155d7de9dSWoojung.Huh@microchip.com 		ret = lan78xx_read_reg(dev, E2P_CMD, &val);
86255d7de9dSWoojung.Huh@microchip.com 		if (unlikely(ret < 0))
86355d7de9dSWoojung.Huh@microchip.com 			return -EIO;
86455d7de9dSWoojung.Huh@microchip.com 
86555d7de9dSWoojung.Huh@microchip.com 		if (!(val & E2P_CMD_EPC_BUSY_) ||
86655d7de9dSWoojung.Huh@microchip.com 		    (val & E2P_CMD_EPC_TIMEOUT_))
86755d7de9dSWoojung.Huh@microchip.com 			break;
86855d7de9dSWoojung.Huh@microchip.com 		usleep_range(40, 100);
86955d7de9dSWoojung.Huh@microchip.com 	} while (!time_after(jiffies, start_time + HZ));
87055d7de9dSWoojung.Huh@microchip.com 
87155d7de9dSWoojung.Huh@microchip.com 	if (val & (E2P_CMD_EPC_TIMEOUT_ | E2P_CMD_EPC_BUSY_)) {
87255d7de9dSWoojung.Huh@microchip.com 		netdev_warn(dev->net, "EEPROM read operation timeout");
87355d7de9dSWoojung.Huh@microchip.com 		return -EIO;
87455d7de9dSWoojung.Huh@microchip.com 	}
87555d7de9dSWoojung.Huh@microchip.com 
87655d7de9dSWoojung.Huh@microchip.com 	return 0;
87755d7de9dSWoojung.Huh@microchip.com }
87855d7de9dSWoojung.Huh@microchip.com 
lan78xx_eeprom_confirm_not_busy(struct lan78xx_net * dev)87955d7de9dSWoojung.Huh@microchip.com static int lan78xx_eeprom_confirm_not_busy(struct lan78xx_net *dev)
88055d7de9dSWoojung.Huh@microchip.com {
88155d7de9dSWoojung.Huh@microchip.com 	unsigned long start_time = jiffies;
88255d7de9dSWoojung.Huh@microchip.com 	u32 val;
88355d7de9dSWoojung.Huh@microchip.com 	int ret;
88455d7de9dSWoojung.Huh@microchip.com 
88555d7de9dSWoojung.Huh@microchip.com 	do {
88655d7de9dSWoojung.Huh@microchip.com 		ret = lan78xx_read_reg(dev, E2P_CMD, &val);
88755d7de9dSWoojung.Huh@microchip.com 		if (unlikely(ret < 0))
88855d7de9dSWoojung.Huh@microchip.com 			return -EIO;
88955d7de9dSWoojung.Huh@microchip.com 
89055d7de9dSWoojung.Huh@microchip.com 		if (!(val & E2P_CMD_EPC_BUSY_))
89155d7de9dSWoojung.Huh@microchip.com 			return 0;
89255d7de9dSWoojung.Huh@microchip.com 
89355d7de9dSWoojung.Huh@microchip.com 		usleep_range(40, 100);
89455d7de9dSWoojung.Huh@microchip.com 	} while (!time_after(jiffies, start_time + HZ));
89555d7de9dSWoojung.Huh@microchip.com 
89655d7de9dSWoojung.Huh@microchip.com 	netdev_warn(dev->net, "EEPROM is busy");
89755d7de9dSWoojung.Huh@microchip.com 	return -EIO;
89855d7de9dSWoojung.Huh@microchip.com }
89955d7de9dSWoojung.Huh@microchip.com 
lan78xx_read_raw_eeprom(struct lan78xx_net * dev,u32 offset,u32 length,u8 * data)90055d7de9dSWoojung.Huh@microchip.com static int lan78xx_read_raw_eeprom(struct lan78xx_net *dev, u32 offset,
90155d7de9dSWoojung.Huh@microchip.com 				   u32 length, u8 *data)
90255d7de9dSWoojung.Huh@microchip.com {
90355d7de9dSWoojung.Huh@microchip.com 	u32 val;
904a0db7d10SWoojung.Huh@microchip.com 	u32 saved;
90555d7de9dSWoojung.Huh@microchip.com 	int i, ret;
906a0db7d10SWoojung.Huh@microchip.com 	int retval;
90755d7de9dSWoojung.Huh@microchip.com 
908a0db7d10SWoojung.Huh@microchip.com 	/* depends on chip, some EEPROM pins are muxed with LED function.
909a0db7d10SWoojung.Huh@microchip.com 	 * disable & restore LED function to access EEPROM.
910a0db7d10SWoojung.Huh@microchip.com 	 */
911a0db7d10SWoojung.Huh@microchip.com 	ret = lan78xx_read_reg(dev, HW_CFG, &val);
912a0db7d10SWoojung.Huh@microchip.com 	saved = val;
91387177ba6SWoojung.Huh@microchip.com 	if (dev->chipid == ID_REV_CHIP_ID_7800_) {
914a0db7d10SWoojung.Huh@microchip.com 		val &= ~(HW_CFG_LED1_EN_ | HW_CFG_LED0_EN_);
915a0db7d10SWoojung.Huh@microchip.com 		ret = lan78xx_write_reg(dev, HW_CFG, val);
916a0db7d10SWoojung.Huh@microchip.com 	}
917a0db7d10SWoojung.Huh@microchip.com 
918a0db7d10SWoojung.Huh@microchip.com 	retval = lan78xx_eeprom_confirm_not_busy(dev);
919a0db7d10SWoojung.Huh@microchip.com 	if (retval)
920a0db7d10SWoojung.Huh@microchip.com 		return retval;
92155d7de9dSWoojung.Huh@microchip.com 
92255d7de9dSWoojung.Huh@microchip.com 	for (i = 0; i < length; i++) {
92355d7de9dSWoojung.Huh@microchip.com 		val = E2P_CMD_EPC_BUSY_ | E2P_CMD_EPC_CMD_READ_;
92455d7de9dSWoojung.Huh@microchip.com 		val |= (offset & E2P_CMD_EPC_ADDR_MASK_);
92555d7de9dSWoojung.Huh@microchip.com 		ret = lan78xx_write_reg(dev, E2P_CMD, val);
926a0db7d10SWoojung.Huh@microchip.com 		if (unlikely(ret < 0)) {
927a0db7d10SWoojung.Huh@microchip.com 			retval = -EIO;
928a0db7d10SWoojung.Huh@microchip.com 			goto exit;
929a0db7d10SWoojung.Huh@microchip.com 		}
93055d7de9dSWoojung.Huh@microchip.com 
931a0db7d10SWoojung.Huh@microchip.com 		retval = lan78xx_wait_eeprom(dev);
932a0db7d10SWoojung.Huh@microchip.com 		if (retval < 0)
933a0db7d10SWoojung.Huh@microchip.com 			goto exit;
93455d7de9dSWoojung.Huh@microchip.com 
93555d7de9dSWoojung.Huh@microchip.com 		ret = lan78xx_read_reg(dev, E2P_DATA, &val);
936a0db7d10SWoojung.Huh@microchip.com 		if (unlikely(ret < 0)) {
937a0db7d10SWoojung.Huh@microchip.com 			retval = -EIO;
938a0db7d10SWoojung.Huh@microchip.com 			goto exit;
939a0db7d10SWoojung.Huh@microchip.com 		}
94055d7de9dSWoojung.Huh@microchip.com 
94155d7de9dSWoojung.Huh@microchip.com 		data[i] = val & 0xFF;
94255d7de9dSWoojung.Huh@microchip.com 		offset++;
94355d7de9dSWoojung.Huh@microchip.com 	}
94455d7de9dSWoojung.Huh@microchip.com 
945a0db7d10SWoojung.Huh@microchip.com 	retval = 0;
946a0db7d10SWoojung.Huh@microchip.com exit:
94787177ba6SWoojung.Huh@microchip.com 	if (dev->chipid == ID_REV_CHIP_ID_7800_)
948a0db7d10SWoojung.Huh@microchip.com 		ret = lan78xx_write_reg(dev, HW_CFG, saved);
949a0db7d10SWoojung.Huh@microchip.com 
950a0db7d10SWoojung.Huh@microchip.com 	return retval;
95155d7de9dSWoojung.Huh@microchip.com }
95255d7de9dSWoojung.Huh@microchip.com 
lan78xx_read_eeprom(struct lan78xx_net * dev,u32 offset,u32 length,u8 * data)95355d7de9dSWoojung.Huh@microchip.com static int lan78xx_read_eeprom(struct lan78xx_net *dev, u32 offset,
95455d7de9dSWoojung.Huh@microchip.com 			       u32 length, u8 *data)
95555d7de9dSWoojung.Huh@microchip.com {
95655d7de9dSWoojung.Huh@microchip.com 	u8 sig;
95755d7de9dSWoojung.Huh@microchip.com 	int ret;
95855d7de9dSWoojung.Huh@microchip.com 
95955d7de9dSWoojung.Huh@microchip.com 	ret = lan78xx_read_raw_eeprom(dev, 0, 1, &sig);
96055d7de9dSWoojung.Huh@microchip.com 	if ((ret == 0) && (sig == EEPROM_INDICATOR))
96155d7de9dSWoojung.Huh@microchip.com 		ret = lan78xx_read_raw_eeprom(dev, offset, length, data);
96255d7de9dSWoojung.Huh@microchip.com 	else
96355d7de9dSWoojung.Huh@microchip.com 		ret = -EINVAL;
96455d7de9dSWoojung.Huh@microchip.com 
96555d7de9dSWoojung.Huh@microchip.com 	return ret;
96655d7de9dSWoojung.Huh@microchip.com }
96755d7de9dSWoojung.Huh@microchip.com 
lan78xx_write_raw_eeprom(struct lan78xx_net * dev,u32 offset,u32 length,u8 * data)96855d7de9dSWoojung.Huh@microchip.com static int lan78xx_write_raw_eeprom(struct lan78xx_net *dev, u32 offset,
96955d7de9dSWoojung.Huh@microchip.com 				    u32 length, u8 *data)
97055d7de9dSWoojung.Huh@microchip.com {
97155d7de9dSWoojung.Huh@microchip.com 	u32 val;
972a0db7d10SWoojung.Huh@microchip.com 	u32 saved;
97355d7de9dSWoojung.Huh@microchip.com 	int i, ret;
974a0db7d10SWoojung.Huh@microchip.com 	int retval;
97555d7de9dSWoojung.Huh@microchip.com 
976a0db7d10SWoojung.Huh@microchip.com 	/* depends on chip, some EEPROM pins are muxed with LED function.
977a0db7d10SWoojung.Huh@microchip.com 	 * disable & restore LED function to access EEPROM.
978a0db7d10SWoojung.Huh@microchip.com 	 */
979a0db7d10SWoojung.Huh@microchip.com 	ret = lan78xx_read_reg(dev, HW_CFG, &val);
980a0db7d10SWoojung.Huh@microchip.com 	saved = val;
98187177ba6SWoojung.Huh@microchip.com 	if (dev->chipid == ID_REV_CHIP_ID_7800_) {
982a0db7d10SWoojung.Huh@microchip.com 		val &= ~(HW_CFG_LED1_EN_ | HW_CFG_LED0_EN_);
983a0db7d10SWoojung.Huh@microchip.com 		ret = lan78xx_write_reg(dev, HW_CFG, val);
984a0db7d10SWoojung.Huh@microchip.com 	}
985a0db7d10SWoojung.Huh@microchip.com 
986a0db7d10SWoojung.Huh@microchip.com 	retval = lan78xx_eeprom_confirm_not_busy(dev);
987a0db7d10SWoojung.Huh@microchip.com 	if (retval)
988a0db7d10SWoojung.Huh@microchip.com 		goto exit;
98955d7de9dSWoojung.Huh@microchip.com 
99055d7de9dSWoojung.Huh@microchip.com 	/* Issue write/erase enable command */
99155d7de9dSWoojung.Huh@microchip.com 	val = E2P_CMD_EPC_BUSY_ | E2P_CMD_EPC_CMD_EWEN_;
99255d7de9dSWoojung.Huh@microchip.com 	ret = lan78xx_write_reg(dev, E2P_CMD, val);
993a0db7d10SWoojung.Huh@microchip.com 	if (unlikely(ret < 0)) {
994a0db7d10SWoojung.Huh@microchip.com 		retval = -EIO;
995a0db7d10SWoojung.Huh@microchip.com 		goto exit;
996a0db7d10SWoojung.Huh@microchip.com 	}
99755d7de9dSWoojung.Huh@microchip.com 
998a0db7d10SWoojung.Huh@microchip.com 	retval = lan78xx_wait_eeprom(dev);
999a0db7d10SWoojung.Huh@microchip.com 	if (retval < 0)
1000a0db7d10SWoojung.Huh@microchip.com 		goto exit;
100155d7de9dSWoojung.Huh@microchip.com 
100255d7de9dSWoojung.Huh@microchip.com 	for (i = 0; i < length; i++) {
100355d7de9dSWoojung.Huh@microchip.com 		/* Fill data register */
100455d7de9dSWoojung.Huh@microchip.com 		val = data[i];
100555d7de9dSWoojung.Huh@microchip.com 		ret = lan78xx_write_reg(dev, E2P_DATA, val);
1006a0db7d10SWoojung.Huh@microchip.com 		if (ret < 0) {
1007a0db7d10SWoojung.Huh@microchip.com 			retval = -EIO;
1008a0db7d10SWoojung.Huh@microchip.com 			goto exit;
1009a0db7d10SWoojung.Huh@microchip.com 		}
101055d7de9dSWoojung.Huh@microchip.com 
101155d7de9dSWoojung.Huh@microchip.com 		/* Send "write" command */
101255d7de9dSWoojung.Huh@microchip.com 		val = E2P_CMD_EPC_BUSY_ | E2P_CMD_EPC_CMD_WRITE_;
101355d7de9dSWoojung.Huh@microchip.com 		val |= (offset & E2P_CMD_EPC_ADDR_MASK_);
101455d7de9dSWoojung.Huh@microchip.com 		ret = lan78xx_write_reg(dev, E2P_CMD, val);
1015a0db7d10SWoojung.Huh@microchip.com 		if (ret < 0) {
1016a0db7d10SWoojung.Huh@microchip.com 			retval = -EIO;
1017a0db7d10SWoojung.Huh@microchip.com 			goto exit;
1018a0db7d10SWoojung.Huh@microchip.com 		}
101955d7de9dSWoojung.Huh@microchip.com 
1020a0db7d10SWoojung.Huh@microchip.com 		retval = lan78xx_wait_eeprom(dev);
1021a0db7d10SWoojung.Huh@microchip.com 		if (retval < 0)
1022a0db7d10SWoojung.Huh@microchip.com 			goto exit;
102355d7de9dSWoojung.Huh@microchip.com 
102455d7de9dSWoojung.Huh@microchip.com 		offset++;
102555d7de9dSWoojung.Huh@microchip.com 	}
102655d7de9dSWoojung.Huh@microchip.com 
1027a0db7d10SWoojung.Huh@microchip.com 	retval = 0;
1028a0db7d10SWoojung.Huh@microchip.com exit:
102987177ba6SWoojung.Huh@microchip.com 	if (dev->chipid == ID_REV_CHIP_ID_7800_)
1030a0db7d10SWoojung.Huh@microchip.com 		ret = lan78xx_write_reg(dev, HW_CFG, saved);
1031a0db7d10SWoojung.Huh@microchip.com 
1032a0db7d10SWoojung.Huh@microchip.com 	return retval;
103355d7de9dSWoojung.Huh@microchip.com }
103455d7de9dSWoojung.Huh@microchip.com 
lan78xx_read_raw_otp(struct lan78xx_net * dev,u32 offset,u32 length,u8 * data)103555d7de9dSWoojung.Huh@microchip.com static int lan78xx_read_raw_otp(struct lan78xx_net *dev, u32 offset,
103655d7de9dSWoojung.Huh@microchip.com 				u32 length, u8 *data)
103755d7de9dSWoojung.Huh@microchip.com {
103855d7de9dSWoojung.Huh@microchip.com 	int i;
103955d7de9dSWoojung.Huh@microchip.com 	u32 buf;
104055d7de9dSWoojung.Huh@microchip.com 	unsigned long timeout;
104155d7de9dSWoojung.Huh@microchip.com 
104206cd7c46SLee Jones 	lan78xx_read_reg(dev, OTP_PWR_DN, &buf);
104355d7de9dSWoojung.Huh@microchip.com 
104455d7de9dSWoojung.Huh@microchip.com 	if (buf & OTP_PWR_DN_PWRDN_N_) {
104555d7de9dSWoojung.Huh@microchip.com 		/* clear it and wait to be cleared */
104606cd7c46SLee Jones 		lan78xx_write_reg(dev, OTP_PWR_DN, 0);
104755d7de9dSWoojung.Huh@microchip.com 
104855d7de9dSWoojung.Huh@microchip.com 		timeout = jiffies + HZ;
104955d7de9dSWoojung.Huh@microchip.com 		do {
105055d7de9dSWoojung.Huh@microchip.com 			usleep_range(1, 10);
105106cd7c46SLee Jones 			lan78xx_read_reg(dev, OTP_PWR_DN, &buf);
105255d7de9dSWoojung.Huh@microchip.com 			if (time_after(jiffies, timeout)) {
105355d7de9dSWoojung.Huh@microchip.com 				netdev_warn(dev->net,
105455d7de9dSWoojung.Huh@microchip.com 					    "timeout on OTP_PWR_DN");
105555d7de9dSWoojung.Huh@microchip.com 				return -EIO;
105655d7de9dSWoojung.Huh@microchip.com 			}
105755d7de9dSWoojung.Huh@microchip.com 		} while (buf & OTP_PWR_DN_PWRDN_N_);
105855d7de9dSWoojung.Huh@microchip.com 	}
105955d7de9dSWoojung.Huh@microchip.com 
106055d7de9dSWoojung.Huh@microchip.com 	for (i = 0; i < length; i++) {
106106cd7c46SLee Jones 		lan78xx_write_reg(dev, OTP_ADDR1,
106255d7de9dSWoojung.Huh@microchip.com 				  ((offset + i) >> 8) & OTP_ADDR1_15_11);
106306cd7c46SLee Jones 		lan78xx_write_reg(dev, OTP_ADDR2,
106455d7de9dSWoojung.Huh@microchip.com 				  ((offset + i) & OTP_ADDR2_10_3));
106555d7de9dSWoojung.Huh@microchip.com 
106606cd7c46SLee Jones 		lan78xx_write_reg(dev, OTP_FUNC_CMD, OTP_FUNC_CMD_READ_);
106706cd7c46SLee Jones 		lan78xx_write_reg(dev, OTP_CMD_GO, OTP_CMD_GO_GO_);
106855d7de9dSWoojung.Huh@microchip.com 
106955d7de9dSWoojung.Huh@microchip.com 		timeout = jiffies + HZ;
107055d7de9dSWoojung.Huh@microchip.com 		do {
107155d7de9dSWoojung.Huh@microchip.com 			udelay(1);
107206cd7c46SLee Jones 			lan78xx_read_reg(dev, OTP_STATUS, &buf);
107355d7de9dSWoojung.Huh@microchip.com 			if (time_after(jiffies, timeout)) {
107455d7de9dSWoojung.Huh@microchip.com 				netdev_warn(dev->net,
107555d7de9dSWoojung.Huh@microchip.com 					    "timeout on OTP_STATUS");
107655d7de9dSWoojung.Huh@microchip.com 				return -EIO;
107755d7de9dSWoojung.Huh@microchip.com 			}
107855d7de9dSWoojung.Huh@microchip.com 		} while (buf & OTP_STATUS_BUSY_);
107955d7de9dSWoojung.Huh@microchip.com 
108006cd7c46SLee Jones 		lan78xx_read_reg(dev, OTP_RD_DATA, &buf);
108155d7de9dSWoojung.Huh@microchip.com 
108255d7de9dSWoojung.Huh@microchip.com 		data[i] = (u8)(buf & 0xFF);
108355d7de9dSWoojung.Huh@microchip.com 	}
108455d7de9dSWoojung.Huh@microchip.com 
108555d7de9dSWoojung.Huh@microchip.com 	return 0;
108655d7de9dSWoojung.Huh@microchip.com }
108755d7de9dSWoojung.Huh@microchip.com 
lan78xx_write_raw_otp(struct lan78xx_net * dev,u32 offset,u32 length,u8 * data)10889fb6066dSWoojung.Huh@microchip.com static int lan78xx_write_raw_otp(struct lan78xx_net *dev, u32 offset,
10899fb6066dSWoojung.Huh@microchip.com 				 u32 length, u8 *data)
10909fb6066dSWoojung.Huh@microchip.com {
10919fb6066dSWoojung.Huh@microchip.com 	int i;
10929fb6066dSWoojung.Huh@microchip.com 	u32 buf;
10939fb6066dSWoojung.Huh@microchip.com 	unsigned long timeout;
10949fb6066dSWoojung.Huh@microchip.com 
109506cd7c46SLee Jones 	lan78xx_read_reg(dev, OTP_PWR_DN, &buf);
10969fb6066dSWoojung.Huh@microchip.com 
10979fb6066dSWoojung.Huh@microchip.com 	if (buf & OTP_PWR_DN_PWRDN_N_) {
10989fb6066dSWoojung.Huh@microchip.com 		/* clear it and wait to be cleared */
109906cd7c46SLee Jones 		lan78xx_write_reg(dev, OTP_PWR_DN, 0);
11009fb6066dSWoojung.Huh@microchip.com 
11019fb6066dSWoojung.Huh@microchip.com 		timeout = jiffies + HZ;
11029fb6066dSWoojung.Huh@microchip.com 		do {
11039fb6066dSWoojung.Huh@microchip.com 			udelay(1);
110406cd7c46SLee Jones 			lan78xx_read_reg(dev, OTP_PWR_DN, &buf);
11059fb6066dSWoojung.Huh@microchip.com 			if (time_after(jiffies, timeout)) {
11069fb6066dSWoojung.Huh@microchip.com 				netdev_warn(dev->net,
11079fb6066dSWoojung.Huh@microchip.com 					    "timeout on OTP_PWR_DN completion");
11089fb6066dSWoojung.Huh@microchip.com 				return -EIO;
11099fb6066dSWoojung.Huh@microchip.com 			}
11109fb6066dSWoojung.Huh@microchip.com 		} while (buf & OTP_PWR_DN_PWRDN_N_);
11119fb6066dSWoojung.Huh@microchip.com 	}
11129fb6066dSWoojung.Huh@microchip.com 
11139fb6066dSWoojung.Huh@microchip.com 	/* set to BYTE program mode */
111406cd7c46SLee Jones 	lan78xx_write_reg(dev, OTP_PRGM_MODE, OTP_PRGM_MODE_BYTE_);
11159fb6066dSWoojung.Huh@microchip.com 
11169fb6066dSWoojung.Huh@microchip.com 	for (i = 0; i < length; i++) {
111706cd7c46SLee Jones 		lan78xx_write_reg(dev, OTP_ADDR1,
11189fb6066dSWoojung.Huh@microchip.com 				  ((offset + i) >> 8) & OTP_ADDR1_15_11);
111906cd7c46SLee Jones 		lan78xx_write_reg(dev, OTP_ADDR2,
11209fb6066dSWoojung.Huh@microchip.com 				  ((offset + i) & OTP_ADDR2_10_3));
112106cd7c46SLee Jones 		lan78xx_write_reg(dev, OTP_PRGM_DATA, data[i]);
112206cd7c46SLee Jones 		lan78xx_write_reg(dev, OTP_TST_CMD, OTP_TST_CMD_PRGVRFY_);
112306cd7c46SLee Jones 		lan78xx_write_reg(dev, OTP_CMD_GO, OTP_CMD_GO_GO_);
11249fb6066dSWoojung.Huh@microchip.com 
11259fb6066dSWoojung.Huh@microchip.com 		timeout = jiffies + HZ;
11269fb6066dSWoojung.Huh@microchip.com 		do {
11279fb6066dSWoojung.Huh@microchip.com 			udelay(1);
112806cd7c46SLee Jones 			lan78xx_read_reg(dev, OTP_STATUS, &buf);
11299fb6066dSWoojung.Huh@microchip.com 			if (time_after(jiffies, timeout)) {
11309fb6066dSWoojung.Huh@microchip.com 				netdev_warn(dev->net,
11319fb6066dSWoojung.Huh@microchip.com 					    "Timeout on OTP_STATUS completion");
11329fb6066dSWoojung.Huh@microchip.com 				return -EIO;
11339fb6066dSWoojung.Huh@microchip.com 			}
11349fb6066dSWoojung.Huh@microchip.com 		} while (buf & OTP_STATUS_BUSY_);
11359fb6066dSWoojung.Huh@microchip.com 	}
11369fb6066dSWoojung.Huh@microchip.com 
11379fb6066dSWoojung.Huh@microchip.com 	return 0;
11389fb6066dSWoojung.Huh@microchip.com }
11399fb6066dSWoojung.Huh@microchip.com 
lan78xx_read_otp(struct lan78xx_net * dev,u32 offset,u32 length,u8 * data)114055d7de9dSWoojung.Huh@microchip.com static int lan78xx_read_otp(struct lan78xx_net *dev, u32 offset,
114155d7de9dSWoojung.Huh@microchip.com 			    u32 length, u8 *data)
114255d7de9dSWoojung.Huh@microchip.com {
114355d7de9dSWoojung.Huh@microchip.com 	u8 sig;
114455d7de9dSWoojung.Huh@microchip.com 	int ret;
114555d7de9dSWoojung.Huh@microchip.com 
114655d7de9dSWoojung.Huh@microchip.com 	ret = lan78xx_read_raw_otp(dev, 0, 1, &sig);
114755d7de9dSWoojung.Huh@microchip.com 
114855d7de9dSWoojung.Huh@microchip.com 	if (ret == 0) {
114994e7c844SNathan Chancellor 		if (sig == OTP_INDICATOR_2)
115055d7de9dSWoojung.Huh@microchip.com 			offset += 0x100;
115194e7c844SNathan Chancellor 		else if (sig != OTP_INDICATOR_1)
115255d7de9dSWoojung.Huh@microchip.com 			ret = -EINVAL;
11534bfc3380SPhil Elwell 		if (!ret)
115455d7de9dSWoojung.Huh@microchip.com 			ret = lan78xx_read_raw_otp(dev, offset, length, data);
115555d7de9dSWoojung.Huh@microchip.com 	}
115655d7de9dSWoojung.Huh@microchip.com 
115755d7de9dSWoojung.Huh@microchip.com 	return ret;
115855d7de9dSWoojung.Huh@microchip.com }
115955d7de9dSWoojung.Huh@microchip.com 
lan78xx_dataport_wait_not_busy(struct lan78xx_net * dev)116055d7de9dSWoojung.Huh@microchip.com static int lan78xx_dataport_wait_not_busy(struct lan78xx_net *dev)
116155d7de9dSWoojung.Huh@microchip.com {
116255d7de9dSWoojung.Huh@microchip.com 	int i, ret;
116355d7de9dSWoojung.Huh@microchip.com 
116455d7de9dSWoojung.Huh@microchip.com 	for (i = 0; i < 100; i++) {
116555d7de9dSWoojung.Huh@microchip.com 		u32 dp_sel;
116655d7de9dSWoojung.Huh@microchip.com 
116755d7de9dSWoojung.Huh@microchip.com 		ret = lan78xx_read_reg(dev, DP_SEL, &dp_sel);
116855d7de9dSWoojung.Huh@microchip.com 		if (unlikely(ret < 0))
116955d7de9dSWoojung.Huh@microchip.com 			return -EIO;
117055d7de9dSWoojung.Huh@microchip.com 
117155d7de9dSWoojung.Huh@microchip.com 		if (dp_sel & DP_SEL_DPRDY_)
117255d7de9dSWoojung.Huh@microchip.com 			return 0;
117355d7de9dSWoojung.Huh@microchip.com 
117455d7de9dSWoojung.Huh@microchip.com 		usleep_range(40, 100);
117555d7de9dSWoojung.Huh@microchip.com 	}
117655d7de9dSWoojung.Huh@microchip.com 
11779ceec7d3SJohn Efstathiades 	netdev_warn(dev->net, "%s timed out", __func__);
117855d7de9dSWoojung.Huh@microchip.com 
117955d7de9dSWoojung.Huh@microchip.com 	return -EIO;
118055d7de9dSWoojung.Huh@microchip.com }
118155d7de9dSWoojung.Huh@microchip.com 
lan78xx_dataport_write(struct lan78xx_net * dev,u32 ram_select,u32 addr,u32 length,u32 * buf)118255d7de9dSWoojung.Huh@microchip.com static int lan78xx_dataport_write(struct lan78xx_net *dev, u32 ram_select,
118355d7de9dSWoojung.Huh@microchip.com 				  u32 addr, u32 length, u32 *buf)
118455d7de9dSWoojung.Huh@microchip.com {
118555d7de9dSWoojung.Huh@microchip.com 	struct lan78xx_priv *pdata = (struct lan78xx_priv *)(dev->data[0]);
118655d7de9dSWoojung.Huh@microchip.com 	u32 dp_sel;
118755d7de9dSWoojung.Huh@microchip.com 	int i, ret;
118855d7de9dSWoojung.Huh@microchip.com 
118955d7de9dSWoojung.Huh@microchip.com 	if (usb_autopm_get_interface(dev->intf) < 0)
119055d7de9dSWoojung.Huh@microchip.com 		return 0;
119155d7de9dSWoojung.Huh@microchip.com 
119255d7de9dSWoojung.Huh@microchip.com 	mutex_lock(&pdata->dataport_mutex);
119355d7de9dSWoojung.Huh@microchip.com 
119455d7de9dSWoojung.Huh@microchip.com 	ret = lan78xx_dataport_wait_not_busy(dev);
119555d7de9dSWoojung.Huh@microchip.com 	if (ret < 0)
119655d7de9dSWoojung.Huh@microchip.com 		goto done;
119755d7de9dSWoojung.Huh@microchip.com 
119855d7de9dSWoojung.Huh@microchip.com 	ret = lan78xx_read_reg(dev, DP_SEL, &dp_sel);
119955d7de9dSWoojung.Huh@microchip.com 
120055d7de9dSWoojung.Huh@microchip.com 	dp_sel &= ~DP_SEL_RSEL_MASK_;
120155d7de9dSWoojung.Huh@microchip.com 	dp_sel |= ram_select;
120255d7de9dSWoojung.Huh@microchip.com 	ret = lan78xx_write_reg(dev, DP_SEL, dp_sel);
120355d7de9dSWoojung.Huh@microchip.com 
120455d7de9dSWoojung.Huh@microchip.com 	for (i = 0; i < length; i++) {
120555d7de9dSWoojung.Huh@microchip.com 		ret = lan78xx_write_reg(dev, DP_ADDR, addr + i);
120655d7de9dSWoojung.Huh@microchip.com 
120755d7de9dSWoojung.Huh@microchip.com 		ret = lan78xx_write_reg(dev, DP_DATA, buf[i]);
120855d7de9dSWoojung.Huh@microchip.com 
120955d7de9dSWoojung.Huh@microchip.com 		ret = lan78xx_write_reg(dev, DP_CMD, DP_CMD_WRITE_);
121055d7de9dSWoojung.Huh@microchip.com 
121155d7de9dSWoojung.Huh@microchip.com 		ret = lan78xx_dataport_wait_not_busy(dev);
121255d7de9dSWoojung.Huh@microchip.com 		if (ret < 0)
121355d7de9dSWoojung.Huh@microchip.com 			goto done;
121455d7de9dSWoojung.Huh@microchip.com 	}
121555d7de9dSWoojung.Huh@microchip.com 
121655d7de9dSWoojung.Huh@microchip.com done:
121755d7de9dSWoojung.Huh@microchip.com 	mutex_unlock(&pdata->dataport_mutex);
121855d7de9dSWoojung.Huh@microchip.com 	usb_autopm_put_interface(dev->intf);
121955d7de9dSWoojung.Huh@microchip.com 
122055d7de9dSWoojung.Huh@microchip.com 	return ret;
122155d7de9dSWoojung.Huh@microchip.com }
122255d7de9dSWoojung.Huh@microchip.com 
lan78xx_set_addr_filter(struct lan78xx_priv * pdata,int index,u8 addr[ETH_ALEN])122355d7de9dSWoojung.Huh@microchip.com static void lan78xx_set_addr_filter(struct lan78xx_priv *pdata,
122455d7de9dSWoojung.Huh@microchip.com 				    int index, u8 addr[ETH_ALEN])
122555d7de9dSWoojung.Huh@microchip.com {
122655d7de9dSWoojung.Huh@microchip.com 	u32 temp;
122755d7de9dSWoojung.Huh@microchip.com 
122855d7de9dSWoojung.Huh@microchip.com 	if ((pdata) && (index > 0) && (index < NUM_OF_MAF)) {
122955d7de9dSWoojung.Huh@microchip.com 		temp = addr[3];
123055d7de9dSWoojung.Huh@microchip.com 		temp = addr[2] | (temp << 8);
123155d7de9dSWoojung.Huh@microchip.com 		temp = addr[1] | (temp << 8);
123255d7de9dSWoojung.Huh@microchip.com 		temp = addr[0] | (temp << 8);
123355d7de9dSWoojung.Huh@microchip.com 		pdata->pfilter_table[index][1] = temp;
123455d7de9dSWoojung.Huh@microchip.com 		temp = addr[5];
123555d7de9dSWoojung.Huh@microchip.com 		temp = addr[4] | (temp << 8);
123655d7de9dSWoojung.Huh@microchip.com 		temp |= MAF_HI_VALID_ | MAF_HI_TYPE_DST_;
123755d7de9dSWoojung.Huh@microchip.com 		pdata->pfilter_table[index][0] = temp;
123855d7de9dSWoojung.Huh@microchip.com 	}
123955d7de9dSWoojung.Huh@microchip.com }
124055d7de9dSWoojung.Huh@microchip.com 
124155d7de9dSWoojung.Huh@microchip.com /* returns hash bit number for given MAC address */
lan78xx_hash(char addr[ETH_ALEN])124255d7de9dSWoojung.Huh@microchip.com static inline u32 lan78xx_hash(char addr[ETH_ALEN])
124355d7de9dSWoojung.Huh@microchip.com {
124455d7de9dSWoojung.Huh@microchip.com 	return (ether_crc(ETH_ALEN, addr) >> 23) & 0x1ff;
124555d7de9dSWoojung.Huh@microchip.com }
124655d7de9dSWoojung.Huh@microchip.com 
lan78xx_deferred_multicast_write(struct work_struct * param)124755d7de9dSWoojung.Huh@microchip.com static void lan78xx_deferred_multicast_write(struct work_struct *param)
124855d7de9dSWoojung.Huh@microchip.com {
124955d7de9dSWoojung.Huh@microchip.com 	struct lan78xx_priv *pdata =
125055d7de9dSWoojung.Huh@microchip.com 			container_of(param, struct lan78xx_priv, set_multicast);
125155d7de9dSWoojung.Huh@microchip.com 	struct lan78xx_net *dev = pdata->dev;
125255d7de9dSWoojung.Huh@microchip.com 	int i;
125355d7de9dSWoojung.Huh@microchip.com 
125455d7de9dSWoojung.Huh@microchip.com 	netif_dbg(dev, drv, dev->net, "deferred multicast write 0x%08x\n",
125555d7de9dSWoojung.Huh@microchip.com 		  pdata->rfe_ctl);
125655d7de9dSWoojung.Huh@microchip.com 
125755d7de9dSWoojung.Huh@microchip.com 	lan78xx_dataport_write(dev, DP_SEL_RSEL_VLAN_DA_, DP_SEL_VHF_VLAN_LEN,
125855d7de9dSWoojung.Huh@microchip.com 			       DP_SEL_VHF_HASH_LEN, pdata->mchash_table);
125955d7de9dSWoojung.Huh@microchip.com 
126055d7de9dSWoojung.Huh@microchip.com 	for (i = 1; i < NUM_OF_MAF; i++) {
126106cd7c46SLee Jones 		lan78xx_write_reg(dev, MAF_HI(i), 0);
126206cd7c46SLee Jones 		lan78xx_write_reg(dev, MAF_LO(i),
126355d7de9dSWoojung.Huh@microchip.com 				  pdata->pfilter_table[i][1]);
126406cd7c46SLee Jones 		lan78xx_write_reg(dev, MAF_HI(i),
126555d7de9dSWoojung.Huh@microchip.com 				  pdata->pfilter_table[i][0]);
126655d7de9dSWoojung.Huh@microchip.com 	}
126755d7de9dSWoojung.Huh@microchip.com 
126806cd7c46SLee Jones 	lan78xx_write_reg(dev, RFE_CTL, pdata->rfe_ctl);
126955d7de9dSWoojung.Huh@microchip.com }
127055d7de9dSWoojung.Huh@microchip.com 
lan78xx_set_multicast(struct net_device * netdev)127155d7de9dSWoojung.Huh@microchip.com static void lan78xx_set_multicast(struct net_device *netdev)
127255d7de9dSWoojung.Huh@microchip.com {
127355d7de9dSWoojung.Huh@microchip.com 	struct lan78xx_net *dev = netdev_priv(netdev);
127455d7de9dSWoojung.Huh@microchip.com 	struct lan78xx_priv *pdata = (struct lan78xx_priv *)(dev->data[0]);
127555d7de9dSWoojung.Huh@microchip.com 	unsigned long flags;
127655d7de9dSWoojung.Huh@microchip.com 	int i;
127755d7de9dSWoojung.Huh@microchip.com 
127855d7de9dSWoojung.Huh@microchip.com 	spin_lock_irqsave(&pdata->rfe_ctl_lock, flags);
127955d7de9dSWoojung.Huh@microchip.com 
128055d7de9dSWoojung.Huh@microchip.com 	pdata->rfe_ctl &= ~(RFE_CTL_UCAST_EN_ | RFE_CTL_MCAST_EN_ |
128155d7de9dSWoojung.Huh@microchip.com 			    RFE_CTL_DA_PERFECT_ | RFE_CTL_MCAST_HASH_);
128255d7de9dSWoojung.Huh@microchip.com 
128355d7de9dSWoojung.Huh@microchip.com 	for (i = 0; i < DP_SEL_VHF_HASH_LEN; i++)
128455d7de9dSWoojung.Huh@microchip.com 		pdata->mchash_table[i] = 0;
12859ceec7d3SJohn Efstathiades 
128655d7de9dSWoojung.Huh@microchip.com 	/* pfilter_table[0] has own HW address */
128755d7de9dSWoojung.Huh@microchip.com 	for (i = 1; i < NUM_OF_MAF; i++) {
12889ceec7d3SJohn Efstathiades 		pdata->pfilter_table[i][0] = 0;
128955d7de9dSWoojung.Huh@microchip.com 		pdata->pfilter_table[i][1] = 0;
129055d7de9dSWoojung.Huh@microchip.com 	}
129155d7de9dSWoojung.Huh@microchip.com 
129255d7de9dSWoojung.Huh@microchip.com 	pdata->rfe_ctl |= RFE_CTL_BCAST_EN_;
129355d7de9dSWoojung.Huh@microchip.com 
129455d7de9dSWoojung.Huh@microchip.com 	if (dev->net->flags & IFF_PROMISC) {
129555d7de9dSWoojung.Huh@microchip.com 		netif_dbg(dev, drv, dev->net, "promiscuous mode enabled");
129655d7de9dSWoojung.Huh@microchip.com 		pdata->rfe_ctl |= RFE_CTL_MCAST_EN_ | RFE_CTL_UCAST_EN_;
129755d7de9dSWoojung.Huh@microchip.com 	} else {
129855d7de9dSWoojung.Huh@microchip.com 		if (dev->net->flags & IFF_ALLMULTI) {
129955d7de9dSWoojung.Huh@microchip.com 			netif_dbg(dev, drv, dev->net,
130055d7de9dSWoojung.Huh@microchip.com 				  "receive all multicast enabled");
130155d7de9dSWoojung.Huh@microchip.com 			pdata->rfe_ctl |= RFE_CTL_MCAST_EN_;
130255d7de9dSWoojung.Huh@microchip.com 		}
130355d7de9dSWoojung.Huh@microchip.com 	}
130455d7de9dSWoojung.Huh@microchip.com 
130555d7de9dSWoojung.Huh@microchip.com 	if (netdev_mc_count(dev->net)) {
130655d7de9dSWoojung.Huh@microchip.com 		struct netdev_hw_addr *ha;
130755d7de9dSWoojung.Huh@microchip.com 		int i;
130855d7de9dSWoojung.Huh@microchip.com 
130955d7de9dSWoojung.Huh@microchip.com 		netif_dbg(dev, drv, dev->net, "receive multicast hash filter");
131055d7de9dSWoojung.Huh@microchip.com 
131155d7de9dSWoojung.Huh@microchip.com 		pdata->rfe_ctl |= RFE_CTL_DA_PERFECT_;
131255d7de9dSWoojung.Huh@microchip.com 
131355d7de9dSWoojung.Huh@microchip.com 		i = 1;
131455d7de9dSWoojung.Huh@microchip.com 		netdev_for_each_mc_addr(ha, netdev) {
131555d7de9dSWoojung.Huh@microchip.com 			/* set first 32 into Perfect Filter */
131655d7de9dSWoojung.Huh@microchip.com 			if (i < 33) {
131755d7de9dSWoojung.Huh@microchip.com 				lan78xx_set_addr_filter(pdata, i, ha->addr);
131855d7de9dSWoojung.Huh@microchip.com 			} else {
131955d7de9dSWoojung.Huh@microchip.com 				u32 bitnum = lan78xx_hash(ha->addr);
132055d7de9dSWoojung.Huh@microchip.com 
132155d7de9dSWoojung.Huh@microchip.com 				pdata->mchash_table[bitnum / 32] |=
132255d7de9dSWoojung.Huh@microchip.com 							(1 << (bitnum % 32));
132355d7de9dSWoojung.Huh@microchip.com 				pdata->rfe_ctl |= RFE_CTL_MCAST_HASH_;
132455d7de9dSWoojung.Huh@microchip.com 			}
132555d7de9dSWoojung.Huh@microchip.com 			i++;
132655d7de9dSWoojung.Huh@microchip.com 		}
132755d7de9dSWoojung.Huh@microchip.com 	}
132855d7de9dSWoojung.Huh@microchip.com 
132955d7de9dSWoojung.Huh@microchip.com 	spin_unlock_irqrestore(&pdata->rfe_ctl_lock, flags);
133055d7de9dSWoojung.Huh@microchip.com 
133155d7de9dSWoojung.Huh@microchip.com 	/* defer register writes to a sleepable context */
133255d7de9dSWoojung.Huh@microchip.com 	schedule_work(&pdata->set_multicast);
133355d7de9dSWoojung.Huh@microchip.com }
133455d7de9dSWoojung.Huh@microchip.com 
lan78xx_update_flowcontrol(struct lan78xx_net * dev,u8 duplex,u16 lcladv,u16 rmtadv)133555d7de9dSWoojung.Huh@microchip.com static int lan78xx_update_flowcontrol(struct lan78xx_net *dev, u8 duplex,
133655d7de9dSWoojung.Huh@microchip.com 				      u16 lcladv, u16 rmtadv)
133755d7de9dSWoojung.Huh@microchip.com {
133855d7de9dSWoojung.Huh@microchip.com 	u32 flow = 0, fct_flow = 0;
1339349e0c5eSWoojung.Huh@microchip.com 	u8 cap;
134055d7de9dSWoojung.Huh@microchip.com 
1341349e0c5eSWoojung.Huh@microchip.com 	if (dev->fc_autoneg)
1342349e0c5eSWoojung.Huh@microchip.com 		cap = mii_resolve_flowctrl_fdx(lcladv, rmtadv);
1343349e0c5eSWoojung.Huh@microchip.com 	else
1344349e0c5eSWoojung.Huh@microchip.com 		cap = dev->fc_request_control;
134555d7de9dSWoojung.Huh@microchip.com 
134655d7de9dSWoojung.Huh@microchip.com 	if (cap & FLOW_CTRL_TX)
1347349e0c5eSWoojung.Huh@microchip.com 		flow |= (FLOW_CR_TX_FCEN_ | 0xFFFF);
134855d7de9dSWoojung.Huh@microchip.com 
134955d7de9dSWoojung.Huh@microchip.com 	if (cap & FLOW_CTRL_RX)
135055d7de9dSWoojung.Huh@microchip.com 		flow |= FLOW_CR_RX_FCEN_;
135155d7de9dSWoojung.Huh@microchip.com 
135255d7de9dSWoojung.Huh@microchip.com 	if (dev->udev->speed == USB_SPEED_SUPER)
1353dc35f854SJohn Efstathiades 		fct_flow = FLOW_CTRL_THRESHOLD(FLOW_ON_SS, FLOW_OFF_SS);
135455d7de9dSWoojung.Huh@microchip.com 	else if (dev->udev->speed == USB_SPEED_HIGH)
1355dc35f854SJohn Efstathiades 		fct_flow = FLOW_CTRL_THRESHOLD(FLOW_ON_HS, FLOW_OFF_HS);
135655d7de9dSWoojung.Huh@microchip.com 
135755d7de9dSWoojung.Huh@microchip.com 	netif_dbg(dev, link, dev->net, "rx pause %s, tx pause %s",
135855d7de9dSWoojung.Huh@microchip.com 		  (cap & FLOW_CTRL_RX ? "enabled" : "disabled"),
135955d7de9dSWoojung.Huh@microchip.com 		  (cap & FLOW_CTRL_TX ? "enabled" : "disabled"));
136055d7de9dSWoojung.Huh@microchip.com 
136106cd7c46SLee Jones 	lan78xx_write_reg(dev, FCT_FLOW, fct_flow);
136255d7de9dSWoojung.Huh@microchip.com 
136355d7de9dSWoojung.Huh@microchip.com 	/* threshold value should be set before enabling flow */
136406cd7c46SLee Jones 	lan78xx_write_reg(dev, FLOW, flow);
136555d7de9dSWoojung.Huh@microchip.com 
136655d7de9dSWoojung.Huh@microchip.com 	return 0;
136755d7de9dSWoojung.Huh@microchip.com }
136855d7de9dSWoojung.Huh@microchip.com 
1369c450a8ebSJohn Efstathiades static void lan78xx_rx_urb_submit_all(struct lan78xx_net *dev);
1370c450a8ebSJohn Efstathiades 
lan78xx_mac_reset(struct lan78xx_net * dev)1371b1f6696dSJohn Efstathiades static int lan78xx_mac_reset(struct lan78xx_net *dev)
1372b1f6696dSJohn Efstathiades {
1373b1f6696dSJohn Efstathiades 	unsigned long start_time = jiffies;
1374b1f6696dSJohn Efstathiades 	u32 val;
1375b1f6696dSJohn Efstathiades 	int ret;
1376b1f6696dSJohn Efstathiades 
1377b1f6696dSJohn Efstathiades 	mutex_lock(&dev->phy_mutex);
1378b1f6696dSJohn Efstathiades 
1379b1f6696dSJohn Efstathiades 	/* Resetting the device while there is activity on the MDIO
1380b1f6696dSJohn Efstathiades 	 * bus can result in the MAC interface locking up and not
1381b1f6696dSJohn Efstathiades 	 * completing register access transactions.
1382b1f6696dSJohn Efstathiades 	 */
1383b1f6696dSJohn Efstathiades 	ret = lan78xx_phy_wait_not_busy(dev);
1384b1f6696dSJohn Efstathiades 	if (ret < 0)
1385b1f6696dSJohn Efstathiades 		goto done;
1386b1f6696dSJohn Efstathiades 
1387b1f6696dSJohn Efstathiades 	ret = lan78xx_read_reg(dev, MAC_CR, &val);
1388b1f6696dSJohn Efstathiades 	if (ret < 0)
1389b1f6696dSJohn Efstathiades 		goto done;
1390b1f6696dSJohn Efstathiades 
1391b1f6696dSJohn Efstathiades 	val |= MAC_CR_RST_;
1392b1f6696dSJohn Efstathiades 	ret = lan78xx_write_reg(dev, MAC_CR, val);
1393b1f6696dSJohn Efstathiades 	if (ret < 0)
1394b1f6696dSJohn Efstathiades 		goto done;
1395b1f6696dSJohn Efstathiades 
1396b1f6696dSJohn Efstathiades 	/* Wait for the reset to complete before allowing any further
1397b1f6696dSJohn Efstathiades 	 * MAC register accesses otherwise the MAC may lock up.
1398b1f6696dSJohn Efstathiades 	 */
1399b1f6696dSJohn Efstathiades 	do {
1400b1f6696dSJohn Efstathiades 		ret = lan78xx_read_reg(dev, MAC_CR, &val);
1401b1f6696dSJohn Efstathiades 		if (ret < 0)
1402b1f6696dSJohn Efstathiades 			goto done;
1403b1f6696dSJohn Efstathiades 
1404b1f6696dSJohn Efstathiades 		if (!(val & MAC_CR_RST_)) {
1405b1f6696dSJohn Efstathiades 			ret = 0;
1406b1f6696dSJohn Efstathiades 			goto done;
1407b1f6696dSJohn Efstathiades 		}
1408b1f6696dSJohn Efstathiades 	} while (!time_after(jiffies, start_time + HZ));
1409b1f6696dSJohn Efstathiades 
1410b1f6696dSJohn Efstathiades 	ret = -ETIMEDOUT;
1411b1f6696dSJohn Efstathiades done:
1412b1f6696dSJohn Efstathiades 	mutex_unlock(&dev->phy_mutex);
1413b1f6696dSJohn Efstathiades 
1414b1f6696dSJohn Efstathiades 	return ret;
1415b1f6696dSJohn Efstathiades }
1416b1f6696dSJohn Efstathiades 
lan78xx_link_reset(struct lan78xx_net * dev)141755d7de9dSWoojung.Huh@microchip.com static int lan78xx_link_reset(struct lan78xx_net *dev)
141855d7de9dSWoojung.Huh@microchip.com {
1419ce85e13aSWoojung.Huh@microchip.com 	struct phy_device *phydev = dev->net->phydev;
14206e76510eSPhilippe Reynes 	struct ethtool_link_ksettings ecmd;
14216b67d4d6SIvan T. Ivanov 	int ladv, radv, ret, link;
142255d7de9dSWoojung.Huh@microchip.com 	u32 buf;
142355d7de9dSWoojung.Huh@microchip.com 
142455d7de9dSWoojung.Huh@microchip.com 	/* clear LAN78xx interrupt status */
142555d7de9dSWoojung.Huh@microchip.com 	ret = lan78xx_write_reg(dev, INT_STS, INT_STS_PHY_INT_);
142655d7de9dSWoojung.Huh@microchip.com 	if (unlikely(ret < 0))
14273415f6baSJohn Efstathiades 		return ret;
142855d7de9dSWoojung.Huh@microchip.com 
14296b67d4d6SIvan T. Ivanov 	mutex_lock(&phydev->lock);
1430ce85e13aSWoojung.Huh@microchip.com 	phy_read_status(phydev);
14316b67d4d6SIvan T. Ivanov 	link = phydev->link;
14326b67d4d6SIvan T. Ivanov 	mutex_unlock(&phydev->lock);
1433ce85e13aSWoojung.Huh@microchip.com 
14346b67d4d6SIvan T. Ivanov 	if (!link && dev->link_on) {
143555d7de9dSWoojung.Huh@microchip.com 		dev->link_on = false;
143655d7de9dSWoojung.Huh@microchip.com 
143755d7de9dSWoojung.Huh@microchip.com 		/* reset MAC */
1438b1f6696dSJohn Efstathiades 		ret = lan78xx_mac_reset(dev);
1439b1f6696dSJohn Efstathiades 		if (ret < 0)
14403415f6baSJohn Efstathiades 			return ret;
1441e4953910SWoojung.Huh@microchip.com 
144220ff5565SWoojung Huh 		del_timer(&dev->stat_monitor);
14436b67d4d6SIvan T. Ivanov 	} else if (link && !dev->link_on) {
144455d7de9dSWoojung.Huh@microchip.com 		dev->link_on = true;
144555d7de9dSWoojung.Huh@microchip.com 
14466e76510eSPhilippe Reynes 		phy_ethtool_ksettings_get(phydev, &ecmd);
144755d7de9dSWoojung.Huh@microchip.com 
144855d7de9dSWoojung.Huh@microchip.com 		if (dev->udev->speed == USB_SPEED_SUPER) {
14496e76510eSPhilippe Reynes 			if (ecmd.base.speed == 1000) {
145055d7de9dSWoojung.Huh@microchip.com 				/* disable U2 */
145155d7de9dSWoojung.Huh@microchip.com 				ret = lan78xx_read_reg(dev, USB_CFG1, &buf);
14523415f6baSJohn Efstathiades 				if (ret < 0)
14533415f6baSJohn Efstathiades 					return ret;
145455d7de9dSWoojung.Huh@microchip.com 				buf &= ~USB_CFG1_DEV_U2_INIT_EN_;
145555d7de9dSWoojung.Huh@microchip.com 				ret = lan78xx_write_reg(dev, USB_CFG1, buf);
14563415f6baSJohn Efstathiades 				if (ret < 0)
14573415f6baSJohn Efstathiades 					return ret;
145855d7de9dSWoojung.Huh@microchip.com 				/* enable U1 */
145955d7de9dSWoojung.Huh@microchip.com 				ret = lan78xx_read_reg(dev, USB_CFG1, &buf);
14603415f6baSJohn Efstathiades 				if (ret < 0)
14613415f6baSJohn Efstathiades 					return ret;
146255d7de9dSWoojung.Huh@microchip.com 				buf |= USB_CFG1_DEV_U1_INIT_EN_;
146355d7de9dSWoojung.Huh@microchip.com 				ret = lan78xx_write_reg(dev, USB_CFG1, buf);
14643415f6baSJohn Efstathiades 				if (ret < 0)
14653415f6baSJohn Efstathiades 					return ret;
146655d7de9dSWoojung.Huh@microchip.com 			} else {
146755d7de9dSWoojung.Huh@microchip.com 				/* enable U1 & U2 */
146855d7de9dSWoojung.Huh@microchip.com 				ret = lan78xx_read_reg(dev, USB_CFG1, &buf);
14693415f6baSJohn Efstathiades 				if (ret < 0)
14703415f6baSJohn Efstathiades 					return ret;
147155d7de9dSWoojung.Huh@microchip.com 				buf |= USB_CFG1_DEV_U2_INIT_EN_;
147255d7de9dSWoojung.Huh@microchip.com 				buf |= USB_CFG1_DEV_U1_INIT_EN_;
147355d7de9dSWoojung.Huh@microchip.com 				ret = lan78xx_write_reg(dev, USB_CFG1, buf);
14743415f6baSJohn Efstathiades 				if (ret < 0)
14753415f6baSJohn Efstathiades 					return ret;
147655d7de9dSWoojung.Huh@microchip.com 			}
147755d7de9dSWoojung.Huh@microchip.com 		}
147855d7de9dSWoojung.Huh@microchip.com 
1479ce85e13aSWoojung.Huh@microchip.com 		ladv = phy_read(phydev, MII_ADVERTISE);
148099c79eceSGeert Uytterhoeven 		if (ladv < 0)
148199c79eceSGeert Uytterhoeven 			return ladv;
148255d7de9dSWoojung.Huh@microchip.com 
1483ce85e13aSWoojung.Huh@microchip.com 		radv = phy_read(phydev, MII_LPA);
148499c79eceSGeert Uytterhoeven 		if (radv < 0)
148599c79eceSGeert Uytterhoeven 			return radv;
148655d7de9dSWoojung.Huh@microchip.com 
148755d7de9dSWoojung.Huh@microchip.com 		netif_dbg(dev, link, dev->net,
148855d7de9dSWoojung.Huh@microchip.com 			  "speed: %u duplex: %d anadv: 0x%04x anlpa: 0x%04x",
14896e76510eSPhilippe Reynes 			  ecmd.base.speed, ecmd.base.duplex, ladv, radv);
149055d7de9dSWoojung.Huh@microchip.com 
14916e76510eSPhilippe Reynes 		ret = lan78xx_update_flowcontrol(dev, ecmd.base.duplex, ladv,
14926e76510eSPhilippe Reynes 						 radv);
14933415f6baSJohn Efstathiades 		if (ret < 0)
14943415f6baSJohn Efstathiades 			return ret;
149520ff5565SWoojung Huh 
149620ff5565SWoojung Huh 		if (!timer_pending(&dev->stat_monitor)) {
149720ff5565SWoojung Huh 			dev->delta = 1;
149820ff5565SWoojung Huh 			mod_timer(&dev->stat_monitor,
149920ff5565SWoojung Huh 				  jiffies + STAT_UPDATE_TIMER);
150020ff5565SWoojung Huh 		}
1501136f55f6SStefan Wahren 
1502c450a8ebSJohn Efstathiades 		lan78xx_rx_urb_submit_all(dev);
1503c450a8ebSJohn Efstathiades 
15042ae3d9d1SOleksij Rempel 		local_bh_disable();
1505ec4c7e12SJohn Efstathiades 		napi_schedule(&dev->napi);
15062ae3d9d1SOleksij Rempel 		local_bh_enable();
150755d7de9dSWoojung.Huh@microchip.com 	}
150855d7de9dSWoojung.Huh@microchip.com 
15093415f6baSJohn Efstathiades 	return 0;
151055d7de9dSWoojung.Huh@microchip.com }
151155d7de9dSWoojung.Huh@microchip.com 
151255d7de9dSWoojung.Huh@microchip.com /* some work can't be done in tasklets, so we use keventd
151355d7de9dSWoojung.Huh@microchip.com  *
151455d7de9dSWoojung.Huh@microchip.com  * NOTE:  annoying asymmetry:  if it's active, schedule_work() fails,
151555d7de9dSWoojung.Huh@microchip.com  * but tasklet_schedule() doesn't.	hope the failure is rare.
151655d7de9dSWoojung.Huh@microchip.com  */
lan78xx_defer_kevent(struct lan78xx_net * dev,int work)1517e0c79ff6SBaoyou Xie static void lan78xx_defer_kevent(struct lan78xx_net *dev, int work)
151855d7de9dSWoojung.Huh@microchip.com {
151955d7de9dSWoojung.Huh@microchip.com 	set_bit(work, &dev->flags);
152055d7de9dSWoojung.Huh@microchip.com 	if (!schedule_delayed_work(&dev->wq, 0))
152155d7de9dSWoojung.Huh@microchip.com 		netdev_err(dev->net, "kevent %d may have been dropped\n", work);
152255d7de9dSWoojung.Huh@microchip.com }
152355d7de9dSWoojung.Huh@microchip.com 
lan78xx_status(struct lan78xx_net * dev,struct urb * urb)152455d7de9dSWoojung.Huh@microchip.com static void lan78xx_status(struct lan78xx_net *dev, struct urb *urb)
152555d7de9dSWoojung.Huh@microchip.com {
152655d7de9dSWoojung.Huh@microchip.com 	u32 intdata;
152755d7de9dSWoojung.Huh@microchip.com 
152855d7de9dSWoojung.Huh@microchip.com 	if (urb->actual_length != 4) {
152955d7de9dSWoojung.Huh@microchip.com 		netdev_warn(dev->net,
153055d7de9dSWoojung.Huh@microchip.com 			    "unexpected urb length %d", urb->actual_length);
153155d7de9dSWoojung.Huh@microchip.com 		return;
153255d7de9dSWoojung.Huh@microchip.com 	}
153355d7de9dSWoojung.Huh@microchip.com 
1534bb448f8aSChuhong Yuan 	intdata = get_unaligned_le32(urb->transfer_buffer);
153555d7de9dSWoojung.Huh@microchip.com 
153655d7de9dSWoojung.Huh@microchip.com 	if (intdata & INT_ENP_PHY_INT) {
153755d7de9dSWoojung.Huh@microchip.com 		netif_dbg(dev, link, dev->net, "PHY INTR: 0x%08x\n", intdata);
153855d7de9dSWoojung.Huh@microchip.com 		lan78xx_defer_kevent(dev, EVENT_LINK_RESET);
1539cc89c323SWoojung Huh 
1540bfe6b967SSebastian Andrzej Siewior 		if (dev->domain_data.phyirq > 0)
1541bfe6b967SSebastian Andrzej Siewior 			generic_handle_irq_safe(dev->domain_data.phyirq);
15429ceec7d3SJohn Efstathiades 	} else {
154355d7de9dSWoojung.Huh@microchip.com 		netdev_warn(dev->net,
154455d7de9dSWoojung.Huh@microchip.com 			    "unexpected interrupt: 0x%08x\n", intdata);
154555d7de9dSWoojung.Huh@microchip.com 	}
15469ceec7d3SJohn Efstathiades }
154755d7de9dSWoojung.Huh@microchip.com 
lan78xx_ethtool_get_eeprom_len(struct net_device * netdev)154855d7de9dSWoojung.Huh@microchip.com static int lan78xx_ethtool_get_eeprom_len(struct net_device *netdev)
154955d7de9dSWoojung.Huh@microchip.com {
155055d7de9dSWoojung.Huh@microchip.com 	return MAX_EEPROM_SIZE;
155155d7de9dSWoojung.Huh@microchip.com }
155255d7de9dSWoojung.Huh@microchip.com 
lan78xx_ethtool_get_eeprom(struct net_device * netdev,struct ethtool_eeprom * ee,u8 * data)155355d7de9dSWoojung.Huh@microchip.com static int lan78xx_ethtool_get_eeprom(struct net_device *netdev,
155455d7de9dSWoojung.Huh@microchip.com 				      struct ethtool_eeprom *ee, u8 *data)
155555d7de9dSWoojung.Huh@microchip.com {
155655d7de9dSWoojung.Huh@microchip.com 	struct lan78xx_net *dev = netdev_priv(netdev);
15578a7ffeb7SNisar Sayed 	int ret;
15588a7ffeb7SNisar Sayed 
15598a7ffeb7SNisar Sayed 	ret = usb_autopm_get_interface(dev->intf);
15608a7ffeb7SNisar Sayed 	if (ret)
15618a7ffeb7SNisar Sayed 		return ret;
156255d7de9dSWoojung.Huh@microchip.com 
156355d7de9dSWoojung.Huh@microchip.com 	ee->magic = LAN78XX_EEPROM_MAGIC;
156455d7de9dSWoojung.Huh@microchip.com 
15658a7ffeb7SNisar Sayed 	ret = lan78xx_read_raw_eeprom(dev, ee->offset, ee->len, data);
15668a7ffeb7SNisar Sayed 
15678a7ffeb7SNisar Sayed 	usb_autopm_put_interface(dev->intf);
15688a7ffeb7SNisar Sayed 
15698a7ffeb7SNisar Sayed 	return ret;
157055d7de9dSWoojung.Huh@microchip.com }
157155d7de9dSWoojung.Huh@microchip.com 
lan78xx_ethtool_set_eeprom(struct net_device * netdev,struct ethtool_eeprom * ee,u8 * data)157255d7de9dSWoojung.Huh@microchip.com static int lan78xx_ethtool_set_eeprom(struct net_device *netdev,
157355d7de9dSWoojung.Huh@microchip.com 				      struct ethtool_eeprom *ee, u8 *data)
157455d7de9dSWoojung.Huh@microchip.com {
157555d7de9dSWoojung.Huh@microchip.com 	struct lan78xx_net *dev = netdev_priv(netdev);
15768a7ffeb7SNisar Sayed 	int ret;
15778a7ffeb7SNisar Sayed 
15788a7ffeb7SNisar Sayed 	ret = usb_autopm_get_interface(dev->intf);
15798a7ffeb7SNisar Sayed 	if (ret)
15808a7ffeb7SNisar Sayed 		return ret;
158155d7de9dSWoojung.Huh@microchip.com 
1582c0776822SNisar Sayed 	/* Invalid EEPROM_INDICATOR at offset zero will result in a failure
1583c0776822SNisar Sayed 	 * to load data from EEPROM
1584c0776822SNisar Sayed 	 */
1585c0776822SNisar Sayed 	if (ee->magic == LAN78XX_EEPROM_MAGIC)
15868a7ffeb7SNisar Sayed 		ret = lan78xx_write_raw_eeprom(dev, ee->offset, ee->len, data);
158755d7de9dSWoojung.Huh@microchip.com 	else if ((ee->magic == LAN78XX_OTP_MAGIC) &&
158855d7de9dSWoojung.Huh@microchip.com 		 (ee->offset == 0) &&
158955d7de9dSWoojung.Huh@microchip.com 		 (ee->len == 512) &&
159055d7de9dSWoojung.Huh@microchip.com 		 (data[0] == OTP_INDICATOR_1))
15918a7ffeb7SNisar Sayed 		ret = lan78xx_write_raw_otp(dev, ee->offset, ee->len, data);
159255d7de9dSWoojung.Huh@microchip.com 
15938a7ffeb7SNisar Sayed 	usb_autopm_put_interface(dev->intf);
15948a7ffeb7SNisar Sayed 
15958a7ffeb7SNisar Sayed 	return ret;
159655d7de9dSWoojung.Huh@microchip.com }
159755d7de9dSWoojung.Huh@microchip.com 
lan78xx_get_strings(struct net_device * netdev,u32 stringset,u8 * data)159855d7de9dSWoojung.Huh@microchip.com static void lan78xx_get_strings(struct net_device *netdev, u32 stringset,
159955d7de9dSWoojung.Huh@microchip.com 				u8 *data)
160055d7de9dSWoojung.Huh@microchip.com {
160155d7de9dSWoojung.Huh@microchip.com 	if (stringset == ETH_SS_STATS)
160255d7de9dSWoojung.Huh@microchip.com 		memcpy(data, lan78xx_gstrings, sizeof(lan78xx_gstrings));
160355d7de9dSWoojung.Huh@microchip.com }
160455d7de9dSWoojung.Huh@microchip.com 
lan78xx_get_sset_count(struct net_device * netdev,int sset)160555d7de9dSWoojung.Huh@microchip.com static int lan78xx_get_sset_count(struct net_device *netdev, int sset)
160655d7de9dSWoojung.Huh@microchip.com {
160755d7de9dSWoojung.Huh@microchip.com 	if (sset == ETH_SS_STATS)
160855d7de9dSWoojung.Huh@microchip.com 		return ARRAY_SIZE(lan78xx_gstrings);
160955d7de9dSWoojung.Huh@microchip.com 	else
161055d7de9dSWoojung.Huh@microchip.com 		return -EOPNOTSUPP;
161155d7de9dSWoojung.Huh@microchip.com }
161255d7de9dSWoojung.Huh@microchip.com 
lan78xx_get_stats(struct net_device * netdev,struct ethtool_stats * stats,u64 * data)161355d7de9dSWoojung.Huh@microchip.com static void lan78xx_get_stats(struct net_device *netdev,
161455d7de9dSWoojung.Huh@microchip.com 			      struct ethtool_stats *stats, u64 *data)
161555d7de9dSWoojung.Huh@microchip.com {
161655d7de9dSWoojung.Huh@microchip.com 	struct lan78xx_net *dev = netdev_priv(netdev);
161755d7de9dSWoojung.Huh@microchip.com 
161820ff5565SWoojung Huh 	lan78xx_update_stats(dev);
161955d7de9dSWoojung.Huh@microchip.com 
162020ff5565SWoojung Huh 	mutex_lock(&dev->stats.access_lock);
162120ff5565SWoojung Huh 	memcpy(data, &dev->stats.curr_stat, sizeof(dev->stats.curr_stat));
162220ff5565SWoojung Huh 	mutex_unlock(&dev->stats.access_lock);
162355d7de9dSWoojung.Huh@microchip.com }
162455d7de9dSWoojung.Huh@microchip.com 
lan78xx_get_wol(struct net_device * netdev,struct ethtool_wolinfo * wol)162555d7de9dSWoojung.Huh@microchip.com static void lan78xx_get_wol(struct net_device *netdev,
162655d7de9dSWoojung.Huh@microchip.com 			    struct ethtool_wolinfo *wol)
162755d7de9dSWoojung.Huh@microchip.com {
162855d7de9dSWoojung.Huh@microchip.com 	struct lan78xx_net *dev = netdev_priv(netdev);
162955d7de9dSWoojung.Huh@microchip.com 	int ret;
163055d7de9dSWoojung.Huh@microchip.com 	u32 buf;
163155d7de9dSWoojung.Huh@microchip.com 	struct lan78xx_priv *pdata = (struct lan78xx_priv *)(dev->data[0]);
163255d7de9dSWoojung.Huh@microchip.com 
163355d7de9dSWoojung.Huh@microchip.com 	if (usb_autopm_get_interface(dev->intf) < 0)
163455d7de9dSWoojung.Huh@microchip.com 		return;
163555d7de9dSWoojung.Huh@microchip.com 
163655d7de9dSWoojung.Huh@microchip.com 	ret = lan78xx_read_reg(dev, USB_CFG0, &buf);
163755d7de9dSWoojung.Huh@microchip.com 	if (unlikely(ret < 0)) {
163855d7de9dSWoojung.Huh@microchip.com 		wol->supported = 0;
163955d7de9dSWoojung.Huh@microchip.com 		wol->wolopts = 0;
164055d7de9dSWoojung.Huh@microchip.com 	} else {
164155d7de9dSWoojung.Huh@microchip.com 		if (buf & USB_CFG_RMT_WKP_) {
164255d7de9dSWoojung.Huh@microchip.com 			wol->supported = WAKE_ALL;
164355d7de9dSWoojung.Huh@microchip.com 			wol->wolopts = pdata->wol;
164455d7de9dSWoojung.Huh@microchip.com 		} else {
164555d7de9dSWoojung.Huh@microchip.com 			wol->supported = 0;
164655d7de9dSWoojung.Huh@microchip.com 			wol->wolopts = 0;
164755d7de9dSWoojung.Huh@microchip.com 		}
164855d7de9dSWoojung.Huh@microchip.com 	}
164955d7de9dSWoojung.Huh@microchip.com 
165055d7de9dSWoojung.Huh@microchip.com 	usb_autopm_put_interface(dev->intf);
165155d7de9dSWoojung.Huh@microchip.com }
165255d7de9dSWoojung.Huh@microchip.com 
lan78xx_set_wol(struct net_device * netdev,struct ethtool_wolinfo * wol)165355d7de9dSWoojung.Huh@microchip.com static int lan78xx_set_wol(struct net_device *netdev,
165455d7de9dSWoojung.Huh@microchip.com 			   struct ethtool_wolinfo *wol)
165555d7de9dSWoojung.Huh@microchip.com {
165655d7de9dSWoojung.Huh@microchip.com 	struct lan78xx_net *dev = netdev_priv(netdev);
165755d7de9dSWoojung.Huh@microchip.com 	struct lan78xx_priv *pdata = (struct lan78xx_priv *)(dev->data[0]);
165855d7de9dSWoojung.Huh@microchip.com 	int ret;
165955d7de9dSWoojung.Huh@microchip.com 
1660*e3bb76feSOleksij Rempel 	if (wol->wolopts & ~WAKE_ALL)
1661*e3bb76feSOleksij Rempel 		return -EINVAL;
1662*e3bb76feSOleksij Rempel 
166355d7de9dSWoojung.Huh@microchip.com 	ret = usb_autopm_get_interface(dev->intf);
166455d7de9dSWoojung.Huh@microchip.com 	if (ret < 0)
166555d7de9dSWoojung.Huh@microchip.com 		return ret;
166655d7de9dSWoojung.Huh@microchip.com 
1667eb9ad088SFlorian Fainelli 	pdata->wol = wol->wolopts;
166855d7de9dSWoojung.Huh@microchip.com 
166955d7de9dSWoojung.Huh@microchip.com 	device_set_wakeup_enable(&dev->udev->dev, (bool)wol->wolopts);
167055d7de9dSWoojung.Huh@microchip.com 
1671ce85e13aSWoojung.Huh@microchip.com 	phy_ethtool_set_wol(netdev->phydev, wol);
1672ce85e13aSWoojung.Huh@microchip.com 
167355d7de9dSWoojung.Huh@microchip.com 	usb_autopm_put_interface(dev->intf);
167455d7de9dSWoojung.Huh@microchip.com 
167555d7de9dSWoojung.Huh@microchip.com 	return ret;
167655d7de9dSWoojung.Huh@microchip.com }
167755d7de9dSWoojung.Huh@microchip.com 
lan78xx_get_eee(struct net_device * net,struct ethtool_eee * edata)167855d7de9dSWoojung.Huh@microchip.com static int lan78xx_get_eee(struct net_device *net, struct ethtool_eee *edata)
167955d7de9dSWoojung.Huh@microchip.com {
168055d7de9dSWoojung.Huh@microchip.com 	struct lan78xx_net *dev = netdev_priv(net);
1681ce85e13aSWoojung.Huh@microchip.com 	struct phy_device *phydev = net->phydev;
168255d7de9dSWoojung.Huh@microchip.com 	int ret;
168355d7de9dSWoojung.Huh@microchip.com 	u32 buf;
168455d7de9dSWoojung.Huh@microchip.com 
168555d7de9dSWoojung.Huh@microchip.com 	ret = usb_autopm_get_interface(dev->intf);
168655d7de9dSWoojung.Huh@microchip.com 	if (ret < 0)
168755d7de9dSWoojung.Huh@microchip.com 		return ret;
168855d7de9dSWoojung.Huh@microchip.com 
1689ce85e13aSWoojung.Huh@microchip.com 	ret = phy_ethtool_get_eee(phydev, edata);
1690ce85e13aSWoojung.Huh@microchip.com 	if (ret < 0)
1691ce85e13aSWoojung.Huh@microchip.com 		goto exit;
1692ce85e13aSWoojung.Huh@microchip.com 
169355d7de9dSWoojung.Huh@microchip.com 	ret = lan78xx_read_reg(dev, MAC_CR, &buf);
169455d7de9dSWoojung.Huh@microchip.com 	if (buf & MAC_CR_EEE_EN_) {
169555d7de9dSWoojung.Huh@microchip.com 		edata->eee_enabled = true;
1696ce85e13aSWoojung.Huh@microchip.com 		edata->eee_active = !!(edata->advertised &
1697ce85e13aSWoojung.Huh@microchip.com 				       edata->lp_advertised);
169855d7de9dSWoojung.Huh@microchip.com 		edata->tx_lpi_enabled = true;
169955d7de9dSWoojung.Huh@microchip.com 		/* EEE_TX_LPI_REQ_DLY & tx_lpi_timer are same uSec unit */
170055d7de9dSWoojung.Huh@microchip.com 		ret = lan78xx_read_reg(dev, EEE_TX_LPI_REQ_DLY, &buf);
170155d7de9dSWoojung.Huh@microchip.com 		edata->tx_lpi_timer = buf;
170255d7de9dSWoojung.Huh@microchip.com 	} else {
170355d7de9dSWoojung.Huh@microchip.com 		edata->eee_enabled = false;
170455d7de9dSWoojung.Huh@microchip.com 		edata->eee_active = false;
170555d7de9dSWoojung.Huh@microchip.com 		edata->tx_lpi_enabled = false;
170655d7de9dSWoojung.Huh@microchip.com 		edata->tx_lpi_timer = 0;
170755d7de9dSWoojung.Huh@microchip.com 	}
170855d7de9dSWoojung.Huh@microchip.com 
1709ce85e13aSWoojung.Huh@microchip.com 	ret = 0;
1710ce85e13aSWoojung.Huh@microchip.com exit:
171155d7de9dSWoojung.Huh@microchip.com 	usb_autopm_put_interface(dev->intf);
171255d7de9dSWoojung.Huh@microchip.com 
1713ce85e13aSWoojung.Huh@microchip.com 	return ret;
171455d7de9dSWoojung.Huh@microchip.com }
171555d7de9dSWoojung.Huh@microchip.com 
lan78xx_set_eee(struct net_device * net,struct ethtool_eee * edata)171655d7de9dSWoojung.Huh@microchip.com static int lan78xx_set_eee(struct net_device *net, struct ethtool_eee *edata)
171755d7de9dSWoojung.Huh@microchip.com {
171855d7de9dSWoojung.Huh@microchip.com 	struct lan78xx_net *dev = netdev_priv(net);
171955d7de9dSWoojung.Huh@microchip.com 	int ret;
172055d7de9dSWoojung.Huh@microchip.com 	u32 buf;
172155d7de9dSWoojung.Huh@microchip.com 
172255d7de9dSWoojung.Huh@microchip.com 	ret = usb_autopm_get_interface(dev->intf);
172355d7de9dSWoojung.Huh@microchip.com 	if (ret < 0)
172455d7de9dSWoojung.Huh@microchip.com 		return ret;
172555d7de9dSWoojung.Huh@microchip.com 
172655d7de9dSWoojung.Huh@microchip.com 	if (edata->eee_enabled) {
172755d7de9dSWoojung.Huh@microchip.com 		ret = lan78xx_read_reg(dev, MAC_CR, &buf);
172855d7de9dSWoojung.Huh@microchip.com 		buf |= MAC_CR_EEE_EN_;
172955d7de9dSWoojung.Huh@microchip.com 		ret = lan78xx_write_reg(dev, MAC_CR, buf);
173055d7de9dSWoojung.Huh@microchip.com 
1731ce85e13aSWoojung.Huh@microchip.com 		phy_ethtool_set_eee(net->phydev, edata);
1732ce85e13aSWoojung.Huh@microchip.com 
1733ce85e13aSWoojung.Huh@microchip.com 		buf = (u32)edata->tx_lpi_timer;
1734ce85e13aSWoojung.Huh@microchip.com 		ret = lan78xx_write_reg(dev, EEE_TX_LPI_REQ_DLY, buf);
173555d7de9dSWoojung.Huh@microchip.com 	} else {
173655d7de9dSWoojung.Huh@microchip.com 		ret = lan78xx_read_reg(dev, MAC_CR, &buf);
173755d7de9dSWoojung.Huh@microchip.com 		buf &= ~MAC_CR_EEE_EN_;
173855d7de9dSWoojung.Huh@microchip.com 		ret = lan78xx_write_reg(dev, MAC_CR, buf);
173955d7de9dSWoojung.Huh@microchip.com 	}
174055d7de9dSWoojung.Huh@microchip.com 
174155d7de9dSWoojung.Huh@microchip.com 	usb_autopm_put_interface(dev->intf);
174255d7de9dSWoojung.Huh@microchip.com 
174355d7de9dSWoojung.Huh@microchip.com 	return 0;
174455d7de9dSWoojung.Huh@microchip.com }
174555d7de9dSWoojung.Huh@microchip.com 
lan78xx_get_link(struct net_device * net)174655d7de9dSWoojung.Huh@microchip.com static u32 lan78xx_get_link(struct net_device *net)
174755d7de9dSWoojung.Huh@microchip.com {
17486b67d4d6SIvan T. Ivanov 	u32 link;
174955d7de9dSWoojung.Huh@microchip.com 
17506b67d4d6SIvan T. Ivanov 	mutex_lock(&net->phydev->lock);
17516b67d4d6SIvan T. Ivanov 	phy_read_status(net->phydev);
17526b67d4d6SIvan T. Ivanov 	link = net->phydev->link;
17536b67d4d6SIvan T. Ivanov 	mutex_unlock(&net->phydev->lock);
17546b67d4d6SIvan T. Ivanov 
17556b67d4d6SIvan T. Ivanov 	return link;
175655d7de9dSWoojung.Huh@microchip.com }
175755d7de9dSWoojung.Huh@microchip.com 
lan78xx_get_drvinfo(struct net_device * net,struct ethtool_drvinfo * info)175855d7de9dSWoojung.Huh@microchip.com static void lan78xx_get_drvinfo(struct net_device *net,
175955d7de9dSWoojung.Huh@microchip.com 				struct ethtool_drvinfo *info)
176055d7de9dSWoojung.Huh@microchip.com {
176155d7de9dSWoojung.Huh@microchip.com 	struct lan78xx_net *dev = netdev_priv(net);
176255d7de9dSWoojung.Huh@microchip.com 
176355d7de9dSWoojung.Huh@microchip.com 	strncpy(info->driver, DRIVER_NAME, sizeof(info->driver));
176455d7de9dSWoojung.Huh@microchip.com 	usb_make_path(dev->udev, info->bus_info, sizeof(info->bus_info));
176555d7de9dSWoojung.Huh@microchip.com }
176655d7de9dSWoojung.Huh@microchip.com 
lan78xx_get_msglevel(struct net_device * net)176755d7de9dSWoojung.Huh@microchip.com static u32 lan78xx_get_msglevel(struct net_device *net)
176855d7de9dSWoojung.Huh@microchip.com {
176955d7de9dSWoojung.Huh@microchip.com 	struct lan78xx_net *dev = netdev_priv(net);
177055d7de9dSWoojung.Huh@microchip.com 
177155d7de9dSWoojung.Huh@microchip.com 	return dev->msg_enable;
177255d7de9dSWoojung.Huh@microchip.com }
177355d7de9dSWoojung.Huh@microchip.com 
lan78xx_set_msglevel(struct net_device * net,u32 level)177455d7de9dSWoojung.Huh@microchip.com static void lan78xx_set_msglevel(struct net_device *net, u32 level)
177555d7de9dSWoojung.Huh@microchip.com {
177655d7de9dSWoojung.Huh@microchip.com 	struct lan78xx_net *dev = netdev_priv(net);
177755d7de9dSWoojung.Huh@microchip.com 
177855d7de9dSWoojung.Huh@microchip.com 	dev->msg_enable = level;
177955d7de9dSWoojung.Huh@microchip.com }
178055d7de9dSWoojung.Huh@microchip.com 
lan78xx_get_link_ksettings(struct net_device * net,struct ethtool_link_ksettings * cmd)17816e76510eSPhilippe Reynes static int lan78xx_get_link_ksettings(struct net_device *net,
17826e76510eSPhilippe Reynes 				      struct ethtool_link_ksettings *cmd)
178355d7de9dSWoojung.Huh@microchip.com {
178455d7de9dSWoojung.Huh@microchip.com 	struct lan78xx_net *dev = netdev_priv(net);
1785ce85e13aSWoojung.Huh@microchip.com 	struct phy_device *phydev = net->phydev;
178655d7de9dSWoojung.Huh@microchip.com 	int ret;
178755d7de9dSWoojung.Huh@microchip.com 
178855d7de9dSWoojung.Huh@microchip.com 	ret = usb_autopm_get_interface(dev->intf);
178955d7de9dSWoojung.Huh@microchip.com 	if (ret < 0)
179055d7de9dSWoojung.Huh@microchip.com 		return ret;
179155d7de9dSWoojung.Huh@microchip.com 
17925514174fSyuval.shaia@oracle.com 	phy_ethtool_ksettings_get(phydev, cmd);
179355d7de9dSWoojung.Huh@microchip.com 
179455d7de9dSWoojung.Huh@microchip.com 	usb_autopm_put_interface(dev->intf);
179555d7de9dSWoojung.Huh@microchip.com 
179655d7de9dSWoojung.Huh@microchip.com 	return ret;
179755d7de9dSWoojung.Huh@microchip.com }
179855d7de9dSWoojung.Huh@microchip.com 
lan78xx_set_link_ksettings(struct net_device * net,const struct ethtool_link_ksettings * cmd)17996e76510eSPhilippe Reynes static int lan78xx_set_link_ksettings(struct net_device *net,
18006e76510eSPhilippe Reynes 				      const struct ethtool_link_ksettings *cmd)
180155d7de9dSWoojung.Huh@microchip.com {
180255d7de9dSWoojung.Huh@microchip.com 	struct lan78xx_net *dev = netdev_priv(net);
1803ce85e13aSWoojung.Huh@microchip.com 	struct phy_device *phydev = net->phydev;
180455d7de9dSWoojung.Huh@microchip.com 	int ret = 0;
180555d7de9dSWoojung.Huh@microchip.com 	int temp;
180655d7de9dSWoojung.Huh@microchip.com 
180755d7de9dSWoojung.Huh@microchip.com 	ret = usb_autopm_get_interface(dev->intf);
180855d7de9dSWoojung.Huh@microchip.com 	if (ret < 0)
180955d7de9dSWoojung.Huh@microchip.com 		return ret;
181055d7de9dSWoojung.Huh@microchip.com 
181155d7de9dSWoojung.Huh@microchip.com 	/* change speed & duplex */
18126e76510eSPhilippe Reynes 	ret = phy_ethtool_ksettings_set(phydev, cmd);
181355d7de9dSWoojung.Huh@microchip.com 
18146e76510eSPhilippe Reynes 	if (!cmd->base.autoneg) {
181555d7de9dSWoojung.Huh@microchip.com 		/* force link down */
1816ce85e13aSWoojung.Huh@microchip.com 		temp = phy_read(phydev, MII_BMCR);
1817ce85e13aSWoojung.Huh@microchip.com 		phy_write(phydev, MII_BMCR, temp | BMCR_LOOPBACK);
181855d7de9dSWoojung.Huh@microchip.com 		mdelay(1);
1819ce85e13aSWoojung.Huh@microchip.com 		phy_write(phydev, MII_BMCR, temp);
182055d7de9dSWoojung.Huh@microchip.com 	}
182155d7de9dSWoojung.Huh@microchip.com 
182255d7de9dSWoojung.Huh@microchip.com 	usb_autopm_put_interface(dev->intf);
182355d7de9dSWoojung.Huh@microchip.com 
182455d7de9dSWoojung.Huh@microchip.com 	return ret;
182555d7de9dSWoojung.Huh@microchip.com }
182655d7de9dSWoojung.Huh@microchip.com 
lan78xx_get_pause(struct net_device * net,struct ethtool_pauseparam * pause)1827349e0c5eSWoojung.Huh@microchip.com static void lan78xx_get_pause(struct net_device *net,
1828349e0c5eSWoojung.Huh@microchip.com 			      struct ethtool_pauseparam *pause)
1829349e0c5eSWoojung.Huh@microchip.com {
1830349e0c5eSWoojung.Huh@microchip.com 	struct lan78xx_net *dev = netdev_priv(net);
1831349e0c5eSWoojung.Huh@microchip.com 	struct phy_device *phydev = net->phydev;
18326e76510eSPhilippe Reynes 	struct ethtool_link_ksettings ecmd;
1833349e0c5eSWoojung.Huh@microchip.com 
18346e76510eSPhilippe Reynes 	phy_ethtool_ksettings_get(phydev, &ecmd);
1835349e0c5eSWoojung.Huh@microchip.com 
1836349e0c5eSWoojung.Huh@microchip.com 	pause->autoneg = dev->fc_autoneg;
1837349e0c5eSWoojung.Huh@microchip.com 
1838349e0c5eSWoojung.Huh@microchip.com 	if (dev->fc_request_control & FLOW_CTRL_TX)
1839349e0c5eSWoojung.Huh@microchip.com 		pause->tx_pause = 1;
1840349e0c5eSWoojung.Huh@microchip.com 
1841349e0c5eSWoojung.Huh@microchip.com 	if (dev->fc_request_control & FLOW_CTRL_RX)
1842349e0c5eSWoojung.Huh@microchip.com 		pause->rx_pause = 1;
1843349e0c5eSWoojung.Huh@microchip.com }
1844349e0c5eSWoojung.Huh@microchip.com 
lan78xx_set_pause(struct net_device * net,struct ethtool_pauseparam * pause)1845349e0c5eSWoojung.Huh@microchip.com static int lan78xx_set_pause(struct net_device *net,
1846349e0c5eSWoojung.Huh@microchip.com 			     struct ethtool_pauseparam *pause)
1847349e0c5eSWoojung.Huh@microchip.com {
1848349e0c5eSWoojung.Huh@microchip.com 	struct lan78xx_net *dev = netdev_priv(net);
1849349e0c5eSWoojung.Huh@microchip.com 	struct phy_device *phydev = net->phydev;
18506e76510eSPhilippe Reynes 	struct ethtool_link_ksettings ecmd;
1851349e0c5eSWoojung.Huh@microchip.com 	int ret;
1852349e0c5eSWoojung.Huh@microchip.com 
18536e76510eSPhilippe Reynes 	phy_ethtool_ksettings_get(phydev, &ecmd);
1854349e0c5eSWoojung.Huh@microchip.com 
18556e76510eSPhilippe Reynes 	if (pause->autoneg && !ecmd.base.autoneg) {
1856349e0c5eSWoojung.Huh@microchip.com 		ret = -EINVAL;
1857349e0c5eSWoojung.Huh@microchip.com 		goto exit;
1858349e0c5eSWoojung.Huh@microchip.com 	}
1859349e0c5eSWoojung.Huh@microchip.com 
1860349e0c5eSWoojung.Huh@microchip.com 	dev->fc_request_control = 0;
1861349e0c5eSWoojung.Huh@microchip.com 	if (pause->rx_pause)
1862349e0c5eSWoojung.Huh@microchip.com 		dev->fc_request_control |= FLOW_CTRL_RX;
1863349e0c5eSWoojung.Huh@microchip.com 
1864349e0c5eSWoojung.Huh@microchip.com 	if (pause->tx_pause)
1865349e0c5eSWoojung.Huh@microchip.com 		dev->fc_request_control |= FLOW_CTRL_TX;
1866349e0c5eSWoojung.Huh@microchip.com 
18676e76510eSPhilippe Reynes 	if (ecmd.base.autoneg) {
18683c1bcc86SAndrew Lunn 		__ETHTOOL_DECLARE_LINK_MODE_MASK(fc) = { 0, };
1869349e0c5eSWoojung.Huh@microchip.com 		u32 mii_adv;
1870349e0c5eSWoojung.Huh@microchip.com 
18713c1bcc86SAndrew Lunn 		linkmode_clear_bit(ETHTOOL_LINK_MODE_Pause_BIT,
18723c1bcc86SAndrew Lunn 				   ecmd.link_modes.advertising);
18733c1bcc86SAndrew Lunn 		linkmode_clear_bit(ETHTOOL_LINK_MODE_Asym_Pause_BIT,
18743c1bcc86SAndrew Lunn 				   ecmd.link_modes.advertising);
1875349e0c5eSWoojung.Huh@microchip.com 		mii_adv = (u32)mii_advertise_flowctrl(dev->fc_request_control);
18763c1bcc86SAndrew Lunn 		mii_adv_to_linkmode_adv_t(fc, mii_adv);
18773c1bcc86SAndrew Lunn 		linkmode_or(ecmd.link_modes.advertising, fc,
18783c1bcc86SAndrew Lunn 			    ecmd.link_modes.advertising);
18796e76510eSPhilippe Reynes 
18806e76510eSPhilippe Reynes 		phy_ethtool_ksettings_set(phydev, &ecmd);
1881349e0c5eSWoojung.Huh@microchip.com 	}
1882349e0c5eSWoojung.Huh@microchip.com 
1883349e0c5eSWoojung.Huh@microchip.com 	dev->fc_autoneg = pause->autoneg;
1884349e0c5eSWoojung.Huh@microchip.com 
1885349e0c5eSWoojung.Huh@microchip.com 	ret = 0;
1886349e0c5eSWoojung.Huh@microchip.com exit:
1887349e0c5eSWoojung.Huh@microchip.com 	return ret;
1888349e0c5eSWoojung.Huh@microchip.com }
1889349e0c5eSWoojung.Huh@microchip.com 
lan78xx_get_regs_len(struct net_device * netdev)189049621865SRaghuram Chary J static int lan78xx_get_regs_len(struct net_device *netdev)
189149621865SRaghuram Chary J {
189249621865SRaghuram Chary J 	if (!netdev->phydev)
189349621865SRaghuram Chary J 		return (sizeof(lan78xx_regs));
189449621865SRaghuram Chary J 	else
189549621865SRaghuram Chary J 		return (sizeof(lan78xx_regs) + PHY_REG_SIZE);
189649621865SRaghuram Chary J }
189749621865SRaghuram Chary J 
189849621865SRaghuram Chary J static void
lan78xx_get_regs(struct net_device * netdev,struct ethtool_regs * regs,void * buf)189949621865SRaghuram Chary J lan78xx_get_regs(struct net_device *netdev, struct ethtool_regs *regs,
190049621865SRaghuram Chary J 		 void *buf)
190149621865SRaghuram Chary J {
190249621865SRaghuram Chary J 	u32 *data = buf;
190349621865SRaghuram Chary J 	int i, j;
190449621865SRaghuram Chary J 	struct lan78xx_net *dev = netdev_priv(netdev);
190549621865SRaghuram Chary J 
190649621865SRaghuram Chary J 	/* Read Device/MAC registers */
1907bf37afceSzhong jiang 	for (i = 0; i < ARRAY_SIZE(lan78xx_regs); i++)
190849621865SRaghuram Chary J 		lan78xx_read_reg(dev, lan78xx_regs[i], &data[i]);
190949621865SRaghuram Chary J 
191049621865SRaghuram Chary J 	if (!netdev->phydev)
191149621865SRaghuram Chary J 		return;
191249621865SRaghuram Chary J 
191349621865SRaghuram Chary J 	/* Read PHY registers */
191449621865SRaghuram Chary J 	for (j = 0; j < 32; i++, j++)
191549621865SRaghuram Chary J 		data[i] = phy_read(netdev->phydev, j);
191649621865SRaghuram Chary J }
191749621865SRaghuram Chary J 
191855d7de9dSWoojung.Huh@microchip.com static const struct ethtool_ops lan78xx_ethtool_ops = {
191955d7de9dSWoojung.Huh@microchip.com 	.get_link	= lan78xx_get_link,
1920860ce4b4SFlorian Fainelli 	.nway_reset	= phy_ethtool_nway_reset,
192155d7de9dSWoojung.Huh@microchip.com 	.get_drvinfo	= lan78xx_get_drvinfo,
192255d7de9dSWoojung.Huh@microchip.com 	.get_msglevel	= lan78xx_get_msglevel,
192355d7de9dSWoojung.Huh@microchip.com 	.set_msglevel	= lan78xx_set_msglevel,
192455d7de9dSWoojung.Huh@microchip.com 	.get_eeprom_len = lan78xx_ethtool_get_eeprom_len,
192555d7de9dSWoojung.Huh@microchip.com 	.get_eeprom	= lan78xx_ethtool_get_eeprom,
192655d7de9dSWoojung.Huh@microchip.com 	.set_eeprom	= lan78xx_ethtool_set_eeprom,
192755d7de9dSWoojung.Huh@microchip.com 	.get_ethtool_stats = lan78xx_get_stats,
192855d7de9dSWoojung.Huh@microchip.com 	.get_sset_count = lan78xx_get_sset_count,
192955d7de9dSWoojung.Huh@microchip.com 	.get_strings	= lan78xx_get_strings,
193055d7de9dSWoojung.Huh@microchip.com 	.get_wol	= lan78xx_get_wol,
193155d7de9dSWoojung.Huh@microchip.com 	.set_wol	= lan78xx_set_wol,
193233e6b167SMarkus Bloechl 	.get_ts_info	= ethtool_op_get_ts_info,
193355d7de9dSWoojung.Huh@microchip.com 	.get_eee	= lan78xx_get_eee,
193455d7de9dSWoojung.Huh@microchip.com 	.set_eee	= lan78xx_set_eee,
1935349e0c5eSWoojung.Huh@microchip.com 	.get_pauseparam	= lan78xx_get_pause,
1936349e0c5eSWoojung.Huh@microchip.com 	.set_pauseparam	= lan78xx_set_pause,
19376e76510eSPhilippe Reynes 	.get_link_ksettings = lan78xx_get_link_ksettings,
19386e76510eSPhilippe Reynes 	.set_link_ksettings = lan78xx_set_link_ksettings,
193949621865SRaghuram Chary J 	.get_regs_len	= lan78xx_get_regs_len,
194049621865SRaghuram Chary J 	.get_regs	= lan78xx_get_regs,
194155d7de9dSWoojung.Huh@microchip.com };
194255d7de9dSWoojung.Huh@microchip.com 
lan78xx_init_mac_address(struct lan78xx_net * dev)194355d7de9dSWoojung.Huh@microchip.com static void lan78xx_init_mac_address(struct lan78xx_net *dev)
194455d7de9dSWoojung.Huh@microchip.com {
194555d7de9dSWoojung.Huh@microchip.com 	u32 addr_lo, addr_hi;
194655d7de9dSWoojung.Huh@microchip.com 	u8 addr[6];
194755d7de9dSWoojung.Huh@microchip.com 
194806cd7c46SLee Jones 	lan78xx_read_reg(dev, RX_ADDRL, &addr_lo);
194906cd7c46SLee Jones 	lan78xx_read_reg(dev, RX_ADDRH, &addr_hi);
195055d7de9dSWoojung.Huh@microchip.com 
195155d7de9dSWoojung.Huh@microchip.com 	addr[0] = addr_lo & 0xFF;
195255d7de9dSWoojung.Huh@microchip.com 	addr[1] = (addr_lo >> 8) & 0xFF;
195355d7de9dSWoojung.Huh@microchip.com 	addr[2] = (addr_lo >> 16) & 0xFF;
195455d7de9dSWoojung.Huh@microchip.com 	addr[3] = (addr_lo >> 24) & 0xFF;
195555d7de9dSWoojung.Huh@microchip.com 	addr[4] = addr_hi & 0xFF;
195655d7de9dSWoojung.Huh@microchip.com 	addr[5] = (addr_hi >> 8) & 0xFF;
195755d7de9dSWoojung.Huh@microchip.com 
195855d7de9dSWoojung.Huh@microchip.com 	if (!is_valid_ether_addr(addr)) {
1959760db29bSPhil Elwell 		if (!eth_platform_get_mac_address(&dev->udev->dev, addr)) {
1960760db29bSPhil Elwell 			/* valid address present in Device Tree */
1961760db29bSPhil Elwell 			netif_dbg(dev, ifup, dev->net,
1962760db29bSPhil Elwell 				  "MAC address read from Device Tree");
1963760db29bSPhil Elwell 		} else if (((lan78xx_read_eeprom(dev, EEPROM_MAC_OFFSET,
1964760db29bSPhil Elwell 						 ETH_ALEN, addr) == 0) ||
1965760db29bSPhil Elwell 			    (lan78xx_read_otp(dev, EEPROM_MAC_OFFSET,
1966760db29bSPhil Elwell 					      ETH_ALEN, addr) == 0)) &&
1967760db29bSPhil Elwell 			   is_valid_ether_addr(addr)) {
196855d7de9dSWoojung.Huh@microchip.com 			/* eeprom values are valid so use them */
196955d7de9dSWoojung.Huh@microchip.com 			netif_dbg(dev, ifup, dev->net,
197055d7de9dSWoojung.Huh@microchip.com 				  "MAC address read from EEPROM");
197155d7de9dSWoojung.Huh@microchip.com 		} else {
197255d7de9dSWoojung.Huh@microchip.com 			/* generate random MAC */
19736c1f0a1fSJoe Perches 			eth_random_addr(addr);
197455d7de9dSWoojung.Huh@microchip.com 			netif_dbg(dev, ifup, dev->net,
197555d7de9dSWoojung.Huh@microchip.com 				  "MAC address set to random addr");
197655d7de9dSWoojung.Huh@microchip.com 		}
197755d7de9dSWoojung.Huh@microchip.com 
197855d7de9dSWoojung.Huh@microchip.com 		addr_lo = addr[0] | (addr[1] << 8) |
197955d7de9dSWoojung.Huh@microchip.com 			  (addr[2] << 16) | (addr[3] << 24);
198055d7de9dSWoojung.Huh@microchip.com 		addr_hi = addr[4] | (addr[5] << 8);
198155d7de9dSWoojung.Huh@microchip.com 
198206cd7c46SLee Jones 		lan78xx_write_reg(dev, RX_ADDRL, addr_lo);
198306cd7c46SLee Jones 		lan78xx_write_reg(dev, RX_ADDRH, addr_hi);
198455d7de9dSWoojung.Huh@microchip.com 	}
198555d7de9dSWoojung.Huh@microchip.com 
198606cd7c46SLee Jones 	lan78xx_write_reg(dev, MAF_LO(0), addr_lo);
198706cd7c46SLee Jones 	lan78xx_write_reg(dev, MAF_HI(0), addr_hi | MAF_HI_VALID_);
198855d7de9dSWoojung.Huh@microchip.com 
1989af804e6dSJakub Kicinski 	eth_hw_addr_set(dev->net, addr);
199055d7de9dSWoojung.Huh@microchip.com }
199155d7de9dSWoojung.Huh@microchip.com 
1992ce85e13aSWoojung.Huh@microchip.com /* MDIO read and write wrappers for phylib */
lan78xx_mdiobus_read(struct mii_bus * bus,int phy_id,int idx)1993ce85e13aSWoojung.Huh@microchip.com static int lan78xx_mdiobus_read(struct mii_bus *bus, int phy_id, int idx)
199455d7de9dSWoojung.Huh@microchip.com {
1995ce85e13aSWoojung.Huh@microchip.com 	struct lan78xx_net *dev = bus->priv;
1996ce85e13aSWoojung.Huh@microchip.com 	u32 val, addr;
1997ce85e13aSWoojung.Huh@microchip.com 	int ret;
1998ce85e13aSWoojung.Huh@microchip.com 
1999ce85e13aSWoojung.Huh@microchip.com 	ret = usb_autopm_get_interface(dev->intf);
2000ce85e13aSWoojung.Huh@microchip.com 	if (ret < 0)
2001ce85e13aSWoojung.Huh@microchip.com 		return ret;
2002ce85e13aSWoojung.Huh@microchip.com 
2003ce85e13aSWoojung.Huh@microchip.com 	mutex_lock(&dev->phy_mutex);
2004ce85e13aSWoojung.Huh@microchip.com 
2005ce85e13aSWoojung.Huh@microchip.com 	/* confirm MII not busy */
2006ce85e13aSWoojung.Huh@microchip.com 	ret = lan78xx_phy_wait_not_busy(dev);
2007ce85e13aSWoojung.Huh@microchip.com 	if (ret < 0)
2008ce85e13aSWoojung.Huh@microchip.com 		goto done;
2009ce85e13aSWoojung.Huh@microchip.com 
2010ce85e13aSWoojung.Huh@microchip.com 	/* set the address, index & direction (read from PHY) */
2011ce85e13aSWoojung.Huh@microchip.com 	addr = mii_access(phy_id, idx, MII_READ);
2012ce85e13aSWoojung.Huh@microchip.com 	ret = lan78xx_write_reg(dev, MII_ACC, addr);
2013ce85e13aSWoojung.Huh@microchip.com 
2014ce85e13aSWoojung.Huh@microchip.com 	ret = lan78xx_phy_wait_not_busy(dev);
2015ce85e13aSWoojung.Huh@microchip.com 	if (ret < 0)
2016ce85e13aSWoojung.Huh@microchip.com 		goto done;
2017ce85e13aSWoojung.Huh@microchip.com 
2018ce85e13aSWoojung.Huh@microchip.com 	ret = lan78xx_read_reg(dev, MII_DATA, &val);
2019ce85e13aSWoojung.Huh@microchip.com 
2020ce85e13aSWoojung.Huh@microchip.com 	ret = (int)(val & 0xFFFF);
2021ce85e13aSWoojung.Huh@microchip.com 
2022ce85e13aSWoojung.Huh@microchip.com done:
2023ce85e13aSWoojung.Huh@microchip.com 	mutex_unlock(&dev->phy_mutex);
2024ce85e13aSWoojung.Huh@microchip.com 	usb_autopm_put_interface(dev->intf);
202502dc1f3dSWoojung Huh 
2026ce85e13aSWoojung.Huh@microchip.com 	return ret;
2027ce85e13aSWoojung.Huh@microchip.com }
2028ce85e13aSWoojung.Huh@microchip.com 
lan78xx_mdiobus_write(struct mii_bus * bus,int phy_id,int idx,u16 regval)2029ce85e13aSWoojung.Huh@microchip.com static int lan78xx_mdiobus_write(struct mii_bus *bus, int phy_id, int idx,
2030ce85e13aSWoojung.Huh@microchip.com 				 u16 regval)
2031ce85e13aSWoojung.Huh@microchip.com {
2032ce85e13aSWoojung.Huh@microchip.com 	struct lan78xx_net *dev = bus->priv;
2033ce85e13aSWoojung.Huh@microchip.com 	u32 val, addr;
2034ce85e13aSWoojung.Huh@microchip.com 	int ret;
2035ce85e13aSWoojung.Huh@microchip.com 
2036ce85e13aSWoojung.Huh@microchip.com 	ret = usb_autopm_get_interface(dev->intf);
2037ce85e13aSWoojung.Huh@microchip.com 	if (ret < 0)
2038ce85e13aSWoojung.Huh@microchip.com 		return ret;
2039ce85e13aSWoojung.Huh@microchip.com 
2040ce85e13aSWoojung.Huh@microchip.com 	mutex_lock(&dev->phy_mutex);
2041ce85e13aSWoojung.Huh@microchip.com 
2042ce85e13aSWoojung.Huh@microchip.com 	/* confirm MII not busy */
2043ce85e13aSWoojung.Huh@microchip.com 	ret = lan78xx_phy_wait_not_busy(dev);
2044ce85e13aSWoojung.Huh@microchip.com 	if (ret < 0)
2045ce85e13aSWoojung.Huh@microchip.com 		goto done;
2046ce85e13aSWoojung.Huh@microchip.com 
2047ce85e13aSWoojung.Huh@microchip.com 	val = (u32)regval;
2048ce85e13aSWoojung.Huh@microchip.com 	ret = lan78xx_write_reg(dev, MII_DATA, val);
2049ce85e13aSWoojung.Huh@microchip.com 
2050ce85e13aSWoojung.Huh@microchip.com 	/* set the address, index & direction (write to PHY) */
2051ce85e13aSWoojung.Huh@microchip.com 	addr = mii_access(phy_id, idx, MII_WRITE);
2052ce85e13aSWoojung.Huh@microchip.com 	ret = lan78xx_write_reg(dev, MII_ACC, addr);
2053ce85e13aSWoojung.Huh@microchip.com 
2054ce85e13aSWoojung.Huh@microchip.com 	ret = lan78xx_phy_wait_not_busy(dev);
2055ce85e13aSWoojung.Huh@microchip.com 	if (ret < 0)
2056ce85e13aSWoojung.Huh@microchip.com 		goto done;
2057ce85e13aSWoojung.Huh@microchip.com 
2058ce85e13aSWoojung.Huh@microchip.com done:
2059ce85e13aSWoojung.Huh@microchip.com 	mutex_unlock(&dev->phy_mutex);
2060ce85e13aSWoojung.Huh@microchip.com 	usb_autopm_put_interface(dev->intf);
2061ce85e13aSWoojung.Huh@microchip.com 	return 0;
2062ce85e13aSWoojung.Huh@microchip.com }
2063ce85e13aSWoojung.Huh@microchip.com 
lan78xx_mdio_init(struct lan78xx_net * dev)2064ce85e13aSWoojung.Huh@microchip.com static int lan78xx_mdio_init(struct lan78xx_net *dev)
2065ce85e13aSWoojung.Huh@microchip.com {
20661827b067SPhil Elwell 	struct device_node *node;
2067ce85e13aSWoojung.Huh@microchip.com 	int ret;
2068ce85e13aSWoojung.Huh@microchip.com 
2069ce85e13aSWoojung.Huh@microchip.com 	dev->mdiobus = mdiobus_alloc();
2070ce85e13aSWoojung.Huh@microchip.com 	if (!dev->mdiobus) {
2071ce85e13aSWoojung.Huh@microchip.com 		netdev_err(dev->net, "can't allocate MDIO bus\n");
2072ce85e13aSWoojung.Huh@microchip.com 		return -ENOMEM;
2073ce85e13aSWoojung.Huh@microchip.com 	}
2074ce85e13aSWoojung.Huh@microchip.com 
2075ce85e13aSWoojung.Huh@microchip.com 	dev->mdiobus->priv = (void *)dev;
2076ce85e13aSWoojung.Huh@microchip.com 	dev->mdiobus->read = lan78xx_mdiobus_read;
2077ce85e13aSWoojung.Huh@microchip.com 	dev->mdiobus->write = lan78xx_mdiobus_write;
2078ce85e13aSWoojung.Huh@microchip.com 	dev->mdiobus->name = "lan78xx-mdiobus";
207920032b63SCristian Birsan 	dev->mdiobus->parent = &dev->udev->dev;
2080ce85e13aSWoojung.Huh@microchip.com 
2081ce85e13aSWoojung.Huh@microchip.com 	snprintf(dev->mdiobus->id, MII_BUS_ID_SIZE, "usb-%03d:%03d",
2082ce85e13aSWoojung.Huh@microchip.com 		 dev->udev->bus->busnum, dev->udev->devnum);
2083ce85e13aSWoojung.Huh@microchip.com 
208487177ba6SWoojung.Huh@microchip.com 	switch (dev->chipid) {
208587177ba6SWoojung.Huh@microchip.com 	case ID_REV_CHIP_ID_7800_:
208687177ba6SWoojung.Huh@microchip.com 	case ID_REV_CHIP_ID_7850_:
2087ce85e13aSWoojung.Huh@microchip.com 		/* set to internal PHY id */
2088ce85e13aSWoojung.Huh@microchip.com 		dev->mdiobus->phy_mask = ~(1 << 1);
2089ce85e13aSWoojung.Huh@microchip.com 		break;
209002dc1f3dSWoojung Huh 	case ID_REV_CHIP_ID_7801_:
209102dc1f3dSWoojung Huh 		/* scan thru PHYAD[2..0] */
209202dc1f3dSWoojung Huh 		dev->mdiobus->phy_mask = ~(0xFF);
209302dc1f3dSWoojung Huh 		break;
2094ce85e13aSWoojung.Huh@microchip.com 	}
2095ce85e13aSWoojung.Huh@microchip.com 
20961827b067SPhil Elwell 	node = of_get_child_by_name(dev->udev->dev.of_node, "mdio");
20971827b067SPhil Elwell 	ret = of_mdiobus_register(dev->mdiobus, node);
20981827b067SPhil Elwell 	of_node_put(node);
2099ce85e13aSWoojung.Huh@microchip.com 	if (ret) {
2100ce85e13aSWoojung.Huh@microchip.com 		netdev_err(dev->net, "can't register MDIO bus\n");
2101e7f4dc35SAndrew Lunn 		goto exit1;
2102ce85e13aSWoojung.Huh@microchip.com 	}
2103ce85e13aSWoojung.Huh@microchip.com 
2104ce85e13aSWoojung.Huh@microchip.com 	netdev_dbg(dev->net, "registered mdiobus bus %s\n", dev->mdiobus->id);
2105ce85e13aSWoojung.Huh@microchip.com 	return 0;
2106ce85e13aSWoojung.Huh@microchip.com exit1:
2107ce85e13aSWoojung.Huh@microchip.com 	mdiobus_free(dev->mdiobus);
2108ce85e13aSWoojung.Huh@microchip.com 	return ret;
2109ce85e13aSWoojung.Huh@microchip.com }
2110ce85e13aSWoojung.Huh@microchip.com 
lan78xx_remove_mdio(struct lan78xx_net * dev)2111ce85e13aSWoojung.Huh@microchip.com static void lan78xx_remove_mdio(struct lan78xx_net *dev)
2112ce85e13aSWoojung.Huh@microchip.com {
2113ce85e13aSWoojung.Huh@microchip.com 	mdiobus_unregister(dev->mdiobus);
2114ce85e13aSWoojung.Huh@microchip.com 	mdiobus_free(dev->mdiobus);
2115ce85e13aSWoojung.Huh@microchip.com }
2116ce85e13aSWoojung.Huh@microchip.com 
lan78xx_link_status_change(struct net_device * net)2117ce85e13aSWoojung.Huh@microchip.com static void lan78xx_link_status_change(struct net_device *net)
2118ce85e13aSWoojung.Huh@microchip.com {
211914437e3fSWoojung Huh 	struct phy_device *phydev = net->phydev;
212014437e3fSWoojung Huh 
2121e57cf363SYuiko Oshino 	phy_print_status(phydev);
212255d7de9dSWoojung.Huh@microchip.com }
212355d7de9dSWoojung.Huh@microchip.com 
irq_map(struct irq_domain * d,unsigned int irq,irq_hw_number_t hwirq)2124cc89c323SWoojung Huh static int irq_map(struct irq_domain *d, unsigned int irq,
2125cc89c323SWoojung Huh 		   irq_hw_number_t hwirq)
2126cc89c323SWoojung Huh {
2127cc89c323SWoojung Huh 	struct irq_domain_data *data = d->host_data;
2128cc89c323SWoojung Huh 
2129cc89c323SWoojung Huh 	irq_set_chip_data(irq, data);
2130cc89c323SWoojung Huh 	irq_set_chip_and_handler(irq, data->irqchip, data->irq_handler);
2131cc89c323SWoojung Huh 	irq_set_noprobe(irq);
2132cc89c323SWoojung Huh 
2133cc89c323SWoojung Huh 	return 0;
2134cc89c323SWoojung Huh }
2135cc89c323SWoojung Huh 
irq_unmap(struct irq_domain * d,unsigned int irq)2136cc89c323SWoojung Huh static void irq_unmap(struct irq_domain *d, unsigned int irq)
2137cc89c323SWoojung Huh {
2138cc89c323SWoojung Huh 	irq_set_chip_and_handler(irq, NULL, NULL);
2139cc89c323SWoojung Huh 	irq_set_chip_data(irq, NULL);
2140cc89c323SWoojung Huh }
2141cc89c323SWoojung Huh 
2142cc89c323SWoojung Huh static const struct irq_domain_ops chip_domain_ops = {
2143cc89c323SWoojung Huh 	.map	= irq_map,
2144cc89c323SWoojung Huh 	.unmap	= irq_unmap,
2145cc89c323SWoojung Huh };
2146cc89c323SWoojung Huh 
lan78xx_irq_mask(struct irq_data * irqd)2147cc89c323SWoojung Huh static void lan78xx_irq_mask(struct irq_data *irqd)
2148cc89c323SWoojung Huh {
2149cc89c323SWoojung Huh 	struct irq_domain_data *data = irq_data_get_irq_chip_data(irqd);
2150cc89c323SWoojung Huh 
2151cc89c323SWoojung Huh 	data->irqenable &= ~BIT(irqd_to_hwirq(irqd));
2152cc89c323SWoojung Huh }
2153cc89c323SWoojung Huh 
lan78xx_irq_unmask(struct irq_data * irqd)2154cc89c323SWoojung Huh static void lan78xx_irq_unmask(struct irq_data *irqd)
2155cc89c323SWoojung Huh {
2156cc89c323SWoojung Huh 	struct irq_domain_data *data = irq_data_get_irq_chip_data(irqd);
2157cc89c323SWoojung Huh 
2158cc89c323SWoojung Huh 	data->irqenable |= BIT(irqd_to_hwirq(irqd));
2159cc89c323SWoojung Huh }
2160cc89c323SWoojung Huh 
lan78xx_irq_bus_lock(struct irq_data * irqd)2161cc89c323SWoojung Huh static void lan78xx_irq_bus_lock(struct irq_data *irqd)
2162cc89c323SWoojung Huh {
2163cc89c323SWoojung Huh 	struct irq_domain_data *data = irq_data_get_irq_chip_data(irqd);
2164cc89c323SWoojung Huh 
2165cc89c323SWoojung Huh 	mutex_lock(&data->irq_lock);
2166cc89c323SWoojung Huh }
2167cc89c323SWoojung Huh 
lan78xx_irq_bus_sync_unlock(struct irq_data * irqd)2168cc89c323SWoojung Huh static void lan78xx_irq_bus_sync_unlock(struct irq_data *irqd)
2169cc89c323SWoojung Huh {
2170cc89c323SWoojung Huh 	struct irq_domain_data *data = irq_data_get_irq_chip_data(irqd);
2171cc89c323SWoojung Huh 	struct lan78xx_net *dev =
2172cc89c323SWoojung Huh 			container_of(data, struct lan78xx_net, domain_data);
2173cc89c323SWoojung Huh 	u32 buf;
2174cc89c323SWoojung Huh 
2175cc89c323SWoojung Huh 	/* call register access here because irq_bus_lock & irq_bus_sync_unlock
2176cc89c323SWoojung Huh 	 * are only two callbacks executed in non-atomic contex.
2177cc89c323SWoojung Huh 	 */
217806cd7c46SLee Jones 	lan78xx_read_reg(dev, INT_EP_CTL, &buf);
2179cc89c323SWoojung Huh 	if (buf != data->irqenable)
218006cd7c46SLee Jones 		lan78xx_write_reg(dev, INT_EP_CTL, data->irqenable);
2181cc89c323SWoojung Huh 
2182cc89c323SWoojung Huh 	mutex_unlock(&data->irq_lock);
2183cc89c323SWoojung Huh }
2184cc89c323SWoojung Huh 
2185cc89c323SWoojung Huh static struct irq_chip lan78xx_irqchip = {
2186cc89c323SWoojung Huh 	.name			= "lan78xx-irqs",
2187cc89c323SWoojung Huh 	.irq_mask		= lan78xx_irq_mask,
2188cc89c323SWoojung Huh 	.irq_unmask		= lan78xx_irq_unmask,
2189cc89c323SWoojung Huh 	.irq_bus_lock		= lan78xx_irq_bus_lock,
2190cc89c323SWoojung Huh 	.irq_bus_sync_unlock	= lan78xx_irq_bus_sync_unlock,
2191cc89c323SWoojung Huh };
2192cc89c323SWoojung Huh 
lan78xx_setup_irq_domain(struct lan78xx_net * dev)2193cc89c323SWoojung Huh static int lan78xx_setup_irq_domain(struct lan78xx_net *dev)
2194cc89c323SWoojung Huh {
2195cc89c323SWoojung Huh 	struct device_node *of_node;
2196cc89c323SWoojung Huh 	struct irq_domain *irqdomain;
2197cc89c323SWoojung Huh 	unsigned int irqmap = 0;
2198cc89c323SWoojung Huh 	u32 buf;
2199cc89c323SWoojung Huh 	int ret = 0;
2200cc89c323SWoojung Huh 
2201cc89c323SWoojung Huh 	of_node = dev->udev->dev.parent->of_node;
2202cc89c323SWoojung Huh 
2203cc89c323SWoojung Huh 	mutex_init(&dev->domain_data.irq_lock);
2204cc89c323SWoojung Huh 
2205cc89c323SWoojung Huh 	lan78xx_read_reg(dev, INT_EP_CTL, &buf);
2206cc89c323SWoojung Huh 	dev->domain_data.irqenable = buf;
2207cc89c323SWoojung Huh 
2208cc89c323SWoojung Huh 	dev->domain_data.irqchip = &lan78xx_irqchip;
2209cc89c323SWoojung Huh 	dev->domain_data.irq_handler = handle_simple_irq;
2210cc89c323SWoojung Huh 
2211cc89c323SWoojung Huh 	irqdomain = irq_domain_add_simple(of_node, MAX_INT_EP, 0,
2212cc89c323SWoojung Huh 					  &chip_domain_ops, &dev->domain_data);
2213cc89c323SWoojung Huh 	if (irqdomain) {
2214cc89c323SWoojung Huh 		/* create mapping for PHY interrupt */
2215cc89c323SWoojung Huh 		irqmap = irq_create_mapping(irqdomain, INT_EP_PHY);
2216cc89c323SWoojung Huh 		if (!irqmap) {
2217cc89c323SWoojung Huh 			irq_domain_remove(irqdomain);
2218cc89c323SWoojung Huh 
2219cc89c323SWoojung Huh 			irqdomain = NULL;
2220cc89c323SWoojung Huh 			ret = -EINVAL;
2221cc89c323SWoojung Huh 		}
2222cc89c323SWoojung Huh 	} else {
2223cc89c323SWoojung Huh 		ret = -EINVAL;
2224cc89c323SWoojung Huh 	}
2225cc89c323SWoojung Huh 
2226cc89c323SWoojung Huh 	dev->domain_data.irqdomain = irqdomain;
2227cc89c323SWoojung Huh 	dev->domain_data.phyirq = irqmap;
2228cc89c323SWoojung Huh 
2229cc89c323SWoojung Huh 	return ret;
2230cc89c323SWoojung Huh }
2231cc89c323SWoojung Huh 
lan78xx_remove_irq_domain(struct lan78xx_net * dev)2232cc89c323SWoojung Huh static void lan78xx_remove_irq_domain(struct lan78xx_net *dev)
2233cc89c323SWoojung Huh {
2234cc89c323SWoojung Huh 	if (dev->domain_data.phyirq > 0) {
2235cc89c323SWoojung Huh 		irq_dispose_mapping(dev->domain_data.phyirq);
2236cc89c323SWoojung Huh 
2237cc89c323SWoojung Huh 		if (dev->domain_data.irqdomain)
2238cc89c323SWoojung Huh 			irq_domain_remove(dev->domain_data.irqdomain);
2239cc89c323SWoojung Huh 	}
2240cc89c323SWoojung Huh 	dev->domain_data.phyirq = 0;
2241cc89c323SWoojung Huh 	dev->domain_data.irqdomain = NULL;
2242cc89c323SWoojung Huh }
2243cc89c323SWoojung Huh 
lan8835_fixup(struct phy_device * phydev)224402dc1f3dSWoojung Huh static int lan8835_fixup(struct phy_device *phydev)
224502dc1f3dSWoojung Huh {
224602dc1f3dSWoojung Huh 	int buf;
224702dc1f3dSWoojung Huh 	struct lan78xx_net *dev = netdev_priv(phydev->attached_dev);
224802dc1f3dSWoojung Huh 
224902dc1f3dSWoojung Huh 	/* LED2/PME_N/IRQ_N/RGMII_ID pin to IRQ_N mode */
22505f613677SRussell King 	buf = phy_read_mmd(phydev, MDIO_MMD_PCS, 0x8010);
225102dc1f3dSWoojung Huh 	buf &= ~0x1800;
225202dc1f3dSWoojung Huh 	buf |= 0x0800;
22535f613677SRussell King 	phy_write_mmd(phydev, MDIO_MMD_PCS, 0x8010, buf);
225402dc1f3dSWoojung Huh 
225502dc1f3dSWoojung Huh 	/* RGMII MAC TXC Delay Enable */
225606cd7c46SLee Jones 	lan78xx_write_reg(dev, MAC_RGMII_ID,
225702dc1f3dSWoojung Huh 			  MAC_RGMII_ID_TXC_DELAY_EN_);
225802dc1f3dSWoojung Huh 
225902dc1f3dSWoojung Huh 	/* RGMII TX DLL Tune Adjust */
226006cd7c46SLee Jones 	lan78xx_write_reg(dev, RGMII_TX_BYP_DLL, 0x3D00);
226102dc1f3dSWoojung Huh 
226202dc1f3dSWoojung Huh 	dev->interface = PHY_INTERFACE_MODE_RGMII_TXID;
226302dc1f3dSWoojung Huh 
226402dc1f3dSWoojung Huh 	return 1;
226502dc1f3dSWoojung Huh }
226602dc1f3dSWoojung Huh 
ksz9031rnx_fixup(struct phy_device * phydev)226702dc1f3dSWoojung Huh static int ksz9031rnx_fixup(struct phy_device *phydev)
226802dc1f3dSWoojung Huh {
226902dc1f3dSWoojung Huh 	struct lan78xx_net *dev = netdev_priv(phydev->attached_dev);
227002dc1f3dSWoojung Huh 
227102dc1f3dSWoojung Huh 	/* Micrel9301RNX PHY configuration */
227202dc1f3dSWoojung Huh 	/* RGMII Control Signal Pad Skew */
22735f613677SRussell King 	phy_write_mmd(phydev, MDIO_MMD_WIS, 4, 0x0077);
227402dc1f3dSWoojung Huh 	/* RGMII RX Data Pad Skew */
22755f613677SRussell King 	phy_write_mmd(phydev, MDIO_MMD_WIS, 5, 0x7777);
227602dc1f3dSWoojung Huh 	/* RGMII RX Clock Pad Skew */
22775f613677SRussell King 	phy_write_mmd(phydev, MDIO_MMD_WIS, 8, 0x1FF);
227802dc1f3dSWoojung Huh 
227902dc1f3dSWoojung Huh 	dev->interface = PHY_INTERFACE_MODE_RGMII_RXID;
228002dc1f3dSWoojung Huh 
228102dc1f3dSWoojung Huh 	return 1;
228202dc1f3dSWoojung Huh }
228302dc1f3dSWoojung Huh 
lan7801_phy_init(struct lan78xx_net * dev)228489b36fb5SRaghuram Chary J static struct phy_device *lan7801_phy_init(struct lan78xx_net *dev)
228555d7de9dSWoojung.Huh@microchip.com {
228689b36fb5SRaghuram Chary J 	u32 buf;
2287ce85e13aSWoojung.Huh@microchip.com 	int ret;
228889b36fb5SRaghuram Chary J 	struct fixed_phy_status fphy_status = {
228989b36fb5SRaghuram Chary J 		.link = 1,
229089b36fb5SRaghuram Chary J 		.speed = SPEED_1000,
229189b36fb5SRaghuram Chary J 		.duplex = DUPLEX_FULL,
229289b36fb5SRaghuram Chary J 	};
22933b51cc75SColin Ian King 	struct phy_device *phydev;
229455d7de9dSWoojung.Huh@microchip.com 
2295ce85e13aSWoojung.Huh@microchip.com 	phydev = phy_find_first(dev->mdiobus);
2296ce85e13aSWoojung.Huh@microchip.com 	if (!phydev) {
229789b36fb5SRaghuram Chary J 		netdev_dbg(dev->net, "PHY Not Found!! Registering Fixed PHY\n");
22985468e82fSLinus Walleij 		phydev = fixed_phy_register(PHY_POLL, &fphy_status, NULL);
229989b36fb5SRaghuram Chary J 		if (IS_ERR(phydev)) {
230089b36fb5SRaghuram Chary J 			netdev_err(dev->net, "No PHY/fixed_PHY found\n");
230189b36fb5SRaghuram Chary J 			return NULL;
2302ce85e13aSWoojung.Huh@microchip.com 		}
230389b36fb5SRaghuram Chary J 		netdev_dbg(dev->net, "Registered FIXED PHY\n");
230489b36fb5SRaghuram Chary J 		dev->interface = PHY_INTERFACE_MODE_RGMII;
230589b36fb5SRaghuram Chary J 		ret = lan78xx_write_reg(dev, MAC_RGMII_ID,
230689b36fb5SRaghuram Chary J 					MAC_RGMII_ID_TXC_DELAY_EN_);
230789b36fb5SRaghuram Chary J 		ret = lan78xx_write_reg(dev, RGMII_TX_BYP_DLL, 0x3D00);
230889b36fb5SRaghuram Chary J 		ret = lan78xx_read_reg(dev, HW_CFG, &buf);
230989b36fb5SRaghuram Chary J 		buf |= HW_CFG_CLK125_EN_;
231089b36fb5SRaghuram Chary J 		buf |= HW_CFG_REFCLK25_EN_;
231189b36fb5SRaghuram Chary J 		ret = lan78xx_write_reg(dev, HW_CFG, buf);
231289b36fb5SRaghuram Chary J 	} else {
231302dc1f3dSWoojung Huh 		if (!phydev->drv) {
231402dc1f3dSWoojung Huh 			netdev_err(dev->net, "no PHY driver found\n");
231589b36fb5SRaghuram Chary J 			return NULL;
231602dc1f3dSWoojung Huh 		}
231702dc1f3dSWoojung Huh 		dev->interface = PHY_INTERFACE_MODE_RGMII;
231802dc1f3dSWoojung Huh 		/* external PHY fixup for KSZ9031RNX */
231902dc1f3dSWoojung Huh 		ret = phy_register_fixup_for_uid(PHY_KSZ9031RNX, 0xfffffff0,
232002dc1f3dSWoojung Huh 						 ksz9031rnx_fixup);
232102dc1f3dSWoojung Huh 		if (ret < 0) {
23227670ed7aSRaghuram Chary J 			netdev_err(dev->net, "Failed to register fixup for PHY_KSZ9031RNX\n");
232389b36fb5SRaghuram Chary J 			return NULL;
232402dc1f3dSWoojung Huh 		}
232502dc1f3dSWoojung Huh 		/* external PHY fixup for LAN8835 */
232602dc1f3dSWoojung Huh 		ret = phy_register_fixup_for_uid(PHY_LAN8835, 0xfffffff0,
232702dc1f3dSWoojung Huh 						 lan8835_fixup);
232802dc1f3dSWoojung Huh 		if (ret < 0) {
23297670ed7aSRaghuram Chary J 			netdev_err(dev->net, "Failed to register fixup for PHY_LAN8835\n");
233089b36fb5SRaghuram Chary J 			return NULL;
233102dc1f3dSWoojung Huh 		}
233202dc1f3dSWoojung Huh 		/* add more external PHY fixup here if needed */
233302dc1f3dSWoojung Huh 
233402dc1f3dSWoojung Huh 		phydev->is_internal = false;
233589b36fb5SRaghuram Chary J 	}
233689b36fb5SRaghuram Chary J 	return phydev;
233789b36fb5SRaghuram Chary J }
233889b36fb5SRaghuram Chary J 
lan78xx_phy_init(struct lan78xx_net * dev)233989b36fb5SRaghuram Chary J static int lan78xx_phy_init(struct lan78xx_net *dev)
234089b36fb5SRaghuram Chary J {
23413c1bcc86SAndrew Lunn 	__ETHTOOL_DECLARE_LINK_MODE_MASK(fc) = { 0, };
234289b36fb5SRaghuram Chary J 	int ret;
234389b36fb5SRaghuram Chary J 	u32 mii_adv;
234489b36fb5SRaghuram Chary J 	struct phy_device *phydev;
234589b36fb5SRaghuram Chary J 
234689b36fb5SRaghuram Chary J 	switch (dev->chipid) {
234789b36fb5SRaghuram Chary J 	case ID_REV_CHIP_ID_7801_:
234889b36fb5SRaghuram Chary J 		phydev = lan7801_phy_init(dev);
234989b36fb5SRaghuram Chary J 		if (!phydev) {
235089b36fb5SRaghuram Chary J 			netdev_err(dev->net, "lan7801: PHY Init Failed");
235189b36fb5SRaghuram Chary J 			return -EIO;
235289b36fb5SRaghuram Chary J 		}
235389b36fb5SRaghuram Chary J 		break;
235489b36fb5SRaghuram Chary J 
235589b36fb5SRaghuram Chary J 	case ID_REV_CHIP_ID_7800_:
235689b36fb5SRaghuram Chary J 	case ID_REV_CHIP_ID_7850_:
235789b36fb5SRaghuram Chary J 		phydev = phy_find_first(dev->mdiobus);
235889b36fb5SRaghuram Chary J 		if (!phydev) {
235989b36fb5SRaghuram Chary J 			netdev_err(dev->net, "no PHY found\n");
236089b36fb5SRaghuram Chary J 			return -EIO;
236189b36fb5SRaghuram Chary J 		}
236289b36fb5SRaghuram Chary J 		phydev->is_internal = true;
236389b36fb5SRaghuram Chary J 		dev->interface = PHY_INTERFACE_MODE_GMII;
236489b36fb5SRaghuram Chary J 		break;
236589b36fb5SRaghuram Chary J 
236689b36fb5SRaghuram Chary J 	default:
236789b36fb5SRaghuram Chary J 		netdev_err(dev->net, "Unknown CHIP ID found\n");
236889b36fb5SRaghuram Chary J 		return -EIO;
236902dc1f3dSWoojung Huh 	}
237002dc1f3dSWoojung Huh 
2371cc89c323SWoojung Huh 	/* if phyirq is not set, use polling mode in phylib */
2372cc89c323SWoojung Huh 	if (dev->domain_data.phyirq > 0)
2373cc89c323SWoojung Huh 		phydev->irq = dev->domain_data.phyirq;
2374cc89c323SWoojung Huh 	else
2375817b6531SSven Schuchmann 		phydev->irq = PHY_POLL;
2376cc89c323SWoojung Huh 	netdev_dbg(dev->net, "phydev->irq = %d\n", phydev->irq);
2377e4953910SWoojung.Huh@microchip.com 
2378f6e3ef3eSWoojung Huh 	/* set to AUTOMDIX */
2379f6e3ef3eSWoojung Huh 	phydev->mdix = ETH_TP_MDI_AUTO;
2380f6e3ef3eSWoojung Huh 
2381ce85e13aSWoojung.Huh@microchip.com 	ret = phy_connect_direct(dev->net, phydev,
2382ce85e13aSWoojung.Huh@microchip.com 				 lan78xx_link_status_change,
238302dc1f3dSWoojung Huh 				 dev->interface);
2384ce85e13aSWoojung.Huh@microchip.com 	if (ret) {
2385ce85e13aSWoojung.Huh@microchip.com 		netdev_err(dev->net, "can't attach PHY to %s\n",
2386ce85e13aSWoojung.Huh@microchip.com 			   dev->mdiobus->id);
238789b36fb5SRaghuram Chary J 		if (dev->chipid == ID_REV_CHIP_ID_7801_) {
238889b36fb5SRaghuram Chary J 			if (phy_is_pseudo_fixed_link(phydev)) {
238989b36fb5SRaghuram Chary J 				fixed_phy_unregister(phydev);
2390bca2c418SOleksij Rempel 				phy_device_free(phydev);
239189b36fb5SRaghuram Chary J 			} else {
239289b36fb5SRaghuram Chary J 				phy_unregister_fixup_for_uid(PHY_KSZ9031RNX,
239389b36fb5SRaghuram Chary J 							     0xfffffff0);
239489b36fb5SRaghuram Chary J 				phy_unregister_fixup_for_uid(PHY_LAN8835,
239589b36fb5SRaghuram Chary J 							     0xfffffff0);
239689b36fb5SRaghuram Chary J 			}
239789b36fb5SRaghuram Chary J 		}
2398ce85e13aSWoojung.Huh@microchip.com 		return -EIO;
2399ce85e13aSWoojung.Huh@microchip.com 	}
240055d7de9dSWoojung.Huh@microchip.com 
2401ce85e13aSWoojung.Huh@microchip.com 	/* MAC doesn't support 1000T Half */
240241124fa6SAndrew Lunn 	phy_remove_link_mode(phydev, ETHTOOL_LINK_MODE_1000baseT_Half_BIT);
2403e270b2dbSWoojung.Huh@microchip.com 
2404349e0c5eSWoojung.Huh@microchip.com 	/* support both flow controls */
2405349e0c5eSWoojung.Huh@microchip.com 	dev->fc_request_control = (FLOW_CTRL_RX | FLOW_CTRL_TX);
24063c1bcc86SAndrew Lunn 	linkmode_clear_bit(ETHTOOL_LINK_MODE_Pause_BIT,
24073c1bcc86SAndrew Lunn 			   phydev->advertising);
24083c1bcc86SAndrew Lunn 	linkmode_clear_bit(ETHTOOL_LINK_MODE_Asym_Pause_BIT,
24093c1bcc86SAndrew Lunn 			   phydev->advertising);
2410349e0c5eSWoojung.Huh@microchip.com 	mii_adv = (u32)mii_advertise_flowctrl(dev->fc_request_control);
24113c1bcc86SAndrew Lunn 	mii_adv_to_linkmode_adv_t(fc, mii_adv);
24123c1bcc86SAndrew Lunn 	linkmode_or(phydev->advertising, fc, phydev->advertising);
2413349e0c5eSWoojung.Huh@microchip.com 
24141827b067SPhil Elwell 	if (phydev->mdio.dev.of_node) {
24151827b067SPhil Elwell 		u32 reg;
24161827b067SPhil Elwell 		int len;
24171827b067SPhil Elwell 
24181827b067SPhil Elwell 		len = of_property_count_elems_of_size(phydev->mdio.dev.of_node,
24191827b067SPhil Elwell 						      "microchip,led-modes",
24201827b067SPhil Elwell 						      sizeof(u32));
24211827b067SPhil Elwell 		if (len >= 0) {
24221827b067SPhil Elwell 			/* Ensure the appropriate LEDs are enabled */
24231827b067SPhil Elwell 			lan78xx_read_reg(dev, HW_CFG, &reg);
24241827b067SPhil Elwell 			reg &= ~(HW_CFG_LED0_EN_ |
24251827b067SPhil Elwell 				 HW_CFG_LED1_EN_ |
24261827b067SPhil Elwell 				 HW_CFG_LED2_EN_ |
24271827b067SPhil Elwell 				 HW_CFG_LED3_EN_);
24281827b067SPhil Elwell 			reg |= (len > 0) * HW_CFG_LED0_EN_ |
24291827b067SPhil Elwell 				(len > 1) * HW_CFG_LED1_EN_ |
24301827b067SPhil Elwell 				(len > 2) * HW_CFG_LED2_EN_ |
24311827b067SPhil Elwell 				(len > 3) * HW_CFG_LED3_EN_;
24321827b067SPhil Elwell 			lan78xx_write_reg(dev, HW_CFG, reg);
24331827b067SPhil Elwell 		}
24341827b067SPhil Elwell 	}
24351827b067SPhil Elwell 
2436ce85e13aSWoojung.Huh@microchip.com 	genphy_config_aneg(phydev);
243755d7de9dSWoojung.Huh@microchip.com 
2438349e0c5eSWoojung.Huh@microchip.com 	dev->fc_autoneg = phydev->autoneg;
2439349e0c5eSWoojung.Huh@microchip.com 
244055d7de9dSWoojung.Huh@microchip.com 	return 0;
244155d7de9dSWoojung.Huh@microchip.com }
244255d7de9dSWoojung.Huh@microchip.com 
lan78xx_set_rx_max_frame_length(struct lan78xx_net * dev,int size)244355d7de9dSWoojung.Huh@microchip.com static int lan78xx_set_rx_max_frame_length(struct lan78xx_net *dev, int size)
244455d7de9dSWoojung.Huh@microchip.com {
244555d7de9dSWoojung.Huh@microchip.com 	u32 buf;
244655d7de9dSWoojung.Huh@microchip.com 	bool rxenabled;
244755d7de9dSWoojung.Huh@microchip.com 
244806cd7c46SLee Jones 	lan78xx_read_reg(dev, MAC_RX, &buf);
244955d7de9dSWoojung.Huh@microchip.com 
245055d7de9dSWoojung.Huh@microchip.com 	rxenabled = ((buf & MAC_RX_RXEN_) != 0);
245155d7de9dSWoojung.Huh@microchip.com 
245255d7de9dSWoojung.Huh@microchip.com 	if (rxenabled) {
245355d7de9dSWoojung.Huh@microchip.com 		buf &= ~MAC_RX_RXEN_;
245406cd7c46SLee Jones 		lan78xx_write_reg(dev, MAC_RX, buf);
245555d7de9dSWoojung.Huh@microchip.com 	}
245655d7de9dSWoojung.Huh@microchip.com 
245755d7de9dSWoojung.Huh@microchip.com 	/* add 4 to size for FCS */
245855d7de9dSWoojung.Huh@microchip.com 	buf &= ~MAC_RX_MAX_SIZE_MASK_;
245955d7de9dSWoojung.Huh@microchip.com 	buf |= (((size + 4) << MAC_RX_MAX_SIZE_SHIFT_) & MAC_RX_MAX_SIZE_MASK_);
246055d7de9dSWoojung.Huh@microchip.com 
246106cd7c46SLee Jones 	lan78xx_write_reg(dev, MAC_RX, buf);
246255d7de9dSWoojung.Huh@microchip.com 
246355d7de9dSWoojung.Huh@microchip.com 	if (rxenabled) {
246455d7de9dSWoojung.Huh@microchip.com 		buf |= MAC_RX_RXEN_;
246506cd7c46SLee Jones 		lan78xx_write_reg(dev, MAC_RX, buf);
246655d7de9dSWoojung.Huh@microchip.com 	}
246755d7de9dSWoojung.Huh@microchip.com 
246855d7de9dSWoojung.Huh@microchip.com 	return 0;
246955d7de9dSWoojung.Huh@microchip.com }
247055d7de9dSWoojung.Huh@microchip.com 
unlink_urbs(struct lan78xx_net * dev,struct sk_buff_head * q)247155d7de9dSWoojung.Huh@microchip.com static int unlink_urbs(struct lan78xx_net *dev, struct sk_buff_head *q)
247255d7de9dSWoojung.Huh@microchip.com {
247355d7de9dSWoojung.Huh@microchip.com 	struct sk_buff *skb;
247455d7de9dSWoojung.Huh@microchip.com 	unsigned long flags;
247555d7de9dSWoojung.Huh@microchip.com 	int count = 0;
247655d7de9dSWoojung.Huh@microchip.com 
247755d7de9dSWoojung.Huh@microchip.com 	spin_lock_irqsave(&q->lock, flags);
247855d7de9dSWoojung.Huh@microchip.com 	while (!skb_queue_empty(q)) {
247955d7de9dSWoojung.Huh@microchip.com 		struct skb_data	*entry;
248055d7de9dSWoojung.Huh@microchip.com 		struct urb *urb;
248155d7de9dSWoojung.Huh@microchip.com 		int ret;
248255d7de9dSWoojung.Huh@microchip.com 
248355d7de9dSWoojung.Huh@microchip.com 		skb_queue_walk(q, skb) {
248455d7de9dSWoojung.Huh@microchip.com 			entry = (struct skb_data *)skb->cb;
248555d7de9dSWoojung.Huh@microchip.com 			if (entry->state != unlink_start)
248655d7de9dSWoojung.Huh@microchip.com 				goto found;
248755d7de9dSWoojung.Huh@microchip.com 		}
248855d7de9dSWoojung.Huh@microchip.com 		break;
248955d7de9dSWoojung.Huh@microchip.com found:
249055d7de9dSWoojung.Huh@microchip.com 		entry->state = unlink_start;
249155d7de9dSWoojung.Huh@microchip.com 		urb = entry->urb;
249255d7de9dSWoojung.Huh@microchip.com 
249355d7de9dSWoojung.Huh@microchip.com 		/* Get reference count of the URB to avoid it to be
249455d7de9dSWoojung.Huh@microchip.com 		 * freed during usb_unlink_urb, which may trigger
249555d7de9dSWoojung.Huh@microchip.com 		 * use-after-free problem inside usb_unlink_urb since
249655d7de9dSWoojung.Huh@microchip.com 		 * usb_unlink_urb is always racing with .complete
249755d7de9dSWoojung.Huh@microchip.com 		 * handler(include defer_bh).
249855d7de9dSWoojung.Huh@microchip.com 		 */
249955d7de9dSWoojung.Huh@microchip.com 		usb_get_urb(urb);
250055d7de9dSWoojung.Huh@microchip.com 		spin_unlock_irqrestore(&q->lock, flags);
250155d7de9dSWoojung.Huh@microchip.com 		/* during some PM-driven resume scenarios,
250255d7de9dSWoojung.Huh@microchip.com 		 * these (async) unlinks complete immediately
250355d7de9dSWoojung.Huh@microchip.com 		 */
250455d7de9dSWoojung.Huh@microchip.com 		ret = usb_unlink_urb(urb);
250555d7de9dSWoojung.Huh@microchip.com 		if (ret != -EINPROGRESS && ret != 0)
250655d7de9dSWoojung.Huh@microchip.com 			netdev_dbg(dev->net, "unlink urb err, %d\n", ret);
250755d7de9dSWoojung.Huh@microchip.com 		else
250855d7de9dSWoojung.Huh@microchip.com 			count++;
250955d7de9dSWoojung.Huh@microchip.com 		usb_put_urb(urb);
251055d7de9dSWoojung.Huh@microchip.com 		spin_lock_irqsave(&q->lock, flags);
251155d7de9dSWoojung.Huh@microchip.com 	}
251255d7de9dSWoojung.Huh@microchip.com 	spin_unlock_irqrestore(&q->lock, flags);
251355d7de9dSWoojung.Huh@microchip.com 	return count;
251455d7de9dSWoojung.Huh@microchip.com }
251555d7de9dSWoojung.Huh@microchip.com 
lan78xx_change_mtu(struct net_device * netdev,int new_mtu)251655d7de9dSWoojung.Huh@microchip.com static int lan78xx_change_mtu(struct net_device *netdev, int new_mtu)
251755d7de9dSWoojung.Huh@microchip.com {
251855d7de9dSWoojung.Huh@microchip.com 	struct lan78xx_net *dev = netdev_priv(netdev);
25190dd87266SJohn Efstathiades 	int max_frame_len = RX_MAX_FRAME_LEN(new_mtu);
25205f4cc6e2SJohn Efstathiades 	int ret;
252155d7de9dSWoojung.Huh@microchip.com 
252255d7de9dSWoojung.Huh@microchip.com 	/* no second zero-length packet read wanted after mtu-sized packets */
25230dd87266SJohn Efstathiades 	if ((max_frame_len % dev->maxpacket) == 0)
252455d7de9dSWoojung.Huh@microchip.com 		return -EDOM;
252555d7de9dSWoojung.Huh@microchip.com 
25265f4cc6e2SJohn Efstathiades 	ret = usb_autopm_get_interface(dev->intf);
25275f4cc6e2SJohn Efstathiades 	if (ret < 0)
25285f4cc6e2SJohn Efstathiades 		return ret;
25295f4cc6e2SJohn Efstathiades 
25300dd87266SJohn Efstathiades 	ret = lan78xx_set_rx_max_frame_length(dev, max_frame_len);
25310dd87266SJohn Efstathiades 	if (!ret)
253255d7de9dSWoojung.Huh@microchip.com 		netdev->mtu = new_mtu;
253355d7de9dSWoojung.Huh@microchip.com 
25345f4cc6e2SJohn Efstathiades 	usb_autopm_put_interface(dev->intf);
25355f4cc6e2SJohn Efstathiades 
25360dd87266SJohn Efstathiades 	return ret;
253755d7de9dSWoojung.Huh@microchip.com }
253855d7de9dSWoojung.Huh@microchip.com 
lan78xx_set_mac_addr(struct net_device * netdev,void * p)2539e0c79ff6SBaoyou Xie static int lan78xx_set_mac_addr(struct net_device *netdev, void *p)
254055d7de9dSWoojung.Huh@microchip.com {
254155d7de9dSWoojung.Huh@microchip.com 	struct lan78xx_net *dev = netdev_priv(netdev);
254255d7de9dSWoojung.Huh@microchip.com 	struct sockaddr *addr = p;
254355d7de9dSWoojung.Huh@microchip.com 	u32 addr_lo, addr_hi;
254455d7de9dSWoojung.Huh@microchip.com 
254555d7de9dSWoojung.Huh@microchip.com 	if (netif_running(netdev))
254655d7de9dSWoojung.Huh@microchip.com 		return -EBUSY;
254755d7de9dSWoojung.Huh@microchip.com 
254855d7de9dSWoojung.Huh@microchip.com 	if (!is_valid_ether_addr(addr->sa_data))
254955d7de9dSWoojung.Huh@microchip.com 		return -EADDRNOTAVAIL;
255055d7de9dSWoojung.Huh@microchip.com 
2551af804e6dSJakub Kicinski 	eth_hw_addr_set(netdev, addr->sa_data);
255255d7de9dSWoojung.Huh@microchip.com 
255355d7de9dSWoojung.Huh@microchip.com 	addr_lo = netdev->dev_addr[0] |
255455d7de9dSWoojung.Huh@microchip.com 		  netdev->dev_addr[1] << 8 |
255555d7de9dSWoojung.Huh@microchip.com 		  netdev->dev_addr[2] << 16 |
255655d7de9dSWoojung.Huh@microchip.com 		  netdev->dev_addr[3] << 24;
255755d7de9dSWoojung.Huh@microchip.com 	addr_hi = netdev->dev_addr[4] |
255855d7de9dSWoojung.Huh@microchip.com 		  netdev->dev_addr[5] << 8;
255955d7de9dSWoojung.Huh@microchip.com 
256006cd7c46SLee Jones 	lan78xx_write_reg(dev, RX_ADDRL, addr_lo);
256106cd7c46SLee Jones 	lan78xx_write_reg(dev, RX_ADDRH, addr_hi);
256255d7de9dSWoojung.Huh@microchip.com 
256315515aaaSJason Martinsen 	/* Added to support MAC address changes */
256406cd7c46SLee Jones 	lan78xx_write_reg(dev, MAF_LO(0), addr_lo);
256506cd7c46SLee Jones 	lan78xx_write_reg(dev, MAF_HI(0), addr_hi | MAF_HI_VALID_);
256615515aaaSJason Martinsen 
256755d7de9dSWoojung.Huh@microchip.com 	return 0;
256855d7de9dSWoojung.Huh@microchip.com }
256955d7de9dSWoojung.Huh@microchip.com 
257055d7de9dSWoojung.Huh@microchip.com /* Enable or disable Rx checksum offload engine */
lan78xx_set_features(struct net_device * netdev,netdev_features_t features)257155d7de9dSWoojung.Huh@microchip.com static int lan78xx_set_features(struct net_device *netdev,
257255d7de9dSWoojung.Huh@microchip.com 				netdev_features_t features)
257355d7de9dSWoojung.Huh@microchip.com {
257455d7de9dSWoojung.Huh@microchip.com 	struct lan78xx_net *dev = netdev_priv(netdev);
257555d7de9dSWoojung.Huh@microchip.com 	struct lan78xx_priv *pdata = (struct lan78xx_priv *)(dev->data[0]);
257655d7de9dSWoojung.Huh@microchip.com 	unsigned long flags;
257755d7de9dSWoojung.Huh@microchip.com 
257855d7de9dSWoojung.Huh@microchip.com 	spin_lock_irqsave(&pdata->rfe_ctl_lock, flags);
257955d7de9dSWoojung.Huh@microchip.com 
258055d7de9dSWoojung.Huh@microchip.com 	if (features & NETIF_F_RXCSUM) {
258155d7de9dSWoojung.Huh@microchip.com 		pdata->rfe_ctl |= RFE_CTL_TCPUDP_COE_ | RFE_CTL_IP_COE_;
258255d7de9dSWoojung.Huh@microchip.com 		pdata->rfe_ctl |= RFE_CTL_ICMP_COE_ | RFE_CTL_IGMP_COE_;
258355d7de9dSWoojung.Huh@microchip.com 	} else {
258455d7de9dSWoojung.Huh@microchip.com 		pdata->rfe_ctl &= ~(RFE_CTL_TCPUDP_COE_ | RFE_CTL_IP_COE_);
258555d7de9dSWoojung.Huh@microchip.com 		pdata->rfe_ctl &= ~(RFE_CTL_ICMP_COE_ | RFE_CTL_IGMP_COE_);
258655d7de9dSWoojung.Huh@microchip.com 	}
258755d7de9dSWoojung.Huh@microchip.com 
258855d7de9dSWoojung.Huh@microchip.com 	if (features & NETIF_F_HW_VLAN_CTAG_RX)
2589ec21ecf0SDave Stevenson 		pdata->rfe_ctl |= RFE_CTL_VLAN_STRIP_;
2590ec21ecf0SDave Stevenson 	else
2591ec21ecf0SDave Stevenson 		pdata->rfe_ctl &= ~RFE_CTL_VLAN_STRIP_;
2592ec21ecf0SDave Stevenson 
25934a27327bSDave Stevenson 	if (features & NETIF_F_HW_VLAN_CTAG_FILTER)
259455d7de9dSWoojung.Huh@microchip.com 		pdata->rfe_ctl |= RFE_CTL_VLAN_FILTER_;
259555d7de9dSWoojung.Huh@microchip.com 	else
259655d7de9dSWoojung.Huh@microchip.com 		pdata->rfe_ctl &= ~RFE_CTL_VLAN_FILTER_;
259755d7de9dSWoojung.Huh@microchip.com 
259855d7de9dSWoojung.Huh@microchip.com 	spin_unlock_irqrestore(&pdata->rfe_ctl_lock, flags);
259955d7de9dSWoojung.Huh@microchip.com 
260006cd7c46SLee Jones 	lan78xx_write_reg(dev, RFE_CTL, pdata->rfe_ctl);
260155d7de9dSWoojung.Huh@microchip.com 
260255d7de9dSWoojung.Huh@microchip.com 	return 0;
260355d7de9dSWoojung.Huh@microchip.com }
260455d7de9dSWoojung.Huh@microchip.com 
lan78xx_deferred_vlan_write(struct work_struct * param)260555d7de9dSWoojung.Huh@microchip.com static void lan78xx_deferred_vlan_write(struct work_struct *param)
260655d7de9dSWoojung.Huh@microchip.com {
260755d7de9dSWoojung.Huh@microchip.com 	struct lan78xx_priv *pdata =
260855d7de9dSWoojung.Huh@microchip.com 			container_of(param, struct lan78xx_priv, set_vlan);
260955d7de9dSWoojung.Huh@microchip.com 	struct lan78xx_net *dev = pdata->dev;
261055d7de9dSWoojung.Huh@microchip.com 
261155d7de9dSWoojung.Huh@microchip.com 	lan78xx_dataport_write(dev, DP_SEL_RSEL_VLAN_DA_, 0,
261255d7de9dSWoojung.Huh@microchip.com 			       DP_SEL_VHF_VLAN_LEN, pdata->vlan_table);
261355d7de9dSWoojung.Huh@microchip.com }
261455d7de9dSWoojung.Huh@microchip.com 
lan78xx_vlan_rx_add_vid(struct net_device * netdev,__be16 proto,u16 vid)261555d7de9dSWoojung.Huh@microchip.com static int lan78xx_vlan_rx_add_vid(struct net_device *netdev,
261655d7de9dSWoojung.Huh@microchip.com 				   __be16 proto, u16 vid)
261755d7de9dSWoojung.Huh@microchip.com {
261855d7de9dSWoojung.Huh@microchip.com 	struct lan78xx_net *dev = netdev_priv(netdev);
261955d7de9dSWoojung.Huh@microchip.com 	struct lan78xx_priv *pdata = (struct lan78xx_priv *)(dev->data[0]);
262055d7de9dSWoojung.Huh@microchip.com 	u16 vid_bit_index;
262155d7de9dSWoojung.Huh@microchip.com 	u16 vid_dword_index;
262255d7de9dSWoojung.Huh@microchip.com 
262355d7de9dSWoojung.Huh@microchip.com 	vid_dword_index = (vid >> 5) & 0x7F;
262455d7de9dSWoojung.Huh@microchip.com 	vid_bit_index = vid & 0x1F;
262555d7de9dSWoojung.Huh@microchip.com 
262655d7de9dSWoojung.Huh@microchip.com 	pdata->vlan_table[vid_dword_index] |= (1 << vid_bit_index);
262755d7de9dSWoojung.Huh@microchip.com 
262855d7de9dSWoojung.Huh@microchip.com 	/* defer register writes to a sleepable context */
262955d7de9dSWoojung.Huh@microchip.com 	schedule_work(&pdata->set_vlan);
263055d7de9dSWoojung.Huh@microchip.com 
263155d7de9dSWoojung.Huh@microchip.com 	return 0;
263255d7de9dSWoojung.Huh@microchip.com }
263355d7de9dSWoojung.Huh@microchip.com 
lan78xx_vlan_rx_kill_vid(struct net_device * netdev,__be16 proto,u16 vid)263455d7de9dSWoojung.Huh@microchip.com static int lan78xx_vlan_rx_kill_vid(struct net_device *netdev,
263555d7de9dSWoojung.Huh@microchip.com 				    __be16 proto, u16 vid)
263655d7de9dSWoojung.Huh@microchip.com {
263755d7de9dSWoojung.Huh@microchip.com 	struct lan78xx_net *dev = netdev_priv(netdev);
263855d7de9dSWoojung.Huh@microchip.com 	struct lan78xx_priv *pdata = (struct lan78xx_priv *)(dev->data[0]);
263955d7de9dSWoojung.Huh@microchip.com 	u16 vid_bit_index;
264055d7de9dSWoojung.Huh@microchip.com 	u16 vid_dword_index;
264155d7de9dSWoojung.Huh@microchip.com 
264255d7de9dSWoojung.Huh@microchip.com 	vid_dword_index = (vid >> 5) & 0x7F;
264355d7de9dSWoojung.Huh@microchip.com 	vid_bit_index = vid & 0x1F;
264455d7de9dSWoojung.Huh@microchip.com 
264555d7de9dSWoojung.Huh@microchip.com 	pdata->vlan_table[vid_dword_index] &= ~(1 << vid_bit_index);
264655d7de9dSWoojung.Huh@microchip.com 
264755d7de9dSWoojung.Huh@microchip.com 	/* defer register writes to a sleepable context */
264855d7de9dSWoojung.Huh@microchip.com 	schedule_work(&pdata->set_vlan);
264955d7de9dSWoojung.Huh@microchip.com 
265055d7de9dSWoojung.Huh@microchip.com 	return 0;
265155d7de9dSWoojung.Huh@microchip.com }
265255d7de9dSWoojung.Huh@microchip.com 
lan78xx_init_ltm(struct lan78xx_net * dev)265355d7de9dSWoojung.Huh@microchip.com static void lan78xx_init_ltm(struct lan78xx_net *dev)
265455d7de9dSWoojung.Huh@microchip.com {
265555d7de9dSWoojung.Huh@microchip.com 	int ret;
265655d7de9dSWoojung.Huh@microchip.com 	u32 buf;
265755d7de9dSWoojung.Huh@microchip.com 	u32 regs[6] = { 0 };
265855d7de9dSWoojung.Huh@microchip.com 
265955d7de9dSWoojung.Huh@microchip.com 	ret = lan78xx_read_reg(dev, USB_CFG1, &buf);
266055d7de9dSWoojung.Huh@microchip.com 	if (buf & USB_CFG1_LTM_ENABLE_) {
266155d7de9dSWoojung.Huh@microchip.com 		u8 temp[2];
266255d7de9dSWoojung.Huh@microchip.com 		/* Get values from EEPROM first */
266355d7de9dSWoojung.Huh@microchip.com 		if (lan78xx_read_eeprom(dev, 0x3F, 2, temp) == 0) {
266455d7de9dSWoojung.Huh@microchip.com 			if (temp[0] == 24) {
266555d7de9dSWoojung.Huh@microchip.com 				ret = lan78xx_read_raw_eeprom(dev,
266655d7de9dSWoojung.Huh@microchip.com 							      temp[1] * 2,
266755d7de9dSWoojung.Huh@microchip.com 							      24,
266855d7de9dSWoojung.Huh@microchip.com 							      (u8 *)regs);
266955d7de9dSWoojung.Huh@microchip.com 				if (ret < 0)
267055d7de9dSWoojung.Huh@microchip.com 					return;
267155d7de9dSWoojung.Huh@microchip.com 			}
267255d7de9dSWoojung.Huh@microchip.com 		} else if (lan78xx_read_otp(dev, 0x3F, 2, temp) == 0) {
267355d7de9dSWoojung.Huh@microchip.com 			if (temp[0] == 24) {
267455d7de9dSWoojung.Huh@microchip.com 				ret = lan78xx_read_raw_otp(dev,
267555d7de9dSWoojung.Huh@microchip.com 							   temp[1] * 2,
267655d7de9dSWoojung.Huh@microchip.com 							   24,
267755d7de9dSWoojung.Huh@microchip.com 							   (u8 *)regs);
267855d7de9dSWoojung.Huh@microchip.com 				if (ret < 0)
267955d7de9dSWoojung.Huh@microchip.com 					return;
268055d7de9dSWoojung.Huh@microchip.com 			}
268155d7de9dSWoojung.Huh@microchip.com 		}
268255d7de9dSWoojung.Huh@microchip.com 	}
268355d7de9dSWoojung.Huh@microchip.com 
268455d7de9dSWoojung.Huh@microchip.com 	lan78xx_write_reg(dev, LTM_BELT_IDLE0, regs[0]);
268555d7de9dSWoojung.Huh@microchip.com 	lan78xx_write_reg(dev, LTM_BELT_IDLE1, regs[1]);
268655d7de9dSWoojung.Huh@microchip.com 	lan78xx_write_reg(dev, LTM_BELT_ACT0, regs[2]);
268755d7de9dSWoojung.Huh@microchip.com 	lan78xx_write_reg(dev, LTM_BELT_ACT1, regs[3]);
268855d7de9dSWoojung.Huh@microchip.com 	lan78xx_write_reg(dev, LTM_INACTIVE0, regs[4]);
268955d7de9dSWoojung.Huh@microchip.com 	lan78xx_write_reg(dev, LTM_INACTIVE1, regs[5]);
269055d7de9dSWoojung.Huh@microchip.com }
269155d7de9dSWoojung.Huh@microchip.com 
lan78xx_urb_config_init(struct lan78xx_net * dev)2692d383216aSJohn Efstathiades static int lan78xx_urb_config_init(struct lan78xx_net *dev)
2693d383216aSJohn Efstathiades {
2694d383216aSJohn Efstathiades 	int result = 0;
2695d383216aSJohn Efstathiades 
2696d383216aSJohn Efstathiades 	switch (dev->udev->speed) {
2697d383216aSJohn Efstathiades 	case USB_SPEED_SUPER:
2698c450a8ebSJohn Efstathiades 		dev->rx_urb_size = RX_SS_URB_SIZE;
2699d383216aSJohn Efstathiades 		dev->tx_urb_size = TX_SS_URB_SIZE;
2700c450a8ebSJohn Efstathiades 		dev->n_rx_urbs = RX_SS_URB_NUM;
2701d383216aSJohn Efstathiades 		dev->n_tx_urbs = TX_SS_URB_NUM;
2702c450a8ebSJohn Efstathiades 		dev->bulk_in_delay = SS_BULK_IN_DELAY;
2703c450a8ebSJohn Efstathiades 		dev->burst_cap = SS_BURST_CAP_SIZE / SS_USB_PKT_SIZE;
2704d383216aSJohn Efstathiades 		break;
2705d383216aSJohn Efstathiades 	case USB_SPEED_HIGH:
2706c450a8ebSJohn Efstathiades 		dev->rx_urb_size = RX_HS_URB_SIZE;
2707d383216aSJohn Efstathiades 		dev->tx_urb_size = TX_HS_URB_SIZE;
2708c450a8ebSJohn Efstathiades 		dev->n_rx_urbs = RX_HS_URB_NUM;
2709d383216aSJohn Efstathiades 		dev->n_tx_urbs = TX_HS_URB_NUM;
2710c450a8ebSJohn Efstathiades 		dev->bulk_in_delay = HS_BULK_IN_DELAY;
2711c450a8ebSJohn Efstathiades 		dev->burst_cap = HS_BURST_CAP_SIZE / HS_USB_PKT_SIZE;
2712d383216aSJohn Efstathiades 		break;
2713d383216aSJohn Efstathiades 	case USB_SPEED_FULL:
2714c450a8ebSJohn Efstathiades 		dev->rx_urb_size = RX_FS_URB_SIZE;
2715d383216aSJohn Efstathiades 		dev->tx_urb_size = TX_FS_URB_SIZE;
2716c450a8ebSJohn Efstathiades 		dev->n_rx_urbs = RX_FS_URB_NUM;
2717d383216aSJohn Efstathiades 		dev->n_tx_urbs = TX_FS_URB_NUM;
2718c450a8ebSJohn Efstathiades 		dev->bulk_in_delay = FS_BULK_IN_DELAY;
2719c450a8ebSJohn Efstathiades 		dev->burst_cap = FS_BURST_CAP_SIZE / FS_USB_PKT_SIZE;
2720d383216aSJohn Efstathiades 		break;
2721d383216aSJohn Efstathiades 	default:
2722d383216aSJohn Efstathiades 		netdev_warn(dev->net, "USB bus speed not supported\n");
2723d383216aSJohn Efstathiades 		result = -EIO;
2724d383216aSJohn Efstathiades 		break;
2725d383216aSJohn Efstathiades 	}
2726d383216aSJohn Efstathiades 
2727d383216aSJohn Efstathiades 	return result;
2728d383216aSJohn Efstathiades }
2729d383216aSJohn Efstathiades 
lan78xx_start_hw(struct lan78xx_net * dev,u32 reg,u32 hw_enable)2730e1210fe6SJohn Efstathiades static int lan78xx_start_hw(struct lan78xx_net *dev, u32 reg, u32 hw_enable)
2731e1210fe6SJohn Efstathiades {
2732e1210fe6SJohn Efstathiades 	return lan78xx_update_reg(dev, reg, hw_enable, hw_enable);
2733e1210fe6SJohn Efstathiades }
2734e1210fe6SJohn Efstathiades 
lan78xx_stop_hw(struct lan78xx_net * dev,u32 reg,u32 hw_enabled,u32 hw_disabled)2735e1210fe6SJohn Efstathiades static int lan78xx_stop_hw(struct lan78xx_net *dev, u32 reg, u32 hw_enabled,
2736e1210fe6SJohn Efstathiades 			   u32 hw_disabled)
2737e1210fe6SJohn Efstathiades {
2738e1210fe6SJohn Efstathiades 	unsigned long timeout;
2739e1210fe6SJohn Efstathiades 	bool stopped = true;
2740e1210fe6SJohn Efstathiades 	int ret;
2741e1210fe6SJohn Efstathiades 	u32 buf;
2742e1210fe6SJohn Efstathiades 
2743e1210fe6SJohn Efstathiades 	/* Stop the h/w block (if not already stopped) */
2744e1210fe6SJohn Efstathiades 
2745e1210fe6SJohn Efstathiades 	ret = lan78xx_read_reg(dev, reg, &buf);
2746e1210fe6SJohn Efstathiades 	if (ret < 0)
2747e1210fe6SJohn Efstathiades 		return ret;
2748e1210fe6SJohn Efstathiades 
2749e1210fe6SJohn Efstathiades 	if (buf & hw_enabled) {
2750e1210fe6SJohn Efstathiades 		buf &= ~hw_enabled;
2751e1210fe6SJohn Efstathiades 
2752e1210fe6SJohn Efstathiades 		ret = lan78xx_write_reg(dev, reg, buf);
2753e1210fe6SJohn Efstathiades 		if (ret < 0)
2754e1210fe6SJohn Efstathiades 			return ret;
2755e1210fe6SJohn Efstathiades 
2756e1210fe6SJohn Efstathiades 		stopped = false;
2757e1210fe6SJohn Efstathiades 		timeout = jiffies + HW_DISABLE_TIMEOUT;
2758e1210fe6SJohn Efstathiades 		do  {
2759e1210fe6SJohn Efstathiades 			ret = lan78xx_read_reg(dev, reg, &buf);
2760e1210fe6SJohn Efstathiades 			if (ret < 0)
2761e1210fe6SJohn Efstathiades 				return ret;
2762e1210fe6SJohn Efstathiades 
2763e1210fe6SJohn Efstathiades 			if (buf & hw_disabled)
2764e1210fe6SJohn Efstathiades 				stopped = true;
2765e1210fe6SJohn Efstathiades 			else
2766e1210fe6SJohn Efstathiades 				msleep(HW_DISABLE_DELAY_MS);
2767e1210fe6SJohn Efstathiades 		} while (!stopped && !time_after(jiffies, timeout));
2768e1210fe6SJohn Efstathiades 	}
2769e1210fe6SJohn Efstathiades 
2770e1210fe6SJohn Efstathiades 	ret = stopped ? 0 : -ETIME;
2771e1210fe6SJohn Efstathiades 
2772e1210fe6SJohn Efstathiades 	return ret;
2773e1210fe6SJohn Efstathiades }
2774e1210fe6SJohn Efstathiades 
lan78xx_flush_fifo(struct lan78xx_net * dev,u32 reg,u32 fifo_flush)2775e1210fe6SJohn Efstathiades static int lan78xx_flush_fifo(struct lan78xx_net *dev, u32 reg, u32 fifo_flush)
2776e1210fe6SJohn Efstathiades {
2777e1210fe6SJohn Efstathiades 	return lan78xx_update_reg(dev, reg, fifo_flush, fifo_flush);
2778e1210fe6SJohn Efstathiades }
2779e1210fe6SJohn Efstathiades 
lan78xx_start_tx_path(struct lan78xx_net * dev)2780e1210fe6SJohn Efstathiades static int lan78xx_start_tx_path(struct lan78xx_net *dev)
2781e1210fe6SJohn Efstathiades {
2782e1210fe6SJohn Efstathiades 	int ret;
2783e1210fe6SJohn Efstathiades 
2784e1210fe6SJohn Efstathiades 	netif_dbg(dev, drv, dev->net, "start tx path");
2785e1210fe6SJohn Efstathiades 
2786e1210fe6SJohn Efstathiades 	/* Start the MAC transmitter */
2787e1210fe6SJohn Efstathiades 
2788e1210fe6SJohn Efstathiades 	ret = lan78xx_start_hw(dev, MAC_TX, MAC_TX_TXEN_);
2789e1210fe6SJohn Efstathiades 	if (ret < 0)
2790e1210fe6SJohn Efstathiades 		return ret;
2791e1210fe6SJohn Efstathiades 
2792e1210fe6SJohn Efstathiades 	/* Start the Tx FIFO */
2793e1210fe6SJohn Efstathiades 
2794e1210fe6SJohn Efstathiades 	ret = lan78xx_start_hw(dev, FCT_TX_CTL, FCT_TX_CTL_EN_);
2795e1210fe6SJohn Efstathiades 	if (ret < 0)
2796e1210fe6SJohn Efstathiades 		return ret;
2797e1210fe6SJohn Efstathiades 
2798e1210fe6SJohn Efstathiades 	return 0;
2799e1210fe6SJohn Efstathiades }
2800e1210fe6SJohn Efstathiades 
lan78xx_stop_tx_path(struct lan78xx_net * dev)2801e1210fe6SJohn Efstathiades static int lan78xx_stop_tx_path(struct lan78xx_net *dev)
2802e1210fe6SJohn Efstathiades {
2803e1210fe6SJohn Efstathiades 	int ret;
2804e1210fe6SJohn Efstathiades 
2805e1210fe6SJohn Efstathiades 	netif_dbg(dev, drv, dev->net, "stop tx path");
2806e1210fe6SJohn Efstathiades 
2807e1210fe6SJohn Efstathiades 	/* Stop the Tx FIFO */
2808e1210fe6SJohn Efstathiades 
2809e1210fe6SJohn Efstathiades 	ret = lan78xx_stop_hw(dev, FCT_TX_CTL, FCT_TX_CTL_EN_, FCT_TX_CTL_DIS_);
2810e1210fe6SJohn Efstathiades 	if (ret < 0)
2811e1210fe6SJohn Efstathiades 		return ret;
2812e1210fe6SJohn Efstathiades 
2813e1210fe6SJohn Efstathiades 	/* Stop the MAC transmitter */
2814e1210fe6SJohn Efstathiades 
2815e1210fe6SJohn Efstathiades 	ret = lan78xx_stop_hw(dev, MAC_TX, MAC_TX_TXEN_, MAC_TX_TXD_);
2816e1210fe6SJohn Efstathiades 	if (ret < 0)
2817e1210fe6SJohn Efstathiades 		return ret;
2818e1210fe6SJohn Efstathiades 
2819e1210fe6SJohn Efstathiades 	return 0;
2820e1210fe6SJohn Efstathiades }
2821e1210fe6SJohn Efstathiades 
2822e1210fe6SJohn Efstathiades /* The caller must ensure the Tx path is stopped before calling
2823e1210fe6SJohn Efstathiades  * lan78xx_flush_tx_fifo().
2824e1210fe6SJohn Efstathiades  */
lan78xx_flush_tx_fifo(struct lan78xx_net * dev)2825e1210fe6SJohn Efstathiades static int lan78xx_flush_tx_fifo(struct lan78xx_net *dev)
2826e1210fe6SJohn Efstathiades {
2827e1210fe6SJohn Efstathiades 	return lan78xx_flush_fifo(dev, FCT_TX_CTL, FCT_TX_CTL_RST_);
2828e1210fe6SJohn Efstathiades }
2829e1210fe6SJohn Efstathiades 
lan78xx_start_rx_path(struct lan78xx_net * dev)2830e1210fe6SJohn Efstathiades static int lan78xx_start_rx_path(struct lan78xx_net *dev)
2831e1210fe6SJohn Efstathiades {
2832e1210fe6SJohn Efstathiades 	int ret;
2833e1210fe6SJohn Efstathiades 
2834e1210fe6SJohn Efstathiades 	netif_dbg(dev, drv, dev->net, "start rx path");
2835e1210fe6SJohn Efstathiades 
2836e1210fe6SJohn Efstathiades 	/* Start the Rx FIFO */
2837e1210fe6SJohn Efstathiades 
2838e1210fe6SJohn Efstathiades 	ret = lan78xx_start_hw(dev, FCT_RX_CTL, FCT_RX_CTL_EN_);
2839e1210fe6SJohn Efstathiades 	if (ret < 0)
2840e1210fe6SJohn Efstathiades 		return ret;
2841e1210fe6SJohn Efstathiades 
2842e1210fe6SJohn Efstathiades 	/* Start the MAC receiver*/
2843e1210fe6SJohn Efstathiades 
2844e1210fe6SJohn Efstathiades 	ret = lan78xx_start_hw(dev, MAC_RX, MAC_RX_RXEN_);
2845e1210fe6SJohn Efstathiades 	if (ret < 0)
2846e1210fe6SJohn Efstathiades 		return ret;
2847e1210fe6SJohn Efstathiades 
2848e1210fe6SJohn Efstathiades 	return 0;
2849e1210fe6SJohn Efstathiades }
2850e1210fe6SJohn Efstathiades 
lan78xx_stop_rx_path(struct lan78xx_net * dev)2851e1210fe6SJohn Efstathiades static int lan78xx_stop_rx_path(struct lan78xx_net *dev)
2852e1210fe6SJohn Efstathiades {
2853e1210fe6SJohn Efstathiades 	int ret;
2854e1210fe6SJohn Efstathiades 
2855e1210fe6SJohn Efstathiades 	netif_dbg(dev, drv, dev->net, "stop rx path");
2856e1210fe6SJohn Efstathiades 
2857e1210fe6SJohn Efstathiades 	/* Stop the MAC receiver */
2858e1210fe6SJohn Efstathiades 
2859e1210fe6SJohn Efstathiades 	ret = lan78xx_stop_hw(dev, MAC_RX, MAC_RX_RXEN_, MAC_RX_RXD_);
2860e1210fe6SJohn Efstathiades 	if (ret < 0)
2861e1210fe6SJohn Efstathiades 		return ret;
2862e1210fe6SJohn Efstathiades 
2863e1210fe6SJohn Efstathiades 	/* Stop the Rx FIFO */
2864e1210fe6SJohn Efstathiades 
2865e1210fe6SJohn Efstathiades 	ret = lan78xx_stop_hw(dev, FCT_RX_CTL, FCT_RX_CTL_EN_, FCT_RX_CTL_DIS_);
2866e1210fe6SJohn Efstathiades 	if (ret < 0)
2867e1210fe6SJohn Efstathiades 		return ret;
2868e1210fe6SJohn Efstathiades 
2869e1210fe6SJohn Efstathiades 	return 0;
2870e1210fe6SJohn Efstathiades }
2871e1210fe6SJohn Efstathiades 
2872e1210fe6SJohn Efstathiades /* The caller must ensure the Rx path is stopped before calling
2873e1210fe6SJohn Efstathiades  * lan78xx_flush_rx_fifo().
2874e1210fe6SJohn Efstathiades  */
lan78xx_flush_rx_fifo(struct lan78xx_net * dev)2875e1210fe6SJohn Efstathiades static int lan78xx_flush_rx_fifo(struct lan78xx_net *dev)
2876e1210fe6SJohn Efstathiades {
2877e1210fe6SJohn Efstathiades 	return lan78xx_flush_fifo(dev, FCT_RX_CTL, FCT_RX_CTL_RST_);
2878e1210fe6SJohn Efstathiades }
2879e1210fe6SJohn Efstathiades 
lan78xx_reset(struct lan78xx_net * dev)288055d7de9dSWoojung.Huh@microchip.com static int lan78xx_reset(struct lan78xx_net *dev)
288155d7de9dSWoojung.Huh@microchip.com {
288255d7de9dSWoojung.Huh@microchip.com 	struct lan78xx_priv *pdata = (struct lan78xx_priv *)(dev->data[0]);
288355d7de9dSWoojung.Huh@microchip.com 	unsigned long timeout;
28843415f6baSJohn Efstathiades 	int ret;
28853415f6baSJohn Efstathiades 	u32 buf;
2886e69647a1SRaghuram Chary J 	u8 sig;
288755d7de9dSWoojung.Huh@microchip.com 
288855d7de9dSWoojung.Huh@microchip.com 	ret = lan78xx_read_reg(dev, HW_CFG, &buf);
28893415f6baSJohn Efstathiades 	if (ret < 0)
28903415f6baSJohn Efstathiades 		return ret;
28913415f6baSJohn Efstathiades 
289255d7de9dSWoojung.Huh@microchip.com 	buf |= HW_CFG_LRST_;
28933415f6baSJohn Efstathiades 
289455d7de9dSWoojung.Huh@microchip.com 	ret = lan78xx_write_reg(dev, HW_CFG, buf);
28953415f6baSJohn Efstathiades 	if (ret < 0)
28963415f6baSJohn Efstathiades 		return ret;
289755d7de9dSWoojung.Huh@microchip.com 
289855d7de9dSWoojung.Huh@microchip.com 	timeout = jiffies + HZ;
289955d7de9dSWoojung.Huh@microchip.com 	do {
290055d7de9dSWoojung.Huh@microchip.com 		mdelay(1);
290155d7de9dSWoojung.Huh@microchip.com 		ret = lan78xx_read_reg(dev, HW_CFG, &buf);
29023415f6baSJohn Efstathiades 		if (ret < 0)
29033415f6baSJohn Efstathiades 			return ret;
29043415f6baSJohn Efstathiades 
290555d7de9dSWoojung.Huh@microchip.com 		if (time_after(jiffies, timeout)) {
290655d7de9dSWoojung.Huh@microchip.com 			netdev_warn(dev->net,
290755d7de9dSWoojung.Huh@microchip.com 				    "timeout on completion of LiteReset");
29083415f6baSJohn Efstathiades 			ret = -ETIMEDOUT;
29093415f6baSJohn Efstathiades 			return ret;
291055d7de9dSWoojung.Huh@microchip.com 		}
291155d7de9dSWoojung.Huh@microchip.com 	} while (buf & HW_CFG_LRST_);
291255d7de9dSWoojung.Huh@microchip.com 
291355d7de9dSWoojung.Huh@microchip.com 	lan78xx_init_mac_address(dev);
291455d7de9dSWoojung.Huh@microchip.com 
2915ce85e13aSWoojung.Huh@microchip.com 	/* save DEVID for later usage */
2916ce85e13aSWoojung.Huh@microchip.com 	ret = lan78xx_read_reg(dev, ID_REV, &buf);
29173415f6baSJohn Efstathiades 	if (ret < 0)
29183415f6baSJohn Efstathiades 		return ret;
29193415f6baSJohn Efstathiades 
292087177ba6SWoojung.Huh@microchip.com 	dev->chipid = (buf & ID_REV_CHIP_ID_MASK_) >> 16;
292187177ba6SWoojung.Huh@microchip.com 	dev->chiprev = buf & ID_REV_CHIP_REV_MASK_;
2922ce85e13aSWoojung.Huh@microchip.com 
292355d7de9dSWoojung.Huh@microchip.com 	/* Respond to the IN token with a NAK */
292455d7de9dSWoojung.Huh@microchip.com 	ret = lan78xx_read_reg(dev, USB_CFG0, &buf);
29253415f6baSJohn Efstathiades 	if (ret < 0)
29263415f6baSJohn Efstathiades 		return ret;
29273415f6baSJohn Efstathiades 
292855d7de9dSWoojung.Huh@microchip.com 	buf |= USB_CFG_BIR_;
29293415f6baSJohn Efstathiades 
293055d7de9dSWoojung.Huh@microchip.com 	ret = lan78xx_write_reg(dev, USB_CFG0, buf);
29313415f6baSJohn Efstathiades 	if (ret < 0)
29323415f6baSJohn Efstathiades 		return ret;
293355d7de9dSWoojung.Huh@microchip.com 
293455d7de9dSWoojung.Huh@microchip.com 	/* Init LTM */
293555d7de9dSWoojung.Huh@microchip.com 	lan78xx_init_ltm(dev);
293655d7de9dSWoojung.Huh@microchip.com 
2937c450a8ebSJohn Efstathiades 	ret = lan78xx_write_reg(dev, BURST_CAP, dev->burst_cap);
29383415f6baSJohn Efstathiades 	if (ret < 0)
29393415f6baSJohn Efstathiades 		return ret;
29403415f6baSJohn Efstathiades 
2941c450a8ebSJohn Efstathiades 	ret = lan78xx_write_reg(dev, BULK_IN_DLY, dev->bulk_in_delay);
29423415f6baSJohn Efstathiades 	if (ret < 0)
29433415f6baSJohn Efstathiades 		return ret;
294455d7de9dSWoojung.Huh@microchip.com 
294555d7de9dSWoojung.Huh@microchip.com 	ret = lan78xx_read_reg(dev, HW_CFG, &buf);
29463415f6baSJohn Efstathiades 	if (ret < 0)
29473415f6baSJohn Efstathiades 		return ret;
29483415f6baSJohn Efstathiades 
294955d7de9dSWoojung.Huh@microchip.com 	buf |= HW_CFG_MEF_;
29503415f6baSJohn Efstathiades 
295155d7de9dSWoojung.Huh@microchip.com 	ret = lan78xx_write_reg(dev, HW_CFG, buf);
29523415f6baSJohn Efstathiades 	if (ret < 0)
29533415f6baSJohn Efstathiades 		return ret;
295455d7de9dSWoojung.Huh@microchip.com 
295555d7de9dSWoojung.Huh@microchip.com 	ret = lan78xx_read_reg(dev, USB_CFG0, &buf);
29563415f6baSJohn Efstathiades 	if (ret < 0)
29573415f6baSJohn Efstathiades 		return ret;
29583415f6baSJohn Efstathiades 
295955d7de9dSWoojung.Huh@microchip.com 	buf |= USB_CFG_BCE_;
29603415f6baSJohn Efstathiades 
296155d7de9dSWoojung.Huh@microchip.com 	ret = lan78xx_write_reg(dev, USB_CFG0, buf);
29623415f6baSJohn Efstathiades 	if (ret < 0)
29633415f6baSJohn Efstathiades 		return ret;
296455d7de9dSWoojung.Huh@microchip.com 
296555d7de9dSWoojung.Huh@microchip.com 	/* set FIFO sizes */
296655d7de9dSWoojung.Huh@microchip.com 	buf = (MAX_RX_FIFO_SIZE - 512) / 512;
29673415f6baSJohn Efstathiades 
296855d7de9dSWoojung.Huh@microchip.com 	ret = lan78xx_write_reg(dev, FCT_RX_FIFO_END, buf);
29693415f6baSJohn Efstathiades 	if (ret < 0)
29703415f6baSJohn Efstathiades 		return ret;
297155d7de9dSWoojung.Huh@microchip.com 
297255d7de9dSWoojung.Huh@microchip.com 	buf = (MAX_TX_FIFO_SIZE - 512) / 512;
29733415f6baSJohn Efstathiades 
297455d7de9dSWoojung.Huh@microchip.com 	ret = lan78xx_write_reg(dev, FCT_TX_FIFO_END, buf);
29753415f6baSJohn Efstathiades 	if (ret < 0)
29763415f6baSJohn Efstathiades 		return ret;
297755d7de9dSWoojung.Huh@microchip.com 
297855d7de9dSWoojung.Huh@microchip.com 	ret = lan78xx_write_reg(dev, INT_STS, INT_STS_CLEAR_ALL_);
29793415f6baSJohn Efstathiades 	if (ret < 0)
29803415f6baSJohn Efstathiades 		return ret;
29813415f6baSJohn Efstathiades 
298255d7de9dSWoojung.Huh@microchip.com 	ret = lan78xx_write_reg(dev, FLOW, 0);
29833415f6baSJohn Efstathiades 	if (ret < 0)
29843415f6baSJohn Efstathiades 		return ret;
29853415f6baSJohn Efstathiades 
298655d7de9dSWoojung.Huh@microchip.com 	ret = lan78xx_write_reg(dev, FCT_FLOW, 0);
29873415f6baSJohn Efstathiades 	if (ret < 0)
29883415f6baSJohn Efstathiades 		return ret;
298955d7de9dSWoojung.Huh@microchip.com 
299055d7de9dSWoojung.Huh@microchip.com 	/* Don't need rfe_ctl_lock during initialisation */
299155d7de9dSWoojung.Huh@microchip.com 	ret = lan78xx_read_reg(dev, RFE_CTL, &pdata->rfe_ctl);
29923415f6baSJohn Efstathiades 	if (ret < 0)
29933415f6baSJohn Efstathiades 		return ret;
29943415f6baSJohn Efstathiades 
299555d7de9dSWoojung.Huh@microchip.com 	pdata->rfe_ctl |= RFE_CTL_BCAST_EN_ | RFE_CTL_DA_PERFECT_;
29963415f6baSJohn Efstathiades 
299755d7de9dSWoojung.Huh@microchip.com 	ret = lan78xx_write_reg(dev, RFE_CTL, pdata->rfe_ctl);
29983415f6baSJohn Efstathiades 	if (ret < 0)
29993415f6baSJohn Efstathiades 		return ret;
300055d7de9dSWoojung.Huh@microchip.com 
300155d7de9dSWoojung.Huh@microchip.com 	/* Enable or disable checksum offload engines */
30023415f6baSJohn Efstathiades 	ret = lan78xx_set_features(dev->net, dev->net->features);
30033415f6baSJohn Efstathiades 	if (ret < 0)
30043415f6baSJohn Efstathiades 		return ret;
300555d7de9dSWoojung.Huh@microchip.com 
300655d7de9dSWoojung.Huh@microchip.com 	lan78xx_set_multicast(dev->net);
300755d7de9dSWoojung.Huh@microchip.com 
300855d7de9dSWoojung.Huh@microchip.com 	/* reset PHY */
300955d7de9dSWoojung.Huh@microchip.com 	ret = lan78xx_read_reg(dev, PMT_CTL, &buf);
30103415f6baSJohn Efstathiades 	if (ret < 0)
30113415f6baSJohn Efstathiades 		return ret;
30123415f6baSJohn Efstathiades 
301355d7de9dSWoojung.Huh@microchip.com 	buf |= PMT_CTL_PHY_RST_;
30143415f6baSJohn Efstathiades 
301555d7de9dSWoojung.Huh@microchip.com 	ret = lan78xx_write_reg(dev, PMT_CTL, buf);
30163415f6baSJohn Efstathiades 	if (ret < 0)
30173415f6baSJohn Efstathiades 		return ret;
301855d7de9dSWoojung.Huh@microchip.com 
301955d7de9dSWoojung.Huh@microchip.com 	timeout = jiffies + HZ;
302055d7de9dSWoojung.Huh@microchip.com 	do {
302155d7de9dSWoojung.Huh@microchip.com 		mdelay(1);
302255d7de9dSWoojung.Huh@microchip.com 		ret = lan78xx_read_reg(dev, PMT_CTL, &buf);
30233415f6baSJohn Efstathiades 		if (ret < 0)
30243415f6baSJohn Efstathiades 			return ret;
30253415f6baSJohn Efstathiades 
302655d7de9dSWoojung.Huh@microchip.com 		if (time_after(jiffies, timeout)) {
302755d7de9dSWoojung.Huh@microchip.com 			netdev_warn(dev->net, "timeout waiting for PHY Reset");
30283415f6baSJohn Efstathiades 			ret = -ETIMEDOUT;
30293415f6baSJohn Efstathiades 			return ret;
303055d7de9dSWoojung.Huh@microchip.com 		}
30316c595b03SWoojung.Huh@microchip.com 	} while ((buf & PMT_CTL_PHY_RST_) || !(buf & PMT_CTL_READY_));
303255d7de9dSWoojung.Huh@microchip.com 
303355d7de9dSWoojung.Huh@microchip.com 	ret = lan78xx_read_reg(dev, MAC_CR, &buf);
30343415f6baSJohn Efstathiades 	if (ret < 0)
30353415f6baSJohn Efstathiades 		return ret;
30363415f6baSJohn Efstathiades 
303702dc1f3dSWoojung Huh 	/* LAN7801 only has RGMII mode */
303802dc1f3dSWoojung Huh 	if (dev->chipid == ID_REV_CHIP_ID_7801_)
303902dc1f3dSWoojung Huh 		buf &= ~MAC_CR_GMII_EN_;
3040e69647a1SRaghuram Chary J 
3041288218b1SOleksij Rempel 	if (dev->chipid == ID_REV_CHIP_ID_7800_ ||
3042288218b1SOleksij Rempel 	    dev->chipid == ID_REV_CHIP_ID_7850_) {
3043e69647a1SRaghuram Chary J 		ret = lan78xx_read_raw_eeprom(dev, 0, 1, &sig);
3044e69647a1SRaghuram Chary J 		if (!ret && sig != EEPROM_INDICATOR) {
3045e69647a1SRaghuram Chary J 			/* Implies there is no external eeprom. Set mac speed */
3046e69647a1SRaghuram Chary J 			netdev_info(dev->net, "No External EEPROM. Setting MAC Speed\n");
3047e69647a1SRaghuram Chary J 			buf |= MAC_CR_AUTO_DUPLEX_ | MAC_CR_AUTO_SPEED_;
3048e69647a1SRaghuram Chary J 		}
3049e69647a1SRaghuram Chary J 	}
305055d7de9dSWoojung.Huh@microchip.com 	ret = lan78xx_write_reg(dev, MAC_CR, buf);
30513415f6baSJohn Efstathiades 	if (ret < 0)
30523415f6baSJohn Efstathiades 		return ret;
305355d7de9dSWoojung.Huh@microchip.com 
30542259b7a6SDave Stevenson 	ret = lan78xx_set_rx_max_frame_length(dev,
30550dd87266SJohn Efstathiades 					      RX_MAX_FRAME_LEN(dev->net->mtu));
3056e1210fe6SJohn Efstathiades 
30573415f6baSJohn Efstathiades 	return ret;
305855d7de9dSWoojung.Huh@microchip.com }
305955d7de9dSWoojung.Huh@microchip.com 
lan78xx_init_stats(struct lan78xx_net * dev)306020ff5565SWoojung Huh static void lan78xx_init_stats(struct lan78xx_net *dev)
306120ff5565SWoojung Huh {
306220ff5565SWoojung Huh 	u32 *p;
306320ff5565SWoojung Huh 	int i;
306420ff5565SWoojung Huh 
306520ff5565SWoojung Huh 	/* initialize for stats update
306620ff5565SWoojung Huh 	 * some counters are 20bits and some are 32bits
306720ff5565SWoojung Huh 	 */
306820ff5565SWoojung Huh 	p = (u32 *)&dev->stats.rollover_max;
306920ff5565SWoojung Huh 	for (i = 0; i < (sizeof(dev->stats.rollover_max) / (sizeof(u32))); i++)
307020ff5565SWoojung Huh 		p[i] = 0xFFFFF;
307120ff5565SWoojung Huh 
307220ff5565SWoojung Huh 	dev->stats.rollover_max.rx_unicast_byte_count = 0xFFFFFFFF;
307320ff5565SWoojung Huh 	dev->stats.rollover_max.rx_broadcast_byte_count = 0xFFFFFFFF;
307420ff5565SWoojung Huh 	dev->stats.rollover_max.rx_multicast_byte_count = 0xFFFFFFFF;
307520ff5565SWoojung Huh 	dev->stats.rollover_max.eee_rx_lpi_transitions = 0xFFFFFFFF;
307620ff5565SWoojung Huh 	dev->stats.rollover_max.eee_rx_lpi_time = 0xFFFFFFFF;
307720ff5565SWoojung Huh 	dev->stats.rollover_max.tx_unicast_byte_count = 0xFFFFFFFF;
307820ff5565SWoojung Huh 	dev->stats.rollover_max.tx_broadcast_byte_count = 0xFFFFFFFF;
307920ff5565SWoojung Huh 	dev->stats.rollover_max.tx_multicast_byte_count = 0xFFFFFFFF;
308020ff5565SWoojung Huh 	dev->stats.rollover_max.eee_tx_lpi_transitions = 0xFFFFFFFF;
308120ff5565SWoojung Huh 	dev->stats.rollover_max.eee_tx_lpi_time = 0xFFFFFFFF;
308220ff5565SWoojung Huh 
3083fed56079SPhil Elwell 	set_bit(EVENT_STAT_UPDATE, &dev->flags);
308420ff5565SWoojung Huh }
308520ff5565SWoojung Huh 
lan78xx_open(struct net_device * net)308655d7de9dSWoojung.Huh@microchip.com static int lan78xx_open(struct net_device *net)
308755d7de9dSWoojung.Huh@microchip.com {
308855d7de9dSWoojung.Huh@microchip.com 	struct lan78xx_net *dev = netdev_priv(net);
308955d7de9dSWoojung.Huh@microchip.com 	int ret;
309055d7de9dSWoojung.Huh@microchip.com 
30915f4cc6e2SJohn Efstathiades 	netif_dbg(dev, ifup, dev->net, "open device");
30925f4cc6e2SJohn Efstathiades 
309355d7de9dSWoojung.Huh@microchip.com 	ret = usb_autopm_get_interface(dev->intf);
309455d7de9dSWoojung.Huh@microchip.com 	if (ret < 0)
30953415f6baSJohn Efstathiades 		return ret;
309655d7de9dSWoojung.Huh@microchip.com 
30975f4cc6e2SJohn Efstathiades 	mutex_lock(&dev->dev_mutex);
30985f4cc6e2SJohn Efstathiades 
309992571a1aSAlexander Graf 	phy_start(net->phydev);
310092571a1aSAlexander Graf 
310192571a1aSAlexander Graf 	netif_dbg(dev, ifup, dev->net, "phy initialised successfully");
3102ce85e13aSWoojung.Huh@microchip.com 
310355d7de9dSWoojung.Huh@microchip.com 	/* for Link Check */
310455d7de9dSWoojung.Huh@microchip.com 	if (dev->urb_intr) {
310555d7de9dSWoojung.Huh@microchip.com 		ret = usb_submit_urb(dev->urb_intr, GFP_KERNEL);
310655d7de9dSWoojung.Huh@microchip.com 		if (ret < 0) {
310755d7de9dSWoojung.Huh@microchip.com 			netif_err(dev, ifup, dev->net,
310855d7de9dSWoojung.Huh@microchip.com 				  "intr submit %d\n", ret);
310955d7de9dSWoojung.Huh@microchip.com 			goto done;
311055d7de9dSWoojung.Huh@microchip.com 		}
311155d7de9dSWoojung.Huh@microchip.com 	}
311255d7de9dSWoojung.Huh@microchip.com 
31135f4cc6e2SJohn Efstathiades 	ret = lan78xx_flush_rx_fifo(dev);
31145f4cc6e2SJohn Efstathiades 	if (ret < 0)
31155f4cc6e2SJohn Efstathiades 		goto done;
31165f4cc6e2SJohn Efstathiades 	ret = lan78xx_flush_tx_fifo(dev);
31175f4cc6e2SJohn Efstathiades 	if (ret < 0)
31185f4cc6e2SJohn Efstathiades 		goto done;
31195f4cc6e2SJohn Efstathiades 
31205f4cc6e2SJohn Efstathiades 	ret = lan78xx_start_tx_path(dev);
31215f4cc6e2SJohn Efstathiades 	if (ret < 0)
31225f4cc6e2SJohn Efstathiades 		goto done;
31235f4cc6e2SJohn Efstathiades 	ret = lan78xx_start_rx_path(dev);
31245f4cc6e2SJohn Efstathiades 	if (ret < 0)
31255f4cc6e2SJohn Efstathiades 		goto done;
31265f4cc6e2SJohn Efstathiades 
312720ff5565SWoojung Huh 	lan78xx_init_stats(dev);
312820ff5565SWoojung Huh 
312955d7de9dSWoojung.Huh@microchip.com 	set_bit(EVENT_DEV_OPEN, &dev->flags);
313055d7de9dSWoojung.Huh@microchip.com 
313155d7de9dSWoojung.Huh@microchip.com 	netif_start_queue(net);
313255d7de9dSWoojung.Huh@microchip.com 
313355d7de9dSWoojung.Huh@microchip.com 	dev->link_on = false;
313455d7de9dSWoojung.Huh@microchip.com 
3135ec4c7e12SJohn Efstathiades 	napi_enable(&dev->napi);
3136ec4c7e12SJohn Efstathiades 
313755d7de9dSWoojung.Huh@microchip.com 	lan78xx_defer_kevent(dev, EVENT_LINK_RESET);
313855d7de9dSWoojung.Huh@microchip.com done:
31395f4cc6e2SJohn Efstathiades 	mutex_unlock(&dev->dev_mutex);
31405f4cc6e2SJohn Efstathiades 
3141550fe716SOleksij Rempel 	if (ret < 0)
314255d7de9dSWoojung.Huh@microchip.com 		usb_autopm_put_interface(dev->intf);
314355d7de9dSWoojung.Huh@microchip.com 
314455d7de9dSWoojung.Huh@microchip.com 	return ret;
314555d7de9dSWoojung.Huh@microchip.com }
314655d7de9dSWoojung.Huh@microchip.com 
lan78xx_terminate_urbs(struct lan78xx_net * dev)314755d7de9dSWoojung.Huh@microchip.com static void lan78xx_terminate_urbs(struct lan78xx_net *dev)
314855d7de9dSWoojung.Huh@microchip.com {
314955d7de9dSWoojung.Huh@microchip.com 	DECLARE_WAIT_QUEUE_HEAD_ONSTACK(unlink_wakeup);
315055d7de9dSWoojung.Huh@microchip.com 	DECLARE_WAITQUEUE(wait, current);
315155d7de9dSWoojung.Huh@microchip.com 	int temp;
315255d7de9dSWoojung.Huh@microchip.com 
315355d7de9dSWoojung.Huh@microchip.com 	/* ensure there are no more active urbs */
315455d7de9dSWoojung.Huh@microchip.com 	add_wait_queue(&unlink_wakeup, &wait);
315555d7de9dSWoojung.Huh@microchip.com 	set_current_state(TASK_UNINTERRUPTIBLE);
315655d7de9dSWoojung.Huh@microchip.com 	dev->wait = &unlink_wakeup;
315755d7de9dSWoojung.Huh@microchip.com 	temp = unlink_urbs(dev, &dev->txq) + unlink_urbs(dev, &dev->rxq);
315855d7de9dSWoojung.Huh@microchip.com 
315955d7de9dSWoojung.Huh@microchip.com 	/* maybe wait for deletions to finish. */
31605f4cc6e2SJohn Efstathiades 	while (!skb_queue_empty(&dev->rxq) ||
31615f4cc6e2SJohn Efstathiades 	       !skb_queue_empty(&dev->txq)) {
316255d7de9dSWoojung.Huh@microchip.com 		schedule_timeout(msecs_to_jiffies(UNLINK_TIMEOUT_MS));
316355d7de9dSWoojung.Huh@microchip.com 		set_current_state(TASK_UNINTERRUPTIBLE);
316455d7de9dSWoojung.Huh@microchip.com 		netif_dbg(dev, ifdown, dev->net,
31655f4cc6e2SJohn Efstathiades 			  "waited for %d urb completions", temp);
316655d7de9dSWoojung.Huh@microchip.com 	}
316755d7de9dSWoojung.Huh@microchip.com 	set_current_state(TASK_RUNNING);
316855d7de9dSWoojung.Huh@microchip.com 	dev->wait = NULL;
316955d7de9dSWoojung.Huh@microchip.com 	remove_wait_queue(&unlink_wakeup, &wait);
31705f4cc6e2SJohn Efstathiades 
3171ec4c7e12SJohn Efstathiades 	/* empty Rx done, Rx overflow and Tx pend queues
3172c450a8ebSJohn Efstathiades 	 */
3173c450a8ebSJohn Efstathiades 	while (!skb_queue_empty(&dev->rxq_done)) {
3174c450a8ebSJohn Efstathiades 		struct sk_buff *skb = skb_dequeue(&dev->rxq_done);
31755f4cc6e2SJohn Efstathiades 
3176c450a8ebSJohn Efstathiades 		lan78xx_release_rx_buf(dev, skb);
31775f4cc6e2SJohn Efstathiades 	}
3178d383216aSJohn Efstathiades 
3179ec4c7e12SJohn Efstathiades 	skb_queue_purge(&dev->rxq_overflow);
3180d383216aSJohn Efstathiades 	skb_queue_purge(&dev->txq_pend);
318155d7de9dSWoojung.Huh@microchip.com }
318255d7de9dSWoojung.Huh@microchip.com 
lan78xx_stop(struct net_device * net)3183e0c79ff6SBaoyou Xie static int lan78xx_stop(struct net_device *net)
318455d7de9dSWoojung.Huh@microchip.com {
318555d7de9dSWoojung.Huh@microchip.com 	struct lan78xx_net *dev = netdev_priv(net);
318655d7de9dSWoojung.Huh@microchip.com 
31875f4cc6e2SJohn Efstathiades 	netif_dbg(dev, ifup, dev->net, "stop device");
31885f4cc6e2SJohn Efstathiades 
31895f4cc6e2SJohn Efstathiades 	mutex_lock(&dev->dev_mutex);
31905f4cc6e2SJohn Efstathiades 
319120ff5565SWoojung Huh 	if (timer_pending(&dev->stat_monitor))
319220ff5565SWoojung Huh 		del_timer_sync(&dev->stat_monitor);
319320ff5565SWoojung Huh 
319455d7de9dSWoojung.Huh@microchip.com 	clear_bit(EVENT_DEV_OPEN, &dev->flags);
319555d7de9dSWoojung.Huh@microchip.com 	netif_stop_queue(net);
3196ec4c7e12SJohn Efstathiades 	napi_disable(&dev->napi);
31975f4cc6e2SJohn Efstathiades 
31985f4cc6e2SJohn Efstathiades 	lan78xx_terminate_urbs(dev);
319955d7de9dSWoojung.Huh@microchip.com 
320055d7de9dSWoojung.Huh@microchip.com 	netif_info(dev, ifdown, dev->net,
320155d7de9dSWoojung.Huh@microchip.com 		   "stop stats: rx/tx %lu/%lu, errs %lu/%lu\n",
320255d7de9dSWoojung.Huh@microchip.com 		   net->stats.rx_packets, net->stats.tx_packets,
320355d7de9dSWoojung.Huh@microchip.com 		   net->stats.rx_errors, net->stats.tx_errors);
320455d7de9dSWoojung.Huh@microchip.com 
32055f4cc6e2SJohn Efstathiades 	/* ignore errors that occur stopping the Tx and Rx data paths */
32065f4cc6e2SJohn Efstathiades 	lan78xx_stop_tx_path(dev);
32075f4cc6e2SJohn Efstathiades 	lan78xx_stop_rx_path(dev);
32085f4cc6e2SJohn Efstathiades 
32095f4cc6e2SJohn Efstathiades 	if (net->phydev)
32105f4cc6e2SJohn Efstathiades 		phy_stop(net->phydev);
321155d7de9dSWoojung.Huh@microchip.com 
321255d7de9dSWoojung.Huh@microchip.com 	usb_kill_urb(dev->urb_intr);
321355d7de9dSWoojung.Huh@microchip.com 
321455d7de9dSWoojung.Huh@microchip.com 	/* deferred work (task, timer, softirq) must also stop.
321555d7de9dSWoojung.Huh@microchip.com 	 * can't flush_scheduled_work() until we drop rtnl (later),
321655d7de9dSWoojung.Huh@microchip.com 	 * else workers could deadlock; so make workers a NOP.
321755d7de9dSWoojung.Huh@microchip.com 	 */
32185f4cc6e2SJohn Efstathiades 	clear_bit(EVENT_TX_HALT, &dev->flags);
32195f4cc6e2SJohn Efstathiades 	clear_bit(EVENT_RX_HALT, &dev->flags);
32205f4cc6e2SJohn Efstathiades 	clear_bit(EVENT_LINK_RESET, &dev->flags);
32215f4cc6e2SJohn Efstathiades 	clear_bit(EVENT_STAT_UPDATE, &dev->flags);
32225f4cc6e2SJohn Efstathiades 
322355d7de9dSWoojung.Huh@microchip.com 	cancel_delayed_work_sync(&dev->wq);
322455d7de9dSWoojung.Huh@microchip.com 
322555d7de9dSWoojung.Huh@microchip.com 	usb_autopm_put_interface(dev->intf);
322655d7de9dSWoojung.Huh@microchip.com 
32275f4cc6e2SJohn Efstathiades 	mutex_unlock(&dev->dev_mutex);
32285f4cc6e2SJohn Efstathiades 
322955d7de9dSWoojung.Huh@microchip.com 	return 0;
323055d7de9dSWoojung.Huh@microchip.com }
323155d7de9dSWoojung.Huh@microchip.com 
defer_bh(struct lan78xx_net * dev,struct sk_buff * skb,struct sk_buff_head * list,enum skb_state state)323255d7de9dSWoojung.Huh@microchip.com static enum skb_state defer_bh(struct lan78xx_net *dev, struct sk_buff *skb,
323355d7de9dSWoojung.Huh@microchip.com 			       struct sk_buff_head *list, enum skb_state state)
323455d7de9dSWoojung.Huh@microchip.com {
323555d7de9dSWoojung.Huh@microchip.com 	unsigned long flags;
323655d7de9dSWoojung.Huh@microchip.com 	enum skb_state old_state;
323755d7de9dSWoojung.Huh@microchip.com 	struct skb_data *entry = (struct skb_data *)skb->cb;
323855d7de9dSWoojung.Huh@microchip.com 
323955d7de9dSWoojung.Huh@microchip.com 	spin_lock_irqsave(&list->lock, flags);
324055d7de9dSWoojung.Huh@microchip.com 	old_state = entry->state;
324155d7de9dSWoojung.Huh@microchip.com 	entry->state = state;
324255d7de9dSWoojung.Huh@microchip.com 
324355d7de9dSWoojung.Huh@microchip.com 	__skb_unlink(skb, list);
324455d7de9dSWoojung.Huh@microchip.com 	spin_unlock(&list->lock);
3245c450a8ebSJohn Efstathiades 	spin_lock(&dev->rxq_done.lock);
324655d7de9dSWoojung.Huh@microchip.com 
3247c450a8ebSJohn Efstathiades 	__skb_queue_tail(&dev->rxq_done, skb);
3248c450a8ebSJohn Efstathiades 	if (skb_queue_len(&dev->rxq_done) == 1)
3249ec4c7e12SJohn Efstathiades 		napi_schedule(&dev->napi);
3250ec4c7e12SJohn Efstathiades 
3251c450a8ebSJohn Efstathiades 	spin_unlock_irqrestore(&dev->rxq_done.lock, flags);
325255d7de9dSWoojung.Huh@microchip.com 
325355d7de9dSWoojung.Huh@microchip.com 	return old_state;
325455d7de9dSWoojung.Huh@microchip.com }
325555d7de9dSWoojung.Huh@microchip.com 
tx_complete(struct urb * urb)325655d7de9dSWoojung.Huh@microchip.com static void tx_complete(struct urb *urb)
325755d7de9dSWoojung.Huh@microchip.com {
325855d7de9dSWoojung.Huh@microchip.com 	struct sk_buff *skb = (struct sk_buff *)urb->context;
325955d7de9dSWoojung.Huh@microchip.com 	struct skb_data *entry = (struct skb_data *)skb->cb;
326055d7de9dSWoojung.Huh@microchip.com 	struct lan78xx_net *dev = entry->dev;
326155d7de9dSWoojung.Huh@microchip.com 
326255d7de9dSWoojung.Huh@microchip.com 	if (urb->status == 0) {
326374d79a2eSWoojung Huh 		dev->net->stats.tx_packets += entry->num_of_packet;
326455d7de9dSWoojung.Huh@microchip.com 		dev->net->stats.tx_bytes += entry->length;
326555d7de9dSWoojung.Huh@microchip.com 	} else {
3266d383216aSJohn Efstathiades 		dev->net->stats.tx_errors += entry->num_of_packet;
326755d7de9dSWoojung.Huh@microchip.com 
326855d7de9dSWoojung.Huh@microchip.com 		switch (urb->status) {
326955d7de9dSWoojung.Huh@microchip.com 		case -EPIPE:
327055d7de9dSWoojung.Huh@microchip.com 			lan78xx_defer_kevent(dev, EVENT_TX_HALT);
327155d7de9dSWoojung.Huh@microchip.com 			break;
327255d7de9dSWoojung.Huh@microchip.com 
327355d7de9dSWoojung.Huh@microchip.com 		/* software-driven interface shutdown */
327455d7de9dSWoojung.Huh@microchip.com 		case -ECONNRESET:
327555d7de9dSWoojung.Huh@microchip.com 		case -ESHUTDOWN:
327677dfff5bSJohn Efstathiades 			netif_dbg(dev, tx_err, dev->net,
327777dfff5bSJohn Efstathiades 				  "tx err interface gone %d\n",
327877dfff5bSJohn Efstathiades 				  entry->urb->status);
327955d7de9dSWoojung.Huh@microchip.com 			break;
328055d7de9dSWoojung.Huh@microchip.com 
328155d7de9dSWoojung.Huh@microchip.com 		case -EPROTO:
328255d7de9dSWoojung.Huh@microchip.com 		case -ETIME:
328355d7de9dSWoojung.Huh@microchip.com 		case -EILSEQ:
328455d7de9dSWoojung.Huh@microchip.com 			netif_stop_queue(dev->net);
328577dfff5bSJohn Efstathiades 			netif_dbg(dev, tx_err, dev->net,
328677dfff5bSJohn Efstathiades 				  "tx err queue stopped %d\n",
328777dfff5bSJohn Efstathiades 				  entry->urb->status);
328855d7de9dSWoojung.Huh@microchip.com 			break;
328955d7de9dSWoojung.Huh@microchip.com 		default:
329055d7de9dSWoojung.Huh@microchip.com 			netif_dbg(dev, tx_err, dev->net,
329177dfff5bSJohn Efstathiades 				  "unknown tx err %d\n",
329277dfff5bSJohn Efstathiades 				  entry->urb->status);
329355d7de9dSWoojung.Huh@microchip.com 			break;
329455d7de9dSWoojung.Huh@microchip.com 		}
329555d7de9dSWoojung.Huh@microchip.com 	}
329655d7de9dSWoojung.Huh@microchip.com 
329755d7de9dSWoojung.Huh@microchip.com 	usb_autopm_put_interface_async(dev->intf);
329855d7de9dSWoojung.Huh@microchip.com 
3299d383216aSJohn Efstathiades 	skb_unlink(skb, &dev->txq);
3300d383216aSJohn Efstathiades 
3301d383216aSJohn Efstathiades 	lan78xx_release_tx_buf(dev, skb);
3302d383216aSJohn Efstathiades 
3303ec4c7e12SJohn Efstathiades 	/* Re-schedule NAPI if Tx data pending but no URBs in progress.
3304d383216aSJohn Efstathiades 	 */
3305d383216aSJohn Efstathiades 	if (skb_queue_empty(&dev->txq) &&
3306d383216aSJohn Efstathiades 	    !skb_queue_empty(&dev->txq_pend))
3307ec4c7e12SJohn Efstathiades 		napi_schedule(&dev->napi);
330855d7de9dSWoojung.Huh@microchip.com }
330955d7de9dSWoojung.Huh@microchip.com 
lan78xx_queue_skb(struct sk_buff_head * list,struct sk_buff * newsk,enum skb_state state)331055d7de9dSWoojung.Huh@microchip.com static void lan78xx_queue_skb(struct sk_buff_head *list,
331155d7de9dSWoojung.Huh@microchip.com 			      struct sk_buff *newsk, enum skb_state state)
331255d7de9dSWoojung.Huh@microchip.com {
331355d7de9dSWoojung.Huh@microchip.com 	struct skb_data *entry = (struct skb_data *)newsk->cb;
331455d7de9dSWoojung.Huh@microchip.com 
331555d7de9dSWoojung.Huh@microchip.com 	__skb_queue_tail(list, newsk);
331655d7de9dSWoojung.Huh@microchip.com 	entry->state = state;
331755d7de9dSWoojung.Huh@microchip.com }
331855d7de9dSWoojung.Huh@microchip.com 
lan78xx_tx_urb_space(struct lan78xx_net * dev)3319d383216aSJohn Efstathiades static unsigned int lan78xx_tx_urb_space(struct lan78xx_net *dev)
3320d383216aSJohn Efstathiades {
3321d383216aSJohn Efstathiades 	return skb_queue_len(&dev->txq_free) * dev->tx_urb_size;
3322d383216aSJohn Efstathiades }
3323d383216aSJohn Efstathiades 
lan78xx_tx_pend_data_len(struct lan78xx_net * dev)3324d383216aSJohn Efstathiades static unsigned int lan78xx_tx_pend_data_len(struct lan78xx_net *dev)
3325d383216aSJohn Efstathiades {
3326d383216aSJohn Efstathiades 	return dev->tx_pend_data_len;
3327d383216aSJohn Efstathiades }
3328d383216aSJohn Efstathiades 
lan78xx_tx_pend_skb_add(struct lan78xx_net * dev,struct sk_buff * skb,unsigned int * tx_pend_data_len)3329d383216aSJohn Efstathiades static void lan78xx_tx_pend_skb_add(struct lan78xx_net *dev,
3330d383216aSJohn Efstathiades 				    struct sk_buff *skb,
3331d383216aSJohn Efstathiades 				    unsigned int *tx_pend_data_len)
3332d383216aSJohn Efstathiades {
3333d383216aSJohn Efstathiades 	unsigned long flags;
3334d383216aSJohn Efstathiades 
3335d383216aSJohn Efstathiades 	spin_lock_irqsave(&dev->txq_pend.lock, flags);
3336d383216aSJohn Efstathiades 
3337d383216aSJohn Efstathiades 	__skb_queue_tail(&dev->txq_pend, skb);
3338d383216aSJohn Efstathiades 
3339d383216aSJohn Efstathiades 	dev->tx_pend_data_len += skb->len;
3340d383216aSJohn Efstathiades 	*tx_pend_data_len = dev->tx_pend_data_len;
3341d383216aSJohn Efstathiades 
3342d383216aSJohn Efstathiades 	spin_unlock_irqrestore(&dev->txq_pend.lock, flags);
3343d383216aSJohn Efstathiades }
3344d383216aSJohn Efstathiades 
lan78xx_tx_pend_skb_head_add(struct lan78xx_net * dev,struct sk_buff * skb,unsigned int * tx_pend_data_len)3345d383216aSJohn Efstathiades static void lan78xx_tx_pend_skb_head_add(struct lan78xx_net *dev,
3346d383216aSJohn Efstathiades 					 struct sk_buff *skb,
3347d383216aSJohn Efstathiades 					 unsigned int *tx_pend_data_len)
3348d383216aSJohn Efstathiades {
3349d383216aSJohn Efstathiades 	unsigned long flags;
3350d383216aSJohn Efstathiades 
3351d383216aSJohn Efstathiades 	spin_lock_irqsave(&dev->txq_pend.lock, flags);
3352d383216aSJohn Efstathiades 
3353d383216aSJohn Efstathiades 	__skb_queue_head(&dev->txq_pend, skb);
3354d383216aSJohn Efstathiades 
3355d383216aSJohn Efstathiades 	dev->tx_pend_data_len += skb->len;
3356d383216aSJohn Efstathiades 	*tx_pend_data_len = dev->tx_pend_data_len;
3357d383216aSJohn Efstathiades 
3358d383216aSJohn Efstathiades 	spin_unlock_irqrestore(&dev->txq_pend.lock, flags);
3359d383216aSJohn Efstathiades }
3360d383216aSJohn Efstathiades 
lan78xx_tx_pend_skb_get(struct lan78xx_net * dev,struct sk_buff ** skb,unsigned int * tx_pend_data_len)3361d383216aSJohn Efstathiades static void lan78xx_tx_pend_skb_get(struct lan78xx_net *dev,
3362d383216aSJohn Efstathiades 				    struct sk_buff **skb,
3363d383216aSJohn Efstathiades 				    unsigned int *tx_pend_data_len)
3364d383216aSJohn Efstathiades {
3365d383216aSJohn Efstathiades 	unsigned long flags;
3366d383216aSJohn Efstathiades 
3367d383216aSJohn Efstathiades 	spin_lock_irqsave(&dev->txq_pend.lock, flags);
3368d383216aSJohn Efstathiades 
3369d383216aSJohn Efstathiades 	*skb = __skb_dequeue(&dev->txq_pend);
3370d383216aSJohn Efstathiades 	if (*skb)
3371d383216aSJohn Efstathiades 		dev->tx_pend_data_len -= (*skb)->len;
3372d383216aSJohn Efstathiades 	*tx_pend_data_len = dev->tx_pend_data_len;
3373d383216aSJohn Efstathiades 
3374d383216aSJohn Efstathiades 	spin_unlock_irqrestore(&dev->txq_pend.lock, flags);
3375d383216aSJohn Efstathiades }
3376d383216aSJohn Efstathiades 
3377e0c79ff6SBaoyou Xie static netdev_tx_t
lan78xx_start_xmit(struct sk_buff * skb,struct net_device * net)3378e0c79ff6SBaoyou Xie lan78xx_start_xmit(struct sk_buff *skb, struct net_device *net)
337955d7de9dSWoojung.Huh@microchip.com {
338055d7de9dSWoojung.Huh@microchip.com 	struct lan78xx_net *dev = netdev_priv(net);
3381d383216aSJohn Efstathiades 	unsigned int tx_pend_data_len;
338255d7de9dSWoojung.Huh@microchip.com 
33835f4cc6e2SJohn Efstathiades 	if (test_bit(EVENT_DEV_ASLEEP, &dev->flags))
33845f4cc6e2SJohn Efstathiades 		schedule_delayed_work(&dev->wq, 0);
33855f4cc6e2SJohn Efstathiades 
338681c38e81SWoojung.Huh@microchip.com 	skb_tx_timestamp(skb);
338781c38e81SWoojung.Huh@microchip.com 
3388d383216aSJohn Efstathiades 	lan78xx_tx_pend_skb_add(dev, skb, &tx_pend_data_len);
338955d7de9dSWoojung.Huh@microchip.com 
3390d383216aSJohn Efstathiades 	/* Set up a Tx URB if none is in progress */
339155d7de9dSWoojung.Huh@microchip.com 
3392d383216aSJohn Efstathiades 	if (skb_queue_empty(&dev->txq))
3393ec4c7e12SJohn Efstathiades 		napi_schedule(&dev->napi);
339455d7de9dSWoojung.Huh@microchip.com 
3395d383216aSJohn Efstathiades 	/* Stop stack Tx queue if we have enough data to fill
3396d383216aSJohn Efstathiades 	 * all the free Tx URBs.
3397d383216aSJohn Efstathiades 	 */
3398d383216aSJohn Efstathiades 	if (tx_pend_data_len > lan78xx_tx_urb_space(dev)) {
3399d383216aSJohn Efstathiades 		netif_stop_queue(net);
3400d383216aSJohn Efstathiades 
3401d383216aSJohn Efstathiades 		netif_dbg(dev, hw, dev->net, "tx data len: %u, urb space %u",
3402d383216aSJohn Efstathiades 			  tx_pend_data_len, lan78xx_tx_urb_space(dev));
3403d383216aSJohn Efstathiades 
3404d383216aSJohn Efstathiades 		/* Kick off transmission of pending data */
3405d383216aSJohn Efstathiades 
3406d383216aSJohn Efstathiades 		if (!skb_queue_empty(&dev->txq_free))
3407ec4c7e12SJohn Efstathiades 			napi_schedule(&dev->napi);
3408d383216aSJohn Efstathiades 	}
3409d383216aSJohn Efstathiades 
341055d7de9dSWoojung.Huh@microchip.com 	return NETDEV_TX_OK;
341155d7de9dSWoojung.Huh@microchip.com }
341255d7de9dSWoojung.Huh@microchip.com 
lan78xx_bind(struct lan78xx_net * dev,struct usb_interface * intf)341355d7de9dSWoojung.Huh@microchip.com static int lan78xx_bind(struct lan78xx_net *dev, struct usb_interface *intf)
341455d7de9dSWoojung.Huh@microchip.com {
341555d7de9dSWoojung.Huh@microchip.com 	struct lan78xx_priv *pdata = NULL;
341655d7de9dSWoojung.Huh@microchip.com 	int ret;
341755d7de9dSWoojung.Huh@microchip.com 	int i;
341855d7de9dSWoojung.Huh@microchip.com 
341955d7de9dSWoojung.Huh@microchip.com 	dev->data[0] = (unsigned long)kzalloc(sizeof(*pdata), GFP_KERNEL);
342055d7de9dSWoojung.Huh@microchip.com 
342155d7de9dSWoojung.Huh@microchip.com 	pdata = (struct lan78xx_priv *)(dev->data[0]);
342255d7de9dSWoojung.Huh@microchip.com 	if (!pdata) {
342355d7de9dSWoojung.Huh@microchip.com 		netdev_warn(dev->net, "Unable to allocate lan78xx_priv");
342455d7de9dSWoojung.Huh@microchip.com 		return -ENOMEM;
342555d7de9dSWoojung.Huh@microchip.com 	}
342655d7de9dSWoojung.Huh@microchip.com 
342755d7de9dSWoojung.Huh@microchip.com 	pdata->dev = dev;
342855d7de9dSWoojung.Huh@microchip.com 
342955d7de9dSWoojung.Huh@microchip.com 	spin_lock_init(&pdata->rfe_ctl_lock);
343055d7de9dSWoojung.Huh@microchip.com 	mutex_init(&pdata->dataport_mutex);
343155d7de9dSWoojung.Huh@microchip.com 
343255d7de9dSWoojung.Huh@microchip.com 	INIT_WORK(&pdata->set_multicast, lan78xx_deferred_multicast_write);
343355d7de9dSWoojung.Huh@microchip.com 
343455d7de9dSWoojung.Huh@microchip.com 	for (i = 0; i < DP_SEL_VHF_VLAN_LEN; i++)
343555d7de9dSWoojung.Huh@microchip.com 		pdata->vlan_table[i] = 0;
343655d7de9dSWoojung.Huh@microchip.com 
343755d7de9dSWoojung.Huh@microchip.com 	INIT_WORK(&pdata->set_vlan, lan78xx_deferred_vlan_write);
343855d7de9dSWoojung.Huh@microchip.com 
343955d7de9dSWoojung.Huh@microchip.com 	dev->net->features = 0;
344055d7de9dSWoojung.Huh@microchip.com 
344155d7de9dSWoojung.Huh@microchip.com 	if (DEFAULT_TX_CSUM_ENABLE)
344255d7de9dSWoojung.Huh@microchip.com 		dev->net->features |= NETIF_F_HW_CSUM;
344355d7de9dSWoojung.Huh@microchip.com 
344455d7de9dSWoojung.Huh@microchip.com 	if (DEFAULT_RX_CSUM_ENABLE)
344555d7de9dSWoojung.Huh@microchip.com 		dev->net->features |= NETIF_F_RXCSUM;
344655d7de9dSWoojung.Huh@microchip.com 
344755d7de9dSWoojung.Huh@microchip.com 	if (DEFAULT_TSO_CSUM_ENABLE)
344855d7de9dSWoojung.Huh@microchip.com 		dev->net->features |= NETIF_F_TSO | NETIF_F_TSO6 | NETIF_F_SG;
344955d7de9dSWoojung.Huh@microchip.com 
3450ec21ecf0SDave Stevenson 	if (DEFAULT_VLAN_RX_OFFLOAD)
3451ec21ecf0SDave Stevenson 		dev->net->features |= NETIF_F_HW_VLAN_CTAG_RX;
3452ec21ecf0SDave Stevenson 
34534a27327bSDave Stevenson 	if (DEFAULT_VLAN_FILTER_ENABLE)
34544a27327bSDave Stevenson 		dev->net->features |= NETIF_F_HW_VLAN_CTAG_FILTER;
34554a27327bSDave Stevenson 
345655d7de9dSWoojung.Huh@microchip.com 	dev->net->hw_features = dev->net->features;
345755d7de9dSWoojung.Huh@microchip.com 
3458cc89c323SWoojung Huh 	ret = lan78xx_setup_irq_domain(dev);
3459cc89c323SWoojung Huh 	if (ret < 0) {
3460cc89c323SWoojung Huh 		netdev_warn(dev->net,
3461cc89c323SWoojung Huh 			    "lan78xx_setup_irq_domain() failed : %d", ret);
34622d2d99ecSRaghuram Chary J 		goto out1;
3463cc89c323SWoojung Huh 	}
3464cc89c323SWoojung Huh 
346555d7de9dSWoojung.Huh@microchip.com 	/* Init all registers */
346655d7de9dSWoojung.Huh@microchip.com 	ret = lan78xx_reset(dev);
34672d2d99ecSRaghuram Chary J 	if (ret) {
34682d2d99ecSRaghuram Chary J 		netdev_warn(dev->net, "Registers INIT FAILED....");
34692d2d99ecSRaghuram Chary J 		goto out2;
34702d2d99ecSRaghuram Chary J 	}
347155d7de9dSWoojung.Huh@microchip.com 
3472fb52c3b5SNisar Sayed 	ret = lan78xx_mdio_init(dev);
34732d2d99ecSRaghuram Chary J 	if (ret) {
34742d2d99ecSRaghuram Chary J 		netdev_warn(dev->net, "MDIO INIT FAILED.....");
34752d2d99ecSRaghuram Chary J 		goto out2;
34762d2d99ecSRaghuram Chary J 	}
3477ce85e13aSWoojung.Huh@microchip.com 
347855d7de9dSWoojung.Huh@microchip.com 	dev->net->flags |= IFF_MULTICAST;
347955d7de9dSWoojung.Huh@microchip.com 
348055d7de9dSWoojung.Huh@microchip.com 	pdata->wol = WAKE_MAGIC;
348155d7de9dSWoojung.Huh@microchip.com 
3482fb52c3b5SNisar Sayed 	return ret;
34832d2d99ecSRaghuram Chary J 
34842d2d99ecSRaghuram Chary J out2:
34852d2d99ecSRaghuram Chary J 	lan78xx_remove_irq_domain(dev);
34862d2d99ecSRaghuram Chary J 
34872d2d99ecSRaghuram Chary J out1:
34882d2d99ecSRaghuram Chary J 	netdev_warn(dev->net, "Bind routine FAILED");
34892d2d99ecSRaghuram Chary J 	cancel_work_sync(&pdata->set_multicast);
34902d2d99ecSRaghuram Chary J 	cancel_work_sync(&pdata->set_vlan);
34912d2d99ecSRaghuram Chary J 	kfree(pdata);
34922d2d99ecSRaghuram Chary J 	return ret;
349355d7de9dSWoojung.Huh@microchip.com }
349455d7de9dSWoojung.Huh@microchip.com 
lan78xx_unbind(struct lan78xx_net * dev,struct usb_interface * intf)349555d7de9dSWoojung.Huh@microchip.com static void lan78xx_unbind(struct lan78xx_net *dev, struct usb_interface *intf)
349655d7de9dSWoojung.Huh@microchip.com {
349755d7de9dSWoojung.Huh@microchip.com 	struct lan78xx_priv *pdata = (struct lan78xx_priv *)(dev->data[0]);
349855d7de9dSWoojung.Huh@microchip.com 
3499cc89c323SWoojung Huh 	lan78xx_remove_irq_domain(dev);
3500cc89c323SWoojung Huh 
3501ce85e13aSWoojung.Huh@microchip.com 	lan78xx_remove_mdio(dev);
3502ce85e13aSWoojung.Huh@microchip.com 
350355d7de9dSWoojung.Huh@microchip.com 	if (pdata) {
35042d2d99ecSRaghuram Chary J 		cancel_work_sync(&pdata->set_multicast);
35052d2d99ecSRaghuram Chary J 		cancel_work_sync(&pdata->set_vlan);
350655d7de9dSWoojung.Huh@microchip.com 		netif_dbg(dev, ifdown, dev->net, "free pdata");
350755d7de9dSWoojung.Huh@microchip.com 		kfree(pdata);
350855d7de9dSWoojung.Huh@microchip.com 		pdata = NULL;
350955d7de9dSWoojung.Huh@microchip.com 		dev->data[0] = 0;
351055d7de9dSWoojung.Huh@microchip.com 	}
351155d7de9dSWoojung.Huh@microchip.com }
351255d7de9dSWoojung.Huh@microchip.com 
lan78xx_rx_csum_offload(struct lan78xx_net * dev,struct sk_buff * skb,u32 rx_cmd_a,u32 rx_cmd_b)351355d7de9dSWoojung.Huh@microchip.com static void lan78xx_rx_csum_offload(struct lan78xx_net *dev,
351455d7de9dSWoojung.Huh@microchip.com 				    struct sk_buff *skb,
351555d7de9dSWoojung.Huh@microchip.com 				    u32 rx_cmd_a, u32 rx_cmd_b)
351655d7de9dSWoojung.Huh@microchip.com {
35179343ac87SDave Stevenson 	/* HW Checksum offload appears to be flawed if used when not stripping
35189343ac87SDave Stevenson 	 * VLAN headers. Drop back to S/W checksums under these conditions.
35199343ac87SDave Stevenson 	 */
352055d7de9dSWoojung.Huh@microchip.com 	if (!(dev->net->features & NETIF_F_RXCSUM) ||
35219343ac87SDave Stevenson 	    unlikely(rx_cmd_a & RX_CMD_A_ICSM_) ||
35229343ac87SDave Stevenson 	    ((rx_cmd_a & RX_CMD_A_FVTG_) &&
35239343ac87SDave Stevenson 	     !(dev->net->features & NETIF_F_HW_VLAN_CTAG_RX))) {
352455d7de9dSWoojung.Huh@microchip.com 		skb->ip_summed = CHECKSUM_NONE;
352555d7de9dSWoojung.Huh@microchip.com 	} else {
352655d7de9dSWoojung.Huh@microchip.com 		skb->csum = ntohs((u16)(rx_cmd_b >> RX_CMD_B_CSUM_SHIFT_));
352755d7de9dSWoojung.Huh@microchip.com 		skb->ip_summed = CHECKSUM_COMPLETE;
352855d7de9dSWoojung.Huh@microchip.com 	}
352955d7de9dSWoojung.Huh@microchip.com }
353055d7de9dSWoojung.Huh@microchip.com 
lan78xx_rx_vlan_offload(struct lan78xx_net * dev,struct sk_buff * skb,u32 rx_cmd_a,u32 rx_cmd_b)3531ec21ecf0SDave Stevenson static void lan78xx_rx_vlan_offload(struct lan78xx_net *dev,
3532ec21ecf0SDave Stevenson 				    struct sk_buff *skb,
3533ec21ecf0SDave Stevenson 				    u32 rx_cmd_a, u32 rx_cmd_b)
3534ec21ecf0SDave Stevenson {
3535ec21ecf0SDave Stevenson 	if ((dev->net->features & NETIF_F_HW_VLAN_CTAG_RX) &&
3536ec21ecf0SDave Stevenson 	    (rx_cmd_a & RX_CMD_A_FVTG_))
3537ec21ecf0SDave Stevenson 		__vlan_hwaccel_put_tag(skb, htons(ETH_P_8021Q),
3538ec21ecf0SDave Stevenson 				       (rx_cmd_b & 0xffff));
3539ec21ecf0SDave Stevenson }
3540ec21ecf0SDave Stevenson 
lan78xx_skb_return(struct lan78xx_net * dev,struct sk_buff * skb)3541e0c79ff6SBaoyou Xie static void lan78xx_skb_return(struct lan78xx_net *dev, struct sk_buff *skb)
354255d7de9dSWoojung.Huh@microchip.com {
354355d7de9dSWoojung.Huh@microchip.com 	dev->net->stats.rx_packets++;
354455d7de9dSWoojung.Huh@microchip.com 	dev->net->stats.rx_bytes += skb->len;
354555d7de9dSWoojung.Huh@microchip.com 
354674d79a2eSWoojung Huh 	skb->protocol = eth_type_trans(skb, dev->net);
354774d79a2eSWoojung Huh 
354855d7de9dSWoojung.Huh@microchip.com 	netif_dbg(dev, rx_status, dev->net, "< rx, len %zu, type 0x%x\n",
354955d7de9dSWoojung.Huh@microchip.com 		  skb->len + sizeof(struct ethhdr), skb->protocol);
355055d7de9dSWoojung.Huh@microchip.com 	memset(skb->cb, 0, sizeof(struct skb_data));
355155d7de9dSWoojung.Huh@microchip.com 
355255d7de9dSWoojung.Huh@microchip.com 	if (skb_defer_rx_timestamp(skb))
355355d7de9dSWoojung.Huh@microchip.com 		return;
355455d7de9dSWoojung.Huh@microchip.com 
3555ec4c7e12SJohn Efstathiades 	napi_gro_receive(&dev->napi, skb);
355655d7de9dSWoojung.Huh@microchip.com }
355755d7de9dSWoojung.Huh@microchip.com 
lan78xx_rx(struct lan78xx_net * dev,struct sk_buff * skb,int budget,int * work_done)3558ec4c7e12SJohn Efstathiades static int lan78xx_rx(struct lan78xx_net *dev, struct sk_buff *skb,
3559ec4c7e12SJohn Efstathiades 		      int budget, int *work_done)
356055d7de9dSWoojung.Huh@microchip.com {
35610dd87266SJohn Efstathiades 	if (skb->len < RX_SKB_MIN_LEN)
356255d7de9dSWoojung.Huh@microchip.com 		return 0;
356355d7de9dSWoojung.Huh@microchip.com 
3564ec4c7e12SJohn Efstathiades 	/* Extract frames from the URB buffer and pass each one to
3565ec4c7e12SJohn Efstathiades 	 * the stack in a new NAPI SKB.
3566ec4c7e12SJohn Efstathiades 	 */
356755d7de9dSWoojung.Huh@microchip.com 	while (skb->len > 0) {
356855d7de9dSWoojung.Huh@microchip.com 		u32 rx_cmd_a, rx_cmd_b, align_count, size;
356955d7de9dSWoojung.Huh@microchip.com 		u16 rx_cmd_c;
357055d7de9dSWoojung.Huh@microchip.com 		unsigned char *packet;
357155d7de9dSWoojung.Huh@microchip.com 
3572bb448f8aSChuhong Yuan 		rx_cmd_a = get_unaligned_le32(skb->data);
357355d7de9dSWoojung.Huh@microchip.com 		skb_pull(skb, sizeof(rx_cmd_a));
357455d7de9dSWoojung.Huh@microchip.com 
3575bb448f8aSChuhong Yuan 		rx_cmd_b = get_unaligned_le32(skb->data);
357655d7de9dSWoojung.Huh@microchip.com 		skb_pull(skb, sizeof(rx_cmd_b));
357755d7de9dSWoojung.Huh@microchip.com 
3578bb448f8aSChuhong Yuan 		rx_cmd_c = get_unaligned_le16(skb->data);
357955d7de9dSWoojung.Huh@microchip.com 		skb_pull(skb, sizeof(rx_cmd_c));
358055d7de9dSWoojung.Huh@microchip.com 
358155d7de9dSWoojung.Huh@microchip.com 		packet = skb->data;
358255d7de9dSWoojung.Huh@microchip.com 
358355d7de9dSWoojung.Huh@microchip.com 		/* get the packet length */
358455d7de9dSWoojung.Huh@microchip.com 		size = (rx_cmd_a & RX_CMD_A_LEN_MASK_);
358555d7de9dSWoojung.Huh@microchip.com 		align_count = (4 - ((size + RXW_PADDING) % 4)) % 4;
358655d7de9dSWoojung.Huh@microchip.com 
35877f247f5aSSzymon Heidrich 		if (unlikely(size > skb->len)) {
35887f247f5aSSzymon Heidrich 			netif_dbg(dev, rx_err, dev->net,
35897f247f5aSSzymon Heidrich 				  "size err rx_cmd_a=0x%08x\n",
35907f247f5aSSzymon Heidrich 				  rx_cmd_a);
35917f247f5aSSzymon Heidrich 			return 0;
35927f247f5aSSzymon Heidrich 		}
35937f247f5aSSzymon Heidrich 
359455d7de9dSWoojung.Huh@microchip.com 		if (unlikely(rx_cmd_a & RX_CMD_A_RED_)) {
359555d7de9dSWoojung.Huh@microchip.com 			netif_dbg(dev, rx_err, dev->net,
359655d7de9dSWoojung.Huh@microchip.com 				  "Error rx_cmd_a=0x%08x", rx_cmd_a);
359755d7de9dSWoojung.Huh@microchip.com 		} else {
35987f247f5aSSzymon Heidrich 			u32 frame_len;
3599ec4c7e12SJohn Efstathiades 			struct sk_buff *skb2;
360055d7de9dSWoojung.Huh@microchip.com 
36017f247f5aSSzymon Heidrich 			if (unlikely(size < ETH_FCS_LEN)) {
36027f247f5aSSzymon Heidrich 				netif_dbg(dev, rx_err, dev->net,
36037f247f5aSSzymon Heidrich 					  "size err rx_cmd_a=0x%08x\n",
36047f247f5aSSzymon Heidrich 					  rx_cmd_a);
36057f247f5aSSzymon Heidrich 				return 0;
36067f247f5aSSzymon Heidrich 			}
36077f247f5aSSzymon Heidrich 
36087f247f5aSSzymon Heidrich 			frame_len = size - ETH_FCS_LEN;
36097f247f5aSSzymon Heidrich 
3610ec4c7e12SJohn Efstathiades 			skb2 = napi_alloc_skb(&dev->napi, frame_len);
3611ec4c7e12SJohn Efstathiades 			if (!skb2)
361255d7de9dSWoojung.Huh@microchip.com 				return 0;
361355d7de9dSWoojung.Huh@microchip.com 
3614ec4c7e12SJohn Efstathiades 			memcpy(skb2->data, packet, frame_len);
3615ec4c7e12SJohn Efstathiades 
3616ec4c7e12SJohn Efstathiades 			skb_put(skb2, frame_len);
361755d7de9dSWoojung.Huh@microchip.com 
361855d7de9dSWoojung.Huh@microchip.com 			lan78xx_rx_csum_offload(dev, skb2, rx_cmd_a, rx_cmd_b);
3619ec21ecf0SDave Stevenson 			lan78xx_rx_vlan_offload(dev, skb2, rx_cmd_a, rx_cmd_b);
362055d7de9dSWoojung.Huh@microchip.com 
3621ec4c7e12SJohn Efstathiades 			/* Processing of the URB buffer must complete once
3622ec4c7e12SJohn Efstathiades 			 * it has started. If the NAPI work budget is exhausted
3623ec4c7e12SJohn Efstathiades 			 * while frames remain they are added to the overflow
3624ec4c7e12SJohn Efstathiades 			 * queue for delivery in the next NAPI polling cycle.
3625ec4c7e12SJohn Efstathiades 			 */
3626ec4c7e12SJohn Efstathiades 			if (*work_done < budget) {
362755d7de9dSWoojung.Huh@microchip.com 				lan78xx_skb_return(dev, skb2);
3628ec4c7e12SJohn Efstathiades 				++(*work_done);
3629ec4c7e12SJohn Efstathiades 			} else {
3630ec4c7e12SJohn Efstathiades 				skb_queue_tail(&dev->rxq_overflow, skb2);
3631ec4c7e12SJohn Efstathiades 			}
363255d7de9dSWoojung.Huh@microchip.com 		}
363355d7de9dSWoojung.Huh@microchip.com 
363455d7de9dSWoojung.Huh@microchip.com 		skb_pull(skb, size);
363555d7de9dSWoojung.Huh@microchip.com 
3636ec4c7e12SJohn Efstathiades 		/* skip padding bytes before the next frame starts */
363755d7de9dSWoojung.Huh@microchip.com 		if (skb->len)
363855d7de9dSWoojung.Huh@microchip.com 			skb_pull(skb, align_count);
363955d7de9dSWoojung.Huh@microchip.com 	}
364055d7de9dSWoojung.Huh@microchip.com 
364155d7de9dSWoojung.Huh@microchip.com 	return 1;
364255d7de9dSWoojung.Huh@microchip.com }
364355d7de9dSWoojung.Huh@microchip.com 
rx_process(struct lan78xx_net * dev,struct sk_buff * skb,int budget,int * work_done)3644ec4c7e12SJohn Efstathiades static inline void rx_process(struct lan78xx_net *dev, struct sk_buff *skb,
3645ec4c7e12SJohn Efstathiades 			      int budget, int *work_done)
364655d7de9dSWoojung.Huh@microchip.com {
3647ec4c7e12SJohn Efstathiades 	if (!lan78xx_rx(dev, skb, budget, work_done)) {
364855d7de9dSWoojung.Huh@microchip.com 		netif_dbg(dev, rx_err, dev->net, "drop\n");
364955d7de9dSWoojung.Huh@microchip.com 		dev->net->stats.rx_errors++;
365055d7de9dSWoojung.Huh@microchip.com 	}
3651ec4c7e12SJohn Efstathiades }
365255d7de9dSWoojung.Huh@microchip.com 
rx_complete(struct urb * urb)365355d7de9dSWoojung.Huh@microchip.com static void rx_complete(struct urb *urb)
365455d7de9dSWoojung.Huh@microchip.com {
365555d7de9dSWoojung.Huh@microchip.com 	struct sk_buff	*skb = (struct sk_buff *)urb->context;
365655d7de9dSWoojung.Huh@microchip.com 	struct skb_data	*entry = (struct skb_data *)skb->cb;
365755d7de9dSWoojung.Huh@microchip.com 	struct lan78xx_net *dev = entry->dev;
365855d7de9dSWoojung.Huh@microchip.com 	int urb_status = urb->status;
365955d7de9dSWoojung.Huh@microchip.com 	enum skb_state state;
366055d7de9dSWoojung.Huh@microchip.com 
3661c450a8ebSJohn Efstathiades 	netif_dbg(dev, rx_status, dev->net,
3662c450a8ebSJohn Efstathiades 		  "rx done: status %d", urb->status);
3663c450a8ebSJohn Efstathiades 
366455d7de9dSWoojung.Huh@microchip.com 	skb_put(skb, urb->actual_length);
366555d7de9dSWoojung.Huh@microchip.com 	state = rx_done;
3666c450a8ebSJohn Efstathiades 
3667c450a8ebSJohn Efstathiades 	if (urb != entry->urb)
3668c450a8ebSJohn Efstathiades 		netif_warn(dev, rx_err, dev->net, "URB pointer mismatch");
366955d7de9dSWoojung.Huh@microchip.com 
367055d7de9dSWoojung.Huh@microchip.com 	switch (urb_status) {
367155d7de9dSWoojung.Huh@microchip.com 	case 0:
36720dd87266SJohn Efstathiades 		if (skb->len < RX_SKB_MIN_LEN) {
367355d7de9dSWoojung.Huh@microchip.com 			state = rx_cleanup;
367455d7de9dSWoojung.Huh@microchip.com 			dev->net->stats.rx_errors++;
367555d7de9dSWoojung.Huh@microchip.com 			dev->net->stats.rx_length_errors++;
367655d7de9dSWoojung.Huh@microchip.com 			netif_dbg(dev, rx_err, dev->net,
367755d7de9dSWoojung.Huh@microchip.com 				  "rx length %d\n", skb->len);
367855d7de9dSWoojung.Huh@microchip.com 		}
367955d7de9dSWoojung.Huh@microchip.com 		usb_mark_last_busy(dev->udev);
368055d7de9dSWoojung.Huh@microchip.com 		break;
368155d7de9dSWoojung.Huh@microchip.com 	case -EPIPE:
368255d7de9dSWoojung.Huh@microchip.com 		dev->net->stats.rx_errors++;
368355d7de9dSWoojung.Huh@microchip.com 		lan78xx_defer_kevent(dev, EVENT_RX_HALT);
3684df561f66SGustavo A. R. Silva 		fallthrough;
368555d7de9dSWoojung.Huh@microchip.com 	case -ECONNRESET:				/* async unlink */
368655d7de9dSWoojung.Huh@microchip.com 	case -ESHUTDOWN:				/* hardware gone */
368755d7de9dSWoojung.Huh@microchip.com 		netif_dbg(dev, ifdown, dev->net,
368855d7de9dSWoojung.Huh@microchip.com 			  "rx shutdown, code %d\n", urb_status);
368955d7de9dSWoojung.Huh@microchip.com 		state = rx_cleanup;
369055d7de9dSWoojung.Huh@microchip.com 		break;
369155d7de9dSWoojung.Huh@microchip.com 	case -EPROTO:
369255d7de9dSWoojung.Huh@microchip.com 	case -ETIME:
369355d7de9dSWoojung.Huh@microchip.com 	case -EILSEQ:
369455d7de9dSWoojung.Huh@microchip.com 		dev->net->stats.rx_errors++;
369555d7de9dSWoojung.Huh@microchip.com 		state = rx_cleanup;
369655d7de9dSWoojung.Huh@microchip.com 		break;
369755d7de9dSWoojung.Huh@microchip.com 
369855d7de9dSWoojung.Huh@microchip.com 	/* data overrun ... flush fifo? */
369955d7de9dSWoojung.Huh@microchip.com 	case -EOVERFLOW:
370055d7de9dSWoojung.Huh@microchip.com 		dev->net->stats.rx_over_errors++;
3701df561f66SGustavo A. R. Silva 		fallthrough;
370255d7de9dSWoojung.Huh@microchip.com 
370355d7de9dSWoojung.Huh@microchip.com 	default:
370455d7de9dSWoojung.Huh@microchip.com 		state = rx_cleanup;
370555d7de9dSWoojung.Huh@microchip.com 		dev->net->stats.rx_errors++;
370655d7de9dSWoojung.Huh@microchip.com 		netif_dbg(dev, rx_err, dev->net, "rx status %d\n", urb_status);
370755d7de9dSWoojung.Huh@microchip.com 		break;
370855d7de9dSWoojung.Huh@microchip.com 	}
370955d7de9dSWoojung.Huh@microchip.com 
371055d7de9dSWoojung.Huh@microchip.com 	state = defer_bh(dev, skb, &dev->rxq, state);
3711c450a8ebSJohn Efstathiades }
371255d7de9dSWoojung.Huh@microchip.com 
rx_submit(struct lan78xx_net * dev,struct sk_buff * skb,gfp_t flags)37139d2da721SJohn Efstathiades static int rx_submit(struct lan78xx_net *dev, struct sk_buff *skb, gfp_t flags)
37149d2da721SJohn Efstathiades {
37159d2da721SJohn Efstathiades 	struct skb_data	*entry = (struct skb_data *)skb->cb;
37169d2da721SJohn Efstathiades 	size_t size = dev->rx_urb_size;
37179d2da721SJohn Efstathiades 	struct urb *urb = entry->urb;
37189d2da721SJohn Efstathiades 	unsigned long lockflags;
37199d2da721SJohn Efstathiades 	int ret = 0;
37209d2da721SJohn Efstathiades 
37219d2da721SJohn Efstathiades 	usb_fill_bulk_urb(urb, dev->udev, dev->pipe_in,
37229d2da721SJohn Efstathiades 			  skb->data, size, rx_complete, skb);
37239d2da721SJohn Efstathiades 
37249d2da721SJohn Efstathiades 	spin_lock_irqsave(&dev->rxq.lock, lockflags);
37259d2da721SJohn Efstathiades 
37269d2da721SJohn Efstathiades 	if (netif_device_present(dev->net) &&
37279d2da721SJohn Efstathiades 	    netif_running(dev->net) &&
37289d2da721SJohn Efstathiades 	    !test_bit(EVENT_RX_HALT, &dev->flags) &&
37299d2da721SJohn Efstathiades 	    !test_bit(EVENT_DEV_ASLEEP, &dev->flags)) {
37309d2da721SJohn Efstathiades 		ret = usb_submit_urb(urb, flags);
37319d2da721SJohn Efstathiades 		switch (ret) {
37329d2da721SJohn Efstathiades 		case 0:
37339d2da721SJohn Efstathiades 			lan78xx_queue_skb(&dev->rxq, skb, rx_start);
37349d2da721SJohn Efstathiades 			break;
37359d2da721SJohn Efstathiades 		case -EPIPE:
37369d2da721SJohn Efstathiades 			lan78xx_defer_kevent(dev, EVENT_RX_HALT);
37379d2da721SJohn Efstathiades 			break;
37389d2da721SJohn Efstathiades 		case -ENODEV:
37399d2da721SJohn Efstathiades 		case -ENOENT:
37409d2da721SJohn Efstathiades 			netif_dbg(dev, ifdown, dev->net, "device gone\n");
37419d2da721SJohn Efstathiades 			netif_device_detach(dev->net);
37429d2da721SJohn Efstathiades 			break;
37439d2da721SJohn Efstathiades 		case -EHOSTUNREACH:
37449d2da721SJohn Efstathiades 			ret = -ENOLINK;
3745ec4c7e12SJohn Efstathiades 			napi_schedule(&dev->napi);
37469d2da721SJohn Efstathiades 			break;
37479d2da721SJohn Efstathiades 		default:
37489d2da721SJohn Efstathiades 			netif_dbg(dev, rx_err, dev->net,
37499d2da721SJohn Efstathiades 				  "rx submit, %d\n", ret);
3750ec4c7e12SJohn Efstathiades 			napi_schedule(&dev->napi);
37519d2da721SJohn Efstathiades 			break;
37529d2da721SJohn Efstathiades 		}
37539d2da721SJohn Efstathiades 	} else {
37549d2da721SJohn Efstathiades 		netif_dbg(dev, ifdown, dev->net, "rx: stopped\n");
37559d2da721SJohn Efstathiades 		ret = -ENOLINK;
37569d2da721SJohn Efstathiades 	}
37579d2da721SJohn Efstathiades 	spin_unlock_irqrestore(&dev->rxq.lock, lockflags);
37589d2da721SJohn Efstathiades 
37599d2da721SJohn Efstathiades 	if (ret)
37609d2da721SJohn Efstathiades 		lan78xx_release_rx_buf(dev, skb);
37619d2da721SJohn Efstathiades 
37629d2da721SJohn Efstathiades 	return ret;
37639d2da721SJohn Efstathiades }
37649d2da721SJohn Efstathiades 
lan78xx_rx_urb_submit_all(struct lan78xx_net * dev)3765c450a8ebSJohn Efstathiades static void lan78xx_rx_urb_submit_all(struct lan78xx_net *dev)
3766c450a8ebSJohn Efstathiades {
3767c450a8ebSJohn Efstathiades 	struct sk_buff *rx_buf;
3768c450a8ebSJohn Efstathiades 
3769c450a8ebSJohn Efstathiades 	/* Ensure the maximum number of Rx URBs is submitted
3770c450a8ebSJohn Efstathiades 	 */
3771c450a8ebSJohn Efstathiades 	while ((rx_buf = lan78xx_get_rx_buf(dev)) != NULL) {
3772c450a8ebSJohn Efstathiades 		if (rx_submit(dev, rx_buf, GFP_ATOMIC) != 0)
3773c450a8ebSJohn Efstathiades 			break;
377455d7de9dSWoojung.Huh@microchip.com 	}
377555d7de9dSWoojung.Huh@microchip.com }
3776c450a8ebSJohn Efstathiades 
lan78xx_rx_urb_resubmit(struct lan78xx_net * dev,struct sk_buff * rx_buf)3777c450a8ebSJohn Efstathiades static void lan78xx_rx_urb_resubmit(struct lan78xx_net *dev,
3778c450a8ebSJohn Efstathiades 				    struct sk_buff *rx_buf)
3779c450a8ebSJohn Efstathiades {
3780c450a8ebSJohn Efstathiades 	/* reset SKB data pointers */
3781c450a8ebSJohn Efstathiades 
3782c450a8ebSJohn Efstathiades 	rx_buf->data = rx_buf->head;
3783c450a8ebSJohn Efstathiades 	skb_reset_tail_pointer(rx_buf);
3784c450a8ebSJohn Efstathiades 	rx_buf->len = 0;
3785c450a8ebSJohn Efstathiades 	rx_buf->data_len = 0;
3786c450a8ebSJohn Efstathiades 
3787c450a8ebSJohn Efstathiades 	rx_submit(dev, rx_buf, GFP_ATOMIC);
378855d7de9dSWoojung.Huh@microchip.com }
378955d7de9dSWoojung.Huh@microchip.com 
lan78xx_fill_tx_cmd_words(struct sk_buff * skb,u8 * buffer)3790d383216aSJohn Efstathiades static void lan78xx_fill_tx_cmd_words(struct sk_buff *skb, u8 *buffer)
3791d383216aSJohn Efstathiades {
3792d383216aSJohn Efstathiades 	u32 tx_cmd_a;
3793d383216aSJohn Efstathiades 	u32 tx_cmd_b;
3794d383216aSJohn Efstathiades 
3795d383216aSJohn Efstathiades 	tx_cmd_a = (u32)(skb->len & TX_CMD_A_LEN_MASK_) | TX_CMD_A_FCS_;
3796d383216aSJohn Efstathiades 
3797d383216aSJohn Efstathiades 	if (skb->ip_summed == CHECKSUM_PARTIAL)
3798d383216aSJohn Efstathiades 		tx_cmd_a |= TX_CMD_A_IPE_ | TX_CMD_A_TPE_;
3799d383216aSJohn Efstathiades 
3800d383216aSJohn Efstathiades 	tx_cmd_b = 0;
3801d383216aSJohn Efstathiades 	if (skb_is_gso(skb)) {
3802d383216aSJohn Efstathiades 		u16 mss = max(skb_shinfo(skb)->gso_size, TX_CMD_B_MSS_MIN_);
3803d383216aSJohn Efstathiades 
3804d383216aSJohn Efstathiades 		tx_cmd_b = (mss << TX_CMD_B_MSS_SHIFT_) & TX_CMD_B_MSS_MASK_;
3805d383216aSJohn Efstathiades 
3806d383216aSJohn Efstathiades 		tx_cmd_a |= TX_CMD_A_LSO_;
3807d383216aSJohn Efstathiades 	}
3808d383216aSJohn Efstathiades 
3809d383216aSJohn Efstathiades 	if (skb_vlan_tag_present(skb)) {
3810d383216aSJohn Efstathiades 		tx_cmd_a |= TX_CMD_A_IVTG_;
3811d383216aSJohn Efstathiades 		tx_cmd_b |= skb_vlan_tag_get(skb) & TX_CMD_B_VTAG_MASK_;
3812d383216aSJohn Efstathiades 	}
3813d383216aSJohn Efstathiades 
3814d383216aSJohn Efstathiades 	put_unaligned_le32(tx_cmd_a, buffer);
3815d383216aSJohn Efstathiades 	put_unaligned_le32(tx_cmd_b, buffer + 4);
3816d383216aSJohn Efstathiades }
3817d383216aSJohn Efstathiades 
lan78xx_tx_buf_fill(struct lan78xx_net * dev,struct sk_buff * tx_buf)3818d383216aSJohn Efstathiades static struct skb_data *lan78xx_tx_buf_fill(struct lan78xx_net *dev,
3819d383216aSJohn Efstathiades 					    struct sk_buff *tx_buf)
3820d383216aSJohn Efstathiades {
3821d383216aSJohn Efstathiades 	struct skb_data *entry = (struct skb_data *)tx_buf->cb;
3822d383216aSJohn Efstathiades 	int remain = dev->tx_urb_size;
3823d383216aSJohn Efstathiades 	u8 *tx_data = tx_buf->data;
3824d383216aSJohn Efstathiades 	u32 urb_len = 0;
3825d383216aSJohn Efstathiades 
3826d383216aSJohn Efstathiades 	entry->num_of_packet = 0;
3827d383216aSJohn Efstathiades 	entry->length = 0;
3828d383216aSJohn Efstathiades 
3829d383216aSJohn Efstathiades 	/* Work through the pending SKBs and copy the data of each SKB into
3830d383216aSJohn Efstathiades 	 * the URB buffer if there room for all the SKB data.
3831d383216aSJohn Efstathiades 	 *
3832d383216aSJohn Efstathiades 	 * There must be at least DST+SRC+TYPE in the SKB (with padding enabled)
3833d383216aSJohn Efstathiades 	 */
3834d383216aSJohn Efstathiades 	while (remain >= TX_SKB_MIN_LEN) {
3835d383216aSJohn Efstathiades 		unsigned int pending_bytes;
3836d383216aSJohn Efstathiades 		unsigned int align_bytes;
3837d383216aSJohn Efstathiades 		struct sk_buff *skb;
3838d383216aSJohn Efstathiades 		unsigned int len;
3839d383216aSJohn Efstathiades 
3840d383216aSJohn Efstathiades 		lan78xx_tx_pend_skb_get(dev, &skb, &pending_bytes);
3841d383216aSJohn Efstathiades 
3842d383216aSJohn Efstathiades 		if (!skb)
3843d383216aSJohn Efstathiades 			break;
3844d383216aSJohn Efstathiades 
3845d383216aSJohn Efstathiades 		align_bytes = (TX_ALIGNMENT - (urb_len % TX_ALIGNMENT)) %
3846d383216aSJohn Efstathiades 			      TX_ALIGNMENT;
3847d383216aSJohn Efstathiades 		len = align_bytes + TX_CMD_LEN + skb->len;
3848d383216aSJohn Efstathiades 		if (len > remain) {
3849d383216aSJohn Efstathiades 			lan78xx_tx_pend_skb_head_add(dev, skb, &pending_bytes);
3850d383216aSJohn Efstathiades 			break;
3851d383216aSJohn Efstathiades 		}
3852d383216aSJohn Efstathiades 
3853d383216aSJohn Efstathiades 		tx_data += align_bytes;
3854d383216aSJohn Efstathiades 
3855d383216aSJohn Efstathiades 		lan78xx_fill_tx_cmd_words(skb, tx_data);
3856d383216aSJohn Efstathiades 		tx_data += TX_CMD_LEN;
3857d383216aSJohn Efstathiades 
3858d383216aSJohn Efstathiades 		len = skb->len;
3859d383216aSJohn Efstathiades 		if (skb_copy_bits(skb, 0, tx_data, len) < 0) {
3860d383216aSJohn Efstathiades 			struct net_device_stats *stats = &dev->net->stats;
3861d383216aSJohn Efstathiades 
3862d383216aSJohn Efstathiades 			stats->tx_dropped++;
3863d383216aSJohn Efstathiades 			dev_kfree_skb_any(skb);
3864d383216aSJohn Efstathiades 			tx_data -= TX_CMD_LEN;
3865d383216aSJohn Efstathiades 			continue;
3866d383216aSJohn Efstathiades 		}
3867d383216aSJohn Efstathiades 
3868d383216aSJohn Efstathiades 		tx_data += len;
3869d383216aSJohn Efstathiades 		entry->length += len;
3870d383216aSJohn Efstathiades 		entry->num_of_packet += skb_shinfo(skb)->gso_segs ?: 1;
3871d383216aSJohn Efstathiades 
3872d383216aSJohn Efstathiades 		dev_kfree_skb_any(skb);
3873d383216aSJohn Efstathiades 
3874d383216aSJohn Efstathiades 		urb_len = (u32)(tx_data - (u8 *)tx_buf->data);
3875d383216aSJohn Efstathiades 
3876d383216aSJohn Efstathiades 		remain = dev->tx_urb_size - urb_len;
3877d383216aSJohn Efstathiades 	}
3878d383216aSJohn Efstathiades 
3879d383216aSJohn Efstathiades 	skb_put(tx_buf, urb_len);
3880d383216aSJohn Efstathiades 
3881d383216aSJohn Efstathiades 	return entry;
3882d383216aSJohn Efstathiades }
3883d383216aSJohn Efstathiades 
lan78xx_tx_bh(struct lan78xx_net * dev)388455d7de9dSWoojung.Huh@microchip.com static void lan78xx_tx_bh(struct lan78xx_net *dev)
388555d7de9dSWoojung.Huh@microchip.com {
388655d7de9dSWoojung.Huh@microchip.com 	int ret;
388755d7de9dSWoojung.Huh@microchip.com 
3888d383216aSJohn Efstathiades 	/* Start the stack Tx queue if it was stopped
3889d383216aSJohn Efstathiades 	 */
3890d383216aSJohn Efstathiades 	netif_tx_lock(dev->net);
3891d383216aSJohn Efstathiades 	if (netif_queue_stopped(dev->net)) {
3892d383216aSJohn Efstathiades 		if (lan78xx_tx_pend_data_len(dev) < lan78xx_tx_urb_space(dev))
3893d383216aSJohn Efstathiades 			netif_wake_queue(dev->net);
3894d383216aSJohn Efstathiades 	}
3895d383216aSJohn Efstathiades 	netif_tx_unlock(dev->net);
3896d383216aSJohn Efstathiades 
3897d383216aSJohn Efstathiades 	/* Go through the Tx pending queue and set up URBs to transfer
3898d383216aSJohn Efstathiades 	 * the data to the device. Stop if no more pending data or URBs,
3899d383216aSJohn Efstathiades 	 * or if an error occurs when a URB is submitted.
3900d383216aSJohn Efstathiades 	 */
3901d383216aSJohn Efstathiades 	do {
3902d383216aSJohn Efstathiades 		struct skb_data *entry;
3903d383216aSJohn Efstathiades 		struct sk_buff *tx_buf;
3904d383216aSJohn Efstathiades 		unsigned long flags;
3905d383216aSJohn Efstathiades 
3906d383216aSJohn Efstathiades 		if (skb_queue_empty(&dev->txq_pend))
390755d7de9dSWoojung.Huh@microchip.com 			break;
390855d7de9dSWoojung.Huh@microchip.com 
3909d383216aSJohn Efstathiades 		tx_buf = lan78xx_get_tx_buf(dev);
3910d383216aSJohn Efstathiades 		if (!tx_buf)
391155d7de9dSWoojung.Huh@microchip.com 			break;
391255d7de9dSWoojung.Huh@microchip.com 
3913d383216aSJohn Efstathiades 		entry = lan78xx_tx_buf_fill(dev, tx_buf);
391455d7de9dSWoojung.Huh@microchip.com 
391555d7de9dSWoojung.Huh@microchip.com 		spin_lock_irqsave(&dev->txq.lock, flags);
391655d7de9dSWoojung.Huh@microchip.com 		ret = usb_autopm_get_interface_async(dev->intf);
391755d7de9dSWoojung.Huh@microchip.com 		if (ret < 0) {
391855d7de9dSWoojung.Huh@microchip.com 			spin_unlock_irqrestore(&dev->txq.lock, flags);
3919d383216aSJohn Efstathiades 			goto out;
392055d7de9dSWoojung.Huh@microchip.com 		}
392155d7de9dSWoojung.Huh@microchip.com 
3922d383216aSJohn Efstathiades 		usb_fill_bulk_urb(entry->urb, dev->udev, dev->pipe_out,
3923d383216aSJohn Efstathiades 				  tx_buf->data, tx_buf->len, tx_complete,
3924d383216aSJohn Efstathiades 				  tx_buf);
392555d7de9dSWoojung.Huh@microchip.com 
3926d383216aSJohn Efstathiades 		if (tx_buf->len % dev->maxpacket == 0) {
392755d7de9dSWoojung.Huh@microchip.com 			/* send USB_ZERO_PACKET */
3928d383216aSJohn Efstathiades 			entry->urb->transfer_flags |= URB_ZERO_PACKET;
392955d7de9dSWoojung.Huh@microchip.com 		}
393055d7de9dSWoojung.Huh@microchip.com 
393155d7de9dSWoojung.Huh@microchip.com #ifdef CONFIG_PM
3932d383216aSJohn Efstathiades 		/* if device is asleep stop outgoing packet processing */
393355d7de9dSWoojung.Huh@microchip.com 		if (test_bit(EVENT_DEV_ASLEEP, &dev->flags)) {
3934d383216aSJohn Efstathiades 			usb_anchor_urb(entry->urb, &dev->deferred);
393555d7de9dSWoojung.Huh@microchip.com 			netif_stop_queue(dev->net);
393655d7de9dSWoojung.Huh@microchip.com 			spin_unlock_irqrestore(&dev->txq.lock, flags);
3937d383216aSJohn Efstathiades 			netdev_dbg(dev->net,
3938d383216aSJohn Efstathiades 				   "Delaying transmission for resumption\n");
393955d7de9dSWoojung.Huh@microchip.com 			return;
394055d7de9dSWoojung.Huh@microchip.com 		}
394155d7de9dSWoojung.Huh@microchip.com #endif
3942d383216aSJohn Efstathiades 		ret = usb_submit_urb(entry->urb, GFP_ATOMIC);
394355d7de9dSWoojung.Huh@microchip.com 		switch (ret) {
394455d7de9dSWoojung.Huh@microchip.com 		case 0:
3945860e9538SFlorian Westphal 			netif_trans_update(dev->net);
3946d383216aSJohn Efstathiades 			lan78xx_queue_skb(&dev->txq, tx_buf, tx_start);
394755d7de9dSWoojung.Huh@microchip.com 			break;
394855d7de9dSWoojung.Huh@microchip.com 		case -EPIPE:
394955d7de9dSWoojung.Huh@microchip.com 			netif_stop_queue(dev->net);
395055d7de9dSWoojung.Huh@microchip.com 			lan78xx_defer_kevent(dev, EVENT_TX_HALT);
395155d7de9dSWoojung.Huh@microchip.com 			usb_autopm_put_interface_async(dev->intf);
395255d7de9dSWoojung.Huh@microchip.com 			break;
395377dfff5bSJohn Efstathiades 		case -ENODEV:
395477dfff5bSJohn Efstathiades 		case -ENOENT:
395577dfff5bSJohn Efstathiades 			netif_dbg(dev, tx_err, dev->net,
3956d383216aSJohn Efstathiades 				  "tx submit urb err %d (disconnected?)", ret);
395777dfff5bSJohn Efstathiades 			netif_device_detach(dev->net);
395877dfff5bSJohn Efstathiades 			break;
395955d7de9dSWoojung.Huh@microchip.com 		default:
396055d7de9dSWoojung.Huh@microchip.com 			usb_autopm_put_interface_async(dev->intf);
396155d7de9dSWoojung.Huh@microchip.com 			netif_dbg(dev, tx_err, dev->net,
3962d383216aSJohn Efstathiades 				  "tx submit urb err %d\n", ret);
396355d7de9dSWoojung.Huh@microchip.com 			break;
396455d7de9dSWoojung.Huh@microchip.com 		}
396555d7de9dSWoojung.Huh@microchip.com 
396655d7de9dSWoojung.Huh@microchip.com 		spin_unlock_irqrestore(&dev->txq.lock, flags);
396755d7de9dSWoojung.Huh@microchip.com 
396855d7de9dSWoojung.Huh@microchip.com 		if (ret) {
3969d383216aSJohn Efstathiades 			netdev_warn(dev->net, "failed to tx urb %d\n", ret);
3970d383216aSJohn Efstathiades out:
3971d383216aSJohn Efstathiades 			dev->net->stats.tx_dropped += entry->num_of_packet;
3972d383216aSJohn Efstathiades 			lan78xx_release_tx_buf(dev, tx_buf);
397355d7de9dSWoojung.Huh@microchip.com 		}
3974d383216aSJohn Efstathiades 	} while (ret == 0);
39759ceec7d3SJohn Efstathiades }
397655d7de9dSWoojung.Huh@microchip.com 
lan78xx_bh(struct lan78xx_net * dev,int budget)3977ec4c7e12SJohn Efstathiades static int lan78xx_bh(struct lan78xx_net *dev, int budget)
397855d7de9dSWoojung.Huh@microchip.com {
3979c450a8ebSJohn Efstathiades 	struct sk_buff_head done;
3980c450a8ebSJohn Efstathiades 	struct sk_buff *rx_buf;
398155d7de9dSWoojung.Huh@microchip.com 	struct skb_data *entry;
3982c450a8ebSJohn Efstathiades 	unsigned long flags;
3983ec4c7e12SJohn Efstathiades 	int work_done = 0;
3984ec4c7e12SJohn Efstathiades 
3985ec4c7e12SJohn Efstathiades 	/* Pass frames received in the last NAPI cycle before
3986ec4c7e12SJohn Efstathiades 	 * working on newly completed URBs.
3987ec4c7e12SJohn Efstathiades 	 */
3988ec4c7e12SJohn Efstathiades 	while (!skb_queue_empty(&dev->rxq_overflow)) {
3989ec4c7e12SJohn Efstathiades 		lan78xx_skb_return(dev, skb_dequeue(&dev->rxq_overflow));
3990ec4c7e12SJohn Efstathiades 		++work_done;
3991ec4c7e12SJohn Efstathiades 	}
399255d7de9dSWoojung.Huh@microchip.com 
3993c450a8ebSJohn Efstathiades 	/* Take a snapshot of the done queue and move items to a
3994c450a8ebSJohn Efstathiades 	 * temporary queue. Rx URB completions will continue to add
3995c450a8ebSJohn Efstathiades 	 * to the done queue.
3996c450a8ebSJohn Efstathiades 	 */
3997c450a8ebSJohn Efstathiades 	__skb_queue_head_init(&done);
3998c450a8ebSJohn Efstathiades 
3999c450a8ebSJohn Efstathiades 	spin_lock_irqsave(&dev->rxq_done.lock, flags);
4000c450a8ebSJohn Efstathiades 	skb_queue_splice_init(&dev->rxq_done, &done);
4001c450a8ebSJohn Efstathiades 	spin_unlock_irqrestore(&dev->rxq_done.lock, flags);
4002c450a8ebSJohn Efstathiades 
4003c450a8ebSJohn Efstathiades 	/* Extract receive frames from completed URBs and
4004c450a8ebSJohn Efstathiades 	 * pass them to the stack. Re-submit each completed URB.
4005c450a8ebSJohn Efstathiades 	 */
4006ec4c7e12SJohn Efstathiades 	while ((work_done < budget) &&
4007ec4c7e12SJohn Efstathiades 	       (rx_buf = __skb_dequeue(&done))) {
4008c450a8ebSJohn Efstathiades 		entry = (struct skb_data *)(rx_buf->cb);
400955d7de9dSWoojung.Huh@microchip.com 		switch (entry->state) {
401055d7de9dSWoojung.Huh@microchip.com 		case rx_done:
4011ec4c7e12SJohn Efstathiades 			rx_process(dev, rx_buf, budget, &work_done);
4012c450a8ebSJohn Efstathiades 			break;
401355d7de9dSWoojung.Huh@microchip.com 		case rx_cleanup:
4014c450a8ebSJohn Efstathiades 			break;
401555d7de9dSWoojung.Huh@microchip.com 		default:
4016ec4c7e12SJohn Efstathiades 			netdev_dbg(dev->net, "rx buf state %d\n",
4017ec4c7e12SJohn Efstathiades 				   entry->state);
4018c450a8ebSJohn Efstathiades 			break;
401955d7de9dSWoojung.Huh@microchip.com 		}
4020c450a8ebSJohn Efstathiades 
4021c450a8ebSJohn Efstathiades 		lan78xx_rx_urb_resubmit(dev, rx_buf);
402255d7de9dSWoojung.Huh@microchip.com 	}
402355d7de9dSWoojung.Huh@microchip.com 
4024ec4c7e12SJohn Efstathiades 	/* If budget was consumed before processing all the URBs put them
4025ec4c7e12SJohn Efstathiades 	 * back on the front of the done queue. They will be first to be
4026ec4c7e12SJohn Efstathiades 	 * processed in the next NAPI cycle.
4027ec4c7e12SJohn Efstathiades 	 */
4028ec4c7e12SJohn Efstathiades 	spin_lock_irqsave(&dev->rxq_done.lock, flags);
4029ec4c7e12SJohn Efstathiades 	skb_queue_splice(&done, &dev->rxq_done);
4030ec4c7e12SJohn Efstathiades 	spin_unlock_irqrestore(&dev->rxq_done.lock, flags);
4031ec4c7e12SJohn Efstathiades 
403255d7de9dSWoojung.Huh@microchip.com 	if (netif_device_present(dev->net) && netif_running(dev->net)) {
403320ff5565SWoojung Huh 		/* reset update timer delta */
403420ff5565SWoojung Huh 		if (timer_pending(&dev->stat_monitor) && (dev->delta != 1)) {
403520ff5565SWoojung Huh 			dev->delta = 1;
403620ff5565SWoojung Huh 			mod_timer(&dev->stat_monitor,
403720ff5565SWoojung Huh 				  jiffies + STAT_UPDATE_TIMER);
403820ff5565SWoojung Huh 		}
403920ff5565SWoojung Huh 
4040ec4c7e12SJohn Efstathiades 		/* Submit all free Rx URBs */
4041ec4c7e12SJohn Efstathiades 
40423bef6b9eSJohn Efstathiades 		if (!test_bit(EVENT_RX_HALT, &dev->flags))
4043c450a8ebSJohn Efstathiades 			lan78xx_rx_urb_submit_all(dev);
4044d383216aSJohn Efstathiades 
4045ec4c7e12SJohn Efstathiades 		/* Submit new Tx URBs */
4046ec4c7e12SJohn Efstathiades 
4047d383216aSJohn Efstathiades 		lan78xx_tx_bh(dev);
4048ec4c7e12SJohn Efstathiades 	}
4049ec4c7e12SJohn Efstathiades 
4050ec4c7e12SJohn Efstathiades 	return work_done;
4051ec4c7e12SJohn Efstathiades }
4052ec4c7e12SJohn Efstathiades 
lan78xx_poll(struct napi_struct * napi,int budget)4053ec4c7e12SJohn Efstathiades static int lan78xx_poll(struct napi_struct *napi, int budget)
4054ec4c7e12SJohn Efstathiades {
4055ec4c7e12SJohn Efstathiades 	struct lan78xx_net *dev = container_of(napi, struct lan78xx_net, napi);
4056ec4c7e12SJohn Efstathiades 	int result = budget;
4057ec4c7e12SJohn Efstathiades 	int work_done;
4058ec4c7e12SJohn Efstathiades 
4059ec4c7e12SJohn Efstathiades 	/* Don't do any work if the device is suspended */
4060ec4c7e12SJohn Efstathiades 
4061ec4c7e12SJohn Efstathiades 	if (test_bit(EVENT_DEV_ASLEEP, &dev->flags)) {
4062ec4c7e12SJohn Efstathiades 		napi_complete_done(napi, 0);
4063ec4c7e12SJohn Efstathiades 		return 0;
4064ec4c7e12SJohn Efstathiades 	}
4065ec4c7e12SJohn Efstathiades 
4066ec4c7e12SJohn Efstathiades 	/* Process completed URBs and submit new URBs */
4067ec4c7e12SJohn Efstathiades 
4068ec4c7e12SJohn Efstathiades 	work_done = lan78xx_bh(dev, budget);
4069ec4c7e12SJohn Efstathiades 
4070ec4c7e12SJohn Efstathiades 	if (work_done < budget) {
4071ec4c7e12SJohn Efstathiades 		napi_complete_done(napi, work_done);
4072d383216aSJohn Efstathiades 
4073c450a8ebSJohn Efstathiades 		/* Start a new polling cycle if data was received or
4074c450a8ebSJohn Efstathiades 		 * data is waiting to be transmitted.
4075c450a8ebSJohn Efstathiades 		 */
4076c450a8ebSJohn Efstathiades 		if (!skb_queue_empty(&dev->rxq_done)) {
4077ec4c7e12SJohn Efstathiades 			napi_schedule(napi);
4078d383216aSJohn Efstathiades 		} else if (netif_carrier_ok(dev->net)) {
4079d383216aSJohn Efstathiades 			if (skb_queue_empty(&dev->txq) &&
4080d383216aSJohn Efstathiades 			    !skb_queue_empty(&dev->txq_pend)) {
4081ec4c7e12SJohn Efstathiades 				napi_schedule(napi);
4082d383216aSJohn Efstathiades 			} else {
4083d383216aSJohn Efstathiades 				netif_tx_lock(dev->net);
4084d383216aSJohn Efstathiades 				if (netif_queue_stopped(dev->net)) {
4085d383216aSJohn Efstathiades 					netif_wake_queue(dev->net);
4086ec4c7e12SJohn Efstathiades 					napi_schedule(napi);
4087d383216aSJohn Efstathiades 				}
4088d383216aSJohn Efstathiades 				netif_tx_unlock(dev->net);
4089d383216aSJohn Efstathiades 			}
4090d383216aSJohn Efstathiades 		}
4091ec4c7e12SJohn Efstathiades 		result = work_done;
409255d7de9dSWoojung.Huh@microchip.com 	}
4093ec4c7e12SJohn Efstathiades 
4094ec4c7e12SJohn Efstathiades 	return result;
409555d7de9dSWoojung.Huh@microchip.com }
409655d7de9dSWoojung.Huh@microchip.com 
lan78xx_delayedwork(struct work_struct * work)409755d7de9dSWoojung.Huh@microchip.com static void lan78xx_delayedwork(struct work_struct *work)
409855d7de9dSWoojung.Huh@microchip.com {
409955d7de9dSWoojung.Huh@microchip.com 	int status;
410055d7de9dSWoojung.Huh@microchip.com 	struct lan78xx_net *dev;
410155d7de9dSWoojung.Huh@microchip.com 
410255d7de9dSWoojung.Huh@microchip.com 	dev = container_of(work, struct lan78xx_net, wq.work);
410355d7de9dSWoojung.Huh@microchip.com 
410477dfff5bSJohn Efstathiades 	if (test_bit(EVENT_DEV_DISCONNECT, &dev->flags))
410577dfff5bSJohn Efstathiades 		return;
410677dfff5bSJohn Efstathiades 
41075f4cc6e2SJohn Efstathiades 	if (usb_autopm_get_interface(dev->intf) < 0)
41085f4cc6e2SJohn Efstathiades 		return;
41095f4cc6e2SJohn Efstathiades 
411055d7de9dSWoojung.Huh@microchip.com 	if (test_bit(EVENT_TX_HALT, &dev->flags)) {
411155d7de9dSWoojung.Huh@microchip.com 		unlink_urbs(dev, &dev->txq);
41125f4cc6e2SJohn Efstathiades 
411355d7de9dSWoojung.Huh@microchip.com 		status = usb_clear_halt(dev->udev, dev->pipe_out);
411455d7de9dSWoojung.Huh@microchip.com 		if (status < 0 &&
411555d7de9dSWoojung.Huh@microchip.com 		    status != -EPIPE &&
411655d7de9dSWoojung.Huh@microchip.com 		    status != -ESHUTDOWN) {
411755d7de9dSWoojung.Huh@microchip.com 			if (netif_msg_tx_err(dev))
411855d7de9dSWoojung.Huh@microchip.com 				netdev_err(dev->net,
411955d7de9dSWoojung.Huh@microchip.com 					   "can't clear tx halt, status %d\n",
412055d7de9dSWoojung.Huh@microchip.com 					   status);
412155d7de9dSWoojung.Huh@microchip.com 		} else {
412255d7de9dSWoojung.Huh@microchip.com 			clear_bit(EVENT_TX_HALT, &dev->flags);
412355d7de9dSWoojung.Huh@microchip.com 			if (status != -ESHUTDOWN)
412455d7de9dSWoojung.Huh@microchip.com 				netif_wake_queue(dev->net);
412555d7de9dSWoojung.Huh@microchip.com 		}
412655d7de9dSWoojung.Huh@microchip.com 	}
41275f4cc6e2SJohn Efstathiades 
412855d7de9dSWoojung.Huh@microchip.com 	if (test_bit(EVENT_RX_HALT, &dev->flags)) {
412955d7de9dSWoojung.Huh@microchip.com 		unlink_urbs(dev, &dev->rxq);
413055d7de9dSWoojung.Huh@microchip.com 		status = usb_clear_halt(dev->udev, dev->pipe_in);
413155d7de9dSWoojung.Huh@microchip.com 		if (status < 0 &&
413255d7de9dSWoojung.Huh@microchip.com 		    status != -EPIPE &&
413355d7de9dSWoojung.Huh@microchip.com 		    status != -ESHUTDOWN) {
413455d7de9dSWoojung.Huh@microchip.com 			if (netif_msg_rx_err(dev))
413555d7de9dSWoojung.Huh@microchip.com 				netdev_err(dev->net,
413655d7de9dSWoojung.Huh@microchip.com 					   "can't clear rx halt, status %d\n",
413755d7de9dSWoojung.Huh@microchip.com 					   status);
413855d7de9dSWoojung.Huh@microchip.com 		} else {
413955d7de9dSWoojung.Huh@microchip.com 			clear_bit(EVENT_RX_HALT, &dev->flags);
4140ec4c7e12SJohn Efstathiades 			napi_schedule(&dev->napi);
414155d7de9dSWoojung.Huh@microchip.com 		}
414255d7de9dSWoojung.Huh@microchip.com 	}
414355d7de9dSWoojung.Huh@microchip.com 
414455d7de9dSWoojung.Huh@microchip.com 	if (test_bit(EVENT_LINK_RESET, &dev->flags)) {
414555d7de9dSWoojung.Huh@microchip.com 		int ret = 0;
414655d7de9dSWoojung.Huh@microchip.com 
414755d7de9dSWoojung.Huh@microchip.com 		clear_bit(EVENT_LINK_RESET, &dev->flags);
414855d7de9dSWoojung.Huh@microchip.com 		if (lan78xx_link_reset(dev) < 0) {
414955d7de9dSWoojung.Huh@microchip.com 			netdev_info(dev->net, "link reset failed (%d)\n",
415055d7de9dSWoojung.Huh@microchip.com 				    ret);
415155d7de9dSWoojung.Huh@microchip.com 		}
415255d7de9dSWoojung.Huh@microchip.com 	}
415320ff5565SWoojung Huh 
415420ff5565SWoojung Huh 	if (test_bit(EVENT_STAT_UPDATE, &dev->flags)) {
415520ff5565SWoojung Huh 		lan78xx_update_stats(dev);
415620ff5565SWoojung Huh 
415720ff5565SWoojung Huh 		clear_bit(EVENT_STAT_UPDATE, &dev->flags);
415820ff5565SWoojung Huh 
415920ff5565SWoojung Huh 		mod_timer(&dev->stat_monitor,
416020ff5565SWoojung Huh 			  jiffies + (STAT_UPDATE_TIMER * dev->delta));
416120ff5565SWoojung Huh 
416220ff5565SWoojung Huh 		dev->delta = min((dev->delta * 2), 50);
416320ff5565SWoojung Huh 	}
41645f4cc6e2SJohn Efstathiades 
41655f4cc6e2SJohn Efstathiades 	usb_autopm_put_interface(dev->intf);
416655d7de9dSWoojung.Huh@microchip.com }
416755d7de9dSWoojung.Huh@microchip.com 
intr_complete(struct urb * urb)416855d7de9dSWoojung.Huh@microchip.com static void intr_complete(struct urb *urb)
416955d7de9dSWoojung.Huh@microchip.com {
417055d7de9dSWoojung.Huh@microchip.com 	struct lan78xx_net *dev = urb->context;
417155d7de9dSWoojung.Huh@microchip.com 	int status = urb->status;
417255d7de9dSWoojung.Huh@microchip.com 
417355d7de9dSWoojung.Huh@microchip.com 	switch (status) {
417455d7de9dSWoojung.Huh@microchip.com 	/* success */
417555d7de9dSWoojung.Huh@microchip.com 	case 0:
417655d7de9dSWoojung.Huh@microchip.com 		lan78xx_status(dev, urb);
417755d7de9dSWoojung.Huh@microchip.com 		break;
417855d7de9dSWoojung.Huh@microchip.com 
417955d7de9dSWoojung.Huh@microchip.com 	/* software-driven interface shutdown */
418055d7de9dSWoojung.Huh@microchip.com 	case -ENOENT:			/* urb killed */
418177dfff5bSJohn Efstathiades 	case -ENODEV:			/* hardware gone */
418255d7de9dSWoojung.Huh@microchip.com 	case -ESHUTDOWN:		/* hardware gone */
418355d7de9dSWoojung.Huh@microchip.com 		netif_dbg(dev, ifdown, dev->net,
418455d7de9dSWoojung.Huh@microchip.com 			  "intr shutdown, code %d\n", status);
418555d7de9dSWoojung.Huh@microchip.com 		return;
418655d7de9dSWoojung.Huh@microchip.com 
418755d7de9dSWoojung.Huh@microchip.com 	/* NOTE:  not throttling like RX/TX, since this endpoint
418855d7de9dSWoojung.Huh@microchip.com 	 * already polls infrequently
418955d7de9dSWoojung.Huh@microchip.com 	 */
419055d7de9dSWoojung.Huh@microchip.com 	default:
419155d7de9dSWoojung.Huh@microchip.com 		netdev_dbg(dev->net, "intr status %d\n", status);
419255d7de9dSWoojung.Huh@microchip.com 		break;
419355d7de9dSWoojung.Huh@microchip.com 	}
419455d7de9dSWoojung.Huh@microchip.com 
419577dfff5bSJohn Efstathiades 	if (!netif_device_present(dev->net) ||
419677dfff5bSJohn Efstathiades 	    !netif_running(dev->net)) {
419777dfff5bSJohn Efstathiades 		netdev_warn(dev->net, "not submitting new status URB");
419855d7de9dSWoojung.Huh@microchip.com 		return;
419977dfff5bSJohn Efstathiades 	}
420055d7de9dSWoojung.Huh@microchip.com 
420155d7de9dSWoojung.Huh@microchip.com 	memset(urb->transfer_buffer, 0, urb->transfer_buffer_length);
420255d7de9dSWoojung.Huh@microchip.com 	status = usb_submit_urb(urb, GFP_ATOMIC);
420377dfff5bSJohn Efstathiades 
420477dfff5bSJohn Efstathiades 	switch (status) {
420577dfff5bSJohn Efstathiades 	case  0:
420677dfff5bSJohn Efstathiades 		break;
420777dfff5bSJohn Efstathiades 	case -ENODEV:
420877dfff5bSJohn Efstathiades 	case -ENOENT:
420977dfff5bSJohn Efstathiades 		netif_dbg(dev, timer, dev->net,
421077dfff5bSJohn Efstathiades 			  "intr resubmit %d (disconnect?)", status);
421177dfff5bSJohn Efstathiades 		netif_device_detach(dev->net);
421277dfff5bSJohn Efstathiades 		break;
421377dfff5bSJohn Efstathiades 	default:
421455d7de9dSWoojung.Huh@microchip.com 		netif_err(dev, timer, dev->net,
421555d7de9dSWoojung.Huh@microchip.com 			  "intr resubmit --> %d\n", status);
421677dfff5bSJohn Efstathiades 		break;
421777dfff5bSJohn Efstathiades 	}
421855d7de9dSWoojung.Huh@microchip.com }
421955d7de9dSWoojung.Huh@microchip.com 
lan78xx_disconnect(struct usb_interface * intf)422055d7de9dSWoojung.Huh@microchip.com static void lan78xx_disconnect(struct usb_interface *intf)
422155d7de9dSWoojung.Huh@microchip.com {
422255d7de9dSWoojung.Huh@microchip.com 	struct lan78xx_net *dev;
422355d7de9dSWoojung.Huh@microchip.com 	struct usb_device *udev;
422455d7de9dSWoojung.Huh@microchip.com 	struct net_device *net;
422589b36fb5SRaghuram Chary J 	struct phy_device *phydev;
422655d7de9dSWoojung.Huh@microchip.com 
422755d7de9dSWoojung.Huh@microchip.com 	dev = usb_get_intfdata(intf);
422855d7de9dSWoojung.Huh@microchip.com 	usb_set_intfdata(intf, NULL);
422955d7de9dSWoojung.Huh@microchip.com 	if (!dev)
423055d7de9dSWoojung.Huh@microchip.com 		return;
423155d7de9dSWoojung.Huh@microchip.com 
4232ec4c7e12SJohn Efstathiades 	netif_napi_del(&dev->napi);
4233ec4c7e12SJohn Efstathiades 
423455d7de9dSWoojung.Huh@microchip.com 	udev = interface_to_usbdev(intf);
423555d7de9dSWoojung.Huh@microchip.com 	net = dev->net;
423677dfff5bSJohn Efstathiades 
423777dfff5bSJohn Efstathiades 	unregister_netdev(net);
423877dfff5bSJohn Efstathiades 
42391e7417c1SDuoming Zhou 	timer_shutdown_sync(&dev->stat_monitor);
42401e7417c1SDuoming Zhou 	set_bit(EVENT_DEV_DISCONNECT, &dev->flags);
424177dfff5bSJohn Efstathiades 	cancel_delayed_work_sync(&dev->wq);
424277dfff5bSJohn Efstathiades 
424389b36fb5SRaghuram Chary J 	phydev = net->phydev;
424492571a1aSAlexander Graf 
424592571a1aSAlexander Graf 	phy_unregister_fixup_for_uid(PHY_KSZ9031RNX, 0xfffffff0);
424692571a1aSAlexander Graf 	phy_unregister_fixup_for_uid(PHY_LAN8835, 0xfffffff0);
424792571a1aSAlexander Graf 
424892571a1aSAlexander Graf 	phy_disconnect(net->phydev);
424992571a1aSAlexander Graf 
4250bca2c418SOleksij Rempel 	if (phy_is_pseudo_fixed_link(phydev)) {
425189b36fb5SRaghuram Chary J 		fixed_phy_unregister(phydev);
4252bca2c418SOleksij Rempel 		phy_device_free(phydev);
4253bca2c418SOleksij Rempel 	}
425489b36fb5SRaghuram Chary J 
425555d7de9dSWoojung.Huh@microchip.com 	usb_scuttle_anchored_urbs(&dev->deferred);
425655d7de9dSWoojung.Huh@microchip.com 
425755d7de9dSWoojung.Huh@microchip.com 	lan78xx_unbind(dev, intf);
425855d7de9dSWoojung.Huh@microchip.com 
4259d383216aSJohn Efstathiades 	lan78xx_free_tx_resources(dev);
4260c450a8ebSJohn Efstathiades 	lan78xx_free_rx_resources(dev);
4261d383216aSJohn Efstathiades 
426255d7de9dSWoojung.Huh@microchip.com 	usb_kill_urb(dev->urb_intr);
426355d7de9dSWoojung.Huh@microchip.com 	usb_free_urb(dev->urb_intr);
426455d7de9dSWoojung.Huh@microchip.com 
426555d7de9dSWoojung.Huh@microchip.com 	free_netdev(net);
426655d7de9dSWoojung.Huh@microchip.com 	usb_put_dev(udev);
426755d7de9dSWoojung.Huh@microchip.com }
426855d7de9dSWoojung.Huh@microchip.com 
lan78xx_tx_timeout(struct net_device * net,unsigned int txqueue)42690290bd29SMichael S. Tsirkin static void lan78xx_tx_timeout(struct net_device *net, unsigned int txqueue)
427055d7de9dSWoojung.Huh@microchip.com {
427155d7de9dSWoojung.Huh@microchip.com 	struct lan78xx_net *dev = netdev_priv(net);
427255d7de9dSWoojung.Huh@microchip.com 
427355d7de9dSWoojung.Huh@microchip.com 	unlink_urbs(dev, &dev->txq);
4274ec4c7e12SJohn Efstathiades 	napi_schedule(&dev->napi);
427555d7de9dSWoojung.Huh@microchip.com }
427655d7de9dSWoojung.Huh@microchip.com 
lan78xx_features_check(struct sk_buff * skb,struct net_device * netdev,netdev_features_t features)4277ce896476SJames Hughes static netdev_features_t lan78xx_features_check(struct sk_buff *skb,
4278ce896476SJames Hughes 						struct net_device *netdev,
4279ce896476SJames Hughes 						netdev_features_t features)
4280ce896476SJames Hughes {
4281d383216aSJohn Efstathiades 	struct lan78xx_net *dev = netdev_priv(netdev);
4282d383216aSJohn Efstathiades 
4283d383216aSJohn Efstathiades 	if (skb->len > LAN78XX_TSO_SIZE(dev))
4284ce896476SJames Hughes 		features &= ~NETIF_F_GSO_MASK;
4285ce896476SJames Hughes 
4286ce896476SJames Hughes 	features = vlan_features_check(skb, features);
4287ce896476SJames Hughes 	features = vxlan_features_check(skb, features);
4288ce896476SJames Hughes 
4289ce896476SJames Hughes 	return features;
4290ce896476SJames Hughes }
4291ce896476SJames Hughes 
429255d7de9dSWoojung.Huh@microchip.com static const struct net_device_ops lan78xx_netdev_ops = {
429355d7de9dSWoojung.Huh@microchip.com 	.ndo_open		= lan78xx_open,
429455d7de9dSWoojung.Huh@microchip.com 	.ndo_stop		= lan78xx_stop,
429555d7de9dSWoojung.Huh@microchip.com 	.ndo_start_xmit		= lan78xx_start_xmit,
429655d7de9dSWoojung.Huh@microchip.com 	.ndo_tx_timeout		= lan78xx_tx_timeout,
429755d7de9dSWoojung.Huh@microchip.com 	.ndo_change_mtu		= lan78xx_change_mtu,
429855d7de9dSWoojung.Huh@microchip.com 	.ndo_set_mac_address	= lan78xx_set_mac_addr,
429955d7de9dSWoojung.Huh@microchip.com 	.ndo_validate_addr	= eth_validate_addr,
4300a7605370SArnd Bergmann 	.ndo_eth_ioctl		= phy_do_ioctl_running,
430155d7de9dSWoojung.Huh@microchip.com 	.ndo_set_rx_mode	= lan78xx_set_multicast,
430255d7de9dSWoojung.Huh@microchip.com 	.ndo_set_features	= lan78xx_set_features,
430355d7de9dSWoojung.Huh@microchip.com 	.ndo_vlan_rx_add_vid	= lan78xx_vlan_rx_add_vid,
430455d7de9dSWoojung.Huh@microchip.com 	.ndo_vlan_rx_kill_vid	= lan78xx_vlan_rx_kill_vid,
4305ce896476SJames Hughes 	.ndo_features_check	= lan78xx_features_check,
430655d7de9dSWoojung.Huh@microchip.com };
430755d7de9dSWoojung.Huh@microchip.com 
lan78xx_stat_monitor(struct timer_list * t)4308d28bb967SKees Cook static void lan78xx_stat_monitor(struct timer_list *t)
430920ff5565SWoojung Huh {
4310d28bb967SKees Cook 	struct lan78xx_net *dev = from_timer(dev, t, stat_monitor);
431120ff5565SWoojung Huh 
431220ff5565SWoojung Huh 	lan78xx_defer_kevent(dev, EVENT_STAT_UPDATE);
431320ff5565SWoojung Huh }
431420ff5565SWoojung Huh 
lan78xx_probe(struct usb_interface * intf,const struct usb_device_id * id)431555d7de9dSWoojung.Huh@microchip.com static int lan78xx_probe(struct usb_interface *intf,
431655d7de9dSWoojung.Huh@microchip.com 			 const struct usb_device_id *id)
431755d7de9dSWoojung.Huh@microchip.com {
4318ea060b35SJohan Hovold 	struct usb_host_endpoint *ep_blkin, *ep_blkout, *ep_intr;
431955d7de9dSWoojung.Huh@microchip.com 	struct lan78xx_net *dev;
432055d7de9dSWoojung.Huh@microchip.com 	struct net_device *netdev;
432155d7de9dSWoojung.Huh@microchip.com 	struct usb_device *udev;
432255d7de9dSWoojung.Huh@microchip.com 	int ret;
43239ceec7d3SJohn Efstathiades 	unsigned int maxp;
43249ceec7d3SJohn Efstathiades 	unsigned int period;
432555d7de9dSWoojung.Huh@microchip.com 	u8 *buf = NULL;
432655d7de9dSWoojung.Huh@microchip.com 
432755d7de9dSWoojung.Huh@microchip.com 	udev = interface_to_usbdev(intf);
432855d7de9dSWoojung.Huh@microchip.com 	udev = usb_get_dev(udev);
432955d7de9dSWoojung.Huh@microchip.com 
433055d7de9dSWoojung.Huh@microchip.com 	netdev = alloc_etherdev(sizeof(struct lan78xx_net));
433155d7de9dSWoojung.Huh@microchip.com 	if (!netdev) {
433255d7de9dSWoojung.Huh@microchip.com 		dev_err(&intf->dev, "Error: OOM\n");
4333fb52c3b5SNisar Sayed 		ret = -ENOMEM;
433455d7de9dSWoojung.Huh@microchip.com 		goto out1;
433555d7de9dSWoojung.Huh@microchip.com 	}
433655d7de9dSWoojung.Huh@microchip.com 
433755d7de9dSWoojung.Huh@microchip.com 	/* netdev_printk() needs this */
433855d7de9dSWoojung.Huh@microchip.com 	SET_NETDEV_DEV(netdev, &intf->dev);
433955d7de9dSWoojung.Huh@microchip.com 
434055d7de9dSWoojung.Huh@microchip.com 	dev = netdev_priv(netdev);
434155d7de9dSWoojung.Huh@microchip.com 	dev->udev = udev;
434255d7de9dSWoojung.Huh@microchip.com 	dev->intf = intf;
434355d7de9dSWoojung.Huh@microchip.com 	dev->net = netdev;
434455d7de9dSWoojung.Huh@microchip.com 	dev->msg_enable = netif_msg_init(msg_level, NETIF_MSG_DRV
434555d7de9dSWoojung.Huh@microchip.com 					| NETIF_MSG_PROBE | NETIF_MSG_LINK);
434655d7de9dSWoojung.Huh@microchip.com 
434755d7de9dSWoojung.Huh@microchip.com 	skb_queue_head_init(&dev->rxq);
434855d7de9dSWoojung.Huh@microchip.com 	skb_queue_head_init(&dev->txq);
4349c450a8ebSJohn Efstathiades 	skb_queue_head_init(&dev->rxq_done);
435055d7de9dSWoojung.Huh@microchip.com 	skb_queue_head_init(&dev->txq_pend);
4351ec4c7e12SJohn Efstathiades 	skb_queue_head_init(&dev->rxq_overflow);
435255d7de9dSWoojung.Huh@microchip.com 	mutex_init(&dev->phy_mutex);
43535f4cc6e2SJohn Efstathiades 	mutex_init(&dev->dev_mutex);
435455d7de9dSWoojung.Huh@microchip.com 
4355d383216aSJohn Efstathiades 	ret = lan78xx_urb_config_init(dev);
4356d383216aSJohn Efstathiades 	if (ret < 0)
4357d383216aSJohn Efstathiades 		goto out2;
4358d383216aSJohn Efstathiades 
4359d383216aSJohn Efstathiades 	ret = lan78xx_alloc_tx_resources(dev);
4360d383216aSJohn Efstathiades 	if (ret < 0)
4361d383216aSJohn Efstathiades 		goto out2;
4362d383216aSJohn Efstathiades 
4363c450a8ebSJohn Efstathiades 	ret = lan78xx_alloc_rx_resources(dev);
4364c450a8ebSJohn Efstathiades 	if (ret < 0)
4365c450a8ebSJohn Efstathiades 		goto out3;
4366c450a8ebSJohn Efstathiades 
43670dd87266SJohn Efstathiades 	/* MTU range: 68 - 9000 */
43680dd87266SJohn Efstathiades 	netdev->max_mtu = MAX_SINGLE_PACKET_SIZE;
43690dd87266SJohn Efstathiades 
4370ee8b7a11SJakub Kicinski 	netif_set_tso_max_size(netdev, LAN78XX_TSO_SIZE(dev));
4371d383216aSJohn Efstathiades 
4372b48b89f9SJakub Kicinski 	netif_napi_add(netdev, &dev->napi, lan78xx_poll);
4373ec4c7e12SJohn Efstathiades 
437455d7de9dSWoojung.Huh@microchip.com 	INIT_DELAYED_WORK(&dev->wq, lan78xx_delayedwork);
437555d7de9dSWoojung.Huh@microchip.com 	init_usb_anchor(&dev->deferred);
437655d7de9dSWoojung.Huh@microchip.com 
437755d7de9dSWoojung.Huh@microchip.com 	netdev->netdev_ops = &lan78xx_netdev_ops;
437855d7de9dSWoojung.Huh@microchip.com 	netdev->watchdog_timeo = TX_TIMEOUT_JIFFIES;
437955d7de9dSWoojung.Huh@microchip.com 	netdev->ethtool_ops = &lan78xx_ethtool_ops;
438055d7de9dSWoojung.Huh@microchip.com 
438120ff5565SWoojung Huh 	dev->delta = 1;
4382d28bb967SKees Cook 	timer_setup(&dev->stat_monitor, lan78xx_stat_monitor, 0);
438320ff5565SWoojung Huh 
438420ff5565SWoojung Huh 	mutex_init(&dev->stats.access_lock);
438520ff5565SWoojung Huh 
4386ea060b35SJohan Hovold 	if (intf->cur_altsetting->desc.bNumEndpoints < 3) {
4387ea060b35SJohan Hovold 		ret = -ENODEV;
4388c450a8ebSJohn Efstathiades 		goto out4;
4389ea060b35SJohan Hovold 	}
4390ea060b35SJohan Hovold 
4391ea060b35SJohan Hovold 	dev->pipe_in = usb_rcvbulkpipe(udev, BULK_IN_PIPE);
4392ea060b35SJohan Hovold 	ep_blkin = usb_pipe_endpoint(udev, dev->pipe_in);
4393ea060b35SJohan Hovold 	if (!ep_blkin || !usb_endpoint_is_bulk_in(&ep_blkin->desc)) {
4394ea060b35SJohan Hovold 		ret = -ENODEV;
4395c450a8ebSJohn Efstathiades 		goto out4;
4396ea060b35SJohan Hovold 	}
4397ea060b35SJohan Hovold 
4398ea060b35SJohan Hovold 	dev->pipe_out = usb_sndbulkpipe(udev, BULK_OUT_PIPE);
4399ea060b35SJohan Hovold 	ep_blkout = usb_pipe_endpoint(udev, dev->pipe_out);
4400ea060b35SJohan Hovold 	if (!ep_blkout || !usb_endpoint_is_bulk_out(&ep_blkout->desc)) {
4401ea060b35SJohan Hovold 		ret = -ENODEV;
4402c450a8ebSJohn Efstathiades 		goto out4;
4403ea060b35SJohan Hovold 	}
4404ea060b35SJohan Hovold 
4405ea060b35SJohan Hovold 	ep_intr = &intf->cur_altsetting->endpoint[2];
4406ea060b35SJohan Hovold 	if (!usb_endpoint_is_int_in(&ep_intr->desc)) {
4407ea060b35SJohan Hovold 		ret = -ENODEV;
4408c450a8ebSJohn Efstathiades 		goto out4;
4409ea060b35SJohan Hovold 	}
4410ea060b35SJohan Hovold 
4411ea060b35SJohan Hovold 	dev->pipe_intr = usb_rcvintpipe(dev->udev,
4412ea060b35SJohan Hovold 					usb_endpoint_num(&ep_intr->desc));
4413ea060b35SJohan Hovold 
441455d7de9dSWoojung.Huh@microchip.com 	ret = lan78xx_bind(dev, intf);
441555d7de9dSWoojung.Huh@microchip.com 	if (ret < 0)
4416c450a8ebSJohn Efstathiades 		goto out4;
441755d7de9dSWoojung.Huh@microchip.com 
4418ea060b35SJohan Hovold 	period = ep_intr->desc.bInterval;
4419e13adbfaSVincent Mailhol 	maxp = usb_maxpacket(dev->udev, dev->pipe_intr);
4420a6df95caSJohn Efstathiades 
442155d7de9dSWoojung.Huh@microchip.com 	dev->urb_intr = usb_alloc_urb(0, GFP_KERNEL);
442255d7de9dSWoojung.Huh@microchip.com 	if (!dev->urb_intr) {
442351920830SPan Bian 		ret = -ENOMEM;
4424a422ebecSOleksij Rempel 		goto out5;
4425a422ebecSOleksij Rempel 	}
4426a422ebecSOleksij Rempel 
4427a422ebecSOleksij Rempel 	buf = kmalloc(maxp, GFP_KERNEL);
4428a422ebecSOleksij Rempel 	if (!buf) {
4429a422ebecSOleksij Rempel 		ret = -ENOMEM;
4430a422ebecSOleksij Rempel 		goto free_urbs;
4431a422ebecSOleksij Rempel 	}
4432a422ebecSOleksij Rempel 
443355d7de9dSWoojung.Huh@microchip.com 	usb_fill_int_urb(dev->urb_intr, dev->udev,
443455d7de9dSWoojung.Huh@microchip.com 			 dev->pipe_intr, buf, maxp,
443555d7de9dSWoojung.Huh@microchip.com 			 intr_complete, dev, period);
443663634aa6SJohan Hovold 	dev->urb_intr->transfer_flags |= URB_FREE_BUFFER;
443755d7de9dSWoojung.Huh@microchip.com 
4438e13adbfaSVincent Mailhol 	dev->maxpacket = usb_maxpacket(dev->udev, dev->pipe_out);
443955d7de9dSWoojung.Huh@microchip.com 
4440db6c3c06SJohan Hovold 	/* Reject broken descriptors. */
4441db6c3c06SJohan Hovold 	if (dev->maxpacket == 0) {
4442db6c3c06SJohan Hovold 		ret = -ENODEV;
4443a422ebecSOleksij Rempel 		goto free_urbs;
4444db6c3c06SJohan Hovold 	}
4445db6c3c06SJohan Hovold 
444655d7de9dSWoojung.Huh@microchip.com 	/* driver requires remote-wakeup capability during autosuspend. */
444755d7de9dSWoojung.Huh@microchip.com 	intf->needs_remote_wakeup = 1;
444855d7de9dSWoojung.Huh@microchip.com 
444938b4fe32SAndrew Lunn 	ret = lan78xx_phy_init(dev);
445038b4fe32SAndrew Lunn 	if (ret < 0)
4451a422ebecSOleksij Rempel 		goto free_urbs;
445238b4fe32SAndrew Lunn 
445355d7de9dSWoojung.Huh@microchip.com 	ret = register_netdev(netdev);
445455d7de9dSWoojung.Huh@microchip.com 	if (ret != 0) {
445555d7de9dSWoojung.Huh@microchip.com 		netif_err(dev, probe, netdev, "couldn't register the device\n");
4456c450a8ebSJohn Efstathiades 		goto out8;
445755d7de9dSWoojung.Huh@microchip.com 	}
445855d7de9dSWoojung.Huh@microchip.com 
445955d7de9dSWoojung.Huh@microchip.com 	usb_set_intfdata(intf, dev);
446055d7de9dSWoojung.Huh@microchip.com 
446155d7de9dSWoojung.Huh@microchip.com 	ret = device_set_wakeup_enable(&udev->dev, true);
446255d7de9dSWoojung.Huh@microchip.com 
446355d7de9dSWoojung.Huh@microchip.com 	 /* Default delay of 2sec has more overhead than advantage.
446455d7de9dSWoojung.Huh@microchip.com 	  * Set to 10sec as default.
446555d7de9dSWoojung.Huh@microchip.com 	  */
446655d7de9dSWoojung.Huh@microchip.com 	pm_runtime_set_autosuspend_delay(&udev->dev,
446755d7de9dSWoojung.Huh@microchip.com 					 DEFAULT_AUTOSUSPEND_DELAY);
446855d7de9dSWoojung.Huh@microchip.com 
446955d7de9dSWoojung.Huh@microchip.com 	return 0;
447055d7de9dSWoojung.Huh@microchip.com 
4471c450a8ebSJohn Efstathiades out8:
447238b4fe32SAndrew Lunn 	phy_disconnect(netdev->phydev);
4473a422ebecSOleksij Rempel free_urbs:
4474b9cbf8a6SWenwen Wang 	usb_free_urb(dev->urb_intr);
4475c450a8ebSJohn Efstathiades out5:
447655d7de9dSWoojung.Huh@microchip.com 	lan78xx_unbind(dev, intf);
4477c450a8ebSJohn Efstathiades out4:
4478ec4c7e12SJohn Efstathiades 	netif_napi_del(&dev->napi);
4479c450a8ebSJohn Efstathiades 	lan78xx_free_rx_resources(dev);
4480d383216aSJohn Efstathiades out3:
4481d383216aSJohn Efstathiades 	lan78xx_free_tx_resources(dev);
448255d7de9dSWoojung.Huh@microchip.com out2:
448355d7de9dSWoojung.Huh@microchip.com 	free_netdev(netdev);
448455d7de9dSWoojung.Huh@microchip.com out1:
448555d7de9dSWoojung.Huh@microchip.com 	usb_put_dev(udev);
448655d7de9dSWoojung.Huh@microchip.com 
448755d7de9dSWoojung.Huh@microchip.com 	return ret;
448855d7de9dSWoojung.Huh@microchip.com }
448955d7de9dSWoojung.Huh@microchip.com 
lan78xx_wakeframe_crc16(const u8 * buf,int len)449055d7de9dSWoojung.Huh@microchip.com static u16 lan78xx_wakeframe_crc16(const u8 *buf, int len)
449155d7de9dSWoojung.Huh@microchip.com {
449255d7de9dSWoojung.Huh@microchip.com 	const u16 crc16poly = 0x8005;
449355d7de9dSWoojung.Huh@microchip.com 	int i;
449455d7de9dSWoojung.Huh@microchip.com 	u16 bit, crc, msb;
449555d7de9dSWoojung.Huh@microchip.com 	u8 data;
449655d7de9dSWoojung.Huh@microchip.com 
449755d7de9dSWoojung.Huh@microchip.com 	crc = 0xFFFF;
449855d7de9dSWoojung.Huh@microchip.com 	for (i = 0; i < len; i++) {
449955d7de9dSWoojung.Huh@microchip.com 		data = *buf++;
450055d7de9dSWoojung.Huh@microchip.com 		for (bit = 0; bit < 8; bit++) {
450155d7de9dSWoojung.Huh@microchip.com 			msb = crc >> 15;
450255d7de9dSWoojung.Huh@microchip.com 			crc <<= 1;
450355d7de9dSWoojung.Huh@microchip.com 
450455d7de9dSWoojung.Huh@microchip.com 			if (msb ^ (u16)(data & 1)) {
450555d7de9dSWoojung.Huh@microchip.com 				crc ^= crc16poly;
450655d7de9dSWoojung.Huh@microchip.com 				crc |= (u16)0x0001U;
450755d7de9dSWoojung.Huh@microchip.com 			}
450855d7de9dSWoojung.Huh@microchip.com 			data >>= 1;
450955d7de9dSWoojung.Huh@microchip.com 		}
451055d7de9dSWoojung.Huh@microchip.com 	}
451155d7de9dSWoojung.Huh@microchip.com 
451255d7de9dSWoojung.Huh@microchip.com 	return crc;
451355d7de9dSWoojung.Huh@microchip.com }
451455d7de9dSWoojung.Huh@microchip.com 
lan78xx_set_auto_suspend(struct lan78xx_net * dev)45155f4cc6e2SJohn Efstathiades static int lan78xx_set_auto_suspend(struct lan78xx_net *dev)
45165f4cc6e2SJohn Efstathiades {
45175f4cc6e2SJohn Efstathiades 	u32 buf;
45185f4cc6e2SJohn Efstathiades 	int ret;
45195f4cc6e2SJohn Efstathiades 
45205f4cc6e2SJohn Efstathiades 	ret = lan78xx_stop_tx_path(dev);
45215f4cc6e2SJohn Efstathiades 	if (ret < 0)
45225f4cc6e2SJohn Efstathiades 		return ret;
45235f4cc6e2SJohn Efstathiades 
45245f4cc6e2SJohn Efstathiades 	ret = lan78xx_stop_rx_path(dev);
45255f4cc6e2SJohn Efstathiades 	if (ret < 0)
45265f4cc6e2SJohn Efstathiades 		return ret;
45275f4cc6e2SJohn Efstathiades 
45285f4cc6e2SJohn Efstathiades 	/* auto suspend (selective suspend) */
45295f4cc6e2SJohn Efstathiades 
45305f4cc6e2SJohn Efstathiades 	ret = lan78xx_write_reg(dev, WUCSR, 0);
45315f4cc6e2SJohn Efstathiades 	if (ret < 0)
45325f4cc6e2SJohn Efstathiades 		return ret;
45335f4cc6e2SJohn Efstathiades 	ret = lan78xx_write_reg(dev, WUCSR2, 0);
45345f4cc6e2SJohn Efstathiades 	if (ret < 0)
45355f4cc6e2SJohn Efstathiades 		return ret;
45365f4cc6e2SJohn Efstathiades 	ret = lan78xx_write_reg(dev, WK_SRC, 0xFFF1FF1FUL);
45375f4cc6e2SJohn Efstathiades 	if (ret < 0)
45385f4cc6e2SJohn Efstathiades 		return ret;
45395f4cc6e2SJohn Efstathiades 
45405f4cc6e2SJohn Efstathiades 	/* set goodframe wakeup */
45415f4cc6e2SJohn Efstathiades 
45425f4cc6e2SJohn Efstathiades 	ret = lan78xx_read_reg(dev, WUCSR, &buf);
45435f4cc6e2SJohn Efstathiades 	if (ret < 0)
45445f4cc6e2SJohn Efstathiades 		return ret;
45455f4cc6e2SJohn Efstathiades 
45465f4cc6e2SJohn Efstathiades 	buf |= WUCSR_RFE_WAKE_EN_;
45475f4cc6e2SJohn Efstathiades 	buf |= WUCSR_STORE_WAKE_;
45485f4cc6e2SJohn Efstathiades 
45495f4cc6e2SJohn Efstathiades 	ret = lan78xx_write_reg(dev, WUCSR, buf);
45505f4cc6e2SJohn Efstathiades 	if (ret < 0)
45515f4cc6e2SJohn Efstathiades 		return ret;
45525f4cc6e2SJohn Efstathiades 
45535f4cc6e2SJohn Efstathiades 	ret = lan78xx_read_reg(dev, PMT_CTL, &buf);
45545f4cc6e2SJohn Efstathiades 	if (ret < 0)
45555f4cc6e2SJohn Efstathiades 		return ret;
45565f4cc6e2SJohn Efstathiades 
45575f4cc6e2SJohn Efstathiades 	buf &= ~PMT_CTL_RES_CLR_WKP_EN_;
45585f4cc6e2SJohn Efstathiades 	buf |= PMT_CTL_RES_CLR_WKP_STS_;
45595f4cc6e2SJohn Efstathiades 	buf |= PMT_CTL_PHY_WAKE_EN_;
45605f4cc6e2SJohn Efstathiades 	buf |= PMT_CTL_WOL_EN_;
45615f4cc6e2SJohn Efstathiades 	buf &= ~PMT_CTL_SUS_MODE_MASK_;
45625f4cc6e2SJohn Efstathiades 	buf |= PMT_CTL_SUS_MODE_3_;
45635f4cc6e2SJohn Efstathiades 
45645f4cc6e2SJohn Efstathiades 	ret = lan78xx_write_reg(dev, PMT_CTL, buf);
45655f4cc6e2SJohn Efstathiades 	if (ret < 0)
45665f4cc6e2SJohn Efstathiades 		return ret;
45675f4cc6e2SJohn Efstathiades 
45685f4cc6e2SJohn Efstathiades 	ret = lan78xx_read_reg(dev, PMT_CTL, &buf);
45695f4cc6e2SJohn Efstathiades 	if (ret < 0)
45705f4cc6e2SJohn Efstathiades 		return ret;
45715f4cc6e2SJohn Efstathiades 
45725f4cc6e2SJohn Efstathiades 	buf |= PMT_CTL_WUPS_MASK_;
45735f4cc6e2SJohn Efstathiades 
45745f4cc6e2SJohn Efstathiades 	ret = lan78xx_write_reg(dev, PMT_CTL, buf);
45755f4cc6e2SJohn Efstathiades 	if (ret < 0)
45765f4cc6e2SJohn Efstathiades 		return ret;
45775f4cc6e2SJohn Efstathiades 
45785f4cc6e2SJohn Efstathiades 	ret = lan78xx_start_rx_path(dev);
45795f4cc6e2SJohn Efstathiades 
45805f4cc6e2SJohn Efstathiades 	return ret;
45815f4cc6e2SJohn Efstathiades }
45825f4cc6e2SJohn Efstathiades 
lan78xx_set_suspend(struct lan78xx_net * dev,u32 wol)458355d7de9dSWoojung.Huh@microchip.com static int lan78xx_set_suspend(struct lan78xx_net *dev, u32 wol)
458455d7de9dSWoojung.Huh@microchip.com {
458555d7de9dSWoojung.Huh@microchip.com 	const u8 ipv4_multicast[3] = { 0x01, 0x00, 0x5E };
458655d7de9dSWoojung.Huh@microchip.com 	const u8 ipv6_multicast[3] = { 0x33, 0x33 };
458755d7de9dSWoojung.Huh@microchip.com 	const u8 arp_type[2] = { 0x08, 0x06 };
45883415f6baSJohn Efstathiades 	u32 temp_pmt_ctl;
45893415f6baSJohn Efstathiades 	int mask_index;
45903415f6baSJohn Efstathiades 	u32 temp_wucsr;
45913415f6baSJohn Efstathiades 	u32 buf;
45923415f6baSJohn Efstathiades 	u16 crc;
45933415f6baSJohn Efstathiades 	int ret;
459455d7de9dSWoojung.Huh@microchip.com 
4595e1210fe6SJohn Efstathiades 	ret = lan78xx_stop_tx_path(dev);
45963415f6baSJohn Efstathiades 	if (ret < 0)
45973415f6baSJohn Efstathiades 		return ret;
4598e1210fe6SJohn Efstathiades 	ret = lan78xx_stop_rx_path(dev);
45993415f6baSJohn Efstathiades 	if (ret < 0)
46003415f6baSJohn Efstathiades 		return ret;
46013415f6baSJohn Efstathiades 
46023415f6baSJohn Efstathiades 	ret = lan78xx_write_reg(dev, WUCSR, 0);
46033415f6baSJohn Efstathiades 	if (ret < 0)
46043415f6baSJohn Efstathiades 		return ret;
46053415f6baSJohn Efstathiades 	ret = lan78xx_write_reg(dev, WUCSR2, 0);
46063415f6baSJohn Efstathiades 	if (ret < 0)
46073415f6baSJohn Efstathiades 		return ret;
46083415f6baSJohn Efstathiades 	ret = lan78xx_write_reg(dev, WK_SRC, 0xFFF1FF1FUL);
46093415f6baSJohn Efstathiades 	if (ret < 0)
46103415f6baSJohn Efstathiades 		return ret;
461155d7de9dSWoojung.Huh@microchip.com 
461255d7de9dSWoojung.Huh@microchip.com 	temp_wucsr = 0;
461355d7de9dSWoojung.Huh@microchip.com 
461455d7de9dSWoojung.Huh@microchip.com 	temp_pmt_ctl = 0;
46153415f6baSJohn Efstathiades 
46163415f6baSJohn Efstathiades 	ret = lan78xx_read_reg(dev, PMT_CTL, &temp_pmt_ctl);
46173415f6baSJohn Efstathiades 	if (ret < 0)
46183415f6baSJohn Efstathiades 		return ret;
46193415f6baSJohn Efstathiades 
462055d7de9dSWoojung.Huh@microchip.com 	temp_pmt_ctl &= ~PMT_CTL_RES_CLR_WKP_EN_;
462155d7de9dSWoojung.Huh@microchip.com 	temp_pmt_ctl |= PMT_CTL_RES_CLR_WKP_STS_;
462255d7de9dSWoojung.Huh@microchip.com 
46233415f6baSJohn Efstathiades 	for (mask_index = 0; mask_index < NUM_OF_WUF_CFG; mask_index++) {
46243415f6baSJohn Efstathiades 		ret = lan78xx_write_reg(dev, WUF_CFG(mask_index), 0);
46253415f6baSJohn Efstathiades 		if (ret < 0)
46263415f6baSJohn Efstathiades 			return ret;
46273415f6baSJohn Efstathiades 	}
462855d7de9dSWoojung.Huh@microchip.com 
462955d7de9dSWoojung.Huh@microchip.com 	mask_index = 0;
463055d7de9dSWoojung.Huh@microchip.com 	if (wol & WAKE_PHY) {
463155d7de9dSWoojung.Huh@microchip.com 		temp_pmt_ctl |= PMT_CTL_PHY_WAKE_EN_;
463255d7de9dSWoojung.Huh@microchip.com 
463355d7de9dSWoojung.Huh@microchip.com 		temp_pmt_ctl |= PMT_CTL_WOL_EN_;
463455d7de9dSWoojung.Huh@microchip.com 		temp_pmt_ctl &= ~PMT_CTL_SUS_MODE_MASK_;
463555d7de9dSWoojung.Huh@microchip.com 		temp_pmt_ctl |= PMT_CTL_SUS_MODE_0_;
463655d7de9dSWoojung.Huh@microchip.com 	}
463755d7de9dSWoojung.Huh@microchip.com 	if (wol & WAKE_MAGIC) {
463855d7de9dSWoojung.Huh@microchip.com 		temp_wucsr |= WUCSR_MPEN_;
463955d7de9dSWoojung.Huh@microchip.com 
464055d7de9dSWoojung.Huh@microchip.com 		temp_pmt_ctl |= PMT_CTL_WOL_EN_;
464155d7de9dSWoojung.Huh@microchip.com 		temp_pmt_ctl &= ~PMT_CTL_SUS_MODE_MASK_;
464255d7de9dSWoojung.Huh@microchip.com 		temp_pmt_ctl |= PMT_CTL_SUS_MODE_3_;
464355d7de9dSWoojung.Huh@microchip.com 	}
464455d7de9dSWoojung.Huh@microchip.com 	if (wol & WAKE_BCAST) {
464555d7de9dSWoojung.Huh@microchip.com 		temp_wucsr |= WUCSR_BCST_EN_;
464655d7de9dSWoojung.Huh@microchip.com 
464755d7de9dSWoojung.Huh@microchip.com 		temp_pmt_ctl |= PMT_CTL_WOL_EN_;
464855d7de9dSWoojung.Huh@microchip.com 		temp_pmt_ctl &= ~PMT_CTL_SUS_MODE_MASK_;
464955d7de9dSWoojung.Huh@microchip.com 		temp_pmt_ctl |= PMT_CTL_SUS_MODE_0_;
465055d7de9dSWoojung.Huh@microchip.com 	}
465155d7de9dSWoojung.Huh@microchip.com 	if (wol & WAKE_MCAST) {
465255d7de9dSWoojung.Huh@microchip.com 		temp_wucsr |= WUCSR_WAKE_EN_;
465355d7de9dSWoojung.Huh@microchip.com 
465455d7de9dSWoojung.Huh@microchip.com 		/* set WUF_CFG & WUF_MASK for IPv4 Multicast */
465555d7de9dSWoojung.Huh@microchip.com 		crc = lan78xx_wakeframe_crc16(ipv4_multicast, 3);
46563415f6baSJohn Efstathiades 		ret = lan78xx_write_reg(dev, WUF_CFG(mask_index),
465755d7de9dSWoojung.Huh@microchip.com 					WUF_CFGX_EN_ |
465855d7de9dSWoojung.Huh@microchip.com 					WUF_CFGX_TYPE_MCAST_ |
465955d7de9dSWoojung.Huh@microchip.com 					(0 << WUF_CFGX_OFFSET_SHIFT_) |
466055d7de9dSWoojung.Huh@microchip.com 					(crc & WUF_CFGX_CRC16_MASK_));
46613415f6baSJohn Efstathiades 		if (ret < 0)
46623415f6baSJohn Efstathiades 			return ret;
466355d7de9dSWoojung.Huh@microchip.com 
46643415f6baSJohn Efstathiades 		ret = lan78xx_write_reg(dev, WUF_MASK0(mask_index), 7);
46653415f6baSJohn Efstathiades 		if (ret < 0)
46663415f6baSJohn Efstathiades 			return ret;
46673415f6baSJohn Efstathiades 		ret = lan78xx_write_reg(dev, WUF_MASK1(mask_index), 0);
46683415f6baSJohn Efstathiades 		if (ret < 0)
46693415f6baSJohn Efstathiades 			return ret;
46703415f6baSJohn Efstathiades 		ret = lan78xx_write_reg(dev, WUF_MASK2(mask_index), 0);
46713415f6baSJohn Efstathiades 		if (ret < 0)
46723415f6baSJohn Efstathiades 			return ret;
46733415f6baSJohn Efstathiades 		ret = lan78xx_write_reg(dev, WUF_MASK3(mask_index), 0);
46743415f6baSJohn Efstathiades 		if (ret < 0)
46753415f6baSJohn Efstathiades 			return ret;
46763415f6baSJohn Efstathiades 
467755d7de9dSWoojung.Huh@microchip.com 		mask_index++;
467855d7de9dSWoojung.Huh@microchip.com 
467955d7de9dSWoojung.Huh@microchip.com 		/* for IPv6 Multicast */
468055d7de9dSWoojung.Huh@microchip.com 		crc = lan78xx_wakeframe_crc16(ipv6_multicast, 2);
46813415f6baSJohn Efstathiades 		ret = lan78xx_write_reg(dev, WUF_CFG(mask_index),
468255d7de9dSWoojung.Huh@microchip.com 					WUF_CFGX_EN_ |
468355d7de9dSWoojung.Huh@microchip.com 					WUF_CFGX_TYPE_MCAST_ |
468455d7de9dSWoojung.Huh@microchip.com 					(0 << WUF_CFGX_OFFSET_SHIFT_) |
468555d7de9dSWoojung.Huh@microchip.com 					(crc & WUF_CFGX_CRC16_MASK_));
46863415f6baSJohn Efstathiades 		if (ret < 0)
46873415f6baSJohn Efstathiades 			return ret;
468855d7de9dSWoojung.Huh@microchip.com 
46893415f6baSJohn Efstathiades 		ret = lan78xx_write_reg(dev, WUF_MASK0(mask_index), 3);
46903415f6baSJohn Efstathiades 		if (ret < 0)
46913415f6baSJohn Efstathiades 			return ret;
46923415f6baSJohn Efstathiades 		ret = lan78xx_write_reg(dev, WUF_MASK1(mask_index), 0);
46933415f6baSJohn Efstathiades 		if (ret < 0)
46943415f6baSJohn Efstathiades 			return ret;
46953415f6baSJohn Efstathiades 		ret = lan78xx_write_reg(dev, WUF_MASK2(mask_index), 0);
46963415f6baSJohn Efstathiades 		if (ret < 0)
46973415f6baSJohn Efstathiades 			return ret;
46983415f6baSJohn Efstathiades 		ret = lan78xx_write_reg(dev, WUF_MASK3(mask_index), 0);
46993415f6baSJohn Efstathiades 		if (ret < 0)
47003415f6baSJohn Efstathiades 			return ret;
47013415f6baSJohn Efstathiades 
470255d7de9dSWoojung.Huh@microchip.com 		mask_index++;
470355d7de9dSWoojung.Huh@microchip.com 
470455d7de9dSWoojung.Huh@microchip.com 		temp_pmt_ctl |= PMT_CTL_WOL_EN_;
470555d7de9dSWoojung.Huh@microchip.com 		temp_pmt_ctl &= ~PMT_CTL_SUS_MODE_MASK_;
470655d7de9dSWoojung.Huh@microchip.com 		temp_pmt_ctl |= PMT_CTL_SUS_MODE_0_;
470755d7de9dSWoojung.Huh@microchip.com 	}
470855d7de9dSWoojung.Huh@microchip.com 	if (wol & WAKE_UCAST) {
470955d7de9dSWoojung.Huh@microchip.com 		temp_wucsr |= WUCSR_PFDA_EN_;
471055d7de9dSWoojung.Huh@microchip.com 
471155d7de9dSWoojung.Huh@microchip.com 		temp_pmt_ctl |= PMT_CTL_WOL_EN_;
471255d7de9dSWoojung.Huh@microchip.com 		temp_pmt_ctl &= ~PMT_CTL_SUS_MODE_MASK_;
471355d7de9dSWoojung.Huh@microchip.com 		temp_pmt_ctl |= PMT_CTL_SUS_MODE_0_;
471455d7de9dSWoojung.Huh@microchip.com 	}
471555d7de9dSWoojung.Huh@microchip.com 	if (wol & WAKE_ARP) {
471655d7de9dSWoojung.Huh@microchip.com 		temp_wucsr |= WUCSR_WAKE_EN_;
471755d7de9dSWoojung.Huh@microchip.com 
471855d7de9dSWoojung.Huh@microchip.com 		/* set WUF_CFG & WUF_MASK
471955d7de9dSWoojung.Huh@microchip.com 		 * for packettype (offset 12,13) = ARP (0x0806)
472055d7de9dSWoojung.Huh@microchip.com 		 */
472155d7de9dSWoojung.Huh@microchip.com 		crc = lan78xx_wakeframe_crc16(arp_type, 2);
47223415f6baSJohn Efstathiades 		ret = lan78xx_write_reg(dev, WUF_CFG(mask_index),
472355d7de9dSWoojung.Huh@microchip.com 					WUF_CFGX_EN_ |
472455d7de9dSWoojung.Huh@microchip.com 					WUF_CFGX_TYPE_ALL_ |
472555d7de9dSWoojung.Huh@microchip.com 					(0 << WUF_CFGX_OFFSET_SHIFT_) |
472655d7de9dSWoojung.Huh@microchip.com 					(crc & WUF_CFGX_CRC16_MASK_));
47273415f6baSJohn Efstathiades 		if (ret < 0)
47283415f6baSJohn Efstathiades 			return ret;
472955d7de9dSWoojung.Huh@microchip.com 
47303415f6baSJohn Efstathiades 		ret = lan78xx_write_reg(dev, WUF_MASK0(mask_index), 0x3000);
47313415f6baSJohn Efstathiades 		if (ret < 0)
47323415f6baSJohn Efstathiades 			return ret;
47333415f6baSJohn Efstathiades 		ret = lan78xx_write_reg(dev, WUF_MASK1(mask_index), 0);
47343415f6baSJohn Efstathiades 		if (ret < 0)
47353415f6baSJohn Efstathiades 			return ret;
47363415f6baSJohn Efstathiades 		ret = lan78xx_write_reg(dev, WUF_MASK2(mask_index), 0);
47373415f6baSJohn Efstathiades 		if (ret < 0)
47383415f6baSJohn Efstathiades 			return ret;
47393415f6baSJohn Efstathiades 		ret = lan78xx_write_reg(dev, WUF_MASK3(mask_index), 0);
47403415f6baSJohn Efstathiades 		if (ret < 0)
47413415f6baSJohn Efstathiades 			return ret;
47423415f6baSJohn Efstathiades 
474355d7de9dSWoojung.Huh@microchip.com 		mask_index++;
474455d7de9dSWoojung.Huh@microchip.com 
474555d7de9dSWoojung.Huh@microchip.com 		temp_pmt_ctl |= PMT_CTL_WOL_EN_;
474655d7de9dSWoojung.Huh@microchip.com 		temp_pmt_ctl &= ~PMT_CTL_SUS_MODE_MASK_;
474755d7de9dSWoojung.Huh@microchip.com 		temp_pmt_ctl |= PMT_CTL_SUS_MODE_0_;
474855d7de9dSWoojung.Huh@microchip.com 	}
474955d7de9dSWoojung.Huh@microchip.com 
47503415f6baSJohn Efstathiades 	ret = lan78xx_write_reg(dev, WUCSR, temp_wucsr);
47513415f6baSJohn Efstathiades 	if (ret < 0)
47523415f6baSJohn Efstathiades 		return ret;
475355d7de9dSWoojung.Huh@microchip.com 
475455d7de9dSWoojung.Huh@microchip.com 	/* when multiple WOL bits are set */
475555d7de9dSWoojung.Huh@microchip.com 	if (hweight_long((unsigned long)wol) > 1) {
475655d7de9dSWoojung.Huh@microchip.com 		temp_pmt_ctl |= PMT_CTL_WOL_EN_;
475755d7de9dSWoojung.Huh@microchip.com 		temp_pmt_ctl &= ~PMT_CTL_SUS_MODE_MASK_;
475855d7de9dSWoojung.Huh@microchip.com 		temp_pmt_ctl |= PMT_CTL_SUS_MODE_0_;
475955d7de9dSWoojung.Huh@microchip.com 	}
47603415f6baSJohn Efstathiades 	ret = lan78xx_write_reg(dev, PMT_CTL, temp_pmt_ctl);
47613415f6baSJohn Efstathiades 	if (ret < 0)
47623415f6baSJohn Efstathiades 		return ret;
476355d7de9dSWoojung.Huh@microchip.com 
476455d7de9dSWoojung.Huh@microchip.com 	/* clear WUPS */
47653415f6baSJohn Efstathiades 	ret = lan78xx_read_reg(dev, PMT_CTL, &buf);
47663415f6baSJohn Efstathiades 	if (ret < 0)
47673415f6baSJohn Efstathiades 		return ret;
47683415f6baSJohn Efstathiades 
476955d7de9dSWoojung.Huh@microchip.com 	buf |= PMT_CTL_WUPS_MASK_;
47703415f6baSJohn Efstathiades 
47713415f6baSJohn Efstathiades 	ret = lan78xx_write_reg(dev, PMT_CTL, buf);
47723415f6baSJohn Efstathiades 	if (ret < 0)
47733415f6baSJohn Efstathiades 		return ret;
477455d7de9dSWoojung.Huh@microchip.com 
4775e1210fe6SJohn Efstathiades 	ret = lan78xx_start_rx_path(dev);
4776e1210fe6SJohn Efstathiades 
47773415f6baSJohn Efstathiades 	return ret;
477855d7de9dSWoojung.Huh@microchip.com }
477955d7de9dSWoojung.Huh@microchip.com 
lan78xx_suspend(struct usb_interface * intf,pm_message_t message)4780e0c79ff6SBaoyou Xie static int lan78xx_suspend(struct usb_interface *intf, pm_message_t message)
478155d7de9dSWoojung.Huh@microchip.com {
478255d7de9dSWoojung.Huh@microchip.com 	struct lan78xx_net *dev = usb_get_intfdata(intf);
47835f4cc6e2SJohn Efstathiades 	bool dev_open;
478455d7de9dSWoojung.Huh@microchip.com 	int ret;
478555d7de9dSWoojung.Huh@microchip.com 
47865f4cc6e2SJohn Efstathiades 	mutex_lock(&dev->dev_mutex);
47875f4cc6e2SJohn Efstathiades 
47885f4cc6e2SJohn Efstathiades 	netif_dbg(dev, ifdown, dev->net,
47895f4cc6e2SJohn Efstathiades 		  "suspending: pm event %#x", message.event);
47905f4cc6e2SJohn Efstathiades 
47915f4cc6e2SJohn Efstathiades 	dev_open = test_bit(EVENT_DEV_OPEN, &dev->flags);
47925f4cc6e2SJohn Efstathiades 
47935f4cc6e2SJohn Efstathiades 	if (dev_open) {
479455d7de9dSWoojung.Huh@microchip.com 		spin_lock_irq(&dev->txq.lock);
479555d7de9dSWoojung.Huh@microchip.com 		/* don't autosuspend while transmitting */
479655d7de9dSWoojung.Huh@microchip.com 		if ((skb_queue_len(&dev->txq) ||
479755d7de9dSWoojung.Huh@microchip.com 		     skb_queue_len(&dev->txq_pend)) &&
479855d7de9dSWoojung.Huh@microchip.com 		    PMSG_IS_AUTO(message)) {
479955d7de9dSWoojung.Huh@microchip.com 			spin_unlock_irq(&dev->txq.lock);
480055d7de9dSWoojung.Huh@microchip.com 			ret = -EBUSY;
480155d7de9dSWoojung.Huh@microchip.com 			goto out;
480255d7de9dSWoojung.Huh@microchip.com 		} else {
480355d7de9dSWoojung.Huh@microchip.com 			set_bit(EVENT_DEV_ASLEEP, &dev->flags);
480455d7de9dSWoojung.Huh@microchip.com 			spin_unlock_irq(&dev->txq.lock);
480555d7de9dSWoojung.Huh@microchip.com 		}
480655d7de9dSWoojung.Huh@microchip.com 
4807e1210fe6SJohn Efstathiades 		/* stop RX */
4808e1210fe6SJohn Efstathiades 		ret = lan78xx_stop_rx_path(dev);
48093415f6baSJohn Efstathiades 		if (ret < 0)
48105f4cc6e2SJohn Efstathiades 			goto out;
48113415f6baSJohn Efstathiades 
4812e1210fe6SJohn Efstathiades 		ret = lan78xx_flush_rx_fifo(dev);
48133415f6baSJohn Efstathiades 		if (ret < 0)
48145f4cc6e2SJohn Efstathiades 			goto out;
48153415f6baSJohn Efstathiades 
4816e1210fe6SJohn Efstathiades 		/* stop Tx */
4817e1210fe6SJohn Efstathiades 		ret = lan78xx_stop_tx_path(dev);
48183415f6baSJohn Efstathiades 		if (ret < 0)
48195f4cc6e2SJohn Efstathiades 			goto out;
482055d7de9dSWoojung.Huh@microchip.com 
48215f4cc6e2SJohn Efstathiades 		/* empty out the Rx and Tx queues */
482255d7de9dSWoojung.Huh@microchip.com 		netif_device_detach(dev->net);
482355d7de9dSWoojung.Huh@microchip.com 		lan78xx_terminate_urbs(dev);
482455d7de9dSWoojung.Huh@microchip.com 		usb_kill_urb(dev->urb_intr);
482555d7de9dSWoojung.Huh@microchip.com 
482655d7de9dSWoojung.Huh@microchip.com 		/* reattach */
482755d7de9dSWoojung.Huh@microchip.com 		netif_device_attach(dev->net);
482855d7de9dSWoojung.Huh@microchip.com 
482920ff5565SWoojung Huh 		del_timer(&dev->stat_monitor);
483020ff5565SWoojung Huh 
483155d7de9dSWoojung.Huh@microchip.com 		if (PMSG_IS_AUTO(message)) {
48325f4cc6e2SJohn Efstathiades 			ret = lan78xx_set_auto_suspend(dev);
48333415f6baSJohn Efstathiades 			if (ret < 0)
48345f4cc6e2SJohn Efstathiades 				goto out;
48355f4cc6e2SJohn Efstathiades 		} else {
48365f4cc6e2SJohn Efstathiades 			struct lan78xx_priv *pdata;
48373415f6baSJohn Efstathiades 
48385f4cc6e2SJohn Efstathiades 			pdata = (struct lan78xx_priv *)(dev->data[0]);
48395f4cc6e2SJohn Efstathiades 			netif_carrier_off(dev->net);
48405f4cc6e2SJohn Efstathiades 			ret = lan78xx_set_suspend(dev, pdata->wol);
48413415f6baSJohn Efstathiades 			if (ret < 0)
48425f4cc6e2SJohn Efstathiades 				goto out;
48435f4cc6e2SJohn Efstathiades 		}
48445f4cc6e2SJohn Efstathiades 	} else {
48455f4cc6e2SJohn Efstathiades 		/* Interface is down; don't allow WOL and PHY
48465f4cc6e2SJohn Efstathiades 		 * events to wake up the host
48475f4cc6e2SJohn Efstathiades 		 */
48485f4cc6e2SJohn Efstathiades 		u32 buf;
48495f4cc6e2SJohn Efstathiades 
48505f4cc6e2SJohn Efstathiades 		set_bit(EVENT_DEV_ASLEEP, &dev->flags);
485155d7de9dSWoojung.Huh@microchip.com 
485255d7de9dSWoojung.Huh@microchip.com 		ret = lan78xx_write_reg(dev, WUCSR, 0);
48533415f6baSJohn Efstathiades 		if (ret < 0)
48545f4cc6e2SJohn Efstathiades 			goto out;
485555d7de9dSWoojung.Huh@microchip.com 		ret = lan78xx_write_reg(dev, WUCSR2, 0);
48563415f6baSJohn Efstathiades 		if (ret < 0)
48575f4cc6e2SJohn Efstathiades 			goto out;
485855d7de9dSWoojung.Huh@microchip.com 
485955d7de9dSWoojung.Huh@microchip.com 		ret = lan78xx_read_reg(dev, PMT_CTL, &buf);
48603415f6baSJohn Efstathiades 		if (ret < 0)
48615f4cc6e2SJohn Efstathiades 			goto out;
486255d7de9dSWoojung.Huh@microchip.com 
486355d7de9dSWoojung.Huh@microchip.com 		buf &= ~PMT_CTL_RES_CLR_WKP_EN_;
486455d7de9dSWoojung.Huh@microchip.com 		buf |= PMT_CTL_RES_CLR_WKP_STS_;
486555d7de9dSWoojung.Huh@microchip.com 		buf &= ~PMT_CTL_SUS_MODE_MASK_;
486655d7de9dSWoojung.Huh@microchip.com 		buf |= PMT_CTL_SUS_MODE_3_;
486755d7de9dSWoojung.Huh@microchip.com 
486855d7de9dSWoojung.Huh@microchip.com 		ret = lan78xx_write_reg(dev, PMT_CTL, buf);
48693415f6baSJohn Efstathiades 		if (ret < 0)
48705f4cc6e2SJohn Efstathiades 			goto out;
487155d7de9dSWoojung.Huh@microchip.com 
487255d7de9dSWoojung.Huh@microchip.com 		ret = lan78xx_read_reg(dev, PMT_CTL, &buf);
48733415f6baSJohn Efstathiades 		if (ret < 0)
48745f4cc6e2SJohn Efstathiades 			goto out;
487555d7de9dSWoojung.Huh@microchip.com 
487655d7de9dSWoojung.Huh@microchip.com 		buf |= PMT_CTL_WUPS_MASK_;
487755d7de9dSWoojung.Huh@microchip.com 
487855d7de9dSWoojung.Huh@microchip.com 		ret = lan78xx_write_reg(dev, PMT_CTL, buf);
48793415f6baSJohn Efstathiades 		if (ret < 0)
48805f4cc6e2SJohn Efstathiades 			goto out;
488155d7de9dSWoojung.Huh@microchip.com 	}
488255d7de9dSWoojung.Huh@microchip.com 
488349d28b56SWoojung.Huh@microchip.com 	ret = 0;
488455d7de9dSWoojung.Huh@microchip.com out:
48855f4cc6e2SJohn Efstathiades 	mutex_unlock(&dev->dev_mutex);
48865f4cc6e2SJohn Efstathiades 
488755d7de9dSWoojung.Huh@microchip.com 	return ret;
488855d7de9dSWoojung.Huh@microchip.com }
488955d7de9dSWoojung.Huh@microchip.com 
lan78xx_submit_deferred_urbs(struct lan78xx_net * dev)48905f4cc6e2SJohn Efstathiades static bool lan78xx_submit_deferred_urbs(struct lan78xx_net *dev)
48915f4cc6e2SJohn Efstathiades {
48925f4cc6e2SJohn Efstathiades 	bool pipe_halted = false;
48935f4cc6e2SJohn Efstathiades 	struct urb *urb;
48945f4cc6e2SJohn Efstathiades 
48955f4cc6e2SJohn Efstathiades 	while ((urb = usb_get_from_anchor(&dev->deferred))) {
48965f4cc6e2SJohn Efstathiades 		struct sk_buff *skb = urb->context;
48975f4cc6e2SJohn Efstathiades 		int ret;
48985f4cc6e2SJohn Efstathiades 
48995f4cc6e2SJohn Efstathiades 		if (!netif_device_present(dev->net) ||
49005f4cc6e2SJohn Efstathiades 		    !netif_carrier_ok(dev->net) ||
49015f4cc6e2SJohn Efstathiades 		    pipe_halted) {
4902d383216aSJohn Efstathiades 			lan78xx_release_tx_buf(dev, skb);
49035f4cc6e2SJohn Efstathiades 			continue;
49045f4cc6e2SJohn Efstathiades 		}
49055f4cc6e2SJohn Efstathiades 
49065f4cc6e2SJohn Efstathiades 		ret = usb_submit_urb(urb, GFP_ATOMIC);
49075f4cc6e2SJohn Efstathiades 
49085f4cc6e2SJohn Efstathiades 		if (ret == 0) {
49095f4cc6e2SJohn Efstathiades 			netif_trans_update(dev->net);
49105f4cc6e2SJohn Efstathiades 			lan78xx_queue_skb(&dev->txq, skb, tx_start);
49115f4cc6e2SJohn Efstathiades 		} else {
49125f4cc6e2SJohn Efstathiades 			if (ret == -EPIPE) {
49135f4cc6e2SJohn Efstathiades 				netif_stop_queue(dev->net);
49145f4cc6e2SJohn Efstathiades 				pipe_halted = true;
49155f4cc6e2SJohn Efstathiades 			} else if (ret == -ENODEV) {
49165f4cc6e2SJohn Efstathiades 				netif_device_detach(dev->net);
49175f4cc6e2SJohn Efstathiades 			}
4918d383216aSJohn Efstathiades 
4919d383216aSJohn Efstathiades 			lan78xx_release_tx_buf(dev, skb);
49205f4cc6e2SJohn Efstathiades 		}
49215f4cc6e2SJohn Efstathiades 	}
49225f4cc6e2SJohn Efstathiades 
49235f4cc6e2SJohn Efstathiades 	return pipe_halted;
49245f4cc6e2SJohn Efstathiades }
49255f4cc6e2SJohn Efstathiades 
lan78xx_resume(struct usb_interface * intf)4926e0c79ff6SBaoyou Xie static int lan78xx_resume(struct usb_interface *intf)
492755d7de9dSWoojung.Huh@microchip.com {
492855d7de9dSWoojung.Huh@microchip.com 	struct lan78xx_net *dev = usb_get_intfdata(intf);
49295f4cc6e2SJohn Efstathiades 	bool dev_open;
493055d7de9dSWoojung.Huh@microchip.com 	int ret;
493155d7de9dSWoojung.Huh@microchip.com 
49325f4cc6e2SJohn Efstathiades 	mutex_lock(&dev->dev_mutex);
49335f4cc6e2SJohn Efstathiades 
49345f4cc6e2SJohn Efstathiades 	netif_dbg(dev, ifup, dev->net, "resuming device");
49355f4cc6e2SJohn Efstathiades 
49365f4cc6e2SJohn Efstathiades 	dev_open = test_bit(EVENT_DEV_OPEN, &dev->flags);
49375f4cc6e2SJohn Efstathiades 
49385f4cc6e2SJohn Efstathiades 	if (dev_open) {
49395f4cc6e2SJohn Efstathiades 		bool pipe_halted = false;
49405f4cc6e2SJohn Efstathiades 
49415f4cc6e2SJohn Efstathiades 		ret = lan78xx_flush_tx_fifo(dev);
49425f4cc6e2SJohn Efstathiades 		if (ret < 0)
49435f4cc6e2SJohn Efstathiades 			goto out;
49445f4cc6e2SJohn Efstathiades 
49455f4cc6e2SJohn Efstathiades 		if (dev->urb_intr) {
49465f4cc6e2SJohn Efstathiades 			int ret = usb_submit_urb(dev->urb_intr, GFP_KERNEL);
49475f4cc6e2SJohn Efstathiades 
49485f4cc6e2SJohn Efstathiades 			if (ret < 0) {
49495f4cc6e2SJohn Efstathiades 				if (ret == -ENODEV)
49505f4cc6e2SJohn Efstathiades 					netif_device_detach(dev->net);
49515f4cc6e2SJohn Efstathiades 				netdev_warn(dev->net, "Failed to submit intr URB");
49525f4cc6e2SJohn Efstathiades 			}
49535f4cc6e2SJohn Efstathiades 		}
49545f4cc6e2SJohn Efstathiades 
49555f4cc6e2SJohn Efstathiades 		spin_lock_irq(&dev->txq.lock);
49565f4cc6e2SJohn Efstathiades 
49575f4cc6e2SJohn Efstathiades 		if (netif_device_present(dev->net)) {
49585f4cc6e2SJohn Efstathiades 			pipe_halted = lan78xx_submit_deferred_urbs(dev);
49595f4cc6e2SJohn Efstathiades 
49605f4cc6e2SJohn Efstathiades 			if (pipe_halted)
49615f4cc6e2SJohn Efstathiades 				lan78xx_defer_kevent(dev, EVENT_TX_HALT);
49625f4cc6e2SJohn Efstathiades 		}
49635f4cc6e2SJohn Efstathiades 
49645f4cc6e2SJohn Efstathiades 		clear_bit(EVENT_DEV_ASLEEP, &dev->flags);
49655f4cc6e2SJohn Efstathiades 
49665f4cc6e2SJohn Efstathiades 		spin_unlock_irq(&dev->txq.lock);
49675f4cc6e2SJohn Efstathiades 
49685f4cc6e2SJohn Efstathiades 		if (!pipe_halted &&
49695f4cc6e2SJohn Efstathiades 		    netif_device_present(dev->net) &&
4970d383216aSJohn Efstathiades 		    (lan78xx_tx_pend_data_len(dev) < lan78xx_tx_urb_space(dev)))
49715f4cc6e2SJohn Efstathiades 			netif_start_queue(dev->net);
49725f4cc6e2SJohn Efstathiades 
49735f4cc6e2SJohn Efstathiades 		ret = lan78xx_start_tx_path(dev);
49745f4cc6e2SJohn Efstathiades 		if (ret < 0)
49755f4cc6e2SJohn Efstathiades 			goto out;
49765f4cc6e2SJohn Efstathiades 
4977ec4c7e12SJohn Efstathiades 		napi_schedule(&dev->napi);
49785f4cc6e2SJohn Efstathiades 
497920ff5565SWoojung Huh 		if (!timer_pending(&dev->stat_monitor)) {
498020ff5565SWoojung Huh 			dev->delta = 1;
498120ff5565SWoojung Huh 			mod_timer(&dev->stat_monitor,
498220ff5565SWoojung Huh 				  jiffies + STAT_UPDATE_TIMER);
498320ff5565SWoojung Huh 		}
498420ff5565SWoojung Huh 
498555d7de9dSWoojung.Huh@microchip.com 	} else {
498655d7de9dSWoojung.Huh@microchip.com 		clear_bit(EVENT_DEV_ASLEEP, &dev->flags);
498755d7de9dSWoojung.Huh@microchip.com 	}
498855d7de9dSWoojung.Huh@microchip.com 
498955d7de9dSWoojung.Huh@microchip.com 	ret = lan78xx_write_reg(dev, WUCSR2, 0);
49903415f6baSJohn Efstathiades 	if (ret < 0)
49915f4cc6e2SJohn Efstathiades 		goto out;
499255d7de9dSWoojung.Huh@microchip.com 	ret = lan78xx_write_reg(dev, WUCSR, 0);
49933415f6baSJohn Efstathiades 	if (ret < 0)
49945f4cc6e2SJohn Efstathiades 		goto out;
499555d7de9dSWoojung.Huh@microchip.com 	ret = lan78xx_write_reg(dev, WK_SRC, 0xFFF1FF1FUL);
49963415f6baSJohn Efstathiades 	if (ret < 0)
49975f4cc6e2SJohn Efstathiades 		goto out;
499855d7de9dSWoojung.Huh@microchip.com 
499955d7de9dSWoojung.Huh@microchip.com 	ret = lan78xx_write_reg(dev, WUCSR2, WUCSR2_NS_RCD_ |
500055d7de9dSWoojung.Huh@microchip.com 					     WUCSR2_ARP_RCD_ |
500155d7de9dSWoojung.Huh@microchip.com 					     WUCSR2_IPV6_TCPSYN_RCD_ |
500255d7de9dSWoojung.Huh@microchip.com 					     WUCSR2_IPV4_TCPSYN_RCD_);
50033415f6baSJohn Efstathiades 	if (ret < 0)
50045f4cc6e2SJohn Efstathiades 		goto out;
500555d7de9dSWoojung.Huh@microchip.com 
500655d7de9dSWoojung.Huh@microchip.com 	ret = lan78xx_write_reg(dev, WUCSR, WUCSR_EEE_TX_WAKE_ |
500755d7de9dSWoojung.Huh@microchip.com 					    WUCSR_EEE_RX_WAKE_ |
500855d7de9dSWoojung.Huh@microchip.com 					    WUCSR_PFDA_FR_ |
500955d7de9dSWoojung.Huh@microchip.com 					    WUCSR_RFE_WAKE_FR_ |
501055d7de9dSWoojung.Huh@microchip.com 					    WUCSR_WUFR_ |
501155d7de9dSWoojung.Huh@microchip.com 					    WUCSR_MPR_ |
501255d7de9dSWoojung.Huh@microchip.com 					    WUCSR_BCST_FR_);
50133415f6baSJohn Efstathiades 	if (ret < 0)
50145f4cc6e2SJohn Efstathiades 		goto out;
501555d7de9dSWoojung.Huh@microchip.com 
50165f4cc6e2SJohn Efstathiades 	ret = 0;
50175f4cc6e2SJohn Efstathiades out:
50185f4cc6e2SJohn Efstathiades 	mutex_unlock(&dev->dev_mutex);
5019e1210fe6SJohn Efstathiades 
50203415f6baSJohn Efstathiades 	return ret;
502155d7de9dSWoojung.Huh@microchip.com }
502255d7de9dSWoojung.Huh@microchip.com 
lan78xx_reset_resume(struct usb_interface * intf)5023e0c79ff6SBaoyou Xie static int lan78xx_reset_resume(struct usb_interface *intf)
502455d7de9dSWoojung.Huh@microchip.com {
502555d7de9dSWoojung.Huh@microchip.com 	struct lan78xx_net *dev = usb_get_intfdata(intf);
50263415f6baSJohn Efstathiades 	int ret;
502755d7de9dSWoojung.Huh@microchip.com 
50285f4cc6e2SJohn Efstathiades 	netif_dbg(dev, ifup, dev->net, "(reset) resuming device");
50295f4cc6e2SJohn Efstathiades 
50303415f6baSJohn Efstathiades 	ret = lan78xx_reset(dev);
50313415f6baSJohn Efstathiades 	if (ret < 0)
50323415f6baSJohn Efstathiades 		return ret;
5033ce85e13aSWoojung.Huh@microchip.com 
503492571a1aSAlexander Graf 	phy_start(dev->net->phydev);
5035ce85e13aSWoojung.Huh@microchip.com 
50363415f6baSJohn Efstathiades 	ret = lan78xx_resume(intf);
50373415f6baSJohn Efstathiades 
50383415f6baSJohn Efstathiades 	return ret;
503955d7de9dSWoojung.Huh@microchip.com }
504055d7de9dSWoojung.Huh@microchip.com 
504155d7de9dSWoojung.Huh@microchip.com static const struct usb_device_id products[] = {
504255d7de9dSWoojung.Huh@microchip.com 	{
504355d7de9dSWoojung.Huh@microchip.com 	/* LAN7800 USB Gigabit Ethernet Device */
504455d7de9dSWoojung.Huh@microchip.com 	USB_DEVICE(LAN78XX_USB_VENDOR_ID, LAN7800_USB_PRODUCT_ID),
504555d7de9dSWoojung.Huh@microchip.com 	},
504655d7de9dSWoojung.Huh@microchip.com 	{
504755d7de9dSWoojung.Huh@microchip.com 	/* LAN7850 USB Gigabit Ethernet Device */
504855d7de9dSWoojung.Huh@microchip.com 	USB_DEVICE(LAN78XX_USB_VENDOR_ID, LAN7850_USB_PRODUCT_ID),
504955d7de9dSWoojung.Huh@microchip.com 	},
505002dc1f3dSWoojung Huh 	{
505102dc1f3dSWoojung Huh 	/* LAN7801 USB Gigabit Ethernet Device */
505202dc1f3dSWoojung Huh 	USB_DEVICE(LAN78XX_USB_VENDOR_ID, LAN7801_USB_PRODUCT_ID),
505302dc1f3dSWoojung Huh 	},
5054ef8a0f6eSGreg Jesionowski 	{
5055ef8a0f6eSGreg Jesionowski 	/* ATM2-AF USB Gigabit Ethernet Device */
5056ef8a0f6eSGreg Jesionowski 	USB_DEVICE(AT29M2AF_USB_VENDOR_ID, AT29M2AF_USB_PRODUCT_ID),
5057ef8a0f6eSGreg Jesionowski 	},
505855d7de9dSWoojung.Huh@microchip.com 	{},
505955d7de9dSWoojung.Huh@microchip.com };
506055d7de9dSWoojung.Huh@microchip.com MODULE_DEVICE_TABLE(usb, products);
506155d7de9dSWoojung.Huh@microchip.com 
506255d7de9dSWoojung.Huh@microchip.com static struct usb_driver lan78xx_driver = {
506355d7de9dSWoojung.Huh@microchip.com 	.name			= DRIVER_NAME,
506455d7de9dSWoojung.Huh@microchip.com 	.id_table		= products,
506555d7de9dSWoojung.Huh@microchip.com 	.probe			= lan78xx_probe,
506655d7de9dSWoojung.Huh@microchip.com 	.disconnect		= lan78xx_disconnect,
506755d7de9dSWoojung.Huh@microchip.com 	.suspend		= lan78xx_suspend,
506855d7de9dSWoojung.Huh@microchip.com 	.resume			= lan78xx_resume,
506955d7de9dSWoojung.Huh@microchip.com 	.reset_resume		= lan78xx_reset_resume,
507055d7de9dSWoojung.Huh@microchip.com 	.supports_autosuspend	= 1,
507155d7de9dSWoojung.Huh@microchip.com 	.disable_hub_initiated_lpm = 1,
507255d7de9dSWoojung.Huh@microchip.com };
507355d7de9dSWoojung.Huh@microchip.com 
507455d7de9dSWoojung.Huh@microchip.com module_usb_driver(lan78xx_driver);
507555d7de9dSWoojung.Huh@microchip.com 
507655d7de9dSWoojung.Huh@microchip.com MODULE_AUTHOR(DRIVER_AUTHOR);
507755d7de9dSWoojung.Huh@microchip.com MODULE_DESCRIPTION(DRIVER_DESC);
507855d7de9dSWoojung.Huh@microchip.com MODULE_LICENSE("GPL");
5079