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, ®);
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