1bcc9736cSJeff Kirsher /**
23396c782SPaul Gortmaker  * drivers/net/ethernet/micrel/ksx884x.c - Micrel KSZ8841/2 PCI Ethernet driver
3bcc9736cSJeff Kirsher  *
4bcc9736cSJeff Kirsher  * Copyright (c) 2009-2010 Micrel, Inc.
5bcc9736cSJeff Kirsher  * 	Tristram Ha <Tristram.Ha@micrel.com>
6bcc9736cSJeff Kirsher  *
7bcc9736cSJeff Kirsher  * This program is free software; you can redistribute it and/or modify
8bcc9736cSJeff Kirsher  * it under the terms of the GNU General Public License version 2 as
9bcc9736cSJeff Kirsher  * published by the Free Software Foundation.
10bcc9736cSJeff Kirsher  *
11bcc9736cSJeff Kirsher  * This program is distributed in the hope that it will be useful,
12bcc9736cSJeff Kirsher  * but WITHOUT ANY WARRANTY; without even the implied warranty of
13bcc9736cSJeff Kirsher  * MERCHANTABILITY or FITNESS FOR A PARTICULAR PURPOSE.  See the
14bcc9736cSJeff Kirsher  * GNU General Public License for more details.
15bcc9736cSJeff Kirsher  */
16bcc9736cSJeff Kirsher 
17bcc9736cSJeff Kirsher #define pr_fmt(fmt) KBUILD_MODNAME ": " fmt
18bcc9736cSJeff Kirsher 
19bcc9736cSJeff Kirsher #include <linux/init.h>
20bcc9736cSJeff Kirsher #include <linux/interrupt.h>
21bcc9736cSJeff Kirsher #include <linux/kernel.h>
22bcc9736cSJeff Kirsher #include <linux/module.h>
23bcc9736cSJeff Kirsher #include <linux/ioport.h>
24bcc9736cSJeff Kirsher #include <linux/pci.h>
25bcc9736cSJeff Kirsher #include <linux/proc_fs.h>
26bcc9736cSJeff Kirsher #include <linux/mii.h>
27bcc9736cSJeff Kirsher #include <linux/platform_device.h>
28bcc9736cSJeff Kirsher #include <linux/ethtool.h>
29bcc9736cSJeff Kirsher #include <linux/etherdevice.h>
30bcc9736cSJeff Kirsher #include <linux/in.h>
31bcc9736cSJeff Kirsher #include <linux/ip.h>
32bcc9736cSJeff Kirsher #include <linux/if_vlan.h>
33bcc9736cSJeff Kirsher #include <linux/crc32.h>
34bcc9736cSJeff Kirsher #include <linux/sched.h>
35bcc9736cSJeff Kirsher #include <linux/slab.h>
36bcc9736cSJeff Kirsher 
37bcc9736cSJeff Kirsher 
38bcc9736cSJeff Kirsher /* DMA Registers */
39bcc9736cSJeff Kirsher 
40bcc9736cSJeff Kirsher #define KS_DMA_TX_CTRL			0x0000
41bcc9736cSJeff Kirsher #define DMA_TX_ENABLE			0x00000001
42bcc9736cSJeff Kirsher #define DMA_TX_CRC_ENABLE		0x00000002
43bcc9736cSJeff Kirsher #define DMA_TX_PAD_ENABLE		0x00000004
44bcc9736cSJeff Kirsher #define DMA_TX_LOOPBACK			0x00000100
45bcc9736cSJeff Kirsher #define DMA_TX_FLOW_ENABLE		0x00000200
46bcc9736cSJeff Kirsher #define DMA_TX_CSUM_IP			0x00010000
47bcc9736cSJeff Kirsher #define DMA_TX_CSUM_TCP			0x00020000
48bcc9736cSJeff Kirsher #define DMA_TX_CSUM_UDP			0x00040000
49bcc9736cSJeff Kirsher #define DMA_TX_BURST_SIZE		0x3F000000
50bcc9736cSJeff Kirsher 
51bcc9736cSJeff Kirsher #define KS_DMA_RX_CTRL			0x0004
52bcc9736cSJeff Kirsher #define DMA_RX_ENABLE			0x00000001
53bcc9736cSJeff Kirsher #define KS884X_DMA_RX_MULTICAST		0x00000002
54bcc9736cSJeff Kirsher #define DMA_RX_PROMISCUOUS		0x00000004
55bcc9736cSJeff Kirsher #define DMA_RX_ERROR			0x00000008
56bcc9736cSJeff Kirsher #define DMA_RX_UNICAST			0x00000010
57bcc9736cSJeff Kirsher #define DMA_RX_ALL_MULTICAST		0x00000020
58bcc9736cSJeff Kirsher #define DMA_RX_BROADCAST		0x00000040
59bcc9736cSJeff Kirsher #define DMA_RX_FLOW_ENABLE		0x00000200
60bcc9736cSJeff Kirsher #define DMA_RX_CSUM_IP			0x00010000
61bcc9736cSJeff Kirsher #define DMA_RX_CSUM_TCP			0x00020000
62bcc9736cSJeff Kirsher #define DMA_RX_CSUM_UDP			0x00040000
63bcc9736cSJeff Kirsher #define DMA_RX_BURST_SIZE		0x3F000000
64bcc9736cSJeff Kirsher 
65bcc9736cSJeff Kirsher #define DMA_BURST_SHIFT			24
66bcc9736cSJeff Kirsher #define DMA_BURST_DEFAULT		8
67bcc9736cSJeff Kirsher 
68bcc9736cSJeff Kirsher #define KS_DMA_TX_START			0x0008
69bcc9736cSJeff Kirsher #define KS_DMA_RX_START			0x000C
70bcc9736cSJeff Kirsher #define DMA_START			0x00000001
71bcc9736cSJeff Kirsher 
72bcc9736cSJeff Kirsher #define KS_DMA_TX_ADDR			0x0010
73bcc9736cSJeff Kirsher #define KS_DMA_RX_ADDR			0x0014
74bcc9736cSJeff Kirsher 
75bcc9736cSJeff Kirsher #define DMA_ADDR_LIST_MASK		0xFFFFFFFC
76bcc9736cSJeff Kirsher #define DMA_ADDR_LIST_SHIFT		2
77bcc9736cSJeff Kirsher 
78bcc9736cSJeff Kirsher /* MTR0 */
79bcc9736cSJeff Kirsher #define KS884X_MULTICAST_0_OFFSET	0x0020
80bcc9736cSJeff Kirsher #define KS884X_MULTICAST_1_OFFSET	0x0021
81bcc9736cSJeff Kirsher #define KS884X_MULTICAST_2_OFFSET	0x0022
82bcc9736cSJeff Kirsher #define KS884x_MULTICAST_3_OFFSET	0x0023
83bcc9736cSJeff Kirsher /* MTR1 */
84bcc9736cSJeff Kirsher #define KS884X_MULTICAST_4_OFFSET	0x0024
85bcc9736cSJeff Kirsher #define KS884X_MULTICAST_5_OFFSET	0x0025
86bcc9736cSJeff Kirsher #define KS884X_MULTICAST_6_OFFSET	0x0026
87bcc9736cSJeff Kirsher #define KS884X_MULTICAST_7_OFFSET	0x0027
88bcc9736cSJeff Kirsher 
89bcc9736cSJeff Kirsher /* Interrupt Registers */
90bcc9736cSJeff Kirsher 
91bcc9736cSJeff Kirsher /* INTEN */
92bcc9736cSJeff Kirsher #define KS884X_INTERRUPTS_ENABLE	0x0028
93bcc9736cSJeff Kirsher /* INTST */
94bcc9736cSJeff Kirsher #define KS884X_INTERRUPTS_STATUS	0x002C
95bcc9736cSJeff Kirsher 
96bcc9736cSJeff Kirsher #define KS884X_INT_RX_STOPPED		0x02000000
97bcc9736cSJeff Kirsher #define KS884X_INT_TX_STOPPED		0x04000000
98bcc9736cSJeff Kirsher #define KS884X_INT_RX_OVERRUN		0x08000000
99bcc9736cSJeff Kirsher #define KS884X_INT_TX_EMPTY		0x10000000
100bcc9736cSJeff Kirsher #define KS884X_INT_RX			0x20000000
101bcc9736cSJeff Kirsher #define KS884X_INT_TX			0x40000000
102bcc9736cSJeff Kirsher #define KS884X_INT_PHY			0x80000000
103bcc9736cSJeff Kirsher 
104bcc9736cSJeff Kirsher #define KS884X_INT_RX_MASK		\
105bcc9736cSJeff Kirsher 	(KS884X_INT_RX | KS884X_INT_RX_OVERRUN)
106bcc9736cSJeff Kirsher #define KS884X_INT_TX_MASK		\
107bcc9736cSJeff Kirsher 	(KS884X_INT_TX | KS884X_INT_TX_EMPTY)
108bcc9736cSJeff Kirsher #define KS884X_INT_MASK	(KS884X_INT_RX | KS884X_INT_TX | KS884X_INT_PHY)
109bcc9736cSJeff Kirsher 
110bcc9736cSJeff Kirsher /* MAC Additional Station Address */
111bcc9736cSJeff Kirsher 
112bcc9736cSJeff Kirsher /* MAAL0 */
113bcc9736cSJeff Kirsher #define KS_ADD_ADDR_0_LO		0x0080
114bcc9736cSJeff Kirsher /* MAAH0 */
115bcc9736cSJeff Kirsher #define KS_ADD_ADDR_0_HI		0x0084
116bcc9736cSJeff Kirsher /* MAAL1 */
117bcc9736cSJeff Kirsher #define KS_ADD_ADDR_1_LO		0x0088
118bcc9736cSJeff Kirsher /* MAAH1 */
119bcc9736cSJeff Kirsher #define KS_ADD_ADDR_1_HI		0x008C
120bcc9736cSJeff Kirsher /* MAAL2 */
121bcc9736cSJeff Kirsher #define KS_ADD_ADDR_2_LO		0x0090
122bcc9736cSJeff Kirsher /* MAAH2 */
123bcc9736cSJeff Kirsher #define KS_ADD_ADDR_2_HI		0x0094
124bcc9736cSJeff Kirsher /* MAAL3 */
125bcc9736cSJeff Kirsher #define KS_ADD_ADDR_3_LO		0x0098
126bcc9736cSJeff Kirsher /* MAAH3 */
127bcc9736cSJeff Kirsher #define KS_ADD_ADDR_3_HI		0x009C
128bcc9736cSJeff Kirsher /* MAAL4 */
129bcc9736cSJeff Kirsher #define KS_ADD_ADDR_4_LO		0x00A0
130bcc9736cSJeff Kirsher /* MAAH4 */
131bcc9736cSJeff Kirsher #define KS_ADD_ADDR_4_HI		0x00A4
132bcc9736cSJeff Kirsher /* MAAL5 */
133bcc9736cSJeff Kirsher #define KS_ADD_ADDR_5_LO		0x00A8
134bcc9736cSJeff Kirsher /* MAAH5 */
135bcc9736cSJeff Kirsher #define KS_ADD_ADDR_5_HI		0x00AC
136bcc9736cSJeff Kirsher /* MAAL6 */
137bcc9736cSJeff Kirsher #define KS_ADD_ADDR_6_LO		0x00B0
138bcc9736cSJeff Kirsher /* MAAH6 */
139bcc9736cSJeff Kirsher #define KS_ADD_ADDR_6_HI		0x00B4
140bcc9736cSJeff Kirsher /* MAAL7 */
141bcc9736cSJeff Kirsher #define KS_ADD_ADDR_7_LO		0x00B8
142bcc9736cSJeff Kirsher /* MAAH7 */
143bcc9736cSJeff Kirsher #define KS_ADD_ADDR_7_HI		0x00BC
144bcc9736cSJeff Kirsher /* MAAL8 */
145bcc9736cSJeff Kirsher #define KS_ADD_ADDR_8_LO		0x00C0
146bcc9736cSJeff Kirsher /* MAAH8 */
147bcc9736cSJeff Kirsher #define KS_ADD_ADDR_8_HI		0x00C4
148bcc9736cSJeff Kirsher /* MAAL9 */
149bcc9736cSJeff Kirsher #define KS_ADD_ADDR_9_LO		0x00C8
150bcc9736cSJeff Kirsher /* MAAH9 */
151bcc9736cSJeff Kirsher #define KS_ADD_ADDR_9_HI		0x00CC
152bcc9736cSJeff Kirsher /* MAAL10 */
153bcc9736cSJeff Kirsher #define KS_ADD_ADDR_A_LO		0x00D0
154bcc9736cSJeff Kirsher /* MAAH10 */
155bcc9736cSJeff Kirsher #define KS_ADD_ADDR_A_HI		0x00D4
156bcc9736cSJeff Kirsher /* MAAL11 */
157bcc9736cSJeff Kirsher #define KS_ADD_ADDR_B_LO		0x00D8
158bcc9736cSJeff Kirsher /* MAAH11 */
159bcc9736cSJeff Kirsher #define KS_ADD_ADDR_B_HI		0x00DC
160bcc9736cSJeff Kirsher /* MAAL12 */
161bcc9736cSJeff Kirsher #define KS_ADD_ADDR_C_LO		0x00E0
162bcc9736cSJeff Kirsher /* MAAH12 */
163bcc9736cSJeff Kirsher #define KS_ADD_ADDR_C_HI		0x00E4
164bcc9736cSJeff Kirsher /* MAAL13 */
165bcc9736cSJeff Kirsher #define KS_ADD_ADDR_D_LO		0x00E8
166bcc9736cSJeff Kirsher /* MAAH13 */
167bcc9736cSJeff Kirsher #define KS_ADD_ADDR_D_HI		0x00EC
168bcc9736cSJeff Kirsher /* MAAL14 */
169bcc9736cSJeff Kirsher #define KS_ADD_ADDR_E_LO		0x00F0
170bcc9736cSJeff Kirsher /* MAAH14 */
171bcc9736cSJeff Kirsher #define KS_ADD_ADDR_E_HI		0x00F4
172bcc9736cSJeff Kirsher /* MAAL15 */
173bcc9736cSJeff Kirsher #define KS_ADD_ADDR_F_LO		0x00F8
174bcc9736cSJeff Kirsher /* MAAH15 */
175bcc9736cSJeff Kirsher #define KS_ADD_ADDR_F_HI		0x00FC
176bcc9736cSJeff Kirsher 
177bcc9736cSJeff Kirsher #define ADD_ADDR_HI_MASK		0x0000FFFF
178bcc9736cSJeff Kirsher #define ADD_ADDR_ENABLE			0x80000000
179bcc9736cSJeff Kirsher #define ADD_ADDR_INCR			8
180bcc9736cSJeff Kirsher 
181bcc9736cSJeff Kirsher /* Miscellaneous Registers */
182bcc9736cSJeff Kirsher 
183bcc9736cSJeff Kirsher /* MARL */
184bcc9736cSJeff Kirsher #define KS884X_ADDR_0_OFFSET		0x0200
185bcc9736cSJeff Kirsher #define KS884X_ADDR_1_OFFSET		0x0201
186bcc9736cSJeff Kirsher /* MARM */
187bcc9736cSJeff Kirsher #define KS884X_ADDR_2_OFFSET		0x0202
188bcc9736cSJeff Kirsher #define KS884X_ADDR_3_OFFSET		0x0203
189bcc9736cSJeff Kirsher /* MARH */
190bcc9736cSJeff Kirsher #define KS884X_ADDR_4_OFFSET		0x0204
191bcc9736cSJeff Kirsher #define KS884X_ADDR_5_OFFSET		0x0205
192bcc9736cSJeff Kirsher 
193bcc9736cSJeff Kirsher /* OBCR */
194bcc9736cSJeff Kirsher #define KS884X_BUS_CTRL_OFFSET		0x0210
195bcc9736cSJeff Kirsher 
196bcc9736cSJeff Kirsher #define BUS_SPEED_125_MHZ		0x0000
197bcc9736cSJeff Kirsher #define BUS_SPEED_62_5_MHZ		0x0001
198bcc9736cSJeff Kirsher #define BUS_SPEED_41_66_MHZ		0x0002
199bcc9736cSJeff Kirsher #define BUS_SPEED_25_MHZ		0x0003
200bcc9736cSJeff Kirsher 
201bcc9736cSJeff Kirsher /* EEPCR */
202bcc9736cSJeff Kirsher #define KS884X_EEPROM_CTRL_OFFSET	0x0212
203bcc9736cSJeff Kirsher 
204bcc9736cSJeff Kirsher #define EEPROM_CHIP_SELECT		0x0001
205bcc9736cSJeff Kirsher #define EEPROM_SERIAL_CLOCK		0x0002
206bcc9736cSJeff Kirsher #define EEPROM_DATA_OUT			0x0004
207bcc9736cSJeff Kirsher #define EEPROM_DATA_IN			0x0008
208bcc9736cSJeff Kirsher #define EEPROM_ACCESS_ENABLE		0x0010
209bcc9736cSJeff Kirsher 
210bcc9736cSJeff Kirsher /* MBIR */
211bcc9736cSJeff Kirsher #define KS884X_MEM_INFO_OFFSET		0x0214
212bcc9736cSJeff Kirsher 
213bcc9736cSJeff Kirsher #define RX_MEM_TEST_FAILED		0x0008
214bcc9736cSJeff Kirsher #define RX_MEM_TEST_FINISHED		0x0010
215bcc9736cSJeff Kirsher #define TX_MEM_TEST_FAILED		0x0800
216bcc9736cSJeff Kirsher #define TX_MEM_TEST_FINISHED		0x1000
217bcc9736cSJeff Kirsher 
218bcc9736cSJeff Kirsher /* GCR */
219bcc9736cSJeff Kirsher #define KS884X_GLOBAL_CTRL_OFFSET	0x0216
220bcc9736cSJeff Kirsher #define GLOBAL_SOFTWARE_RESET		0x0001
221bcc9736cSJeff Kirsher 
222bcc9736cSJeff Kirsher #define KS8841_POWER_MANAGE_OFFSET	0x0218
223bcc9736cSJeff Kirsher 
224bcc9736cSJeff Kirsher /* WFCR */
225bcc9736cSJeff Kirsher #define KS8841_WOL_CTRL_OFFSET		0x021A
226bcc9736cSJeff Kirsher #define KS8841_WOL_MAGIC_ENABLE		0x0080
227bcc9736cSJeff Kirsher #define KS8841_WOL_FRAME3_ENABLE	0x0008
228bcc9736cSJeff Kirsher #define KS8841_WOL_FRAME2_ENABLE	0x0004
229bcc9736cSJeff Kirsher #define KS8841_WOL_FRAME1_ENABLE	0x0002
230bcc9736cSJeff Kirsher #define KS8841_WOL_FRAME0_ENABLE	0x0001
231bcc9736cSJeff Kirsher 
232bcc9736cSJeff Kirsher /* WF0 */
233bcc9736cSJeff Kirsher #define KS8841_WOL_FRAME_CRC_OFFSET	0x0220
234bcc9736cSJeff Kirsher #define KS8841_WOL_FRAME_BYTE0_OFFSET	0x0224
235bcc9736cSJeff Kirsher #define KS8841_WOL_FRAME_BYTE2_OFFSET	0x0228
236bcc9736cSJeff Kirsher 
237bcc9736cSJeff Kirsher /* IACR */
238bcc9736cSJeff Kirsher #define KS884X_IACR_P			0x04A0
239bcc9736cSJeff Kirsher #define KS884X_IACR_OFFSET		KS884X_IACR_P
240bcc9736cSJeff Kirsher 
241bcc9736cSJeff Kirsher /* IADR1 */
242bcc9736cSJeff Kirsher #define KS884X_IADR1_P			0x04A2
243bcc9736cSJeff Kirsher #define KS884X_IADR2_P			0x04A4
244bcc9736cSJeff Kirsher #define KS884X_IADR3_P			0x04A6
245bcc9736cSJeff Kirsher #define KS884X_IADR4_P			0x04A8
246bcc9736cSJeff Kirsher #define KS884X_IADR5_P			0x04AA
247bcc9736cSJeff Kirsher 
248bcc9736cSJeff Kirsher #define KS884X_ACC_CTRL_SEL_OFFSET	KS884X_IACR_P
249bcc9736cSJeff Kirsher #define KS884X_ACC_CTRL_INDEX_OFFSET	(KS884X_ACC_CTRL_SEL_OFFSET + 1)
250bcc9736cSJeff Kirsher 
251bcc9736cSJeff Kirsher #define KS884X_ACC_DATA_0_OFFSET	KS884X_IADR4_P
252bcc9736cSJeff Kirsher #define KS884X_ACC_DATA_1_OFFSET	(KS884X_ACC_DATA_0_OFFSET + 1)
253bcc9736cSJeff Kirsher #define KS884X_ACC_DATA_2_OFFSET	KS884X_IADR5_P
254bcc9736cSJeff Kirsher #define KS884X_ACC_DATA_3_OFFSET	(KS884X_ACC_DATA_2_OFFSET + 1)
255bcc9736cSJeff Kirsher #define KS884X_ACC_DATA_4_OFFSET	KS884X_IADR2_P
256bcc9736cSJeff Kirsher #define KS884X_ACC_DATA_5_OFFSET	(KS884X_ACC_DATA_4_OFFSET + 1)
257bcc9736cSJeff Kirsher #define KS884X_ACC_DATA_6_OFFSET	KS884X_IADR3_P
258bcc9736cSJeff Kirsher #define KS884X_ACC_DATA_7_OFFSET	(KS884X_ACC_DATA_6_OFFSET + 1)
259bcc9736cSJeff Kirsher #define KS884X_ACC_DATA_8_OFFSET	KS884X_IADR1_P
260bcc9736cSJeff Kirsher 
261bcc9736cSJeff Kirsher /* P1MBCR */
262bcc9736cSJeff Kirsher #define KS884X_P1MBCR_P			0x04D0
263bcc9736cSJeff Kirsher #define KS884X_P1MBSR_P			0x04D2
264bcc9736cSJeff Kirsher #define KS884X_PHY1ILR_P		0x04D4
265bcc9736cSJeff Kirsher #define KS884X_PHY1IHR_P		0x04D6
266bcc9736cSJeff Kirsher #define KS884X_P1ANAR_P			0x04D8
267bcc9736cSJeff Kirsher #define KS884X_P1ANLPR_P		0x04DA
268bcc9736cSJeff Kirsher 
269bcc9736cSJeff Kirsher /* P2MBCR */
270bcc9736cSJeff Kirsher #define KS884X_P2MBCR_P			0x04E0
271bcc9736cSJeff Kirsher #define KS884X_P2MBSR_P			0x04E2
272bcc9736cSJeff Kirsher #define KS884X_PHY2ILR_P		0x04E4
273bcc9736cSJeff Kirsher #define KS884X_PHY2IHR_P		0x04E6
274bcc9736cSJeff Kirsher #define KS884X_P2ANAR_P			0x04E8
275bcc9736cSJeff Kirsher #define KS884X_P2ANLPR_P		0x04EA
276bcc9736cSJeff Kirsher 
277bcc9736cSJeff Kirsher #define KS884X_PHY_1_CTRL_OFFSET	KS884X_P1MBCR_P
278bcc9736cSJeff Kirsher #define PHY_CTRL_INTERVAL		(KS884X_P2MBCR_P - KS884X_P1MBCR_P)
279bcc9736cSJeff Kirsher 
280bcc9736cSJeff Kirsher #define KS884X_PHY_CTRL_OFFSET		0x00
281bcc9736cSJeff Kirsher 
282bcc9736cSJeff Kirsher /* Mode Control Register */
283bcc9736cSJeff Kirsher #define PHY_REG_CTRL			0
284bcc9736cSJeff Kirsher 
285bcc9736cSJeff Kirsher #define PHY_RESET			0x8000
286bcc9736cSJeff Kirsher #define PHY_LOOPBACK			0x4000
287bcc9736cSJeff Kirsher #define PHY_SPEED_100MBIT		0x2000
288bcc9736cSJeff Kirsher #define PHY_AUTO_NEG_ENABLE		0x1000
289bcc9736cSJeff Kirsher #define PHY_POWER_DOWN			0x0800
290bcc9736cSJeff Kirsher #define PHY_MII_DISABLE			0x0400
291bcc9736cSJeff Kirsher #define PHY_AUTO_NEG_RESTART		0x0200
292bcc9736cSJeff Kirsher #define PHY_FULL_DUPLEX			0x0100
293bcc9736cSJeff Kirsher #define PHY_COLLISION_TEST		0x0080
294bcc9736cSJeff Kirsher #define PHY_HP_MDIX			0x0020
295bcc9736cSJeff Kirsher #define PHY_FORCE_MDIX			0x0010
296bcc9736cSJeff Kirsher #define PHY_AUTO_MDIX_DISABLE		0x0008
297bcc9736cSJeff Kirsher #define PHY_REMOTE_FAULT_DISABLE	0x0004
298bcc9736cSJeff Kirsher #define PHY_TRANSMIT_DISABLE		0x0002
299bcc9736cSJeff Kirsher #define PHY_LED_DISABLE			0x0001
300bcc9736cSJeff Kirsher 
301bcc9736cSJeff Kirsher #define KS884X_PHY_STATUS_OFFSET	0x02
302bcc9736cSJeff Kirsher 
303bcc9736cSJeff Kirsher /* Mode Status Register */
304bcc9736cSJeff Kirsher #define PHY_REG_STATUS			1
305bcc9736cSJeff Kirsher 
306bcc9736cSJeff Kirsher #define PHY_100BT4_CAPABLE		0x8000
307bcc9736cSJeff Kirsher #define PHY_100BTX_FD_CAPABLE		0x4000
308bcc9736cSJeff Kirsher #define PHY_100BTX_CAPABLE		0x2000
309bcc9736cSJeff Kirsher #define PHY_10BT_FD_CAPABLE		0x1000
310bcc9736cSJeff Kirsher #define PHY_10BT_CAPABLE		0x0800
311bcc9736cSJeff Kirsher #define PHY_MII_SUPPRESS_CAPABLE	0x0040
312bcc9736cSJeff Kirsher #define PHY_AUTO_NEG_ACKNOWLEDGE	0x0020
313bcc9736cSJeff Kirsher #define PHY_REMOTE_FAULT		0x0010
314bcc9736cSJeff Kirsher #define PHY_AUTO_NEG_CAPABLE		0x0008
315bcc9736cSJeff Kirsher #define PHY_LINK_STATUS			0x0004
316bcc9736cSJeff Kirsher #define PHY_JABBER_DETECT		0x0002
317bcc9736cSJeff Kirsher #define PHY_EXTENDED_CAPABILITY		0x0001
318bcc9736cSJeff Kirsher 
319bcc9736cSJeff Kirsher #define KS884X_PHY_ID_1_OFFSET		0x04
320bcc9736cSJeff Kirsher #define KS884X_PHY_ID_2_OFFSET		0x06
321bcc9736cSJeff Kirsher 
322bcc9736cSJeff Kirsher /* PHY Identifier Registers */
323bcc9736cSJeff Kirsher #define PHY_REG_ID_1			2
324bcc9736cSJeff Kirsher #define PHY_REG_ID_2			3
325bcc9736cSJeff Kirsher 
326bcc9736cSJeff Kirsher #define KS884X_PHY_AUTO_NEG_OFFSET	0x08
327bcc9736cSJeff Kirsher 
328bcc9736cSJeff Kirsher /* Auto-Negotiation Advertisement Register */
329bcc9736cSJeff Kirsher #define PHY_REG_AUTO_NEGOTIATION	4
330bcc9736cSJeff Kirsher 
331bcc9736cSJeff Kirsher #define PHY_AUTO_NEG_NEXT_PAGE		0x8000
332bcc9736cSJeff Kirsher #define PHY_AUTO_NEG_REMOTE_FAULT	0x2000
333bcc9736cSJeff Kirsher /* Not supported. */
334bcc9736cSJeff Kirsher #define PHY_AUTO_NEG_ASYM_PAUSE		0x0800
335bcc9736cSJeff Kirsher #define PHY_AUTO_NEG_SYM_PAUSE		0x0400
336bcc9736cSJeff Kirsher #define PHY_AUTO_NEG_100BT4		0x0200
337bcc9736cSJeff Kirsher #define PHY_AUTO_NEG_100BTX_FD		0x0100
338bcc9736cSJeff Kirsher #define PHY_AUTO_NEG_100BTX		0x0080
339bcc9736cSJeff Kirsher #define PHY_AUTO_NEG_10BT_FD		0x0040
340bcc9736cSJeff Kirsher #define PHY_AUTO_NEG_10BT		0x0020
341bcc9736cSJeff Kirsher #define PHY_AUTO_NEG_SELECTOR		0x001F
342bcc9736cSJeff Kirsher #define PHY_AUTO_NEG_802_3		0x0001
343bcc9736cSJeff Kirsher 
344bcc9736cSJeff Kirsher #define PHY_AUTO_NEG_PAUSE  (PHY_AUTO_NEG_SYM_PAUSE | PHY_AUTO_NEG_ASYM_PAUSE)
345bcc9736cSJeff Kirsher 
346bcc9736cSJeff Kirsher #define KS884X_PHY_REMOTE_CAP_OFFSET	0x0A
347bcc9736cSJeff Kirsher 
348bcc9736cSJeff Kirsher /* Auto-Negotiation Link Partner Ability Register */
349bcc9736cSJeff Kirsher #define PHY_REG_REMOTE_CAPABILITY	5
350bcc9736cSJeff Kirsher 
351bcc9736cSJeff Kirsher #define PHY_REMOTE_NEXT_PAGE		0x8000
352bcc9736cSJeff Kirsher #define PHY_REMOTE_ACKNOWLEDGE		0x4000
353bcc9736cSJeff Kirsher #define PHY_REMOTE_REMOTE_FAULT		0x2000
354bcc9736cSJeff Kirsher #define PHY_REMOTE_SYM_PAUSE		0x0400
355bcc9736cSJeff Kirsher #define PHY_REMOTE_100BTX_FD		0x0100
356bcc9736cSJeff Kirsher #define PHY_REMOTE_100BTX		0x0080
357bcc9736cSJeff Kirsher #define PHY_REMOTE_10BT_FD		0x0040
358bcc9736cSJeff Kirsher #define PHY_REMOTE_10BT			0x0020
359bcc9736cSJeff Kirsher 
360bcc9736cSJeff Kirsher /* P1VCT */
361bcc9736cSJeff Kirsher #define KS884X_P1VCT_P			0x04F0
362bcc9736cSJeff Kirsher #define KS884X_P1PHYCTRL_P		0x04F2
363bcc9736cSJeff Kirsher 
364bcc9736cSJeff Kirsher /* P2VCT */
365bcc9736cSJeff Kirsher #define KS884X_P2VCT_P			0x04F4
366bcc9736cSJeff Kirsher #define KS884X_P2PHYCTRL_P		0x04F6
367bcc9736cSJeff Kirsher 
368bcc9736cSJeff Kirsher #define KS884X_PHY_SPECIAL_OFFSET	KS884X_P1VCT_P
369bcc9736cSJeff Kirsher #define PHY_SPECIAL_INTERVAL		(KS884X_P2VCT_P - KS884X_P1VCT_P)
370bcc9736cSJeff Kirsher 
371bcc9736cSJeff Kirsher #define KS884X_PHY_LINK_MD_OFFSET	0x00
372bcc9736cSJeff Kirsher 
373bcc9736cSJeff Kirsher #define PHY_START_CABLE_DIAG		0x8000
374bcc9736cSJeff Kirsher #define PHY_CABLE_DIAG_RESULT		0x6000
375bcc9736cSJeff Kirsher #define PHY_CABLE_STAT_NORMAL		0x0000
376bcc9736cSJeff Kirsher #define PHY_CABLE_STAT_OPEN		0x2000
377bcc9736cSJeff Kirsher #define PHY_CABLE_STAT_SHORT		0x4000
378bcc9736cSJeff Kirsher #define PHY_CABLE_STAT_FAILED		0x6000
379bcc9736cSJeff Kirsher #define PHY_CABLE_10M_SHORT		0x1000
380bcc9736cSJeff Kirsher #define PHY_CABLE_FAULT_COUNTER		0x01FF
381bcc9736cSJeff Kirsher 
382bcc9736cSJeff Kirsher #define KS884X_PHY_PHY_CTRL_OFFSET	0x02
383bcc9736cSJeff Kirsher 
384bcc9736cSJeff Kirsher #define PHY_STAT_REVERSED_POLARITY	0x0020
385bcc9736cSJeff Kirsher #define PHY_STAT_MDIX			0x0010
386bcc9736cSJeff Kirsher #define PHY_FORCE_LINK			0x0008
387bcc9736cSJeff Kirsher #define PHY_POWER_SAVING_DISABLE	0x0004
388bcc9736cSJeff Kirsher #define PHY_REMOTE_LOOPBACK		0x0002
389bcc9736cSJeff Kirsher 
390bcc9736cSJeff Kirsher /* SIDER */
391bcc9736cSJeff Kirsher #define KS884X_SIDER_P			0x0400
392bcc9736cSJeff Kirsher #define KS884X_CHIP_ID_OFFSET		KS884X_SIDER_P
393bcc9736cSJeff Kirsher #define KS884X_FAMILY_ID_OFFSET		(KS884X_CHIP_ID_OFFSET + 1)
394bcc9736cSJeff Kirsher 
395bcc9736cSJeff Kirsher #define REG_FAMILY_ID			0x88
396bcc9736cSJeff Kirsher 
397bcc9736cSJeff Kirsher #define REG_CHIP_ID_41			0x8810
398bcc9736cSJeff Kirsher #define REG_CHIP_ID_42			0x8800
399bcc9736cSJeff Kirsher 
400bcc9736cSJeff Kirsher #define KS884X_CHIP_ID_MASK_41		0xFF10
401bcc9736cSJeff Kirsher #define KS884X_CHIP_ID_MASK		0xFFF0
402bcc9736cSJeff Kirsher #define KS884X_CHIP_ID_SHIFT		4
403bcc9736cSJeff Kirsher #define KS884X_REVISION_MASK		0x000E
404bcc9736cSJeff Kirsher #define KS884X_REVISION_SHIFT		1
405bcc9736cSJeff Kirsher #define KS8842_START			0x0001
406bcc9736cSJeff Kirsher 
407bcc9736cSJeff Kirsher #define CHIP_IP_41_M			0x8810
408bcc9736cSJeff Kirsher #define CHIP_IP_42_M			0x8800
409bcc9736cSJeff Kirsher #define CHIP_IP_61_M			0x8890
410bcc9736cSJeff Kirsher #define CHIP_IP_62_M			0x8880
411bcc9736cSJeff Kirsher 
412bcc9736cSJeff Kirsher #define CHIP_IP_41_P			0x8850
413bcc9736cSJeff Kirsher #define CHIP_IP_42_P			0x8840
414bcc9736cSJeff Kirsher #define CHIP_IP_61_P			0x88D0
415bcc9736cSJeff Kirsher #define CHIP_IP_62_P			0x88C0
416bcc9736cSJeff Kirsher 
417bcc9736cSJeff Kirsher /* SGCR1 */
418bcc9736cSJeff Kirsher #define KS8842_SGCR1_P			0x0402
419bcc9736cSJeff Kirsher #define KS8842_SWITCH_CTRL_1_OFFSET	KS8842_SGCR1_P
420bcc9736cSJeff Kirsher 
421bcc9736cSJeff Kirsher #define SWITCH_PASS_ALL			0x8000
422bcc9736cSJeff Kirsher #define SWITCH_TX_FLOW_CTRL		0x2000
423bcc9736cSJeff Kirsher #define SWITCH_RX_FLOW_CTRL		0x1000
424bcc9736cSJeff Kirsher #define SWITCH_CHECK_LENGTH		0x0800
425bcc9736cSJeff Kirsher #define SWITCH_AGING_ENABLE		0x0400
426bcc9736cSJeff Kirsher #define SWITCH_FAST_AGING		0x0200
427bcc9736cSJeff Kirsher #define SWITCH_AGGR_BACKOFF		0x0100
428bcc9736cSJeff Kirsher #define SWITCH_PASS_PAUSE		0x0008
429bcc9736cSJeff Kirsher #define SWITCH_LINK_AUTO_AGING		0x0001
430bcc9736cSJeff Kirsher 
431bcc9736cSJeff Kirsher /* SGCR2 */
432bcc9736cSJeff Kirsher #define KS8842_SGCR2_P			0x0404
433bcc9736cSJeff Kirsher #define KS8842_SWITCH_CTRL_2_OFFSET	KS8842_SGCR2_P
434bcc9736cSJeff Kirsher 
435bcc9736cSJeff Kirsher #define SWITCH_VLAN_ENABLE		0x8000
436bcc9736cSJeff Kirsher #define SWITCH_IGMP_SNOOP		0x4000
437bcc9736cSJeff Kirsher #define IPV6_MLD_SNOOP_ENABLE		0x2000
438bcc9736cSJeff Kirsher #define IPV6_MLD_SNOOP_OPTION		0x1000
439bcc9736cSJeff Kirsher #define PRIORITY_SCHEME_SELECT		0x0800
440bcc9736cSJeff Kirsher #define SWITCH_MIRROR_RX_TX		0x0100
441bcc9736cSJeff Kirsher #define UNICAST_VLAN_BOUNDARY		0x0080
442bcc9736cSJeff Kirsher #define MULTICAST_STORM_DISABLE		0x0040
443bcc9736cSJeff Kirsher #define SWITCH_BACK_PRESSURE		0x0020
444bcc9736cSJeff Kirsher #define FAIR_FLOW_CTRL			0x0010
445bcc9736cSJeff Kirsher #define NO_EXC_COLLISION_DROP		0x0008
446bcc9736cSJeff Kirsher #define SWITCH_HUGE_PACKET		0x0004
447bcc9736cSJeff Kirsher #define SWITCH_LEGAL_PACKET		0x0002
448bcc9736cSJeff Kirsher #define SWITCH_BUF_RESERVE		0x0001
449bcc9736cSJeff Kirsher 
450bcc9736cSJeff Kirsher /* SGCR3 */
451bcc9736cSJeff Kirsher #define KS8842_SGCR3_P			0x0406
452bcc9736cSJeff Kirsher #define KS8842_SWITCH_CTRL_3_OFFSET	KS8842_SGCR3_P
453bcc9736cSJeff Kirsher 
454bcc9736cSJeff Kirsher #define BROADCAST_STORM_RATE_LO		0xFF00
455bcc9736cSJeff Kirsher #define SWITCH_REPEATER			0x0080
456bcc9736cSJeff Kirsher #define SWITCH_HALF_DUPLEX		0x0040
457bcc9736cSJeff Kirsher #define SWITCH_FLOW_CTRL		0x0020
458bcc9736cSJeff Kirsher #define SWITCH_10_MBIT			0x0010
459bcc9736cSJeff Kirsher #define SWITCH_REPLACE_NULL_VID		0x0008
460bcc9736cSJeff Kirsher #define BROADCAST_STORM_RATE_HI		0x0007
461bcc9736cSJeff Kirsher 
462bcc9736cSJeff Kirsher #define BROADCAST_STORM_RATE		0x07FF
463bcc9736cSJeff Kirsher 
464bcc9736cSJeff Kirsher /* SGCR4 */
465bcc9736cSJeff Kirsher #define KS8842_SGCR4_P			0x0408
466bcc9736cSJeff Kirsher 
467bcc9736cSJeff Kirsher /* SGCR5 */
468bcc9736cSJeff Kirsher #define KS8842_SGCR5_P			0x040A
469bcc9736cSJeff Kirsher #define KS8842_SWITCH_CTRL_5_OFFSET	KS8842_SGCR5_P
470bcc9736cSJeff Kirsher 
471bcc9736cSJeff Kirsher #define LED_MODE			0x8200
472bcc9736cSJeff Kirsher #define LED_SPEED_DUPLEX_ACT		0x0000
473bcc9736cSJeff Kirsher #define LED_SPEED_DUPLEX_LINK_ACT	0x8000
474bcc9736cSJeff Kirsher #define LED_DUPLEX_10_100		0x0200
475bcc9736cSJeff Kirsher 
476bcc9736cSJeff Kirsher /* SGCR6 */
477bcc9736cSJeff Kirsher #define KS8842_SGCR6_P			0x0410
478bcc9736cSJeff Kirsher #define KS8842_SWITCH_CTRL_6_OFFSET	KS8842_SGCR6_P
479bcc9736cSJeff Kirsher 
480bcc9736cSJeff Kirsher #define KS8842_PRIORITY_MASK		3
481bcc9736cSJeff Kirsher #define KS8842_PRIORITY_SHIFT		2
482bcc9736cSJeff Kirsher 
483bcc9736cSJeff Kirsher /* SGCR7 */
484bcc9736cSJeff Kirsher #define KS8842_SGCR7_P			0x0412
485bcc9736cSJeff Kirsher #define KS8842_SWITCH_CTRL_7_OFFSET	KS8842_SGCR7_P
486bcc9736cSJeff Kirsher 
487bcc9736cSJeff Kirsher #define SWITCH_UNK_DEF_PORT_ENABLE	0x0008
488bcc9736cSJeff Kirsher #define SWITCH_UNK_DEF_PORT_3		0x0004
489bcc9736cSJeff Kirsher #define SWITCH_UNK_DEF_PORT_2		0x0002
490bcc9736cSJeff Kirsher #define SWITCH_UNK_DEF_PORT_1		0x0001
491bcc9736cSJeff Kirsher 
492bcc9736cSJeff Kirsher /* MACAR1 */
493bcc9736cSJeff Kirsher #define KS8842_MACAR1_P			0x0470
494bcc9736cSJeff Kirsher #define KS8842_MACAR2_P			0x0472
495bcc9736cSJeff Kirsher #define KS8842_MACAR3_P			0x0474
496bcc9736cSJeff Kirsher #define KS8842_MAC_ADDR_1_OFFSET	KS8842_MACAR1_P
497bcc9736cSJeff Kirsher #define KS8842_MAC_ADDR_0_OFFSET	(KS8842_MAC_ADDR_1_OFFSET + 1)
498bcc9736cSJeff Kirsher #define KS8842_MAC_ADDR_3_OFFSET	KS8842_MACAR2_P
499bcc9736cSJeff Kirsher #define KS8842_MAC_ADDR_2_OFFSET	(KS8842_MAC_ADDR_3_OFFSET + 1)
500bcc9736cSJeff Kirsher #define KS8842_MAC_ADDR_5_OFFSET	KS8842_MACAR3_P
501bcc9736cSJeff Kirsher #define KS8842_MAC_ADDR_4_OFFSET	(KS8842_MAC_ADDR_5_OFFSET + 1)
502bcc9736cSJeff Kirsher 
503bcc9736cSJeff Kirsher /* TOSR1 */
504bcc9736cSJeff Kirsher #define KS8842_TOSR1_P			0x0480
505bcc9736cSJeff Kirsher #define KS8842_TOSR2_P			0x0482
506bcc9736cSJeff Kirsher #define KS8842_TOSR3_P			0x0484
507bcc9736cSJeff Kirsher #define KS8842_TOSR4_P			0x0486
508bcc9736cSJeff Kirsher #define KS8842_TOSR5_P			0x0488
509bcc9736cSJeff Kirsher #define KS8842_TOSR6_P			0x048A
510bcc9736cSJeff Kirsher #define KS8842_TOSR7_P			0x0490
511bcc9736cSJeff Kirsher #define KS8842_TOSR8_P			0x0492
512bcc9736cSJeff Kirsher #define KS8842_TOS_1_OFFSET		KS8842_TOSR1_P
513bcc9736cSJeff Kirsher #define KS8842_TOS_2_OFFSET		KS8842_TOSR2_P
514bcc9736cSJeff Kirsher #define KS8842_TOS_3_OFFSET		KS8842_TOSR3_P
515bcc9736cSJeff Kirsher #define KS8842_TOS_4_OFFSET		KS8842_TOSR4_P
516bcc9736cSJeff Kirsher #define KS8842_TOS_5_OFFSET		KS8842_TOSR5_P
517bcc9736cSJeff Kirsher #define KS8842_TOS_6_OFFSET		KS8842_TOSR6_P
518bcc9736cSJeff Kirsher 
519bcc9736cSJeff Kirsher #define KS8842_TOS_7_OFFSET		KS8842_TOSR7_P
520bcc9736cSJeff Kirsher #define KS8842_TOS_8_OFFSET		KS8842_TOSR8_P
521bcc9736cSJeff Kirsher 
522bcc9736cSJeff Kirsher /* P1CR1 */
523bcc9736cSJeff Kirsher #define KS8842_P1CR1_P			0x0500
524bcc9736cSJeff Kirsher #define KS8842_P1CR2_P			0x0502
525bcc9736cSJeff Kirsher #define KS8842_P1VIDR_P			0x0504
526bcc9736cSJeff Kirsher #define KS8842_P1CR3_P			0x0506
527bcc9736cSJeff Kirsher #define KS8842_P1IRCR_P			0x0508
528bcc9736cSJeff Kirsher #define KS8842_P1ERCR_P			0x050A
529bcc9736cSJeff Kirsher #define KS884X_P1SCSLMD_P		0x0510
530bcc9736cSJeff Kirsher #define KS884X_P1CR4_P			0x0512
531bcc9736cSJeff Kirsher #define KS884X_P1SR_P			0x0514
532bcc9736cSJeff Kirsher 
533bcc9736cSJeff Kirsher /* P2CR1 */
534bcc9736cSJeff Kirsher #define KS8842_P2CR1_P			0x0520
535bcc9736cSJeff Kirsher #define KS8842_P2CR2_P			0x0522
536bcc9736cSJeff Kirsher #define KS8842_P2VIDR_P			0x0524
537bcc9736cSJeff Kirsher #define KS8842_P2CR3_P			0x0526
538bcc9736cSJeff Kirsher #define KS8842_P2IRCR_P			0x0528
539bcc9736cSJeff Kirsher #define KS8842_P2ERCR_P			0x052A
540bcc9736cSJeff Kirsher #define KS884X_P2SCSLMD_P		0x0530
541bcc9736cSJeff Kirsher #define KS884X_P2CR4_P			0x0532
542bcc9736cSJeff Kirsher #define KS884X_P2SR_P			0x0534
543bcc9736cSJeff Kirsher 
544bcc9736cSJeff Kirsher /* P3CR1 */
545bcc9736cSJeff Kirsher #define KS8842_P3CR1_P			0x0540
546bcc9736cSJeff Kirsher #define KS8842_P3CR2_P			0x0542
547bcc9736cSJeff Kirsher #define KS8842_P3VIDR_P			0x0544
548bcc9736cSJeff Kirsher #define KS8842_P3CR3_P			0x0546
549bcc9736cSJeff Kirsher #define KS8842_P3IRCR_P			0x0548
550bcc9736cSJeff Kirsher #define KS8842_P3ERCR_P			0x054A
551bcc9736cSJeff Kirsher 
552bcc9736cSJeff Kirsher #define KS8842_PORT_1_CTRL_1		KS8842_P1CR1_P
553bcc9736cSJeff Kirsher #define KS8842_PORT_2_CTRL_1		KS8842_P2CR1_P
554bcc9736cSJeff Kirsher #define KS8842_PORT_3_CTRL_1		KS8842_P3CR1_P
555bcc9736cSJeff Kirsher 
556bcc9736cSJeff Kirsher #define PORT_CTRL_ADDR(port, addr)		\
557bcc9736cSJeff Kirsher 	(addr = KS8842_PORT_1_CTRL_1 + (port) *	\
558bcc9736cSJeff Kirsher 		(KS8842_PORT_2_CTRL_1 - KS8842_PORT_1_CTRL_1))
559bcc9736cSJeff Kirsher 
560bcc9736cSJeff Kirsher #define KS8842_PORT_CTRL_1_OFFSET	0x00
561bcc9736cSJeff Kirsher 
562bcc9736cSJeff Kirsher #define PORT_BROADCAST_STORM		0x0080
563bcc9736cSJeff Kirsher #define PORT_DIFFSERV_ENABLE		0x0040
564bcc9736cSJeff Kirsher #define PORT_802_1P_ENABLE		0x0020
565bcc9736cSJeff Kirsher #define PORT_BASED_PRIORITY_MASK	0x0018
566bcc9736cSJeff Kirsher #define PORT_BASED_PRIORITY_BASE	0x0003
567bcc9736cSJeff Kirsher #define PORT_BASED_PRIORITY_SHIFT	3
568bcc9736cSJeff Kirsher #define PORT_BASED_PRIORITY_0		0x0000
569bcc9736cSJeff Kirsher #define PORT_BASED_PRIORITY_1		0x0008
570bcc9736cSJeff Kirsher #define PORT_BASED_PRIORITY_2		0x0010
571bcc9736cSJeff Kirsher #define PORT_BASED_PRIORITY_3		0x0018
572bcc9736cSJeff Kirsher #define PORT_INSERT_TAG			0x0004
573bcc9736cSJeff Kirsher #define PORT_REMOVE_TAG			0x0002
574bcc9736cSJeff Kirsher #define PORT_PRIO_QUEUE_ENABLE		0x0001
575bcc9736cSJeff Kirsher 
576bcc9736cSJeff Kirsher #define KS8842_PORT_CTRL_2_OFFSET	0x02
577bcc9736cSJeff Kirsher 
578bcc9736cSJeff Kirsher #define PORT_INGRESS_VLAN_FILTER	0x4000
579bcc9736cSJeff Kirsher #define PORT_DISCARD_NON_VID		0x2000
580bcc9736cSJeff Kirsher #define PORT_FORCE_FLOW_CTRL		0x1000
581bcc9736cSJeff Kirsher #define PORT_BACK_PRESSURE		0x0800
582bcc9736cSJeff Kirsher #define PORT_TX_ENABLE			0x0400
583bcc9736cSJeff Kirsher #define PORT_RX_ENABLE			0x0200
584bcc9736cSJeff Kirsher #define PORT_LEARN_DISABLE		0x0100
585bcc9736cSJeff Kirsher #define PORT_MIRROR_SNIFFER		0x0080
586bcc9736cSJeff Kirsher #define PORT_MIRROR_RX			0x0040
587bcc9736cSJeff Kirsher #define PORT_MIRROR_TX			0x0020
588bcc9736cSJeff Kirsher #define PORT_USER_PRIORITY_CEILING	0x0008
589bcc9736cSJeff Kirsher #define PORT_VLAN_MEMBERSHIP		0x0007
590bcc9736cSJeff Kirsher 
591bcc9736cSJeff Kirsher #define KS8842_PORT_CTRL_VID_OFFSET	0x04
592bcc9736cSJeff Kirsher 
593bcc9736cSJeff Kirsher #define PORT_DEFAULT_VID		0x0001
594bcc9736cSJeff Kirsher 
595bcc9736cSJeff Kirsher #define KS8842_PORT_CTRL_3_OFFSET	0x06
596bcc9736cSJeff Kirsher 
597bcc9736cSJeff Kirsher #define PORT_INGRESS_LIMIT_MODE		0x000C
598bcc9736cSJeff Kirsher #define PORT_INGRESS_ALL		0x0000
599bcc9736cSJeff Kirsher #define PORT_INGRESS_UNICAST		0x0004
600bcc9736cSJeff Kirsher #define PORT_INGRESS_MULTICAST		0x0008
601bcc9736cSJeff Kirsher #define PORT_INGRESS_BROADCAST		0x000C
602bcc9736cSJeff Kirsher #define PORT_COUNT_IFG			0x0002
603bcc9736cSJeff Kirsher #define PORT_COUNT_PREAMBLE		0x0001
604bcc9736cSJeff Kirsher 
605bcc9736cSJeff Kirsher #define KS8842_PORT_IN_RATE_OFFSET	0x08
606bcc9736cSJeff Kirsher #define KS8842_PORT_OUT_RATE_OFFSET	0x0A
607bcc9736cSJeff Kirsher 
608bcc9736cSJeff Kirsher #define PORT_PRIORITY_RATE		0x0F
609bcc9736cSJeff Kirsher #define PORT_PRIORITY_RATE_SHIFT	4
610bcc9736cSJeff Kirsher 
611bcc9736cSJeff Kirsher #define KS884X_PORT_LINK_MD		0x10
612bcc9736cSJeff Kirsher 
613bcc9736cSJeff Kirsher #define PORT_CABLE_10M_SHORT		0x8000
614bcc9736cSJeff Kirsher #define PORT_CABLE_DIAG_RESULT		0x6000
615bcc9736cSJeff Kirsher #define PORT_CABLE_STAT_NORMAL		0x0000
616bcc9736cSJeff Kirsher #define PORT_CABLE_STAT_OPEN		0x2000
617bcc9736cSJeff Kirsher #define PORT_CABLE_STAT_SHORT		0x4000
618bcc9736cSJeff Kirsher #define PORT_CABLE_STAT_FAILED		0x6000
619bcc9736cSJeff Kirsher #define PORT_START_CABLE_DIAG		0x1000
620bcc9736cSJeff Kirsher #define PORT_FORCE_LINK			0x0800
621bcc9736cSJeff Kirsher #define PORT_POWER_SAVING_DISABLE	0x0400
622bcc9736cSJeff Kirsher #define PORT_PHY_REMOTE_LOOPBACK	0x0200
623bcc9736cSJeff Kirsher #define PORT_CABLE_FAULT_COUNTER	0x01FF
624bcc9736cSJeff Kirsher 
625bcc9736cSJeff Kirsher #define KS884X_PORT_CTRL_4_OFFSET	0x12
626bcc9736cSJeff Kirsher 
627bcc9736cSJeff Kirsher #define PORT_LED_OFF			0x8000
628bcc9736cSJeff Kirsher #define PORT_TX_DISABLE			0x4000
629bcc9736cSJeff Kirsher #define PORT_AUTO_NEG_RESTART		0x2000
630bcc9736cSJeff Kirsher #define PORT_REMOTE_FAULT_DISABLE	0x1000
631bcc9736cSJeff Kirsher #define PORT_POWER_DOWN			0x0800
632bcc9736cSJeff Kirsher #define PORT_AUTO_MDIX_DISABLE		0x0400
633bcc9736cSJeff Kirsher #define PORT_FORCE_MDIX			0x0200
634bcc9736cSJeff Kirsher #define PORT_LOOPBACK			0x0100
635bcc9736cSJeff Kirsher #define PORT_AUTO_NEG_ENABLE		0x0080
636bcc9736cSJeff Kirsher #define PORT_FORCE_100_MBIT		0x0040
637bcc9736cSJeff Kirsher #define PORT_FORCE_FULL_DUPLEX		0x0020
638bcc9736cSJeff Kirsher #define PORT_AUTO_NEG_SYM_PAUSE		0x0010
639bcc9736cSJeff Kirsher #define PORT_AUTO_NEG_100BTX_FD		0x0008
640bcc9736cSJeff Kirsher #define PORT_AUTO_NEG_100BTX		0x0004
641bcc9736cSJeff Kirsher #define PORT_AUTO_NEG_10BT_FD		0x0002
642bcc9736cSJeff Kirsher #define PORT_AUTO_NEG_10BT		0x0001
643bcc9736cSJeff Kirsher 
644bcc9736cSJeff Kirsher #define KS884X_PORT_STATUS_OFFSET	0x14
645bcc9736cSJeff Kirsher 
646bcc9736cSJeff Kirsher #define PORT_HP_MDIX			0x8000
647bcc9736cSJeff Kirsher #define PORT_REVERSED_POLARITY		0x2000
648bcc9736cSJeff Kirsher #define PORT_RX_FLOW_CTRL		0x0800
649bcc9736cSJeff Kirsher #define PORT_TX_FLOW_CTRL		0x1000
650bcc9736cSJeff Kirsher #define PORT_STATUS_SPEED_100MBIT	0x0400
651bcc9736cSJeff Kirsher #define PORT_STATUS_FULL_DUPLEX		0x0200
652bcc9736cSJeff Kirsher #define PORT_REMOTE_FAULT		0x0100
653bcc9736cSJeff Kirsher #define PORT_MDIX_STATUS		0x0080
654bcc9736cSJeff Kirsher #define PORT_AUTO_NEG_COMPLETE		0x0040
655bcc9736cSJeff Kirsher #define PORT_STATUS_LINK_GOOD		0x0020
656bcc9736cSJeff Kirsher #define PORT_REMOTE_SYM_PAUSE		0x0010
657bcc9736cSJeff Kirsher #define PORT_REMOTE_100BTX_FD		0x0008
658bcc9736cSJeff Kirsher #define PORT_REMOTE_100BTX		0x0004
659bcc9736cSJeff Kirsher #define PORT_REMOTE_10BT_FD		0x0002
660bcc9736cSJeff Kirsher #define PORT_REMOTE_10BT		0x0001
661bcc9736cSJeff Kirsher 
662bcc9736cSJeff Kirsher /*
663bcc9736cSJeff Kirsher #define STATIC_MAC_TABLE_ADDR		00-0000FFFF-FFFFFFFF
664bcc9736cSJeff Kirsher #define STATIC_MAC_TABLE_FWD_PORTS	00-00070000-00000000
665bcc9736cSJeff Kirsher #define STATIC_MAC_TABLE_VALID		00-00080000-00000000
666bcc9736cSJeff Kirsher #define STATIC_MAC_TABLE_OVERRIDE	00-00100000-00000000
667bcc9736cSJeff Kirsher #define STATIC_MAC_TABLE_USE_FID	00-00200000-00000000
668bcc9736cSJeff Kirsher #define STATIC_MAC_TABLE_FID		00-03C00000-00000000
669bcc9736cSJeff Kirsher */
670bcc9736cSJeff Kirsher 
671bcc9736cSJeff Kirsher #define STATIC_MAC_TABLE_ADDR		0x0000FFFF
672bcc9736cSJeff Kirsher #define STATIC_MAC_TABLE_FWD_PORTS	0x00070000
673bcc9736cSJeff Kirsher #define STATIC_MAC_TABLE_VALID		0x00080000
674bcc9736cSJeff Kirsher #define STATIC_MAC_TABLE_OVERRIDE	0x00100000
675bcc9736cSJeff Kirsher #define STATIC_MAC_TABLE_USE_FID	0x00200000
676bcc9736cSJeff Kirsher #define STATIC_MAC_TABLE_FID		0x03C00000
677bcc9736cSJeff Kirsher 
678bcc9736cSJeff Kirsher #define STATIC_MAC_FWD_PORTS_SHIFT	16
679bcc9736cSJeff Kirsher #define STATIC_MAC_FID_SHIFT		22
680bcc9736cSJeff Kirsher 
681bcc9736cSJeff Kirsher /*
682bcc9736cSJeff Kirsher #define VLAN_TABLE_VID			00-00000000-00000FFF
683bcc9736cSJeff Kirsher #define VLAN_TABLE_FID			00-00000000-0000F000
684bcc9736cSJeff Kirsher #define VLAN_TABLE_MEMBERSHIP		00-00000000-00070000
685bcc9736cSJeff Kirsher #define VLAN_TABLE_VALID		00-00000000-00080000
686bcc9736cSJeff Kirsher */
687bcc9736cSJeff Kirsher 
688bcc9736cSJeff Kirsher #define VLAN_TABLE_VID			0x00000FFF
689bcc9736cSJeff Kirsher #define VLAN_TABLE_FID			0x0000F000
690bcc9736cSJeff Kirsher #define VLAN_TABLE_MEMBERSHIP		0x00070000
691bcc9736cSJeff Kirsher #define VLAN_TABLE_VALID		0x00080000
692bcc9736cSJeff Kirsher 
693bcc9736cSJeff Kirsher #define VLAN_TABLE_FID_SHIFT		12
694bcc9736cSJeff Kirsher #define VLAN_TABLE_MEMBERSHIP_SHIFT	16
695bcc9736cSJeff Kirsher 
696bcc9736cSJeff Kirsher /*
697bcc9736cSJeff Kirsher #define DYNAMIC_MAC_TABLE_ADDR		00-0000FFFF-FFFFFFFF
698bcc9736cSJeff Kirsher #define DYNAMIC_MAC_TABLE_FID		00-000F0000-00000000
699bcc9736cSJeff Kirsher #define DYNAMIC_MAC_TABLE_SRC_PORT	00-00300000-00000000
700bcc9736cSJeff Kirsher #define DYNAMIC_MAC_TABLE_TIMESTAMP	00-00C00000-00000000
701bcc9736cSJeff Kirsher #define DYNAMIC_MAC_TABLE_ENTRIES	03-FF000000-00000000
702bcc9736cSJeff Kirsher #define DYNAMIC_MAC_TABLE_MAC_EMPTY	04-00000000-00000000
703bcc9736cSJeff Kirsher #define DYNAMIC_MAC_TABLE_RESERVED	78-00000000-00000000
704bcc9736cSJeff Kirsher #define DYNAMIC_MAC_TABLE_NOT_READY	80-00000000-00000000
705bcc9736cSJeff Kirsher */
706bcc9736cSJeff Kirsher 
707bcc9736cSJeff Kirsher #define DYNAMIC_MAC_TABLE_ADDR		0x0000FFFF
708bcc9736cSJeff Kirsher #define DYNAMIC_MAC_TABLE_FID		0x000F0000
709bcc9736cSJeff Kirsher #define DYNAMIC_MAC_TABLE_SRC_PORT	0x00300000
710bcc9736cSJeff Kirsher #define DYNAMIC_MAC_TABLE_TIMESTAMP	0x00C00000
711bcc9736cSJeff Kirsher #define DYNAMIC_MAC_TABLE_ENTRIES	0xFF000000
712bcc9736cSJeff Kirsher 
713bcc9736cSJeff Kirsher #define DYNAMIC_MAC_TABLE_ENTRIES_H	0x03
714bcc9736cSJeff Kirsher #define DYNAMIC_MAC_TABLE_MAC_EMPTY	0x04
715bcc9736cSJeff Kirsher #define DYNAMIC_MAC_TABLE_RESERVED	0x78
716bcc9736cSJeff Kirsher #define DYNAMIC_MAC_TABLE_NOT_READY	0x80
717bcc9736cSJeff Kirsher 
718bcc9736cSJeff Kirsher #define DYNAMIC_MAC_FID_SHIFT		16
719bcc9736cSJeff Kirsher #define DYNAMIC_MAC_SRC_PORT_SHIFT	20
720bcc9736cSJeff Kirsher #define DYNAMIC_MAC_TIMESTAMP_SHIFT	22
721bcc9736cSJeff Kirsher #define DYNAMIC_MAC_ENTRIES_SHIFT	24
722bcc9736cSJeff Kirsher #define DYNAMIC_MAC_ENTRIES_H_SHIFT	8
723bcc9736cSJeff Kirsher 
724bcc9736cSJeff Kirsher /*
725bcc9736cSJeff Kirsher #define MIB_COUNTER_VALUE		00-00000000-3FFFFFFF
726bcc9736cSJeff Kirsher #define MIB_COUNTER_VALID		00-00000000-40000000
727bcc9736cSJeff Kirsher #define MIB_COUNTER_OVERFLOW		00-00000000-80000000
728bcc9736cSJeff Kirsher */
729bcc9736cSJeff Kirsher 
730bcc9736cSJeff Kirsher #define MIB_COUNTER_VALUE		0x3FFFFFFF
731bcc9736cSJeff Kirsher #define MIB_COUNTER_VALID		0x40000000
732bcc9736cSJeff Kirsher #define MIB_COUNTER_OVERFLOW		0x80000000
733bcc9736cSJeff Kirsher 
734bcc9736cSJeff Kirsher #define MIB_PACKET_DROPPED		0x0000FFFF
735bcc9736cSJeff Kirsher 
736bcc9736cSJeff Kirsher #define KS_MIB_PACKET_DROPPED_TX_0	0x100
737bcc9736cSJeff Kirsher #define KS_MIB_PACKET_DROPPED_TX_1	0x101
738bcc9736cSJeff Kirsher #define KS_MIB_PACKET_DROPPED_TX	0x102
739bcc9736cSJeff Kirsher #define KS_MIB_PACKET_DROPPED_RX_0	0x103
740bcc9736cSJeff Kirsher #define KS_MIB_PACKET_DROPPED_RX_1	0x104
741bcc9736cSJeff Kirsher #define KS_MIB_PACKET_DROPPED_RX	0x105
742bcc9736cSJeff Kirsher 
743bcc9736cSJeff Kirsher /* Change default LED mode. */
744bcc9736cSJeff Kirsher #define SET_DEFAULT_LED			LED_SPEED_DUPLEX_ACT
745bcc9736cSJeff Kirsher 
7466a3c910cSJoe Perches #define MAC_ADDR_ORDER(i)		(ETH_ALEN - 1 - (i))
747bcc9736cSJeff Kirsher 
748bcc9736cSJeff Kirsher #define MAX_ETHERNET_BODY_SIZE		1500
74983636580SDoug Kehn #define ETHERNET_HEADER_SIZE		(14 + VLAN_HLEN)
750bcc9736cSJeff Kirsher 
751bcc9736cSJeff Kirsher #define MAX_ETHERNET_PACKET_SIZE	\
752bcc9736cSJeff Kirsher 	(MAX_ETHERNET_BODY_SIZE + ETHERNET_HEADER_SIZE)
753bcc9736cSJeff Kirsher 
754bcc9736cSJeff Kirsher #define REGULAR_RX_BUF_SIZE		(MAX_ETHERNET_PACKET_SIZE + 4)
755bcc9736cSJeff Kirsher #define MAX_RX_BUF_SIZE			(1912 + 4)
756bcc9736cSJeff Kirsher 
757bcc9736cSJeff Kirsher #define ADDITIONAL_ENTRIES		16
758bcc9736cSJeff Kirsher #define MAX_MULTICAST_LIST		32
759bcc9736cSJeff Kirsher 
760bcc9736cSJeff Kirsher #define HW_MULTICAST_SIZE		8
761bcc9736cSJeff Kirsher 
762bcc9736cSJeff Kirsher #define HW_TO_DEV_PORT(port)		(port - 1)
763bcc9736cSJeff Kirsher 
764bcc9736cSJeff Kirsher enum {
765bcc9736cSJeff Kirsher 	media_connected,
766bcc9736cSJeff Kirsher 	media_disconnected
767bcc9736cSJeff Kirsher };
768bcc9736cSJeff Kirsher 
769bcc9736cSJeff Kirsher enum {
770bcc9736cSJeff Kirsher 	OID_COUNTER_UNKOWN,
771bcc9736cSJeff Kirsher 
772bcc9736cSJeff Kirsher 	OID_COUNTER_FIRST,
773bcc9736cSJeff Kirsher 
774bcc9736cSJeff Kirsher 	/* total transmit errors */
775bcc9736cSJeff Kirsher 	OID_COUNTER_XMIT_ERROR,
776bcc9736cSJeff Kirsher 
777bcc9736cSJeff Kirsher 	/* total receive errors */
778bcc9736cSJeff Kirsher 	OID_COUNTER_RCV_ERROR,
779bcc9736cSJeff Kirsher 
780bcc9736cSJeff Kirsher 	OID_COUNTER_LAST
781bcc9736cSJeff Kirsher };
782bcc9736cSJeff Kirsher 
783bcc9736cSJeff Kirsher /*
784bcc9736cSJeff Kirsher  * Hardware descriptor definitions
785bcc9736cSJeff Kirsher  */
786bcc9736cSJeff Kirsher 
787bcc9736cSJeff Kirsher #define DESC_ALIGNMENT			16
788bcc9736cSJeff Kirsher #define BUFFER_ALIGNMENT		8
789bcc9736cSJeff Kirsher 
790bcc9736cSJeff Kirsher #define NUM_OF_RX_DESC			64
791bcc9736cSJeff Kirsher #define NUM_OF_TX_DESC			64
792bcc9736cSJeff Kirsher 
793bcc9736cSJeff Kirsher #define KS_DESC_RX_FRAME_LEN		0x000007FF
794bcc9736cSJeff Kirsher #define KS_DESC_RX_FRAME_TYPE		0x00008000
795bcc9736cSJeff Kirsher #define KS_DESC_RX_ERROR_CRC		0x00010000
796bcc9736cSJeff Kirsher #define KS_DESC_RX_ERROR_RUNT		0x00020000
797bcc9736cSJeff Kirsher #define KS_DESC_RX_ERROR_TOO_LONG	0x00040000
798bcc9736cSJeff Kirsher #define KS_DESC_RX_ERROR_PHY		0x00080000
799bcc9736cSJeff Kirsher #define KS884X_DESC_RX_PORT_MASK	0x00300000
800bcc9736cSJeff Kirsher #define KS_DESC_RX_MULTICAST		0x01000000
801bcc9736cSJeff Kirsher #define KS_DESC_RX_ERROR		0x02000000
802bcc9736cSJeff Kirsher #define KS_DESC_RX_ERROR_CSUM_UDP	0x04000000
803bcc9736cSJeff Kirsher #define KS_DESC_RX_ERROR_CSUM_TCP	0x08000000
804bcc9736cSJeff Kirsher #define KS_DESC_RX_ERROR_CSUM_IP	0x10000000
805bcc9736cSJeff Kirsher #define KS_DESC_RX_LAST			0x20000000
806bcc9736cSJeff Kirsher #define KS_DESC_RX_FIRST		0x40000000
807bcc9736cSJeff Kirsher #define KS_DESC_RX_ERROR_COND		\
808bcc9736cSJeff Kirsher 	(KS_DESC_RX_ERROR_CRC |		\
809bcc9736cSJeff Kirsher 	KS_DESC_RX_ERROR_RUNT |		\
810bcc9736cSJeff Kirsher 	KS_DESC_RX_ERROR_PHY |		\
811bcc9736cSJeff Kirsher 	KS_DESC_RX_ERROR_TOO_LONG)
812bcc9736cSJeff Kirsher 
813bcc9736cSJeff Kirsher #define KS_DESC_HW_OWNED		0x80000000
814bcc9736cSJeff Kirsher 
815bcc9736cSJeff Kirsher #define KS_DESC_BUF_SIZE		0x000007FF
816bcc9736cSJeff Kirsher #define KS884X_DESC_TX_PORT_MASK	0x00300000
817bcc9736cSJeff Kirsher #define KS_DESC_END_OF_RING		0x02000000
818bcc9736cSJeff Kirsher #define KS_DESC_TX_CSUM_GEN_UDP		0x04000000
819bcc9736cSJeff Kirsher #define KS_DESC_TX_CSUM_GEN_TCP		0x08000000
820bcc9736cSJeff Kirsher #define KS_DESC_TX_CSUM_GEN_IP		0x10000000
821bcc9736cSJeff Kirsher #define KS_DESC_TX_LAST			0x20000000
822bcc9736cSJeff Kirsher #define KS_DESC_TX_FIRST		0x40000000
823bcc9736cSJeff Kirsher #define KS_DESC_TX_INTERRUPT		0x80000000
824bcc9736cSJeff Kirsher 
825bcc9736cSJeff Kirsher #define KS_DESC_PORT_SHIFT		20
826bcc9736cSJeff Kirsher 
827bcc9736cSJeff Kirsher #define KS_DESC_RX_MASK			(KS_DESC_BUF_SIZE)
828bcc9736cSJeff Kirsher 
829bcc9736cSJeff Kirsher #define KS_DESC_TX_MASK			\
830bcc9736cSJeff Kirsher 	(KS_DESC_TX_INTERRUPT |		\
831bcc9736cSJeff Kirsher 	KS_DESC_TX_FIRST |		\
832bcc9736cSJeff Kirsher 	KS_DESC_TX_LAST |		\
833bcc9736cSJeff Kirsher 	KS_DESC_TX_CSUM_GEN_IP |	\
834bcc9736cSJeff Kirsher 	KS_DESC_TX_CSUM_GEN_TCP |	\
835bcc9736cSJeff Kirsher 	KS_DESC_TX_CSUM_GEN_UDP |	\
836bcc9736cSJeff Kirsher 	KS_DESC_BUF_SIZE)
837bcc9736cSJeff Kirsher 
838bcc9736cSJeff Kirsher struct ksz_desc_rx_stat {
839bcc9736cSJeff Kirsher #ifdef __BIG_ENDIAN_BITFIELD
840bcc9736cSJeff Kirsher 	u32 hw_owned:1;
841bcc9736cSJeff Kirsher 	u32 first_desc:1;
842bcc9736cSJeff Kirsher 	u32 last_desc:1;
843bcc9736cSJeff Kirsher 	u32 csum_err_ip:1;
844bcc9736cSJeff Kirsher 	u32 csum_err_tcp:1;
845bcc9736cSJeff Kirsher 	u32 csum_err_udp:1;
846bcc9736cSJeff Kirsher 	u32 error:1;
847bcc9736cSJeff Kirsher 	u32 multicast:1;
848bcc9736cSJeff Kirsher 	u32 src_port:4;
849bcc9736cSJeff Kirsher 	u32 err_phy:1;
850bcc9736cSJeff Kirsher 	u32 err_too_long:1;
851bcc9736cSJeff Kirsher 	u32 err_runt:1;
852bcc9736cSJeff Kirsher 	u32 err_crc:1;
853bcc9736cSJeff Kirsher 	u32 frame_type:1;
854bcc9736cSJeff Kirsher 	u32 reserved1:4;
855bcc9736cSJeff Kirsher 	u32 frame_len:11;
856bcc9736cSJeff Kirsher #else
857bcc9736cSJeff Kirsher 	u32 frame_len:11;
858bcc9736cSJeff Kirsher 	u32 reserved1:4;
859bcc9736cSJeff Kirsher 	u32 frame_type:1;
860bcc9736cSJeff Kirsher 	u32 err_crc:1;
861bcc9736cSJeff Kirsher 	u32 err_runt:1;
862bcc9736cSJeff Kirsher 	u32 err_too_long:1;
863bcc9736cSJeff Kirsher 	u32 err_phy:1;
864bcc9736cSJeff Kirsher 	u32 src_port:4;
865bcc9736cSJeff Kirsher 	u32 multicast:1;
866bcc9736cSJeff Kirsher 	u32 error:1;
867bcc9736cSJeff Kirsher 	u32 csum_err_udp:1;
868bcc9736cSJeff Kirsher 	u32 csum_err_tcp:1;
869bcc9736cSJeff Kirsher 	u32 csum_err_ip:1;
870bcc9736cSJeff Kirsher 	u32 last_desc:1;
871bcc9736cSJeff Kirsher 	u32 first_desc:1;
872bcc9736cSJeff Kirsher 	u32 hw_owned:1;
873bcc9736cSJeff Kirsher #endif
874bcc9736cSJeff Kirsher };
875bcc9736cSJeff Kirsher 
876bcc9736cSJeff Kirsher struct ksz_desc_tx_stat {
877bcc9736cSJeff Kirsher #ifdef __BIG_ENDIAN_BITFIELD
878bcc9736cSJeff Kirsher 	u32 hw_owned:1;
879bcc9736cSJeff Kirsher 	u32 reserved1:31;
880bcc9736cSJeff Kirsher #else
881bcc9736cSJeff Kirsher 	u32 reserved1:31;
882bcc9736cSJeff Kirsher 	u32 hw_owned:1;
883bcc9736cSJeff Kirsher #endif
884bcc9736cSJeff Kirsher };
885bcc9736cSJeff Kirsher 
886bcc9736cSJeff Kirsher struct ksz_desc_rx_buf {
887bcc9736cSJeff Kirsher #ifdef __BIG_ENDIAN_BITFIELD
888bcc9736cSJeff Kirsher 	u32 reserved4:6;
889bcc9736cSJeff Kirsher 	u32 end_of_ring:1;
890bcc9736cSJeff Kirsher 	u32 reserved3:14;
891bcc9736cSJeff Kirsher 	u32 buf_size:11;
892bcc9736cSJeff Kirsher #else
893bcc9736cSJeff Kirsher 	u32 buf_size:11;
894bcc9736cSJeff Kirsher 	u32 reserved3:14;
895bcc9736cSJeff Kirsher 	u32 end_of_ring:1;
896bcc9736cSJeff Kirsher 	u32 reserved4:6;
897bcc9736cSJeff Kirsher #endif
898bcc9736cSJeff Kirsher };
899bcc9736cSJeff Kirsher 
900bcc9736cSJeff Kirsher struct ksz_desc_tx_buf {
901bcc9736cSJeff Kirsher #ifdef __BIG_ENDIAN_BITFIELD
902bcc9736cSJeff Kirsher 	u32 intr:1;
903bcc9736cSJeff Kirsher 	u32 first_seg:1;
904bcc9736cSJeff Kirsher 	u32 last_seg:1;
905bcc9736cSJeff Kirsher 	u32 csum_gen_ip:1;
906bcc9736cSJeff Kirsher 	u32 csum_gen_tcp:1;
907bcc9736cSJeff Kirsher 	u32 csum_gen_udp:1;
908bcc9736cSJeff Kirsher 	u32 end_of_ring:1;
909bcc9736cSJeff Kirsher 	u32 reserved4:1;
910bcc9736cSJeff Kirsher 	u32 dest_port:4;
911bcc9736cSJeff Kirsher 	u32 reserved3:9;
912bcc9736cSJeff Kirsher 	u32 buf_size:11;
913bcc9736cSJeff Kirsher #else
914bcc9736cSJeff Kirsher 	u32 buf_size:11;
915bcc9736cSJeff Kirsher 	u32 reserved3:9;
916bcc9736cSJeff Kirsher 	u32 dest_port:4;
917bcc9736cSJeff Kirsher 	u32 reserved4:1;
918bcc9736cSJeff Kirsher 	u32 end_of_ring:1;
919bcc9736cSJeff Kirsher 	u32 csum_gen_udp:1;
920bcc9736cSJeff Kirsher 	u32 csum_gen_tcp:1;
921bcc9736cSJeff Kirsher 	u32 csum_gen_ip:1;
922bcc9736cSJeff Kirsher 	u32 last_seg:1;
923bcc9736cSJeff Kirsher 	u32 first_seg:1;
924bcc9736cSJeff Kirsher 	u32 intr:1;
925bcc9736cSJeff Kirsher #endif
926bcc9736cSJeff Kirsher };
927bcc9736cSJeff Kirsher 
928bcc9736cSJeff Kirsher union desc_stat {
929bcc9736cSJeff Kirsher 	struct ksz_desc_rx_stat rx;
930bcc9736cSJeff Kirsher 	struct ksz_desc_tx_stat tx;
931bcc9736cSJeff Kirsher 	u32 data;
932bcc9736cSJeff Kirsher };
933bcc9736cSJeff Kirsher 
934bcc9736cSJeff Kirsher union desc_buf {
935bcc9736cSJeff Kirsher 	struct ksz_desc_rx_buf rx;
936bcc9736cSJeff Kirsher 	struct ksz_desc_tx_buf tx;
937bcc9736cSJeff Kirsher 	u32 data;
938bcc9736cSJeff Kirsher };
939bcc9736cSJeff Kirsher 
940bcc9736cSJeff Kirsher /**
941bcc9736cSJeff Kirsher  * struct ksz_hw_desc - Hardware descriptor data structure
942bcc9736cSJeff Kirsher  * @ctrl:	Descriptor control value.
943bcc9736cSJeff Kirsher  * @buf:	Descriptor buffer value.
944bcc9736cSJeff Kirsher  * @addr:	Physical address of memory buffer.
945bcc9736cSJeff Kirsher  * @next:	Pointer to next hardware descriptor.
946bcc9736cSJeff Kirsher  */
947bcc9736cSJeff Kirsher struct ksz_hw_desc {
948bcc9736cSJeff Kirsher 	union desc_stat ctrl;
949bcc9736cSJeff Kirsher 	union desc_buf buf;
950bcc9736cSJeff Kirsher 	u32 addr;
951bcc9736cSJeff Kirsher 	u32 next;
952bcc9736cSJeff Kirsher };
953bcc9736cSJeff Kirsher 
954bcc9736cSJeff Kirsher /**
955bcc9736cSJeff Kirsher  * struct ksz_sw_desc - Software descriptor data structure
956bcc9736cSJeff Kirsher  * @ctrl:	Descriptor control value.
957bcc9736cSJeff Kirsher  * @buf:	Descriptor buffer value.
958bcc9736cSJeff Kirsher  * @buf_size:	Current buffers size value in hardware descriptor.
959bcc9736cSJeff Kirsher  */
960bcc9736cSJeff Kirsher struct ksz_sw_desc {
961bcc9736cSJeff Kirsher 	union desc_stat ctrl;
962bcc9736cSJeff Kirsher 	union desc_buf buf;
963bcc9736cSJeff Kirsher 	u32 buf_size;
964bcc9736cSJeff Kirsher };
965bcc9736cSJeff Kirsher 
966bcc9736cSJeff Kirsher /**
967bcc9736cSJeff Kirsher  * struct ksz_dma_buf - OS dependent DMA buffer data structure
968bcc9736cSJeff Kirsher  * @skb:	Associated socket buffer.
969bcc9736cSJeff Kirsher  * @dma:	Associated physical DMA address.
970bcc9736cSJeff Kirsher  * len:		Actual len used.
971bcc9736cSJeff Kirsher  */
972bcc9736cSJeff Kirsher struct ksz_dma_buf {
973bcc9736cSJeff Kirsher 	struct sk_buff *skb;
974bcc9736cSJeff Kirsher 	dma_addr_t dma;
975bcc9736cSJeff Kirsher 	int len;
976bcc9736cSJeff Kirsher };
977bcc9736cSJeff Kirsher 
978bcc9736cSJeff Kirsher /**
979bcc9736cSJeff Kirsher  * struct ksz_desc - Descriptor structure
980bcc9736cSJeff Kirsher  * @phw:	Hardware descriptor pointer to uncached physical memory.
981bcc9736cSJeff Kirsher  * @sw:		Cached memory to hold hardware descriptor values for
982bcc9736cSJeff Kirsher  * 		manipulation.
983bcc9736cSJeff Kirsher  * @dma_buf:	Operating system dependent data structure to hold physical
984bcc9736cSJeff Kirsher  * 		memory buffer allocation information.
985bcc9736cSJeff Kirsher  */
986bcc9736cSJeff Kirsher struct ksz_desc {
987bcc9736cSJeff Kirsher 	struct ksz_hw_desc *phw;
988bcc9736cSJeff Kirsher 	struct ksz_sw_desc sw;
989bcc9736cSJeff Kirsher 	struct ksz_dma_buf dma_buf;
990bcc9736cSJeff Kirsher };
991bcc9736cSJeff Kirsher 
992bcc9736cSJeff Kirsher #define DMA_BUFFER(desc)  ((struct ksz_dma_buf *)(&(desc)->dma_buf))
993bcc9736cSJeff Kirsher 
994bcc9736cSJeff Kirsher /**
995bcc9736cSJeff Kirsher  * struct ksz_desc_info - Descriptor information data structure
996bcc9736cSJeff Kirsher  * @ring:	First descriptor in the ring.
997bcc9736cSJeff Kirsher  * @cur:	Current descriptor being manipulated.
998bcc9736cSJeff Kirsher  * @ring_virt:	First hardware descriptor in the ring.
999bcc9736cSJeff Kirsher  * @ring_phys:	The physical address of the first descriptor of the ring.
1000bcc9736cSJeff Kirsher  * @size:	Size of hardware descriptor.
1001bcc9736cSJeff Kirsher  * @alloc:	Number of descriptors allocated.
1002bcc9736cSJeff Kirsher  * @avail:	Number of descriptors available for use.
1003bcc9736cSJeff Kirsher  * @last:	Index for last descriptor released to hardware.
1004bcc9736cSJeff Kirsher  * @next:	Index for next descriptor available for use.
1005bcc9736cSJeff Kirsher  * @mask:	Mask for index wrapping.
1006bcc9736cSJeff Kirsher  */
1007bcc9736cSJeff Kirsher struct ksz_desc_info {
1008bcc9736cSJeff Kirsher 	struct ksz_desc *ring;
1009bcc9736cSJeff Kirsher 	struct ksz_desc *cur;
1010bcc9736cSJeff Kirsher 	struct ksz_hw_desc *ring_virt;
1011bcc9736cSJeff Kirsher 	u32 ring_phys;
1012bcc9736cSJeff Kirsher 	int size;
1013bcc9736cSJeff Kirsher 	int alloc;
1014bcc9736cSJeff Kirsher 	int avail;
1015bcc9736cSJeff Kirsher 	int last;
1016bcc9736cSJeff Kirsher 	int next;
1017bcc9736cSJeff Kirsher 	int mask;
1018bcc9736cSJeff Kirsher };
1019bcc9736cSJeff Kirsher 
1020bcc9736cSJeff Kirsher /*
1021bcc9736cSJeff Kirsher  * KSZ8842 switch definitions
1022bcc9736cSJeff Kirsher  */
1023bcc9736cSJeff Kirsher 
1024bcc9736cSJeff Kirsher enum {
1025bcc9736cSJeff Kirsher 	TABLE_STATIC_MAC = 0,
1026bcc9736cSJeff Kirsher 	TABLE_VLAN,
1027bcc9736cSJeff Kirsher 	TABLE_DYNAMIC_MAC,
1028bcc9736cSJeff Kirsher 	TABLE_MIB
1029bcc9736cSJeff Kirsher };
1030bcc9736cSJeff Kirsher 
1031bcc9736cSJeff Kirsher #define LEARNED_MAC_TABLE_ENTRIES	1024
1032bcc9736cSJeff Kirsher #define STATIC_MAC_TABLE_ENTRIES	8
1033bcc9736cSJeff Kirsher 
1034bcc9736cSJeff Kirsher /**
1035bcc9736cSJeff Kirsher  * struct ksz_mac_table - Static MAC table data structure
1036bcc9736cSJeff Kirsher  * @mac_addr:	MAC address to filter.
1037bcc9736cSJeff Kirsher  * @vid:	VID value.
1038bcc9736cSJeff Kirsher  * @fid:	FID value.
1039bcc9736cSJeff Kirsher  * @ports:	Port membership.
1040bcc9736cSJeff Kirsher  * @override:	Override setting.
1041bcc9736cSJeff Kirsher  * @use_fid:	FID use setting.
1042bcc9736cSJeff Kirsher  * @valid:	Valid setting indicating the entry is being used.
1043bcc9736cSJeff Kirsher  */
1044bcc9736cSJeff Kirsher struct ksz_mac_table {
10456a3c910cSJoe Perches 	u8 mac_addr[ETH_ALEN];
1046bcc9736cSJeff Kirsher 	u16 vid;
1047bcc9736cSJeff Kirsher 	u8 fid;
1048bcc9736cSJeff Kirsher 	u8 ports;
1049bcc9736cSJeff Kirsher 	u8 override:1;
1050bcc9736cSJeff Kirsher 	u8 use_fid:1;
1051bcc9736cSJeff Kirsher 	u8 valid:1;
1052bcc9736cSJeff Kirsher };
1053bcc9736cSJeff Kirsher 
1054bcc9736cSJeff Kirsher #define VLAN_TABLE_ENTRIES		16
1055bcc9736cSJeff Kirsher 
1056bcc9736cSJeff Kirsher /**
1057bcc9736cSJeff Kirsher  * struct ksz_vlan_table - VLAN table data structure
1058bcc9736cSJeff Kirsher  * @vid:	VID value.
1059bcc9736cSJeff Kirsher  * @fid:	FID value.
1060bcc9736cSJeff Kirsher  * @member:	Port membership.
1061bcc9736cSJeff Kirsher  */
1062bcc9736cSJeff Kirsher struct ksz_vlan_table {
1063bcc9736cSJeff Kirsher 	u16 vid;
1064bcc9736cSJeff Kirsher 	u8 fid;
1065bcc9736cSJeff Kirsher 	u8 member;
1066bcc9736cSJeff Kirsher };
1067bcc9736cSJeff Kirsher 
1068bcc9736cSJeff Kirsher #define DIFFSERV_ENTRIES		64
1069bcc9736cSJeff Kirsher #define PRIO_802_1P_ENTRIES		8
1070bcc9736cSJeff Kirsher #define PRIO_QUEUES			4
1071bcc9736cSJeff Kirsher 
1072bcc9736cSJeff Kirsher #define SWITCH_PORT_NUM			2
1073bcc9736cSJeff Kirsher #define TOTAL_PORT_NUM			(SWITCH_PORT_NUM + 1)
1074bcc9736cSJeff Kirsher #define HOST_MASK			(1 << SWITCH_PORT_NUM)
1075bcc9736cSJeff Kirsher #define PORT_MASK			7
1076bcc9736cSJeff Kirsher 
1077bcc9736cSJeff Kirsher #define MAIN_PORT			0
1078bcc9736cSJeff Kirsher #define OTHER_PORT			1
1079bcc9736cSJeff Kirsher #define HOST_PORT			SWITCH_PORT_NUM
1080bcc9736cSJeff Kirsher 
1081bcc9736cSJeff Kirsher #define PORT_COUNTER_NUM		0x20
1082bcc9736cSJeff Kirsher #define TOTAL_PORT_COUNTER_NUM		(PORT_COUNTER_NUM + 2)
1083bcc9736cSJeff Kirsher 
1084bcc9736cSJeff Kirsher #define MIB_COUNTER_RX_LO_PRIORITY	0x00
1085bcc9736cSJeff Kirsher #define MIB_COUNTER_RX_HI_PRIORITY	0x01
1086bcc9736cSJeff Kirsher #define MIB_COUNTER_RX_UNDERSIZE	0x02
1087bcc9736cSJeff Kirsher #define MIB_COUNTER_RX_FRAGMENT		0x03
1088bcc9736cSJeff Kirsher #define MIB_COUNTER_RX_OVERSIZE		0x04
1089bcc9736cSJeff Kirsher #define MIB_COUNTER_RX_JABBER		0x05
1090bcc9736cSJeff Kirsher #define MIB_COUNTER_RX_SYMBOL_ERR	0x06
1091bcc9736cSJeff Kirsher #define MIB_COUNTER_RX_CRC_ERR		0x07
1092bcc9736cSJeff Kirsher #define MIB_COUNTER_RX_ALIGNMENT_ERR	0x08
1093bcc9736cSJeff Kirsher #define MIB_COUNTER_RX_CTRL_8808	0x09
1094bcc9736cSJeff Kirsher #define MIB_COUNTER_RX_PAUSE		0x0A
1095bcc9736cSJeff Kirsher #define MIB_COUNTER_RX_BROADCAST	0x0B
1096bcc9736cSJeff Kirsher #define MIB_COUNTER_RX_MULTICAST	0x0C
1097bcc9736cSJeff Kirsher #define MIB_COUNTER_RX_UNICAST		0x0D
1098bcc9736cSJeff Kirsher #define MIB_COUNTER_RX_OCTET_64		0x0E
1099bcc9736cSJeff Kirsher #define MIB_COUNTER_RX_OCTET_65_127	0x0F
1100bcc9736cSJeff Kirsher #define MIB_COUNTER_RX_OCTET_128_255	0x10
1101bcc9736cSJeff Kirsher #define MIB_COUNTER_RX_OCTET_256_511	0x11
1102bcc9736cSJeff Kirsher #define MIB_COUNTER_RX_OCTET_512_1023	0x12
1103bcc9736cSJeff Kirsher #define MIB_COUNTER_RX_OCTET_1024_1522	0x13
1104bcc9736cSJeff Kirsher #define MIB_COUNTER_TX_LO_PRIORITY	0x14
1105bcc9736cSJeff Kirsher #define MIB_COUNTER_TX_HI_PRIORITY	0x15
1106bcc9736cSJeff Kirsher #define MIB_COUNTER_TX_LATE_COLLISION	0x16
1107bcc9736cSJeff Kirsher #define MIB_COUNTER_TX_PAUSE		0x17
1108bcc9736cSJeff Kirsher #define MIB_COUNTER_TX_BROADCAST	0x18
1109bcc9736cSJeff Kirsher #define MIB_COUNTER_TX_MULTICAST	0x19
1110bcc9736cSJeff Kirsher #define MIB_COUNTER_TX_UNICAST		0x1A
1111bcc9736cSJeff Kirsher #define MIB_COUNTER_TX_DEFERRED		0x1B
1112bcc9736cSJeff Kirsher #define MIB_COUNTER_TX_TOTAL_COLLISION	0x1C
1113bcc9736cSJeff Kirsher #define MIB_COUNTER_TX_EXCESS_COLLISION	0x1D
1114bcc9736cSJeff Kirsher #define MIB_COUNTER_TX_SINGLE_COLLISION	0x1E
1115bcc9736cSJeff Kirsher #define MIB_COUNTER_TX_MULTI_COLLISION	0x1F
1116bcc9736cSJeff Kirsher 
1117bcc9736cSJeff Kirsher #define MIB_COUNTER_RX_DROPPED_PACKET	0x20
1118bcc9736cSJeff Kirsher #define MIB_COUNTER_TX_DROPPED_PACKET	0x21
1119bcc9736cSJeff Kirsher 
1120bcc9736cSJeff Kirsher /**
1121bcc9736cSJeff Kirsher  * struct ksz_port_mib - Port MIB data structure
1122bcc9736cSJeff Kirsher  * @cnt_ptr:	Current pointer to MIB counter index.
1123bcc9736cSJeff Kirsher  * @link_down:	Indication the link has just gone down.
1124bcc9736cSJeff Kirsher  * @state:	Connection status of the port.
1125bcc9736cSJeff Kirsher  * @mib_start:	The starting counter index.  Some ports do not start at 0.
1126bcc9736cSJeff Kirsher  * @counter:	64-bit MIB counter value.
1127bcc9736cSJeff Kirsher  * @dropped:	Temporary buffer to remember last read packet dropped values.
1128bcc9736cSJeff Kirsher  *
1129bcc9736cSJeff Kirsher  * MIB counters needs to be read periodically so that counters do not get
1130bcc9736cSJeff Kirsher  * overflowed and give incorrect values.  A right balance is needed to
1131bcc9736cSJeff Kirsher  * satisfy this condition and not waste too much CPU time.
1132bcc9736cSJeff Kirsher  *
1133bcc9736cSJeff Kirsher  * It is pointless to read MIB counters when the port is disconnected.  The
1134bcc9736cSJeff Kirsher  * @state provides the connection status so that MIB counters are read only
1135bcc9736cSJeff Kirsher  * when the port is connected.  The @link_down indicates the port is just
1136bcc9736cSJeff Kirsher  * disconnected so that all MIB counters are read one last time to update the
1137bcc9736cSJeff Kirsher  * information.
1138bcc9736cSJeff Kirsher  */
1139bcc9736cSJeff Kirsher struct ksz_port_mib {
1140bcc9736cSJeff Kirsher 	u8 cnt_ptr;
1141bcc9736cSJeff Kirsher 	u8 link_down;
1142bcc9736cSJeff Kirsher 	u8 state;
1143bcc9736cSJeff Kirsher 	u8 mib_start;
1144bcc9736cSJeff Kirsher 
1145bcc9736cSJeff Kirsher 	u64 counter[TOTAL_PORT_COUNTER_NUM];
1146bcc9736cSJeff Kirsher 	u32 dropped[2];
1147bcc9736cSJeff Kirsher };
1148bcc9736cSJeff Kirsher 
1149bcc9736cSJeff Kirsher /**
1150bcc9736cSJeff Kirsher  * struct ksz_port_cfg - Port configuration data structure
1151bcc9736cSJeff Kirsher  * @vid:	VID value.
1152bcc9736cSJeff Kirsher  * @member:	Port membership.
1153bcc9736cSJeff Kirsher  * @port_prio:	Port priority.
1154bcc9736cSJeff Kirsher  * @rx_rate:	Receive priority rate.
1155bcc9736cSJeff Kirsher  * @tx_rate:	Transmit priority rate.
1156bcc9736cSJeff Kirsher  * @stp_state:	Current Spanning Tree Protocol state.
1157bcc9736cSJeff Kirsher  */
1158bcc9736cSJeff Kirsher struct ksz_port_cfg {
1159bcc9736cSJeff Kirsher 	u16 vid;
1160bcc9736cSJeff Kirsher 	u8 member;
1161bcc9736cSJeff Kirsher 	u8 port_prio;
1162bcc9736cSJeff Kirsher 	u32 rx_rate[PRIO_QUEUES];
1163bcc9736cSJeff Kirsher 	u32 tx_rate[PRIO_QUEUES];
1164bcc9736cSJeff Kirsher 	int stp_state;
1165bcc9736cSJeff Kirsher };
1166bcc9736cSJeff Kirsher 
1167bcc9736cSJeff Kirsher /**
1168bcc9736cSJeff Kirsher  * struct ksz_switch - KSZ8842 switch data structure
1169bcc9736cSJeff Kirsher  * @mac_table:	MAC table entries information.
1170bcc9736cSJeff Kirsher  * @vlan_table:	VLAN table entries information.
1171bcc9736cSJeff Kirsher  * @port_cfg:	Port configuration information.
1172bcc9736cSJeff Kirsher  * @diffserv:	DiffServ priority settings.  Possible values from 6-bit of ToS
1173bcc9736cSJeff Kirsher  * 		(bit7 ~ bit2) field.
1174bcc9736cSJeff Kirsher  * @p_802_1p:	802.1P priority settings.  Possible values from 3-bit of 802.1p
1175bcc9736cSJeff Kirsher  * 		Tag priority field.
1176bcc9736cSJeff Kirsher  * @br_addr:	Bridge address.  Used for STP.
1177bcc9736cSJeff Kirsher  * @other_addr:	Other MAC address.  Used for multiple network device mode.
1178bcc9736cSJeff Kirsher  * @broad_per:	Broadcast storm percentage.
1179bcc9736cSJeff Kirsher  * @member:	Current port membership.  Used for STP.
1180bcc9736cSJeff Kirsher  */
1181bcc9736cSJeff Kirsher struct ksz_switch {
1182bcc9736cSJeff Kirsher 	struct ksz_mac_table mac_table[STATIC_MAC_TABLE_ENTRIES];
1183bcc9736cSJeff Kirsher 	struct ksz_vlan_table vlan_table[VLAN_TABLE_ENTRIES];
1184bcc9736cSJeff Kirsher 	struct ksz_port_cfg port_cfg[TOTAL_PORT_NUM];
1185bcc9736cSJeff Kirsher 
1186bcc9736cSJeff Kirsher 	u8 diffserv[DIFFSERV_ENTRIES];
1187bcc9736cSJeff Kirsher 	u8 p_802_1p[PRIO_802_1P_ENTRIES];
1188bcc9736cSJeff Kirsher 
11896a3c910cSJoe Perches 	u8 br_addr[ETH_ALEN];
11906a3c910cSJoe Perches 	u8 other_addr[ETH_ALEN];
1191bcc9736cSJeff Kirsher 
1192bcc9736cSJeff Kirsher 	u8 broad_per;
1193bcc9736cSJeff Kirsher 	u8 member;
1194bcc9736cSJeff Kirsher };
1195bcc9736cSJeff Kirsher 
1196bcc9736cSJeff Kirsher #define TX_RATE_UNIT			10000
1197bcc9736cSJeff Kirsher 
1198bcc9736cSJeff Kirsher /**
1199bcc9736cSJeff Kirsher  * struct ksz_port_info - Port information data structure
1200bcc9736cSJeff Kirsher  * @state:	Connection status of the port.
1201bcc9736cSJeff Kirsher  * @tx_rate:	Transmit rate divided by 10000 to get Mbit.
1202bcc9736cSJeff Kirsher  * @duplex:	Duplex mode.
1203bcc9736cSJeff Kirsher  * @advertised:	Advertised auto-negotiation setting.  Used to determine link.
1204bcc9736cSJeff Kirsher  * @partner:	Auto-negotiation partner setting.  Used to determine link.
1205bcc9736cSJeff Kirsher  * @port_id:	Port index to access actual hardware register.
1206bcc9736cSJeff Kirsher  * @pdev:	Pointer to OS dependent network device.
1207bcc9736cSJeff Kirsher  */
1208bcc9736cSJeff Kirsher struct ksz_port_info {
1209bcc9736cSJeff Kirsher 	uint state;
1210bcc9736cSJeff Kirsher 	uint tx_rate;
1211bcc9736cSJeff Kirsher 	u8 duplex;
1212bcc9736cSJeff Kirsher 	u8 advertised;
1213bcc9736cSJeff Kirsher 	u8 partner;
1214bcc9736cSJeff Kirsher 	u8 port_id;
1215bcc9736cSJeff Kirsher 	void *pdev;
1216bcc9736cSJeff Kirsher };
1217bcc9736cSJeff Kirsher 
1218bcc9736cSJeff Kirsher #define MAX_TX_HELD_SIZE		52000
1219bcc9736cSJeff Kirsher 
1220bcc9736cSJeff Kirsher /* Hardware features and bug fixes. */
1221bcc9736cSJeff Kirsher #define LINK_INT_WORKING		(1 << 0)
1222bcc9736cSJeff Kirsher #define SMALL_PACKET_TX_BUG		(1 << 1)
1223bcc9736cSJeff Kirsher #define HALF_DUPLEX_SIGNAL_BUG		(1 << 2)
1224bcc9736cSJeff Kirsher #define RX_HUGE_FRAME			(1 << 4)
1225bcc9736cSJeff Kirsher #define STP_SUPPORT			(1 << 8)
1226bcc9736cSJeff Kirsher 
1227bcc9736cSJeff Kirsher /* Software overrides. */
1228bcc9736cSJeff Kirsher #define PAUSE_FLOW_CTRL			(1 << 0)
1229bcc9736cSJeff Kirsher #define FAST_AGING			(1 << 1)
1230bcc9736cSJeff Kirsher 
1231bcc9736cSJeff Kirsher /**
1232bcc9736cSJeff Kirsher  * struct ksz_hw - KSZ884X hardware data structure
1233bcc9736cSJeff Kirsher  * @io:			Virtual address assigned.
1234bcc9736cSJeff Kirsher  * @ksz_switch:		Pointer to KSZ8842 switch.
1235bcc9736cSJeff Kirsher  * @port_info:		Port information.
1236bcc9736cSJeff Kirsher  * @port_mib:		Port MIB information.
1237bcc9736cSJeff Kirsher  * @dev_count:		Number of network devices this hardware supports.
1238bcc9736cSJeff Kirsher  * @dst_ports:		Destination ports in switch for transmission.
1239bcc9736cSJeff Kirsher  * @id:			Hardware ID.  Used for display only.
1240bcc9736cSJeff Kirsher  * @mib_cnt:		Number of MIB counters this hardware has.
1241bcc9736cSJeff Kirsher  * @mib_port_cnt:	Number of ports with MIB counters.
1242bcc9736cSJeff Kirsher  * @tx_cfg:		Cached transmit control settings.
1243bcc9736cSJeff Kirsher  * @rx_cfg:		Cached receive control settings.
1244bcc9736cSJeff Kirsher  * @intr_mask:		Current interrupt mask.
1245bcc9736cSJeff Kirsher  * @intr_set:		Current interrup set.
1246bcc9736cSJeff Kirsher  * @intr_blocked:	Interrupt blocked.
1247bcc9736cSJeff Kirsher  * @rx_desc_info:	Receive descriptor information.
1248bcc9736cSJeff Kirsher  * @tx_desc_info:	Transmit descriptor information.
1249bcc9736cSJeff Kirsher  * @tx_int_cnt:		Transmit interrupt count.  Used for TX optimization.
1250bcc9736cSJeff Kirsher  * @tx_int_mask:	Transmit interrupt mask.  Used for TX optimization.
1251bcc9736cSJeff Kirsher  * @tx_size:		Transmit data size.  Used for TX optimization.
1252bcc9736cSJeff Kirsher  * 			The maximum is defined by MAX_TX_HELD_SIZE.
1253bcc9736cSJeff Kirsher  * @perm_addr:		Permanent MAC address.
1254bcc9736cSJeff Kirsher  * @override_addr:	Overrided MAC address.
1255bcc9736cSJeff Kirsher  * @address:		Additional MAC address entries.
1256bcc9736cSJeff Kirsher  * @addr_list_size:	Additional MAC address list size.
1257bcc9736cSJeff Kirsher  * @mac_override:	Indication of MAC address overrided.
1258bcc9736cSJeff Kirsher  * @promiscuous:	Counter to keep track of promiscuous mode set.
1259bcc9736cSJeff Kirsher  * @all_multi:		Counter to keep track of all multicast mode set.
1260bcc9736cSJeff Kirsher  * @multi_list:		Multicast address entries.
1261bcc9736cSJeff Kirsher  * @multi_bits:		Cached multicast hash table settings.
1262bcc9736cSJeff Kirsher  * @multi_list_size:	Multicast address list size.
1263bcc9736cSJeff Kirsher  * @enabled:		Indication of hardware enabled.
1264bcc9736cSJeff Kirsher  * @rx_stop:		Indication of receive process stop.
1265bcc9736cSJeff Kirsher  * @features:		Hardware features to enable.
1266bcc9736cSJeff Kirsher  * @overrides:		Hardware features to override.
1267bcc9736cSJeff Kirsher  * @parent:		Pointer to parent, network device private structure.
1268bcc9736cSJeff Kirsher  */
1269bcc9736cSJeff Kirsher struct ksz_hw {
1270bcc9736cSJeff Kirsher 	void __iomem *io;
1271bcc9736cSJeff Kirsher 
1272bcc9736cSJeff Kirsher 	struct ksz_switch *ksz_switch;
1273bcc9736cSJeff Kirsher 	struct ksz_port_info port_info[SWITCH_PORT_NUM];
1274bcc9736cSJeff Kirsher 	struct ksz_port_mib port_mib[TOTAL_PORT_NUM];
1275bcc9736cSJeff Kirsher 	int dev_count;
1276bcc9736cSJeff Kirsher 	int dst_ports;
1277bcc9736cSJeff Kirsher 	int id;
1278bcc9736cSJeff Kirsher 	int mib_cnt;
1279bcc9736cSJeff Kirsher 	int mib_port_cnt;
1280bcc9736cSJeff Kirsher 
1281bcc9736cSJeff Kirsher 	u32 tx_cfg;
1282bcc9736cSJeff Kirsher 	u32 rx_cfg;
1283bcc9736cSJeff Kirsher 	u32 intr_mask;
1284bcc9736cSJeff Kirsher 	u32 intr_set;
1285bcc9736cSJeff Kirsher 	uint intr_blocked;
1286bcc9736cSJeff Kirsher 
1287bcc9736cSJeff Kirsher 	struct ksz_desc_info rx_desc_info;
1288bcc9736cSJeff Kirsher 	struct ksz_desc_info tx_desc_info;
1289bcc9736cSJeff Kirsher 
1290bcc9736cSJeff Kirsher 	int tx_int_cnt;
1291bcc9736cSJeff Kirsher 	int tx_int_mask;
1292bcc9736cSJeff Kirsher 	int tx_size;
1293bcc9736cSJeff Kirsher 
12946a3c910cSJoe Perches 	u8 perm_addr[ETH_ALEN];
12956a3c910cSJoe Perches 	u8 override_addr[ETH_ALEN];
12966a3c910cSJoe Perches 	u8 address[ADDITIONAL_ENTRIES][ETH_ALEN];
1297bcc9736cSJeff Kirsher 	u8 addr_list_size;
1298bcc9736cSJeff Kirsher 	u8 mac_override;
1299bcc9736cSJeff Kirsher 	u8 promiscuous;
1300bcc9736cSJeff Kirsher 	u8 all_multi;
13016a3c910cSJoe Perches 	u8 multi_list[MAX_MULTICAST_LIST][ETH_ALEN];
1302bcc9736cSJeff Kirsher 	u8 multi_bits[HW_MULTICAST_SIZE];
1303bcc9736cSJeff Kirsher 	u8 multi_list_size;
1304bcc9736cSJeff Kirsher 
1305bcc9736cSJeff Kirsher 	u8 enabled;
1306bcc9736cSJeff Kirsher 	u8 rx_stop;
1307bcc9736cSJeff Kirsher 	u8 reserved2[1];
1308bcc9736cSJeff Kirsher 
1309bcc9736cSJeff Kirsher 	uint features;
1310bcc9736cSJeff Kirsher 	uint overrides;
1311bcc9736cSJeff Kirsher 
1312bcc9736cSJeff Kirsher 	void *parent;
1313bcc9736cSJeff Kirsher };
1314bcc9736cSJeff Kirsher 
1315bcc9736cSJeff Kirsher enum {
1316bcc9736cSJeff Kirsher 	PHY_NO_FLOW_CTRL,
1317bcc9736cSJeff Kirsher 	PHY_FLOW_CTRL,
1318bcc9736cSJeff Kirsher 	PHY_TX_ONLY,
1319bcc9736cSJeff Kirsher 	PHY_RX_ONLY
1320bcc9736cSJeff Kirsher };
1321bcc9736cSJeff Kirsher 
1322bcc9736cSJeff Kirsher /**
1323bcc9736cSJeff Kirsher  * struct ksz_port - Virtual port data structure
1324bcc9736cSJeff Kirsher  * @duplex:		Duplex mode setting.  1 for half duplex, 2 for full
1325bcc9736cSJeff Kirsher  * 			duplex, and 0 for auto, which normally results in full
1326bcc9736cSJeff Kirsher  * 			duplex.
1327bcc9736cSJeff Kirsher  * @speed:		Speed setting.  10 for 10 Mbit, 100 for 100 Mbit, and
1328bcc9736cSJeff Kirsher  * 			0 for auto, which normally results in 100 Mbit.
1329bcc9736cSJeff Kirsher  * @force_link:		Force link setting.  0 for auto-negotiation, and 1 for
1330bcc9736cSJeff Kirsher  * 			force.
1331bcc9736cSJeff Kirsher  * @flow_ctrl:		Flow control setting.  PHY_NO_FLOW_CTRL for no flow
1332bcc9736cSJeff Kirsher  * 			control, and PHY_FLOW_CTRL for flow control.
1333bcc9736cSJeff Kirsher  * 			PHY_TX_ONLY and PHY_RX_ONLY are not supported for 100
1334bcc9736cSJeff Kirsher  * 			Mbit PHY.
1335bcc9736cSJeff Kirsher  * @first_port:		Index of first port this port supports.
1336bcc9736cSJeff Kirsher  * @mib_port_cnt:	Number of ports with MIB counters.
1337bcc9736cSJeff Kirsher  * @port_cnt:		Number of ports this port supports.
1338bcc9736cSJeff Kirsher  * @counter:		Port statistics counter.
1339bcc9736cSJeff Kirsher  * @hw:			Pointer to hardware structure.
1340bcc9736cSJeff Kirsher  * @linked:		Pointer to port information linked to this port.
1341bcc9736cSJeff Kirsher  */
1342bcc9736cSJeff Kirsher struct ksz_port {
1343bcc9736cSJeff Kirsher 	u8 duplex;
1344bcc9736cSJeff Kirsher 	u8 speed;
1345bcc9736cSJeff Kirsher 	u8 force_link;
1346bcc9736cSJeff Kirsher 	u8 flow_ctrl;
1347bcc9736cSJeff Kirsher 
1348bcc9736cSJeff Kirsher 	int first_port;
1349bcc9736cSJeff Kirsher 	int mib_port_cnt;
1350bcc9736cSJeff Kirsher 	int port_cnt;
1351bcc9736cSJeff Kirsher 	u64 counter[OID_COUNTER_LAST];
1352bcc9736cSJeff Kirsher 
1353bcc9736cSJeff Kirsher 	struct ksz_hw *hw;
1354bcc9736cSJeff Kirsher 	struct ksz_port_info *linked;
1355bcc9736cSJeff Kirsher };
1356bcc9736cSJeff Kirsher 
1357bcc9736cSJeff Kirsher /**
1358bcc9736cSJeff Kirsher  * struct ksz_timer_info - Timer information data structure
1359bcc9736cSJeff Kirsher  * @timer:	Kernel timer.
1360bcc9736cSJeff Kirsher  * @cnt:	Running timer counter.
1361bcc9736cSJeff Kirsher  * @max:	Number of times to run timer; -1 for infinity.
1362bcc9736cSJeff Kirsher  * @period:	Timer period in jiffies.
1363bcc9736cSJeff Kirsher  */
1364bcc9736cSJeff Kirsher struct ksz_timer_info {
1365bcc9736cSJeff Kirsher 	struct timer_list timer;
1366bcc9736cSJeff Kirsher 	int cnt;
1367bcc9736cSJeff Kirsher 	int max;
1368bcc9736cSJeff Kirsher 	int period;
1369bcc9736cSJeff Kirsher };
1370bcc9736cSJeff Kirsher 
1371bcc9736cSJeff Kirsher /**
1372bcc9736cSJeff Kirsher  * struct ksz_shared_mem - OS dependent shared memory data structure
1373bcc9736cSJeff Kirsher  * @dma_addr:	Physical DMA address allocated.
1374bcc9736cSJeff Kirsher  * @alloc_size:	Allocation size.
1375bcc9736cSJeff Kirsher  * @phys:	Actual physical address used.
1376bcc9736cSJeff Kirsher  * @alloc_virt:	Virtual address allocated.
1377bcc9736cSJeff Kirsher  * @virt:	Actual virtual address used.
1378bcc9736cSJeff Kirsher  */
1379bcc9736cSJeff Kirsher struct ksz_shared_mem {
1380bcc9736cSJeff Kirsher 	dma_addr_t dma_addr;
1381bcc9736cSJeff Kirsher 	uint alloc_size;
1382bcc9736cSJeff Kirsher 	uint phys;
1383bcc9736cSJeff Kirsher 	u8 *alloc_virt;
1384bcc9736cSJeff Kirsher 	u8 *virt;
1385bcc9736cSJeff Kirsher };
1386bcc9736cSJeff Kirsher 
1387bcc9736cSJeff Kirsher /**
1388bcc9736cSJeff Kirsher  * struct ksz_counter_info - OS dependent counter information data structure
1389bcc9736cSJeff Kirsher  * @counter:	Wait queue to wakeup after counters are read.
1390bcc9736cSJeff Kirsher  * @time:	Next time in jiffies to read counter.
1391bcc9736cSJeff Kirsher  * @read:	Indication of counters read in full or not.
1392bcc9736cSJeff Kirsher  */
1393bcc9736cSJeff Kirsher struct ksz_counter_info {
1394bcc9736cSJeff Kirsher 	wait_queue_head_t counter;
1395bcc9736cSJeff Kirsher 	unsigned long time;
1396bcc9736cSJeff Kirsher 	int read;
1397bcc9736cSJeff Kirsher };
1398bcc9736cSJeff Kirsher 
1399bcc9736cSJeff Kirsher /**
1400bcc9736cSJeff Kirsher  * struct dev_info - Network device information data structure
1401bcc9736cSJeff Kirsher  * @dev:		Pointer to network device.
1402bcc9736cSJeff Kirsher  * @pdev:		Pointer to PCI device.
1403bcc9736cSJeff Kirsher  * @hw:			Hardware structure.
1404bcc9736cSJeff Kirsher  * @desc_pool:		Physical memory used for descriptor pool.
1405bcc9736cSJeff Kirsher  * @hwlock:		Spinlock to prevent hardware from accessing.
1406bcc9736cSJeff Kirsher  * @lock:		Mutex lock to prevent device from accessing.
1407bcc9736cSJeff Kirsher  * @dev_rcv:		Receive process function used.
1408bcc9736cSJeff Kirsher  * @last_skb:		Socket buffer allocated for descriptor rx fragments.
1409bcc9736cSJeff Kirsher  * @skb_index:		Buffer index for receiving fragments.
1410bcc9736cSJeff Kirsher  * @skb_len:		Buffer length for receiving fragments.
1411bcc9736cSJeff Kirsher  * @mib_read:		Workqueue to read MIB counters.
1412bcc9736cSJeff Kirsher  * @mib_timer_info:	Timer to read MIB counters.
1413bcc9736cSJeff Kirsher  * @counter:		Used for MIB reading.
1414bcc9736cSJeff Kirsher  * @mtu:		Current MTU used.  The default is REGULAR_RX_BUF_SIZE;
1415bcc9736cSJeff Kirsher  * 			the maximum is MAX_RX_BUF_SIZE.
1416bcc9736cSJeff Kirsher  * @opened:		Counter to keep track of device open.
1417bcc9736cSJeff Kirsher  * @rx_tasklet:		Receive processing tasklet.
1418bcc9736cSJeff Kirsher  * @tx_tasklet:		Transmit processing tasklet.
1419bcc9736cSJeff Kirsher  * @wol_enable:		Wake-on-LAN enable set by ethtool.
1420bcc9736cSJeff Kirsher  * @wol_support:	Wake-on-LAN support used by ethtool.
1421bcc9736cSJeff Kirsher  * @pme_wait:		Used for KSZ8841 power management.
1422bcc9736cSJeff Kirsher  */
1423bcc9736cSJeff Kirsher struct dev_info {
1424bcc9736cSJeff Kirsher 	struct net_device *dev;
1425bcc9736cSJeff Kirsher 	struct pci_dev *pdev;
1426bcc9736cSJeff Kirsher 
1427bcc9736cSJeff Kirsher 	struct ksz_hw hw;
1428bcc9736cSJeff Kirsher 	struct ksz_shared_mem desc_pool;
1429bcc9736cSJeff Kirsher 
1430bcc9736cSJeff Kirsher 	spinlock_t hwlock;
1431bcc9736cSJeff Kirsher 	struct mutex lock;
1432bcc9736cSJeff Kirsher 
1433bcc9736cSJeff Kirsher 	int (*dev_rcv)(struct dev_info *);
1434bcc9736cSJeff Kirsher 
1435bcc9736cSJeff Kirsher 	struct sk_buff *last_skb;
1436bcc9736cSJeff Kirsher 	int skb_index;
1437bcc9736cSJeff Kirsher 	int skb_len;
1438bcc9736cSJeff Kirsher 
1439bcc9736cSJeff Kirsher 	struct work_struct mib_read;
1440bcc9736cSJeff Kirsher 	struct ksz_timer_info mib_timer_info;
1441bcc9736cSJeff Kirsher 	struct ksz_counter_info counter[TOTAL_PORT_NUM];
1442bcc9736cSJeff Kirsher 
1443bcc9736cSJeff Kirsher 	int mtu;
1444bcc9736cSJeff Kirsher 	int opened;
1445bcc9736cSJeff Kirsher 
1446bcc9736cSJeff Kirsher 	struct tasklet_struct rx_tasklet;
1447bcc9736cSJeff Kirsher 	struct tasklet_struct tx_tasklet;
1448bcc9736cSJeff Kirsher 
1449bcc9736cSJeff Kirsher 	int wol_enable;
1450bcc9736cSJeff Kirsher 	int wol_support;
1451bcc9736cSJeff Kirsher 	unsigned long pme_wait;
1452bcc9736cSJeff Kirsher };
1453bcc9736cSJeff Kirsher 
1454bcc9736cSJeff Kirsher /**
1455bcc9736cSJeff Kirsher  * struct dev_priv - Network device private data structure
1456bcc9736cSJeff Kirsher  * @adapter:		Adapter device information.
1457bcc9736cSJeff Kirsher  * @port:		Port information.
1458bcc9736cSJeff Kirsher  * @monitor_time_info:	Timer to monitor ports.
1459bcc9736cSJeff Kirsher  * @proc_sem:		Semaphore for proc accessing.
1460bcc9736cSJeff Kirsher  * @id:			Device ID.
1461bcc9736cSJeff Kirsher  * @mii_if:		MII interface information.
1462bcc9736cSJeff Kirsher  * @advertising:	Temporary variable to store advertised settings.
1463bcc9736cSJeff Kirsher  * @msg_enable:		The message flags controlling driver output.
1464bcc9736cSJeff Kirsher  * @media_state:	The connection status of the device.
1465bcc9736cSJeff Kirsher  * @multicast:		The all multicast state of the device.
1466bcc9736cSJeff Kirsher  * @promiscuous:	The promiscuous state of the device.
1467bcc9736cSJeff Kirsher  */
1468bcc9736cSJeff Kirsher struct dev_priv {
1469bcc9736cSJeff Kirsher 	struct dev_info *adapter;
1470bcc9736cSJeff Kirsher 	struct ksz_port port;
1471bcc9736cSJeff Kirsher 	struct ksz_timer_info monitor_timer_info;
1472bcc9736cSJeff Kirsher 
1473bcc9736cSJeff Kirsher 	struct semaphore proc_sem;
1474bcc9736cSJeff Kirsher 	int id;
1475bcc9736cSJeff Kirsher 
1476bcc9736cSJeff Kirsher 	struct mii_if_info mii_if;
1477bcc9736cSJeff Kirsher 	u32 advertising;
1478bcc9736cSJeff Kirsher 
1479bcc9736cSJeff Kirsher 	u32 msg_enable;
1480bcc9736cSJeff Kirsher 	int media_state;
1481bcc9736cSJeff Kirsher 	int multicast;
1482bcc9736cSJeff Kirsher 	int promiscuous;
1483bcc9736cSJeff Kirsher };
1484bcc9736cSJeff Kirsher 
1485bcc9736cSJeff Kirsher #define DRV_NAME		"KSZ884X PCI"
1486bcc9736cSJeff Kirsher #define DEVICE_NAME		"KSZ884x PCI"
1487bcc9736cSJeff Kirsher #define DRV_VERSION		"1.0.0"
1488bcc9736cSJeff Kirsher #define DRV_RELDATE		"Feb 8, 2010"
1489bcc9736cSJeff Kirsher 
1490bcc9736cSJeff Kirsher static char version[] __devinitdata =
1491bcc9736cSJeff Kirsher 	"Micrel " DEVICE_NAME " " DRV_VERSION " (" DRV_RELDATE ")";
1492bcc9736cSJeff Kirsher 
1493bcc9736cSJeff Kirsher static u8 DEFAULT_MAC_ADDRESS[] = { 0x00, 0x10, 0xA1, 0x88, 0x42, 0x01 };
1494bcc9736cSJeff Kirsher 
1495bcc9736cSJeff Kirsher /*
1496bcc9736cSJeff Kirsher  * Interrupt processing primary routines
1497bcc9736cSJeff Kirsher  */
1498bcc9736cSJeff Kirsher 
1499bcc9736cSJeff Kirsher static inline void hw_ack_intr(struct ksz_hw *hw, uint interrupt)
1500bcc9736cSJeff Kirsher {
1501bcc9736cSJeff Kirsher 	writel(interrupt, hw->io + KS884X_INTERRUPTS_STATUS);
1502bcc9736cSJeff Kirsher }
1503bcc9736cSJeff Kirsher 
1504bcc9736cSJeff Kirsher static inline void hw_dis_intr(struct ksz_hw *hw)
1505bcc9736cSJeff Kirsher {
1506bcc9736cSJeff Kirsher 	hw->intr_blocked = hw->intr_mask;
1507bcc9736cSJeff Kirsher 	writel(0, hw->io + KS884X_INTERRUPTS_ENABLE);
1508bcc9736cSJeff Kirsher 	hw->intr_set = readl(hw->io + KS884X_INTERRUPTS_ENABLE);
1509bcc9736cSJeff Kirsher }
1510bcc9736cSJeff Kirsher 
1511bcc9736cSJeff Kirsher static inline void hw_set_intr(struct ksz_hw *hw, uint interrupt)
1512bcc9736cSJeff Kirsher {
1513bcc9736cSJeff Kirsher 	hw->intr_set = interrupt;
1514bcc9736cSJeff Kirsher 	writel(interrupt, hw->io + KS884X_INTERRUPTS_ENABLE);
1515bcc9736cSJeff Kirsher }
1516bcc9736cSJeff Kirsher 
1517bcc9736cSJeff Kirsher static inline void hw_ena_intr(struct ksz_hw *hw)
1518bcc9736cSJeff Kirsher {
1519bcc9736cSJeff Kirsher 	hw->intr_blocked = 0;
1520bcc9736cSJeff Kirsher 	hw_set_intr(hw, hw->intr_mask);
1521bcc9736cSJeff Kirsher }
1522bcc9736cSJeff Kirsher 
1523bcc9736cSJeff Kirsher static inline void hw_dis_intr_bit(struct ksz_hw *hw, uint bit)
1524bcc9736cSJeff Kirsher {
1525bcc9736cSJeff Kirsher 	hw->intr_mask &= ~(bit);
1526bcc9736cSJeff Kirsher }
1527bcc9736cSJeff Kirsher 
1528bcc9736cSJeff Kirsher static inline void hw_turn_off_intr(struct ksz_hw *hw, uint interrupt)
1529bcc9736cSJeff Kirsher {
1530bcc9736cSJeff Kirsher 	u32 read_intr;
1531bcc9736cSJeff Kirsher 
1532bcc9736cSJeff Kirsher 	read_intr = readl(hw->io + KS884X_INTERRUPTS_ENABLE);
1533bcc9736cSJeff Kirsher 	hw->intr_set = read_intr & ~interrupt;
1534bcc9736cSJeff Kirsher 	writel(hw->intr_set, hw->io + KS884X_INTERRUPTS_ENABLE);
1535bcc9736cSJeff Kirsher 	hw_dis_intr_bit(hw, interrupt);
1536bcc9736cSJeff Kirsher }
1537bcc9736cSJeff Kirsher 
1538bcc9736cSJeff Kirsher /**
1539bcc9736cSJeff Kirsher  * hw_turn_on_intr - turn on specified interrupts
1540bcc9736cSJeff Kirsher  * @hw: 	The hardware instance.
1541bcc9736cSJeff Kirsher  * @bit:	The interrupt bits to be on.
1542bcc9736cSJeff Kirsher  *
1543bcc9736cSJeff Kirsher  * This routine turns on the specified interrupts in the interrupt mask so that
1544bcc9736cSJeff Kirsher  * those interrupts will be enabled.
1545bcc9736cSJeff Kirsher  */
1546bcc9736cSJeff Kirsher static void hw_turn_on_intr(struct ksz_hw *hw, u32 bit)
1547bcc9736cSJeff Kirsher {
1548bcc9736cSJeff Kirsher 	hw->intr_mask |= bit;
1549bcc9736cSJeff Kirsher 
1550bcc9736cSJeff Kirsher 	if (!hw->intr_blocked)
1551bcc9736cSJeff Kirsher 		hw_set_intr(hw, hw->intr_mask);
1552bcc9736cSJeff Kirsher }
1553bcc9736cSJeff Kirsher 
1554bcc9736cSJeff Kirsher static inline void hw_ena_intr_bit(struct ksz_hw *hw, uint interrupt)
1555bcc9736cSJeff Kirsher {
1556bcc9736cSJeff Kirsher 	u32 read_intr;
1557bcc9736cSJeff Kirsher 
1558bcc9736cSJeff Kirsher 	read_intr = readl(hw->io + KS884X_INTERRUPTS_ENABLE);
1559bcc9736cSJeff Kirsher 	hw->intr_set = read_intr | interrupt;
1560bcc9736cSJeff Kirsher 	writel(hw->intr_set, hw->io + KS884X_INTERRUPTS_ENABLE);
1561bcc9736cSJeff Kirsher }
1562bcc9736cSJeff Kirsher 
1563bcc9736cSJeff Kirsher static inline void hw_read_intr(struct ksz_hw *hw, uint *status)
1564bcc9736cSJeff Kirsher {
1565bcc9736cSJeff Kirsher 	*status = readl(hw->io + KS884X_INTERRUPTS_STATUS);
1566bcc9736cSJeff Kirsher 	*status = *status & hw->intr_set;
1567bcc9736cSJeff Kirsher }
1568bcc9736cSJeff Kirsher 
1569bcc9736cSJeff Kirsher static inline void hw_restore_intr(struct ksz_hw *hw, uint interrupt)
1570bcc9736cSJeff Kirsher {
1571bcc9736cSJeff Kirsher 	if (interrupt)
1572bcc9736cSJeff Kirsher 		hw_ena_intr(hw);
1573bcc9736cSJeff Kirsher }
1574bcc9736cSJeff Kirsher 
1575bcc9736cSJeff Kirsher /**
1576bcc9736cSJeff Kirsher  * hw_block_intr - block hardware interrupts
1577bcc9736cSJeff Kirsher  *
1578bcc9736cSJeff Kirsher  * This function blocks all interrupts of the hardware and returns the current
1579bcc9736cSJeff Kirsher  * interrupt enable mask so that interrupts can be restored later.
1580bcc9736cSJeff Kirsher  *
1581bcc9736cSJeff Kirsher  * Return the current interrupt enable mask.
1582bcc9736cSJeff Kirsher  */
1583bcc9736cSJeff Kirsher static uint hw_block_intr(struct ksz_hw *hw)
1584bcc9736cSJeff Kirsher {
1585bcc9736cSJeff Kirsher 	uint interrupt = 0;
1586bcc9736cSJeff Kirsher 
1587bcc9736cSJeff Kirsher 	if (!hw->intr_blocked) {
1588bcc9736cSJeff Kirsher 		hw_dis_intr(hw);
1589bcc9736cSJeff Kirsher 		interrupt = hw->intr_blocked;
1590bcc9736cSJeff Kirsher 	}
1591bcc9736cSJeff Kirsher 	return interrupt;
1592bcc9736cSJeff Kirsher }
1593bcc9736cSJeff Kirsher 
1594bcc9736cSJeff Kirsher /*
1595bcc9736cSJeff Kirsher  * Hardware descriptor routines
1596bcc9736cSJeff Kirsher  */
1597bcc9736cSJeff Kirsher 
1598bcc9736cSJeff Kirsher static inline void reset_desc(struct ksz_desc *desc, union desc_stat status)
1599bcc9736cSJeff Kirsher {
1600bcc9736cSJeff Kirsher 	status.rx.hw_owned = 0;
1601bcc9736cSJeff Kirsher 	desc->phw->ctrl.data = cpu_to_le32(status.data);
1602bcc9736cSJeff Kirsher }
1603bcc9736cSJeff Kirsher 
1604bcc9736cSJeff Kirsher static inline void release_desc(struct ksz_desc *desc)
1605bcc9736cSJeff Kirsher {
1606bcc9736cSJeff Kirsher 	desc->sw.ctrl.tx.hw_owned = 1;
1607bcc9736cSJeff Kirsher 	if (desc->sw.buf_size != desc->sw.buf.data) {
1608bcc9736cSJeff Kirsher 		desc->sw.buf_size = desc->sw.buf.data;
1609bcc9736cSJeff Kirsher 		desc->phw->buf.data = cpu_to_le32(desc->sw.buf.data);
1610bcc9736cSJeff Kirsher 	}
1611bcc9736cSJeff Kirsher 	desc->phw->ctrl.data = cpu_to_le32(desc->sw.ctrl.data);
1612bcc9736cSJeff Kirsher }
1613bcc9736cSJeff Kirsher 
1614bcc9736cSJeff Kirsher static void get_rx_pkt(struct ksz_desc_info *info, struct ksz_desc **desc)
1615bcc9736cSJeff Kirsher {
1616bcc9736cSJeff Kirsher 	*desc = &info->ring[info->last];
1617bcc9736cSJeff Kirsher 	info->last++;
1618bcc9736cSJeff Kirsher 	info->last &= info->mask;
1619bcc9736cSJeff Kirsher 	info->avail--;
1620bcc9736cSJeff Kirsher 	(*desc)->sw.buf.data &= ~KS_DESC_RX_MASK;
1621bcc9736cSJeff Kirsher }
1622bcc9736cSJeff Kirsher 
1623bcc9736cSJeff Kirsher static inline void set_rx_buf(struct ksz_desc *desc, u32 addr)
1624bcc9736cSJeff Kirsher {
1625bcc9736cSJeff Kirsher 	desc->phw->addr = cpu_to_le32(addr);
1626bcc9736cSJeff Kirsher }
1627bcc9736cSJeff Kirsher 
1628bcc9736cSJeff Kirsher static inline void set_rx_len(struct ksz_desc *desc, u32 len)
1629bcc9736cSJeff Kirsher {
1630bcc9736cSJeff Kirsher 	desc->sw.buf.rx.buf_size = len;
1631bcc9736cSJeff Kirsher }
1632bcc9736cSJeff Kirsher 
1633bcc9736cSJeff Kirsher static inline void get_tx_pkt(struct ksz_desc_info *info,
1634bcc9736cSJeff Kirsher 	struct ksz_desc **desc)
1635bcc9736cSJeff Kirsher {
1636bcc9736cSJeff Kirsher 	*desc = &info->ring[info->next];
1637bcc9736cSJeff Kirsher 	info->next++;
1638bcc9736cSJeff Kirsher 	info->next &= info->mask;
1639bcc9736cSJeff Kirsher 	info->avail--;
1640bcc9736cSJeff Kirsher 	(*desc)->sw.buf.data &= ~KS_DESC_TX_MASK;
1641bcc9736cSJeff Kirsher }
1642bcc9736cSJeff Kirsher 
1643bcc9736cSJeff Kirsher static inline void set_tx_buf(struct ksz_desc *desc, u32 addr)
1644bcc9736cSJeff Kirsher {
1645bcc9736cSJeff Kirsher 	desc->phw->addr = cpu_to_le32(addr);
1646bcc9736cSJeff Kirsher }
1647bcc9736cSJeff Kirsher 
1648bcc9736cSJeff Kirsher static inline void set_tx_len(struct ksz_desc *desc, u32 len)
1649bcc9736cSJeff Kirsher {
1650bcc9736cSJeff Kirsher 	desc->sw.buf.tx.buf_size = len;
1651bcc9736cSJeff Kirsher }
1652bcc9736cSJeff Kirsher 
1653bcc9736cSJeff Kirsher /* Switch functions */
1654bcc9736cSJeff Kirsher 
1655bcc9736cSJeff Kirsher #define TABLE_READ			0x10
1656bcc9736cSJeff Kirsher #define TABLE_SEL_SHIFT			2
1657bcc9736cSJeff Kirsher 
1658bcc9736cSJeff Kirsher #define HW_DELAY(hw, reg)			\
1659bcc9736cSJeff Kirsher 	do {					\
1660bcc9736cSJeff Kirsher 		u16 dummy;			\
1661bcc9736cSJeff Kirsher 		dummy = readw(hw->io + reg);	\
1662bcc9736cSJeff Kirsher 	} while (0)
1663bcc9736cSJeff Kirsher 
1664bcc9736cSJeff Kirsher /**
1665bcc9736cSJeff Kirsher  * sw_r_table - read 4 bytes of data from switch table
1666bcc9736cSJeff Kirsher  * @hw:		The hardware instance.
1667bcc9736cSJeff Kirsher  * @table:	The table selector.
1668bcc9736cSJeff Kirsher  * @addr:	The address of the table entry.
1669bcc9736cSJeff Kirsher  * @data:	Buffer to store the read data.
1670bcc9736cSJeff Kirsher  *
1671bcc9736cSJeff Kirsher  * This routine reads 4 bytes of data from the table of the switch.
1672bcc9736cSJeff Kirsher  * Hardware interrupts are disabled to minimize corruption of read data.
1673bcc9736cSJeff Kirsher  */
1674bcc9736cSJeff Kirsher static void sw_r_table(struct ksz_hw *hw, int table, u16 addr, u32 *data)
1675bcc9736cSJeff Kirsher {
1676bcc9736cSJeff Kirsher 	u16 ctrl_addr;
1677bcc9736cSJeff Kirsher 	uint interrupt;
1678bcc9736cSJeff Kirsher 
1679bcc9736cSJeff Kirsher 	ctrl_addr = (((table << TABLE_SEL_SHIFT) | TABLE_READ) << 8) | addr;
1680bcc9736cSJeff Kirsher 
1681bcc9736cSJeff Kirsher 	interrupt = hw_block_intr(hw);
1682bcc9736cSJeff Kirsher 
1683bcc9736cSJeff Kirsher 	writew(ctrl_addr, hw->io + KS884X_IACR_OFFSET);
1684bcc9736cSJeff Kirsher 	HW_DELAY(hw, KS884X_IACR_OFFSET);
1685bcc9736cSJeff Kirsher 	*data = readl(hw->io + KS884X_ACC_DATA_0_OFFSET);
1686bcc9736cSJeff Kirsher 
1687bcc9736cSJeff Kirsher 	hw_restore_intr(hw, interrupt);
1688bcc9736cSJeff Kirsher }
1689bcc9736cSJeff Kirsher 
1690bcc9736cSJeff Kirsher /**
1691bcc9736cSJeff Kirsher  * sw_w_table_64 - write 8 bytes of data to the switch table
1692bcc9736cSJeff Kirsher  * @hw:		The hardware instance.
1693bcc9736cSJeff Kirsher  * @table:	The table selector.
1694bcc9736cSJeff Kirsher  * @addr:	The address of the table entry.
1695bcc9736cSJeff Kirsher  * @data_hi:	The high part of data to be written (bit63 ~ bit32).
1696bcc9736cSJeff Kirsher  * @data_lo:	The low part of data to be written (bit31 ~ bit0).
1697bcc9736cSJeff Kirsher  *
1698bcc9736cSJeff Kirsher  * This routine writes 8 bytes of data to the table of the switch.
1699bcc9736cSJeff Kirsher  * Hardware interrupts are disabled to minimize corruption of written data.
1700bcc9736cSJeff Kirsher  */
1701bcc9736cSJeff Kirsher static void sw_w_table_64(struct ksz_hw *hw, int table, u16 addr, u32 data_hi,
1702bcc9736cSJeff Kirsher 	u32 data_lo)
1703bcc9736cSJeff Kirsher {
1704bcc9736cSJeff Kirsher 	u16 ctrl_addr;
1705bcc9736cSJeff Kirsher 	uint interrupt;
1706bcc9736cSJeff Kirsher 
1707bcc9736cSJeff Kirsher 	ctrl_addr = ((table << TABLE_SEL_SHIFT) << 8) | addr;
1708bcc9736cSJeff Kirsher 
1709bcc9736cSJeff Kirsher 	interrupt = hw_block_intr(hw);
1710bcc9736cSJeff Kirsher 
1711bcc9736cSJeff Kirsher 	writel(data_hi, hw->io + KS884X_ACC_DATA_4_OFFSET);
1712bcc9736cSJeff Kirsher 	writel(data_lo, hw->io + KS884X_ACC_DATA_0_OFFSET);
1713bcc9736cSJeff Kirsher 
1714bcc9736cSJeff Kirsher 	writew(ctrl_addr, hw->io + KS884X_IACR_OFFSET);
1715bcc9736cSJeff Kirsher 	HW_DELAY(hw, KS884X_IACR_OFFSET);
1716bcc9736cSJeff Kirsher 
1717bcc9736cSJeff Kirsher 	hw_restore_intr(hw, interrupt);
1718bcc9736cSJeff Kirsher }
1719bcc9736cSJeff Kirsher 
1720bcc9736cSJeff Kirsher /**
1721bcc9736cSJeff Kirsher  * sw_w_sta_mac_table - write to the static MAC table
1722bcc9736cSJeff Kirsher  * @hw: 	The hardware instance.
1723bcc9736cSJeff Kirsher  * @addr:	The address of the table entry.
1724bcc9736cSJeff Kirsher  * @mac_addr:	The MAC address.
1725bcc9736cSJeff Kirsher  * @ports:	The port members.
1726bcc9736cSJeff Kirsher  * @override:	The flag to override the port receive/transmit settings.
1727bcc9736cSJeff Kirsher  * @valid:	The flag to indicate entry is valid.
1728bcc9736cSJeff Kirsher  * @use_fid:	The flag to indicate the FID is valid.
1729bcc9736cSJeff Kirsher  * @fid:	The FID value.
1730bcc9736cSJeff Kirsher  *
1731bcc9736cSJeff Kirsher  * This routine writes an entry of the static MAC table of the switch.  It
1732bcc9736cSJeff Kirsher  * calls sw_w_table_64() to write the data.
1733bcc9736cSJeff Kirsher  */
1734bcc9736cSJeff Kirsher static void sw_w_sta_mac_table(struct ksz_hw *hw, u16 addr, u8 *mac_addr,
1735bcc9736cSJeff Kirsher 	u8 ports, int override, int valid, int use_fid, u8 fid)
1736bcc9736cSJeff Kirsher {
1737bcc9736cSJeff Kirsher 	u32 data_hi;
1738bcc9736cSJeff Kirsher 	u32 data_lo;
1739bcc9736cSJeff Kirsher 
1740bcc9736cSJeff Kirsher 	data_lo = ((u32) mac_addr[2] << 24) |
1741bcc9736cSJeff Kirsher 		((u32) mac_addr[3] << 16) |
1742bcc9736cSJeff Kirsher 		((u32) mac_addr[4] << 8) | mac_addr[5];
1743bcc9736cSJeff Kirsher 	data_hi = ((u32) mac_addr[0] << 8) | mac_addr[1];
1744bcc9736cSJeff Kirsher 	data_hi |= (u32) ports << STATIC_MAC_FWD_PORTS_SHIFT;
1745bcc9736cSJeff Kirsher 
1746bcc9736cSJeff Kirsher 	if (override)
1747bcc9736cSJeff Kirsher 		data_hi |= STATIC_MAC_TABLE_OVERRIDE;
1748bcc9736cSJeff Kirsher 	if (use_fid) {
1749bcc9736cSJeff Kirsher 		data_hi |= STATIC_MAC_TABLE_USE_FID;
1750bcc9736cSJeff Kirsher 		data_hi |= (u32) fid << STATIC_MAC_FID_SHIFT;
1751bcc9736cSJeff Kirsher 	}
1752bcc9736cSJeff Kirsher 	if (valid)
1753bcc9736cSJeff Kirsher 		data_hi |= STATIC_MAC_TABLE_VALID;
1754bcc9736cSJeff Kirsher 
1755bcc9736cSJeff Kirsher 	sw_w_table_64(hw, TABLE_STATIC_MAC, addr, data_hi, data_lo);
1756bcc9736cSJeff Kirsher }
1757bcc9736cSJeff Kirsher 
1758bcc9736cSJeff Kirsher /**
1759bcc9736cSJeff Kirsher  * sw_r_vlan_table - read from the VLAN table
1760bcc9736cSJeff Kirsher  * @hw: 	The hardware instance.
1761bcc9736cSJeff Kirsher  * @addr:	The address of the table entry.
1762bcc9736cSJeff Kirsher  * @vid:	Buffer to store the VID.
1763bcc9736cSJeff Kirsher  * @fid:	Buffer to store the VID.
1764bcc9736cSJeff Kirsher  * @member:	Buffer to store the port membership.
1765bcc9736cSJeff Kirsher  *
1766bcc9736cSJeff Kirsher  * This function reads an entry of the VLAN table of the switch.  It calls
1767bcc9736cSJeff Kirsher  * sw_r_table() to get the data.
1768bcc9736cSJeff Kirsher  *
1769bcc9736cSJeff Kirsher  * Return 0 if the entry is valid; otherwise -1.
1770bcc9736cSJeff Kirsher  */
1771bcc9736cSJeff Kirsher static int sw_r_vlan_table(struct ksz_hw *hw, u16 addr, u16 *vid, u8 *fid,
1772bcc9736cSJeff Kirsher 	u8 *member)
1773bcc9736cSJeff Kirsher {
1774bcc9736cSJeff Kirsher 	u32 data;
1775bcc9736cSJeff Kirsher 
1776bcc9736cSJeff Kirsher 	sw_r_table(hw, TABLE_VLAN, addr, &data);
1777bcc9736cSJeff Kirsher 	if (data & VLAN_TABLE_VALID) {
1778bcc9736cSJeff Kirsher 		*vid = (u16)(data & VLAN_TABLE_VID);
1779bcc9736cSJeff Kirsher 		*fid = (u8)((data & VLAN_TABLE_FID) >> VLAN_TABLE_FID_SHIFT);
1780bcc9736cSJeff Kirsher 		*member = (u8)((data & VLAN_TABLE_MEMBERSHIP) >>
1781bcc9736cSJeff Kirsher 			VLAN_TABLE_MEMBERSHIP_SHIFT);
1782bcc9736cSJeff Kirsher 		return 0;
1783bcc9736cSJeff Kirsher 	}
1784bcc9736cSJeff Kirsher 	return -1;
1785bcc9736cSJeff Kirsher }
1786bcc9736cSJeff Kirsher 
1787bcc9736cSJeff Kirsher /**
1788bcc9736cSJeff Kirsher  * port_r_mib_cnt - read MIB counter
1789bcc9736cSJeff Kirsher  * @hw: 	The hardware instance.
1790bcc9736cSJeff Kirsher  * @port:	The port index.
1791bcc9736cSJeff Kirsher  * @addr:	The address of the counter.
1792bcc9736cSJeff Kirsher  * @cnt:	Buffer to store the counter.
1793bcc9736cSJeff Kirsher  *
1794bcc9736cSJeff Kirsher  * This routine reads a MIB counter of the port.
1795bcc9736cSJeff Kirsher  * Hardware interrupts are disabled to minimize corruption of read data.
1796bcc9736cSJeff Kirsher  */
1797bcc9736cSJeff Kirsher static void port_r_mib_cnt(struct ksz_hw *hw, int port, u16 addr, u64 *cnt)
1798bcc9736cSJeff Kirsher {
1799bcc9736cSJeff Kirsher 	u32 data;
1800bcc9736cSJeff Kirsher 	u16 ctrl_addr;
1801bcc9736cSJeff Kirsher 	uint interrupt;
1802bcc9736cSJeff Kirsher 	int timeout;
1803bcc9736cSJeff Kirsher 
1804bcc9736cSJeff Kirsher 	ctrl_addr = addr + PORT_COUNTER_NUM * port;
1805bcc9736cSJeff Kirsher 
1806bcc9736cSJeff Kirsher 	interrupt = hw_block_intr(hw);
1807bcc9736cSJeff Kirsher 
1808bcc9736cSJeff Kirsher 	ctrl_addr |= (((TABLE_MIB << TABLE_SEL_SHIFT) | TABLE_READ) << 8);
1809bcc9736cSJeff Kirsher 	writew(ctrl_addr, hw->io + KS884X_IACR_OFFSET);
1810bcc9736cSJeff Kirsher 	HW_DELAY(hw, KS884X_IACR_OFFSET);
1811bcc9736cSJeff Kirsher 
1812bcc9736cSJeff Kirsher 	for (timeout = 100; timeout > 0; timeout--) {
1813bcc9736cSJeff Kirsher 		data = readl(hw->io + KS884X_ACC_DATA_0_OFFSET);
1814bcc9736cSJeff Kirsher 
1815bcc9736cSJeff Kirsher 		if (data & MIB_COUNTER_VALID) {
1816bcc9736cSJeff Kirsher 			if (data & MIB_COUNTER_OVERFLOW)
1817bcc9736cSJeff Kirsher 				*cnt += MIB_COUNTER_VALUE + 1;
1818bcc9736cSJeff Kirsher 			*cnt += data & MIB_COUNTER_VALUE;
1819bcc9736cSJeff Kirsher 			break;
1820bcc9736cSJeff Kirsher 		}
1821bcc9736cSJeff Kirsher 	}
1822bcc9736cSJeff Kirsher 
1823bcc9736cSJeff Kirsher 	hw_restore_intr(hw, interrupt);
1824bcc9736cSJeff Kirsher }
1825bcc9736cSJeff Kirsher 
1826bcc9736cSJeff Kirsher /**
1827bcc9736cSJeff Kirsher  * port_r_mib_pkt - read dropped packet counts
1828bcc9736cSJeff Kirsher  * @hw: 	The hardware instance.
1829bcc9736cSJeff Kirsher  * @port:	The port index.
1830bcc9736cSJeff Kirsher  * @cnt:	Buffer to store the receive and transmit dropped packet counts.
1831bcc9736cSJeff Kirsher  *
1832bcc9736cSJeff Kirsher  * This routine reads the dropped packet counts of the port.
1833bcc9736cSJeff Kirsher  * Hardware interrupts are disabled to minimize corruption of read data.
1834bcc9736cSJeff Kirsher  */
1835bcc9736cSJeff Kirsher static void port_r_mib_pkt(struct ksz_hw *hw, int port, u32 *last, u64 *cnt)
1836bcc9736cSJeff Kirsher {
1837bcc9736cSJeff Kirsher 	u32 cur;
1838bcc9736cSJeff Kirsher 	u32 data;
1839bcc9736cSJeff Kirsher 	u16 ctrl_addr;
1840bcc9736cSJeff Kirsher 	uint interrupt;
1841bcc9736cSJeff Kirsher 	int index;
1842bcc9736cSJeff Kirsher 
1843bcc9736cSJeff Kirsher 	index = KS_MIB_PACKET_DROPPED_RX_0 + port;
1844bcc9736cSJeff Kirsher 	do {
1845bcc9736cSJeff Kirsher 		interrupt = hw_block_intr(hw);
1846bcc9736cSJeff Kirsher 
1847bcc9736cSJeff Kirsher 		ctrl_addr = (u16) index;
1848bcc9736cSJeff Kirsher 		ctrl_addr |= (((TABLE_MIB << TABLE_SEL_SHIFT) | TABLE_READ)
1849bcc9736cSJeff Kirsher 			<< 8);
1850bcc9736cSJeff Kirsher 		writew(ctrl_addr, hw->io + KS884X_IACR_OFFSET);
1851bcc9736cSJeff Kirsher 		HW_DELAY(hw, KS884X_IACR_OFFSET);
1852bcc9736cSJeff Kirsher 		data = readl(hw->io + KS884X_ACC_DATA_0_OFFSET);
1853bcc9736cSJeff Kirsher 
1854bcc9736cSJeff Kirsher 		hw_restore_intr(hw, interrupt);
1855bcc9736cSJeff Kirsher 
1856bcc9736cSJeff Kirsher 		data &= MIB_PACKET_DROPPED;
1857bcc9736cSJeff Kirsher 		cur = *last;
1858bcc9736cSJeff Kirsher 		if (data != cur) {
1859bcc9736cSJeff Kirsher 			*last = data;
1860bcc9736cSJeff Kirsher 			if (data < cur)
1861bcc9736cSJeff Kirsher 				data += MIB_PACKET_DROPPED + 1;
1862bcc9736cSJeff Kirsher 			data -= cur;
1863bcc9736cSJeff Kirsher 			*cnt += data;
1864bcc9736cSJeff Kirsher 		}
1865bcc9736cSJeff Kirsher 		++last;
1866bcc9736cSJeff Kirsher 		++cnt;
1867bcc9736cSJeff Kirsher 		index -= KS_MIB_PACKET_DROPPED_TX -
1868bcc9736cSJeff Kirsher 			KS_MIB_PACKET_DROPPED_TX_0 + 1;
1869bcc9736cSJeff Kirsher 	} while (index >= KS_MIB_PACKET_DROPPED_TX_0 + port);
1870bcc9736cSJeff Kirsher }
1871bcc9736cSJeff Kirsher 
1872bcc9736cSJeff Kirsher /**
1873bcc9736cSJeff Kirsher  * port_r_cnt - read MIB counters periodically
1874bcc9736cSJeff Kirsher  * @hw: 	The hardware instance.
1875bcc9736cSJeff Kirsher  * @port:	The port index.
1876bcc9736cSJeff Kirsher  *
1877bcc9736cSJeff Kirsher  * This routine is used to read the counters of the port periodically to avoid
1878bcc9736cSJeff Kirsher  * counter overflow.  The hardware should be acquired first before calling this
1879bcc9736cSJeff Kirsher  * routine.
1880bcc9736cSJeff Kirsher  *
1881bcc9736cSJeff Kirsher  * Return non-zero when not all counters not read.
1882bcc9736cSJeff Kirsher  */
1883bcc9736cSJeff Kirsher static int port_r_cnt(struct ksz_hw *hw, int port)
1884bcc9736cSJeff Kirsher {
1885bcc9736cSJeff Kirsher 	struct ksz_port_mib *mib = &hw->port_mib[port];
1886bcc9736cSJeff Kirsher 
1887bcc9736cSJeff Kirsher 	if (mib->mib_start < PORT_COUNTER_NUM)
1888bcc9736cSJeff Kirsher 		while (mib->cnt_ptr < PORT_COUNTER_NUM) {
1889bcc9736cSJeff Kirsher 			port_r_mib_cnt(hw, port, mib->cnt_ptr,
1890bcc9736cSJeff Kirsher 				&mib->counter[mib->cnt_ptr]);
1891bcc9736cSJeff Kirsher 			++mib->cnt_ptr;
1892bcc9736cSJeff Kirsher 		}
1893bcc9736cSJeff Kirsher 	if (hw->mib_cnt > PORT_COUNTER_NUM)
1894bcc9736cSJeff Kirsher 		port_r_mib_pkt(hw, port, mib->dropped,
1895bcc9736cSJeff Kirsher 			&mib->counter[PORT_COUNTER_NUM]);
1896bcc9736cSJeff Kirsher 	mib->cnt_ptr = 0;
1897bcc9736cSJeff Kirsher 	return 0;
1898bcc9736cSJeff Kirsher }
1899bcc9736cSJeff Kirsher 
1900bcc9736cSJeff Kirsher /**
1901bcc9736cSJeff Kirsher  * port_init_cnt - initialize MIB counter values
1902bcc9736cSJeff Kirsher  * @hw: 	The hardware instance.
1903bcc9736cSJeff Kirsher  * @port:	The port index.
1904bcc9736cSJeff Kirsher  *
1905bcc9736cSJeff Kirsher  * This routine is used to initialize all counters to zero if the hardware
1906bcc9736cSJeff Kirsher  * cannot do it after reset.
1907bcc9736cSJeff Kirsher  */
1908bcc9736cSJeff Kirsher static void port_init_cnt(struct ksz_hw *hw, int port)
1909bcc9736cSJeff Kirsher {
1910bcc9736cSJeff Kirsher 	struct ksz_port_mib *mib = &hw->port_mib[port];
1911bcc9736cSJeff Kirsher 
1912bcc9736cSJeff Kirsher 	mib->cnt_ptr = 0;
1913bcc9736cSJeff Kirsher 	if (mib->mib_start < PORT_COUNTER_NUM)
1914bcc9736cSJeff Kirsher 		do {
1915bcc9736cSJeff Kirsher 			port_r_mib_cnt(hw, port, mib->cnt_ptr,
1916bcc9736cSJeff Kirsher 				&mib->counter[mib->cnt_ptr]);
1917bcc9736cSJeff Kirsher 			++mib->cnt_ptr;
1918bcc9736cSJeff Kirsher 		} while (mib->cnt_ptr < PORT_COUNTER_NUM);
1919bcc9736cSJeff Kirsher 	if (hw->mib_cnt > PORT_COUNTER_NUM)
1920bcc9736cSJeff Kirsher 		port_r_mib_pkt(hw, port, mib->dropped,
1921bcc9736cSJeff Kirsher 			&mib->counter[PORT_COUNTER_NUM]);
1922bcc9736cSJeff Kirsher 	memset((void *) mib->counter, 0, sizeof(u64) * TOTAL_PORT_COUNTER_NUM);
1923bcc9736cSJeff Kirsher 	mib->cnt_ptr = 0;
1924bcc9736cSJeff Kirsher }
1925bcc9736cSJeff Kirsher 
1926bcc9736cSJeff Kirsher /*
1927bcc9736cSJeff Kirsher  * Port functions
1928bcc9736cSJeff Kirsher  */
1929bcc9736cSJeff Kirsher 
1930bcc9736cSJeff Kirsher /**
1931bcc9736cSJeff Kirsher  * port_chk - check port register bits
1932bcc9736cSJeff Kirsher  * @hw: 	The hardware instance.
1933bcc9736cSJeff Kirsher  * @port:	The port index.
1934bcc9736cSJeff Kirsher  * @offset:	The offset of the port register.
1935bcc9736cSJeff Kirsher  * @bits:	The data bits to check.
1936bcc9736cSJeff Kirsher  *
1937bcc9736cSJeff Kirsher  * This function checks whether the specified bits of the port register are set
1938bcc9736cSJeff Kirsher  * or not.
1939bcc9736cSJeff Kirsher  *
1940bcc9736cSJeff Kirsher  * Return 0 if the bits are not set.
1941bcc9736cSJeff Kirsher  */
1942bcc9736cSJeff Kirsher static int port_chk(struct ksz_hw *hw, int port, int offset, u16 bits)
1943bcc9736cSJeff Kirsher {
1944bcc9736cSJeff Kirsher 	u32 addr;
1945bcc9736cSJeff Kirsher 	u16 data;
1946bcc9736cSJeff Kirsher 
1947bcc9736cSJeff Kirsher 	PORT_CTRL_ADDR(port, addr);
1948bcc9736cSJeff Kirsher 	addr += offset;
1949bcc9736cSJeff Kirsher 	data = readw(hw->io + addr);
1950bcc9736cSJeff Kirsher 	return (data & bits) == bits;
1951bcc9736cSJeff Kirsher }
1952bcc9736cSJeff Kirsher 
1953bcc9736cSJeff Kirsher /**
1954bcc9736cSJeff Kirsher  * port_cfg - set port register bits
1955bcc9736cSJeff Kirsher  * @hw: 	The hardware instance.
1956bcc9736cSJeff Kirsher  * @port:	The port index.
1957bcc9736cSJeff Kirsher  * @offset:	The offset of the port register.
1958bcc9736cSJeff Kirsher  * @bits:	The data bits to set.
1959bcc9736cSJeff Kirsher  * @set:	The flag indicating whether the bits are to be set or not.
1960bcc9736cSJeff Kirsher  *
1961bcc9736cSJeff Kirsher  * This routine sets or resets the specified bits of the port register.
1962bcc9736cSJeff Kirsher  */
1963bcc9736cSJeff Kirsher static void port_cfg(struct ksz_hw *hw, int port, int offset, u16 bits,
1964bcc9736cSJeff Kirsher 	int set)
1965bcc9736cSJeff Kirsher {
1966bcc9736cSJeff Kirsher 	u32 addr;
1967bcc9736cSJeff Kirsher 	u16 data;
1968bcc9736cSJeff Kirsher 
1969bcc9736cSJeff Kirsher 	PORT_CTRL_ADDR(port, addr);
1970bcc9736cSJeff Kirsher 	addr += offset;
1971bcc9736cSJeff Kirsher 	data = readw(hw->io + addr);
1972bcc9736cSJeff Kirsher 	if (set)
1973bcc9736cSJeff Kirsher 		data |= bits;
1974bcc9736cSJeff Kirsher 	else
1975bcc9736cSJeff Kirsher 		data &= ~bits;
1976bcc9736cSJeff Kirsher 	writew(data, hw->io + addr);
1977bcc9736cSJeff Kirsher }
1978bcc9736cSJeff Kirsher 
1979bcc9736cSJeff Kirsher /**
1980bcc9736cSJeff Kirsher  * port_chk_shift - check port bit
1981bcc9736cSJeff Kirsher  * @hw: 	The hardware instance.
1982bcc9736cSJeff Kirsher  * @port:	The port index.
1983bcc9736cSJeff Kirsher  * @offset:	The offset of the register.
1984bcc9736cSJeff Kirsher  * @shift:	Number of bits to shift.
1985bcc9736cSJeff Kirsher  *
1986bcc9736cSJeff Kirsher  * This function checks whether the specified port is set in the register or
1987bcc9736cSJeff Kirsher  * not.
1988bcc9736cSJeff Kirsher  *
1989bcc9736cSJeff Kirsher  * Return 0 if the port is not set.
1990bcc9736cSJeff Kirsher  */
1991bcc9736cSJeff Kirsher static int port_chk_shift(struct ksz_hw *hw, int port, u32 addr, int shift)
1992bcc9736cSJeff Kirsher {
1993bcc9736cSJeff Kirsher 	u16 data;
1994bcc9736cSJeff Kirsher 	u16 bit = 1 << port;
1995bcc9736cSJeff Kirsher 
1996bcc9736cSJeff Kirsher 	data = readw(hw->io + addr);
1997bcc9736cSJeff Kirsher 	data >>= shift;
1998bcc9736cSJeff Kirsher 	return (data & bit) == bit;
1999bcc9736cSJeff Kirsher }
2000bcc9736cSJeff Kirsher 
2001bcc9736cSJeff Kirsher /**
2002bcc9736cSJeff Kirsher  * port_cfg_shift - set port bit
2003bcc9736cSJeff Kirsher  * @hw: 	The hardware instance.
2004bcc9736cSJeff Kirsher  * @port:	The port index.
2005bcc9736cSJeff Kirsher  * @offset:	The offset of the register.
2006bcc9736cSJeff Kirsher  * @shift:	Number of bits to shift.
2007bcc9736cSJeff Kirsher  * @set:	The flag indicating whether the port is to be set or not.
2008bcc9736cSJeff Kirsher  *
2009bcc9736cSJeff Kirsher  * This routine sets or resets the specified port in the register.
2010bcc9736cSJeff Kirsher  */
2011bcc9736cSJeff Kirsher static void port_cfg_shift(struct ksz_hw *hw, int port, u32 addr, int shift,
2012bcc9736cSJeff Kirsher 	int set)
2013bcc9736cSJeff Kirsher {
2014bcc9736cSJeff Kirsher 	u16 data;
2015bcc9736cSJeff Kirsher 	u16 bits = 1 << port;
2016bcc9736cSJeff Kirsher 
2017bcc9736cSJeff Kirsher 	data = readw(hw->io + addr);
2018bcc9736cSJeff Kirsher 	bits <<= shift;
2019bcc9736cSJeff Kirsher 	if (set)
2020bcc9736cSJeff Kirsher 		data |= bits;
2021bcc9736cSJeff Kirsher 	else
2022bcc9736cSJeff Kirsher 		data &= ~bits;
2023bcc9736cSJeff Kirsher 	writew(data, hw->io + addr);
2024bcc9736cSJeff Kirsher }
2025bcc9736cSJeff Kirsher 
2026bcc9736cSJeff Kirsher /**
2027bcc9736cSJeff Kirsher  * port_r8 - read byte from port register
2028bcc9736cSJeff Kirsher  * @hw: 	The hardware instance.
2029bcc9736cSJeff Kirsher  * @port:	The port index.
2030bcc9736cSJeff Kirsher  * @offset:	The offset of the port register.
2031bcc9736cSJeff Kirsher  * @data:	Buffer to store the data.
2032bcc9736cSJeff Kirsher  *
2033bcc9736cSJeff Kirsher  * This routine reads a byte from the port register.
2034bcc9736cSJeff Kirsher  */
2035bcc9736cSJeff Kirsher static void port_r8(struct ksz_hw *hw, int port, int offset, u8 *data)
2036bcc9736cSJeff Kirsher {
2037bcc9736cSJeff Kirsher 	u32 addr;
2038bcc9736cSJeff Kirsher 
2039bcc9736cSJeff Kirsher 	PORT_CTRL_ADDR(port, addr);
2040bcc9736cSJeff Kirsher 	addr += offset;
2041bcc9736cSJeff Kirsher 	*data = readb(hw->io + addr);
2042bcc9736cSJeff Kirsher }
2043bcc9736cSJeff Kirsher 
2044bcc9736cSJeff Kirsher /**
2045bcc9736cSJeff Kirsher  * port_r16 - read word from port register.
2046bcc9736cSJeff Kirsher  * @hw: 	The hardware instance.
2047bcc9736cSJeff Kirsher  * @port:	The port index.
2048bcc9736cSJeff Kirsher  * @offset:	The offset of the port register.
2049bcc9736cSJeff Kirsher  * @data:	Buffer to store the data.
2050bcc9736cSJeff Kirsher  *
2051bcc9736cSJeff Kirsher  * This routine reads a word from the port register.
2052bcc9736cSJeff Kirsher  */
2053bcc9736cSJeff Kirsher static void port_r16(struct ksz_hw *hw, int port, int offset, u16 *data)
2054bcc9736cSJeff Kirsher {
2055bcc9736cSJeff Kirsher 	u32 addr;
2056bcc9736cSJeff Kirsher 
2057bcc9736cSJeff Kirsher 	PORT_CTRL_ADDR(port, addr);
2058bcc9736cSJeff Kirsher 	addr += offset;
2059bcc9736cSJeff Kirsher 	*data = readw(hw->io + addr);
2060bcc9736cSJeff Kirsher }
2061bcc9736cSJeff Kirsher 
2062bcc9736cSJeff Kirsher /**
2063bcc9736cSJeff Kirsher  * port_w16 - write word to port register.
2064bcc9736cSJeff Kirsher  * @hw: 	The hardware instance.
2065bcc9736cSJeff Kirsher  * @port:	The port index.
2066bcc9736cSJeff Kirsher  * @offset:	The offset of the port register.
2067bcc9736cSJeff Kirsher  * @data:	Data to write.
2068bcc9736cSJeff Kirsher  *
2069bcc9736cSJeff Kirsher  * This routine writes a word to the port register.
2070bcc9736cSJeff Kirsher  */
2071bcc9736cSJeff Kirsher static void port_w16(struct ksz_hw *hw, int port, int offset, u16 data)
2072bcc9736cSJeff Kirsher {
2073bcc9736cSJeff Kirsher 	u32 addr;
2074bcc9736cSJeff Kirsher 
2075bcc9736cSJeff Kirsher 	PORT_CTRL_ADDR(port, addr);
2076bcc9736cSJeff Kirsher 	addr += offset;
2077bcc9736cSJeff Kirsher 	writew(data, hw->io + addr);
2078bcc9736cSJeff Kirsher }
2079bcc9736cSJeff Kirsher 
2080bcc9736cSJeff Kirsher /**
2081bcc9736cSJeff Kirsher  * sw_chk - check switch register bits
2082bcc9736cSJeff Kirsher  * @hw: 	The hardware instance.
2083bcc9736cSJeff Kirsher  * @addr:	The address of the switch register.
2084bcc9736cSJeff Kirsher  * @bits:	The data bits to check.
2085bcc9736cSJeff Kirsher  *
2086bcc9736cSJeff Kirsher  * This function checks whether the specified bits of the switch register are
2087bcc9736cSJeff Kirsher  * set or not.
2088bcc9736cSJeff Kirsher  *
2089bcc9736cSJeff Kirsher  * Return 0 if the bits are not set.
2090bcc9736cSJeff Kirsher  */
2091bcc9736cSJeff Kirsher static int sw_chk(struct ksz_hw *hw, u32 addr, u16 bits)
2092bcc9736cSJeff Kirsher {
2093bcc9736cSJeff Kirsher 	u16 data;
2094bcc9736cSJeff Kirsher 
2095bcc9736cSJeff Kirsher 	data = readw(hw->io + addr);
2096bcc9736cSJeff Kirsher 	return (data & bits) == bits;
2097bcc9736cSJeff Kirsher }
2098bcc9736cSJeff Kirsher 
2099bcc9736cSJeff Kirsher /**
2100bcc9736cSJeff Kirsher  * sw_cfg - set switch register bits
2101bcc9736cSJeff Kirsher  * @hw: 	The hardware instance.
2102bcc9736cSJeff Kirsher  * @addr:	The address of the switch register.
2103bcc9736cSJeff Kirsher  * @bits:	The data bits to set.
2104bcc9736cSJeff Kirsher  * @set:	The flag indicating whether the bits are to be set or not.
2105bcc9736cSJeff Kirsher  *
2106bcc9736cSJeff Kirsher  * This function sets or resets the specified bits of the switch register.
2107bcc9736cSJeff Kirsher  */
2108bcc9736cSJeff Kirsher static void sw_cfg(struct ksz_hw *hw, u32 addr, u16 bits, int set)
2109bcc9736cSJeff Kirsher {
2110bcc9736cSJeff Kirsher 	u16 data;
2111bcc9736cSJeff Kirsher 
2112bcc9736cSJeff Kirsher 	data = readw(hw->io + addr);
2113bcc9736cSJeff Kirsher 	if (set)
2114bcc9736cSJeff Kirsher 		data |= bits;
2115bcc9736cSJeff Kirsher 	else
2116bcc9736cSJeff Kirsher 		data &= ~bits;
2117bcc9736cSJeff Kirsher 	writew(data, hw->io + addr);
2118bcc9736cSJeff Kirsher }
2119bcc9736cSJeff Kirsher 
2120bcc9736cSJeff Kirsher /* Bandwidth */
2121bcc9736cSJeff Kirsher 
2122bcc9736cSJeff Kirsher static inline void port_cfg_broad_storm(struct ksz_hw *hw, int p, int set)
2123bcc9736cSJeff Kirsher {
2124bcc9736cSJeff Kirsher 	port_cfg(hw, p,
2125bcc9736cSJeff Kirsher 		KS8842_PORT_CTRL_1_OFFSET, PORT_BROADCAST_STORM, set);
2126bcc9736cSJeff Kirsher }
2127bcc9736cSJeff Kirsher 
2128bcc9736cSJeff Kirsher static inline int port_chk_broad_storm(struct ksz_hw *hw, int p)
2129bcc9736cSJeff Kirsher {
2130bcc9736cSJeff Kirsher 	return port_chk(hw, p,
2131bcc9736cSJeff Kirsher 		KS8842_PORT_CTRL_1_OFFSET, PORT_BROADCAST_STORM);
2132bcc9736cSJeff Kirsher }
2133bcc9736cSJeff Kirsher 
2134bcc9736cSJeff Kirsher /* Driver set switch broadcast storm protection at 10% rate. */
2135bcc9736cSJeff Kirsher #define BROADCAST_STORM_PROTECTION_RATE	10
2136bcc9736cSJeff Kirsher 
2137bcc9736cSJeff Kirsher /* 148,800 frames * 67 ms / 100 */
2138bcc9736cSJeff Kirsher #define BROADCAST_STORM_VALUE		9969
2139bcc9736cSJeff Kirsher 
2140bcc9736cSJeff Kirsher /**
2141bcc9736cSJeff Kirsher  * sw_cfg_broad_storm - configure broadcast storm threshold
2142bcc9736cSJeff Kirsher  * @hw: 	The hardware instance.
2143bcc9736cSJeff Kirsher  * @percent:	Broadcast storm threshold in percent of transmit rate.
2144bcc9736cSJeff Kirsher  *
2145bcc9736cSJeff Kirsher  * This routine configures the broadcast storm threshold of the switch.
2146bcc9736cSJeff Kirsher  */
2147bcc9736cSJeff Kirsher static void sw_cfg_broad_storm(struct ksz_hw *hw, u8 percent)
2148bcc9736cSJeff Kirsher {
2149bcc9736cSJeff Kirsher 	u16 data;
2150bcc9736cSJeff Kirsher 	u32 value = ((u32) BROADCAST_STORM_VALUE * (u32) percent / 100);
2151bcc9736cSJeff Kirsher 
2152bcc9736cSJeff Kirsher 	if (value > BROADCAST_STORM_RATE)
2153bcc9736cSJeff Kirsher 		value = BROADCAST_STORM_RATE;
2154bcc9736cSJeff Kirsher 
2155bcc9736cSJeff Kirsher 	data = readw(hw->io + KS8842_SWITCH_CTRL_3_OFFSET);
2156bcc9736cSJeff Kirsher 	data &= ~(BROADCAST_STORM_RATE_LO | BROADCAST_STORM_RATE_HI);
2157bcc9736cSJeff Kirsher 	data |= ((value & 0x00FF) << 8) | ((value & 0xFF00) >> 8);
2158bcc9736cSJeff Kirsher 	writew(data, hw->io + KS8842_SWITCH_CTRL_3_OFFSET);
2159bcc9736cSJeff Kirsher }
2160bcc9736cSJeff Kirsher 
2161bcc9736cSJeff Kirsher /**
2162bcc9736cSJeff Kirsher  * sw_get_board_storm - get broadcast storm threshold
2163bcc9736cSJeff Kirsher  * @hw: 	The hardware instance.
2164bcc9736cSJeff Kirsher  * @percent:	Buffer to store the broadcast storm threshold percentage.
2165bcc9736cSJeff Kirsher  *
2166bcc9736cSJeff Kirsher  * This routine retrieves the broadcast storm threshold of the switch.
2167bcc9736cSJeff Kirsher  */
2168bcc9736cSJeff Kirsher static void sw_get_broad_storm(struct ksz_hw *hw, u8 *percent)
2169bcc9736cSJeff Kirsher {
2170bcc9736cSJeff Kirsher 	int num;
2171bcc9736cSJeff Kirsher 	u16 data;
2172bcc9736cSJeff Kirsher 
2173bcc9736cSJeff Kirsher 	data = readw(hw->io + KS8842_SWITCH_CTRL_3_OFFSET);
2174bcc9736cSJeff Kirsher 	num = (data & BROADCAST_STORM_RATE_HI);
2175bcc9736cSJeff Kirsher 	num <<= 8;
2176bcc9736cSJeff Kirsher 	num |= (data & BROADCAST_STORM_RATE_LO) >> 8;
2177bcc9736cSJeff Kirsher 	num = (num * 100 + BROADCAST_STORM_VALUE / 2) / BROADCAST_STORM_VALUE;
2178bcc9736cSJeff Kirsher 	*percent = (u8) num;
2179bcc9736cSJeff Kirsher }
2180bcc9736cSJeff Kirsher 
2181bcc9736cSJeff Kirsher /**
2182bcc9736cSJeff Kirsher  * sw_dis_broad_storm - disable broadstorm
2183bcc9736cSJeff Kirsher  * @hw: 	The hardware instance.
2184bcc9736cSJeff Kirsher  * @port:	The port index.
2185bcc9736cSJeff Kirsher  *
2186bcc9736cSJeff Kirsher  * This routine disables the broadcast storm limit function of the switch.
2187bcc9736cSJeff Kirsher  */
2188bcc9736cSJeff Kirsher static void sw_dis_broad_storm(struct ksz_hw *hw, int port)
2189bcc9736cSJeff Kirsher {
2190bcc9736cSJeff Kirsher 	port_cfg_broad_storm(hw, port, 0);
2191bcc9736cSJeff Kirsher }
2192bcc9736cSJeff Kirsher 
2193bcc9736cSJeff Kirsher /**
2194bcc9736cSJeff Kirsher  * sw_ena_broad_storm - enable broadcast storm
2195bcc9736cSJeff Kirsher  * @hw: 	The hardware instance.
2196bcc9736cSJeff Kirsher  * @port:	The port index.
2197bcc9736cSJeff Kirsher  *
2198bcc9736cSJeff Kirsher  * This routine enables the broadcast storm limit function of the switch.
2199bcc9736cSJeff Kirsher  */
2200bcc9736cSJeff Kirsher static void sw_ena_broad_storm(struct ksz_hw *hw, int port)
2201bcc9736cSJeff Kirsher {
2202bcc9736cSJeff Kirsher 	sw_cfg_broad_storm(hw, hw->ksz_switch->broad_per);
2203bcc9736cSJeff Kirsher 	port_cfg_broad_storm(hw, port, 1);
2204bcc9736cSJeff Kirsher }
2205bcc9736cSJeff Kirsher 
2206bcc9736cSJeff Kirsher /**
2207bcc9736cSJeff Kirsher  * sw_init_broad_storm - initialize broadcast storm
2208bcc9736cSJeff Kirsher  * @hw: 	The hardware instance.
2209bcc9736cSJeff Kirsher  *
2210bcc9736cSJeff Kirsher  * This routine initializes the broadcast storm limit function of the switch.
2211bcc9736cSJeff Kirsher  */
2212bcc9736cSJeff Kirsher static void sw_init_broad_storm(struct ksz_hw *hw)
2213bcc9736cSJeff Kirsher {
2214bcc9736cSJeff Kirsher 	int port;
2215bcc9736cSJeff Kirsher 
2216bcc9736cSJeff Kirsher 	hw->ksz_switch->broad_per = 1;
2217bcc9736cSJeff Kirsher 	sw_cfg_broad_storm(hw, hw->ksz_switch->broad_per);
2218bcc9736cSJeff Kirsher 	for (port = 0; port < TOTAL_PORT_NUM; port++)
2219bcc9736cSJeff Kirsher 		sw_dis_broad_storm(hw, port);
2220bcc9736cSJeff Kirsher 	sw_cfg(hw, KS8842_SWITCH_CTRL_2_OFFSET, MULTICAST_STORM_DISABLE, 1);
2221bcc9736cSJeff Kirsher }
2222bcc9736cSJeff Kirsher 
2223bcc9736cSJeff Kirsher /**
2224bcc9736cSJeff Kirsher  * hw_cfg_broad_storm - configure broadcast storm
2225bcc9736cSJeff Kirsher  * @hw: 	The hardware instance.
2226bcc9736cSJeff Kirsher  * @percent:	Broadcast storm threshold in percent of transmit rate.
2227bcc9736cSJeff Kirsher  *
2228bcc9736cSJeff Kirsher  * This routine configures the broadcast storm threshold of the switch.
2229bcc9736cSJeff Kirsher  * It is called by user functions.  The hardware should be acquired first.
2230bcc9736cSJeff Kirsher  */
2231bcc9736cSJeff Kirsher static void hw_cfg_broad_storm(struct ksz_hw *hw, u8 percent)
2232bcc9736cSJeff Kirsher {
2233bcc9736cSJeff Kirsher 	if (percent > 100)
2234bcc9736cSJeff Kirsher 		percent = 100;
2235bcc9736cSJeff Kirsher 
2236bcc9736cSJeff Kirsher 	sw_cfg_broad_storm(hw, percent);
2237bcc9736cSJeff Kirsher 	sw_get_broad_storm(hw, &percent);
2238bcc9736cSJeff Kirsher 	hw->ksz_switch->broad_per = percent;
2239bcc9736cSJeff Kirsher }
2240bcc9736cSJeff Kirsher 
2241bcc9736cSJeff Kirsher /**
2242bcc9736cSJeff Kirsher  * sw_dis_prio_rate - disable switch priority rate
2243bcc9736cSJeff Kirsher  * @hw: 	The hardware instance.
2244bcc9736cSJeff Kirsher  * @port:	The port index.
2245bcc9736cSJeff Kirsher  *
2246bcc9736cSJeff Kirsher  * This routine disables the priority rate function of the switch.
2247bcc9736cSJeff Kirsher  */
2248bcc9736cSJeff Kirsher static void sw_dis_prio_rate(struct ksz_hw *hw, int port)
2249bcc9736cSJeff Kirsher {
2250bcc9736cSJeff Kirsher 	u32 addr;
2251bcc9736cSJeff Kirsher 
2252bcc9736cSJeff Kirsher 	PORT_CTRL_ADDR(port, addr);
2253bcc9736cSJeff Kirsher 	addr += KS8842_PORT_IN_RATE_OFFSET;
2254bcc9736cSJeff Kirsher 	writel(0, hw->io + addr);
2255bcc9736cSJeff Kirsher }
2256bcc9736cSJeff Kirsher 
2257bcc9736cSJeff Kirsher /**
2258bcc9736cSJeff Kirsher  * sw_init_prio_rate - initialize switch prioirty rate
2259bcc9736cSJeff Kirsher  * @hw: 	The hardware instance.
2260bcc9736cSJeff Kirsher  *
2261bcc9736cSJeff Kirsher  * This routine initializes the priority rate function of the switch.
2262bcc9736cSJeff Kirsher  */
2263bcc9736cSJeff Kirsher static void sw_init_prio_rate(struct ksz_hw *hw)
2264bcc9736cSJeff Kirsher {
2265bcc9736cSJeff Kirsher 	int port;
2266bcc9736cSJeff Kirsher 	int prio;
2267bcc9736cSJeff Kirsher 	struct ksz_switch *sw = hw->ksz_switch;
2268bcc9736cSJeff Kirsher 
2269bcc9736cSJeff Kirsher 	for (port = 0; port < TOTAL_PORT_NUM; port++) {
2270bcc9736cSJeff Kirsher 		for (prio = 0; prio < PRIO_QUEUES; prio++) {
2271bcc9736cSJeff Kirsher 			sw->port_cfg[port].rx_rate[prio] =
2272bcc9736cSJeff Kirsher 			sw->port_cfg[port].tx_rate[prio] = 0;
2273bcc9736cSJeff Kirsher 		}
2274bcc9736cSJeff Kirsher 		sw_dis_prio_rate(hw, port);
2275bcc9736cSJeff Kirsher 	}
2276bcc9736cSJeff Kirsher }
2277bcc9736cSJeff Kirsher 
2278bcc9736cSJeff Kirsher /* Communication */
2279bcc9736cSJeff Kirsher 
2280bcc9736cSJeff Kirsher static inline void port_cfg_back_pressure(struct ksz_hw *hw, int p, int set)
2281bcc9736cSJeff Kirsher {
2282bcc9736cSJeff Kirsher 	port_cfg(hw, p,
2283bcc9736cSJeff Kirsher 		KS8842_PORT_CTRL_2_OFFSET, PORT_BACK_PRESSURE, set);
2284bcc9736cSJeff Kirsher }
2285bcc9736cSJeff Kirsher 
2286bcc9736cSJeff Kirsher static inline void port_cfg_force_flow_ctrl(struct ksz_hw *hw, int p, int set)
2287bcc9736cSJeff Kirsher {
2288bcc9736cSJeff Kirsher 	port_cfg(hw, p,
2289bcc9736cSJeff Kirsher 		KS8842_PORT_CTRL_2_OFFSET, PORT_FORCE_FLOW_CTRL, set);
2290bcc9736cSJeff Kirsher }
2291bcc9736cSJeff Kirsher 
2292bcc9736cSJeff Kirsher static inline int port_chk_back_pressure(struct ksz_hw *hw, int p)
2293bcc9736cSJeff Kirsher {
2294bcc9736cSJeff Kirsher 	return port_chk(hw, p,
2295bcc9736cSJeff Kirsher 		KS8842_PORT_CTRL_2_OFFSET, PORT_BACK_PRESSURE);
2296bcc9736cSJeff Kirsher }
2297bcc9736cSJeff Kirsher 
2298bcc9736cSJeff Kirsher static inline int port_chk_force_flow_ctrl(struct ksz_hw *hw, int p)
2299bcc9736cSJeff Kirsher {
2300bcc9736cSJeff Kirsher 	return port_chk(hw, p,
2301bcc9736cSJeff Kirsher 		KS8842_PORT_CTRL_2_OFFSET, PORT_FORCE_FLOW_CTRL);
2302bcc9736cSJeff Kirsher }
2303bcc9736cSJeff Kirsher 
2304bcc9736cSJeff Kirsher /* Spanning Tree */
2305bcc9736cSJeff Kirsher 
2306bcc9736cSJeff Kirsher static inline void port_cfg_dis_learn(struct ksz_hw *hw, int p, int set)
2307bcc9736cSJeff Kirsher {
2308bcc9736cSJeff Kirsher 	port_cfg(hw, p,
2309bcc9736cSJeff Kirsher 		KS8842_PORT_CTRL_2_OFFSET, PORT_LEARN_DISABLE, set);
2310bcc9736cSJeff Kirsher }
2311bcc9736cSJeff Kirsher 
2312bcc9736cSJeff Kirsher static inline void port_cfg_rx(struct ksz_hw *hw, int p, int set)
2313bcc9736cSJeff Kirsher {
2314bcc9736cSJeff Kirsher 	port_cfg(hw, p,
2315bcc9736cSJeff Kirsher 		KS8842_PORT_CTRL_2_OFFSET, PORT_RX_ENABLE, set);
2316bcc9736cSJeff Kirsher }
2317bcc9736cSJeff Kirsher 
2318bcc9736cSJeff Kirsher static inline void port_cfg_tx(struct ksz_hw *hw, int p, int set)
2319bcc9736cSJeff Kirsher {
2320bcc9736cSJeff Kirsher 	port_cfg(hw, p,
2321bcc9736cSJeff Kirsher 		KS8842_PORT_CTRL_2_OFFSET, PORT_TX_ENABLE, set);
2322bcc9736cSJeff Kirsher }
2323bcc9736cSJeff Kirsher 
2324bcc9736cSJeff Kirsher static inline void sw_cfg_fast_aging(struct ksz_hw *hw, int set)
2325bcc9736cSJeff Kirsher {
2326bcc9736cSJeff Kirsher 	sw_cfg(hw, KS8842_SWITCH_CTRL_1_OFFSET, SWITCH_FAST_AGING, set);
2327bcc9736cSJeff Kirsher }
2328bcc9736cSJeff Kirsher 
2329bcc9736cSJeff Kirsher static inline void sw_flush_dyn_mac_table(struct ksz_hw *hw)
2330bcc9736cSJeff Kirsher {
2331bcc9736cSJeff Kirsher 	if (!(hw->overrides & FAST_AGING)) {
2332bcc9736cSJeff Kirsher 		sw_cfg_fast_aging(hw, 1);
2333bcc9736cSJeff Kirsher 		mdelay(1);
2334bcc9736cSJeff Kirsher 		sw_cfg_fast_aging(hw, 0);
2335bcc9736cSJeff Kirsher 	}
2336bcc9736cSJeff Kirsher }
2337bcc9736cSJeff Kirsher 
2338bcc9736cSJeff Kirsher /* VLAN */
2339bcc9736cSJeff Kirsher 
2340bcc9736cSJeff Kirsher static inline void port_cfg_ins_tag(struct ksz_hw *hw, int p, int insert)
2341bcc9736cSJeff Kirsher {
2342bcc9736cSJeff Kirsher 	port_cfg(hw, p,
2343bcc9736cSJeff Kirsher 		KS8842_PORT_CTRL_1_OFFSET, PORT_INSERT_TAG, insert);
2344bcc9736cSJeff Kirsher }
2345bcc9736cSJeff Kirsher 
2346bcc9736cSJeff Kirsher static inline void port_cfg_rmv_tag(struct ksz_hw *hw, int p, int remove)
2347bcc9736cSJeff Kirsher {
2348bcc9736cSJeff Kirsher 	port_cfg(hw, p,
2349bcc9736cSJeff Kirsher 		KS8842_PORT_CTRL_1_OFFSET, PORT_REMOVE_TAG, remove);
2350bcc9736cSJeff Kirsher }
2351bcc9736cSJeff Kirsher 
2352bcc9736cSJeff Kirsher static inline int port_chk_ins_tag(struct ksz_hw *hw, int p)
2353bcc9736cSJeff Kirsher {
2354bcc9736cSJeff Kirsher 	return port_chk(hw, p,
2355bcc9736cSJeff Kirsher 		KS8842_PORT_CTRL_1_OFFSET, PORT_INSERT_TAG);
2356bcc9736cSJeff Kirsher }
2357bcc9736cSJeff Kirsher 
2358bcc9736cSJeff Kirsher static inline int port_chk_rmv_tag(struct ksz_hw *hw, int p)
2359bcc9736cSJeff Kirsher {
2360bcc9736cSJeff Kirsher 	return port_chk(hw, p,
2361bcc9736cSJeff Kirsher 		KS8842_PORT_CTRL_1_OFFSET, PORT_REMOVE_TAG);
2362bcc9736cSJeff Kirsher }
2363bcc9736cSJeff Kirsher 
2364bcc9736cSJeff Kirsher static inline void port_cfg_dis_non_vid(struct ksz_hw *hw, int p, int set)
2365bcc9736cSJeff Kirsher {
2366bcc9736cSJeff Kirsher 	port_cfg(hw, p,
2367bcc9736cSJeff Kirsher 		KS8842_PORT_CTRL_2_OFFSET, PORT_DISCARD_NON_VID, set);
2368bcc9736cSJeff Kirsher }
2369bcc9736cSJeff Kirsher 
2370bcc9736cSJeff Kirsher static inline void port_cfg_in_filter(struct ksz_hw *hw, int p, int set)
2371bcc9736cSJeff Kirsher {
2372bcc9736cSJeff Kirsher 	port_cfg(hw, p,
2373bcc9736cSJeff Kirsher 		KS8842_PORT_CTRL_2_OFFSET, PORT_INGRESS_VLAN_FILTER, set);
2374bcc9736cSJeff Kirsher }
2375bcc9736cSJeff Kirsher 
2376bcc9736cSJeff Kirsher static inline int port_chk_dis_non_vid(struct ksz_hw *hw, int p)
2377bcc9736cSJeff Kirsher {
2378bcc9736cSJeff Kirsher 	return port_chk(hw, p,
2379bcc9736cSJeff Kirsher 		KS8842_PORT_CTRL_2_OFFSET, PORT_DISCARD_NON_VID);
2380bcc9736cSJeff Kirsher }
2381bcc9736cSJeff Kirsher 
2382bcc9736cSJeff Kirsher static inline int port_chk_in_filter(struct ksz_hw *hw, int p)
2383bcc9736cSJeff Kirsher {
2384bcc9736cSJeff Kirsher 	return port_chk(hw, p,
2385bcc9736cSJeff Kirsher 		KS8842_PORT_CTRL_2_OFFSET, PORT_INGRESS_VLAN_FILTER);
2386bcc9736cSJeff Kirsher }
2387bcc9736cSJeff Kirsher 
2388bcc9736cSJeff Kirsher /* Mirroring */
2389bcc9736cSJeff Kirsher 
2390bcc9736cSJeff Kirsher static inline void port_cfg_mirror_sniffer(struct ksz_hw *hw, int p, int set)
2391bcc9736cSJeff Kirsher {
2392bcc9736cSJeff Kirsher 	port_cfg(hw, p,
2393bcc9736cSJeff Kirsher 		KS8842_PORT_CTRL_2_OFFSET, PORT_MIRROR_SNIFFER, set);
2394bcc9736cSJeff Kirsher }
2395bcc9736cSJeff Kirsher 
2396bcc9736cSJeff Kirsher static inline void port_cfg_mirror_rx(struct ksz_hw *hw, int p, int set)
2397bcc9736cSJeff Kirsher {
2398bcc9736cSJeff Kirsher 	port_cfg(hw, p,
2399bcc9736cSJeff Kirsher 		KS8842_PORT_CTRL_2_OFFSET, PORT_MIRROR_RX, set);
2400bcc9736cSJeff Kirsher }
2401bcc9736cSJeff Kirsher 
2402bcc9736cSJeff Kirsher static inline void port_cfg_mirror_tx(struct ksz_hw *hw, int p, int set)
2403bcc9736cSJeff Kirsher {
2404bcc9736cSJeff Kirsher 	port_cfg(hw, p,
2405bcc9736cSJeff Kirsher 		KS8842_PORT_CTRL_2_OFFSET, PORT_MIRROR_TX, set);
2406bcc9736cSJeff Kirsher }
2407bcc9736cSJeff Kirsher 
2408bcc9736cSJeff Kirsher static inline void sw_cfg_mirror_rx_tx(struct ksz_hw *hw, int set)
2409bcc9736cSJeff Kirsher {
2410bcc9736cSJeff Kirsher 	sw_cfg(hw, KS8842_SWITCH_CTRL_2_OFFSET, SWITCH_MIRROR_RX_TX, set);
2411bcc9736cSJeff Kirsher }
2412bcc9736cSJeff Kirsher 
2413bcc9736cSJeff Kirsher static void sw_init_mirror(struct ksz_hw *hw)
2414bcc9736cSJeff Kirsher {
2415bcc9736cSJeff Kirsher 	int port;
2416bcc9736cSJeff Kirsher 
2417bcc9736cSJeff Kirsher 	for (port = 0; port < TOTAL_PORT_NUM; port++) {
2418bcc9736cSJeff Kirsher 		port_cfg_mirror_sniffer(hw, port, 0);
2419bcc9736cSJeff Kirsher 		port_cfg_mirror_rx(hw, port, 0);
2420bcc9736cSJeff Kirsher 		port_cfg_mirror_tx(hw, port, 0);
2421bcc9736cSJeff Kirsher 	}
2422bcc9736cSJeff Kirsher 	sw_cfg_mirror_rx_tx(hw, 0);
2423bcc9736cSJeff Kirsher }
2424bcc9736cSJeff Kirsher 
2425bcc9736cSJeff Kirsher static inline void sw_cfg_unk_def_deliver(struct ksz_hw *hw, int set)
2426bcc9736cSJeff Kirsher {
2427bcc9736cSJeff Kirsher 	sw_cfg(hw, KS8842_SWITCH_CTRL_7_OFFSET,
2428bcc9736cSJeff Kirsher 		SWITCH_UNK_DEF_PORT_ENABLE, set);
2429bcc9736cSJeff Kirsher }
2430bcc9736cSJeff Kirsher 
2431bcc9736cSJeff Kirsher static inline int sw_cfg_chk_unk_def_deliver(struct ksz_hw *hw)
2432bcc9736cSJeff Kirsher {
2433bcc9736cSJeff Kirsher 	return sw_chk(hw, KS8842_SWITCH_CTRL_7_OFFSET,
2434bcc9736cSJeff Kirsher 		SWITCH_UNK_DEF_PORT_ENABLE);
2435bcc9736cSJeff Kirsher }
2436bcc9736cSJeff Kirsher 
2437bcc9736cSJeff Kirsher static inline void sw_cfg_unk_def_port(struct ksz_hw *hw, int port, int set)
2438bcc9736cSJeff Kirsher {
2439bcc9736cSJeff Kirsher 	port_cfg_shift(hw, port, KS8842_SWITCH_CTRL_7_OFFSET, 0, set);
2440bcc9736cSJeff Kirsher }
2441bcc9736cSJeff Kirsher 
2442bcc9736cSJeff Kirsher static inline int sw_chk_unk_def_port(struct ksz_hw *hw, int port)
2443bcc9736cSJeff Kirsher {
2444bcc9736cSJeff Kirsher 	return port_chk_shift(hw, port, KS8842_SWITCH_CTRL_7_OFFSET, 0);
2445bcc9736cSJeff Kirsher }
2446bcc9736cSJeff Kirsher 
2447bcc9736cSJeff Kirsher /* Priority */
2448bcc9736cSJeff Kirsher 
2449bcc9736cSJeff Kirsher static inline void port_cfg_diffserv(struct ksz_hw *hw, int p, int set)
2450bcc9736cSJeff Kirsher {
2451bcc9736cSJeff Kirsher 	port_cfg(hw, p,
2452bcc9736cSJeff Kirsher 		KS8842_PORT_CTRL_1_OFFSET, PORT_DIFFSERV_ENABLE, set);
2453bcc9736cSJeff Kirsher }
2454bcc9736cSJeff Kirsher 
2455bcc9736cSJeff Kirsher static inline void port_cfg_802_1p(struct ksz_hw *hw, int p, int set)
2456bcc9736cSJeff Kirsher {
2457bcc9736cSJeff Kirsher 	port_cfg(hw, p,
2458bcc9736cSJeff Kirsher 		KS8842_PORT_CTRL_1_OFFSET, PORT_802_1P_ENABLE, set);
2459bcc9736cSJeff Kirsher }
2460bcc9736cSJeff Kirsher 
2461bcc9736cSJeff Kirsher static inline void port_cfg_replace_vid(struct ksz_hw *hw, int p, int set)
2462bcc9736cSJeff Kirsher {
2463bcc9736cSJeff Kirsher 	port_cfg(hw, p,
2464bcc9736cSJeff Kirsher 		KS8842_PORT_CTRL_2_OFFSET, PORT_USER_PRIORITY_CEILING, set);
2465bcc9736cSJeff Kirsher }
2466bcc9736cSJeff Kirsher 
2467bcc9736cSJeff Kirsher static inline void port_cfg_prio(struct ksz_hw *hw, int p, int set)
2468bcc9736cSJeff Kirsher {
2469bcc9736cSJeff Kirsher 	port_cfg(hw, p,
2470bcc9736cSJeff Kirsher 		KS8842_PORT_CTRL_1_OFFSET, PORT_PRIO_QUEUE_ENABLE, set);
2471bcc9736cSJeff Kirsher }
2472bcc9736cSJeff Kirsher 
2473bcc9736cSJeff Kirsher static inline int port_chk_diffserv(struct ksz_hw *hw, int p)
2474bcc9736cSJeff Kirsher {
2475bcc9736cSJeff Kirsher 	return port_chk(hw, p,
2476bcc9736cSJeff Kirsher 		KS8842_PORT_CTRL_1_OFFSET, PORT_DIFFSERV_ENABLE);
2477bcc9736cSJeff Kirsher }
2478bcc9736cSJeff Kirsher 
2479bcc9736cSJeff Kirsher static inline int port_chk_802_1p(struct ksz_hw *hw, int p)
2480bcc9736cSJeff Kirsher {
2481bcc9736cSJeff Kirsher 	return port_chk(hw, p,
2482bcc9736cSJeff Kirsher 		KS8842_PORT_CTRL_1_OFFSET, PORT_802_1P_ENABLE);
2483bcc9736cSJeff Kirsher }
2484bcc9736cSJeff Kirsher 
2485bcc9736cSJeff Kirsher static inline int port_chk_replace_vid(struct ksz_hw *hw, int p)
2486bcc9736cSJeff Kirsher {
2487bcc9736cSJeff Kirsher 	return port_chk(hw, p,
2488bcc9736cSJeff Kirsher 		KS8842_PORT_CTRL_2_OFFSET, PORT_USER_PRIORITY_CEILING);
2489bcc9736cSJeff Kirsher }
2490bcc9736cSJeff Kirsher 
2491bcc9736cSJeff Kirsher static inline int port_chk_prio(struct ksz_hw *hw, int p)
2492bcc9736cSJeff Kirsher {
2493bcc9736cSJeff Kirsher 	return port_chk(hw, p,
2494bcc9736cSJeff Kirsher 		KS8842_PORT_CTRL_1_OFFSET, PORT_PRIO_QUEUE_ENABLE);
2495bcc9736cSJeff Kirsher }
2496bcc9736cSJeff Kirsher 
2497bcc9736cSJeff Kirsher /**
2498bcc9736cSJeff Kirsher  * sw_dis_diffserv - disable switch DiffServ priority
2499bcc9736cSJeff Kirsher  * @hw: 	The hardware instance.
2500bcc9736cSJeff Kirsher  * @port:	The port index.
2501bcc9736cSJeff Kirsher  *
2502bcc9736cSJeff Kirsher  * This routine disables the DiffServ priority function of the switch.
2503bcc9736cSJeff Kirsher  */
2504bcc9736cSJeff Kirsher static void sw_dis_diffserv(struct ksz_hw *hw, int port)
2505bcc9736cSJeff Kirsher {
2506bcc9736cSJeff Kirsher 	port_cfg_diffserv(hw, port, 0);
2507bcc9736cSJeff Kirsher }
2508bcc9736cSJeff Kirsher 
2509bcc9736cSJeff Kirsher /**
2510bcc9736cSJeff Kirsher  * sw_dis_802_1p - disable switch 802.1p priority
2511bcc9736cSJeff Kirsher  * @hw: 	The hardware instance.
2512bcc9736cSJeff Kirsher  * @port:	The port index.
2513bcc9736cSJeff Kirsher  *
2514bcc9736cSJeff Kirsher  * This routine disables the 802.1p priority function of the switch.
2515bcc9736cSJeff Kirsher  */
2516bcc9736cSJeff Kirsher static void sw_dis_802_1p(struct ksz_hw *hw, int port)
2517bcc9736cSJeff Kirsher {
2518bcc9736cSJeff Kirsher 	port_cfg_802_1p(hw, port, 0);
2519bcc9736cSJeff Kirsher }
2520bcc9736cSJeff Kirsher 
2521bcc9736cSJeff Kirsher /**
2522bcc9736cSJeff Kirsher  * sw_cfg_replace_null_vid -
2523bcc9736cSJeff Kirsher  * @hw: 	The hardware instance.
2524bcc9736cSJeff Kirsher  * @set:	The flag to disable or enable.
2525bcc9736cSJeff Kirsher  *
2526bcc9736cSJeff Kirsher  */
2527bcc9736cSJeff Kirsher static void sw_cfg_replace_null_vid(struct ksz_hw *hw, int set)
2528bcc9736cSJeff Kirsher {
2529bcc9736cSJeff Kirsher 	sw_cfg(hw, KS8842_SWITCH_CTRL_3_OFFSET, SWITCH_REPLACE_NULL_VID, set);
2530bcc9736cSJeff Kirsher }
2531bcc9736cSJeff Kirsher 
2532bcc9736cSJeff Kirsher /**
2533bcc9736cSJeff Kirsher  * sw_cfg_replace_vid - enable switch 802.10 priority re-mapping
2534bcc9736cSJeff Kirsher  * @hw: 	The hardware instance.
2535bcc9736cSJeff Kirsher  * @port:	The port index.
2536bcc9736cSJeff Kirsher  * @set:	The flag to disable or enable.
2537bcc9736cSJeff Kirsher  *
2538bcc9736cSJeff Kirsher  * This routine enables the 802.1p priority re-mapping function of the switch.
2539bcc9736cSJeff Kirsher  * That allows 802.1p priority field to be replaced with the port's default
2540bcc9736cSJeff Kirsher  * tag's priority value if the ingress packet's 802.1p priority has a higher
2541bcc9736cSJeff Kirsher  * priority than port's default tag's priority.
2542bcc9736cSJeff Kirsher  */
2543bcc9736cSJeff Kirsher static void sw_cfg_replace_vid(struct ksz_hw *hw, int port, int set)
2544bcc9736cSJeff Kirsher {
2545bcc9736cSJeff Kirsher 	port_cfg_replace_vid(hw, port, set);
2546bcc9736cSJeff Kirsher }
2547bcc9736cSJeff Kirsher 
2548bcc9736cSJeff Kirsher /**
2549bcc9736cSJeff Kirsher  * sw_cfg_port_based - configure switch port based priority
2550bcc9736cSJeff Kirsher  * @hw: 	The hardware instance.
2551bcc9736cSJeff Kirsher  * @port:	The port index.
2552bcc9736cSJeff Kirsher  * @prio:	The priority to set.
2553bcc9736cSJeff Kirsher  *
2554bcc9736cSJeff Kirsher  * This routine configures the port based priority of the switch.
2555bcc9736cSJeff Kirsher  */
2556bcc9736cSJeff Kirsher static void sw_cfg_port_based(struct ksz_hw *hw, int port, u8 prio)
2557bcc9736cSJeff Kirsher {
2558bcc9736cSJeff Kirsher 	u16 data;
2559bcc9736cSJeff Kirsher 
2560bcc9736cSJeff Kirsher 	if (prio > PORT_BASED_PRIORITY_BASE)
2561bcc9736cSJeff Kirsher 		prio = PORT_BASED_PRIORITY_BASE;
2562bcc9736cSJeff Kirsher 
2563bcc9736cSJeff Kirsher 	hw->ksz_switch->port_cfg[port].port_prio = prio;
2564bcc9736cSJeff Kirsher 
2565bcc9736cSJeff Kirsher 	port_r16(hw, port, KS8842_PORT_CTRL_1_OFFSET, &data);
2566bcc9736cSJeff Kirsher 	data &= ~PORT_BASED_PRIORITY_MASK;
2567bcc9736cSJeff Kirsher 	data |= prio << PORT_BASED_PRIORITY_SHIFT;
2568bcc9736cSJeff Kirsher 	port_w16(hw, port, KS8842_PORT_CTRL_1_OFFSET, data);
2569bcc9736cSJeff Kirsher }
2570bcc9736cSJeff Kirsher 
2571bcc9736cSJeff Kirsher /**
2572bcc9736cSJeff Kirsher  * sw_dis_multi_queue - disable transmit multiple queues
2573bcc9736cSJeff Kirsher  * @hw: 	The hardware instance.
2574bcc9736cSJeff Kirsher  * @port:	The port index.
2575bcc9736cSJeff Kirsher  *
2576bcc9736cSJeff Kirsher  * This routine disables the transmit multiple queues selection of the switch
2577bcc9736cSJeff Kirsher  * port.  Only single transmit queue on the port.
2578bcc9736cSJeff Kirsher  */
2579bcc9736cSJeff Kirsher static void sw_dis_multi_queue(struct ksz_hw *hw, int port)
2580bcc9736cSJeff Kirsher {
2581bcc9736cSJeff Kirsher 	port_cfg_prio(hw, port, 0);
2582bcc9736cSJeff Kirsher }
2583bcc9736cSJeff Kirsher 
2584bcc9736cSJeff Kirsher /**
2585bcc9736cSJeff Kirsher  * sw_init_prio - initialize switch priority
2586bcc9736cSJeff Kirsher  * @hw: 	The hardware instance.
2587bcc9736cSJeff Kirsher  *
2588bcc9736cSJeff Kirsher  * This routine initializes the switch QoS priority functions.
2589bcc9736cSJeff Kirsher  */
2590bcc9736cSJeff Kirsher static void sw_init_prio(struct ksz_hw *hw)
2591bcc9736cSJeff Kirsher {
2592bcc9736cSJeff Kirsher 	int port;
2593bcc9736cSJeff Kirsher 	int tos;
2594bcc9736cSJeff Kirsher 	struct ksz_switch *sw = hw->ksz_switch;
2595bcc9736cSJeff Kirsher 
2596bcc9736cSJeff Kirsher 	/*
2597bcc9736cSJeff Kirsher 	 * Init all the 802.1p tag priority value to be assigned to different
2598bcc9736cSJeff Kirsher 	 * priority queue.
2599bcc9736cSJeff Kirsher 	 */
2600bcc9736cSJeff Kirsher 	sw->p_802_1p[0] = 0;
2601bcc9736cSJeff Kirsher 	sw->p_802_1p[1] = 0;
2602bcc9736cSJeff Kirsher 	sw->p_802_1p[2] = 1;
2603bcc9736cSJeff Kirsher 	sw->p_802_1p[3] = 1;
2604bcc9736cSJeff Kirsher 	sw->p_802_1p[4] = 2;
2605bcc9736cSJeff Kirsher 	sw->p_802_1p[5] = 2;
2606bcc9736cSJeff Kirsher 	sw->p_802_1p[6] = 3;
2607bcc9736cSJeff Kirsher 	sw->p_802_1p[7] = 3;
2608bcc9736cSJeff Kirsher 
2609bcc9736cSJeff Kirsher 	/*
2610bcc9736cSJeff Kirsher 	 * Init all the DiffServ priority value to be assigned to priority
2611bcc9736cSJeff Kirsher 	 * queue 0.
2612bcc9736cSJeff Kirsher 	 */
2613bcc9736cSJeff Kirsher 	for (tos = 0; tos < DIFFSERV_ENTRIES; tos++)
2614bcc9736cSJeff Kirsher 		sw->diffserv[tos] = 0;
2615bcc9736cSJeff Kirsher 
2616bcc9736cSJeff Kirsher 	/* All QoS functions disabled. */
2617bcc9736cSJeff Kirsher 	for (port = 0; port < TOTAL_PORT_NUM; port++) {
2618bcc9736cSJeff Kirsher 		sw_dis_multi_queue(hw, port);
2619bcc9736cSJeff Kirsher 		sw_dis_diffserv(hw, port);
2620bcc9736cSJeff Kirsher 		sw_dis_802_1p(hw, port);
2621bcc9736cSJeff Kirsher 		sw_cfg_replace_vid(hw, port, 0);
2622bcc9736cSJeff Kirsher 
2623bcc9736cSJeff Kirsher 		sw->port_cfg[port].port_prio = 0;
2624bcc9736cSJeff Kirsher 		sw_cfg_port_based(hw, port, sw->port_cfg[port].port_prio);
2625bcc9736cSJeff Kirsher 	}
2626bcc9736cSJeff Kirsher 	sw_cfg_replace_null_vid(hw, 0);
2627bcc9736cSJeff Kirsher }
2628bcc9736cSJeff Kirsher 
2629bcc9736cSJeff Kirsher /**
2630bcc9736cSJeff Kirsher  * port_get_def_vid - get port default VID.
2631bcc9736cSJeff Kirsher  * @hw: 	The hardware instance.
2632bcc9736cSJeff Kirsher  * @port:	The port index.
2633bcc9736cSJeff Kirsher  * @vid:	Buffer to store the VID.
2634bcc9736cSJeff Kirsher  *
2635bcc9736cSJeff Kirsher  * This routine retrieves the default VID of the port.
2636bcc9736cSJeff Kirsher  */
2637bcc9736cSJeff Kirsher static void port_get_def_vid(struct ksz_hw *hw, int port, u16 *vid)
2638bcc9736cSJeff Kirsher {
2639bcc9736cSJeff Kirsher 	u32 addr;
2640bcc9736cSJeff Kirsher 
2641bcc9736cSJeff Kirsher 	PORT_CTRL_ADDR(port, addr);
2642bcc9736cSJeff Kirsher 	addr += KS8842_PORT_CTRL_VID_OFFSET;
2643bcc9736cSJeff Kirsher 	*vid = readw(hw->io + addr);
2644bcc9736cSJeff Kirsher }
2645bcc9736cSJeff Kirsher 
2646bcc9736cSJeff Kirsher /**
2647bcc9736cSJeff Kirsher  * sw_init_vlan - initialize switch VLAN
2648bcc9736cSJeff Kirsher  * @hw: 	The hardware instance.
2649bcc9736cSJeff Kirsher  *
2650bcc9736cSJeff Kirsher  * This routine initializes the VLAN function of the switch.
2651bcc9736cSJeff Kirsher  */
2652bcc9736cSJeff Kirsher static void sw_init_vlan(struct ksz_hw *hw)
2653bcc9736cSJeff Kirsher {
2654bcc9736cSJeff Kirsher 	int port;
2655bcc9736cSJeff Kirsher 	int entry;
2656bcc9736cSJeff Kirsher 	struct ksz_switch *sw = hw->ksz_switch;
2657bcc9736cSJeff Kirsher 
2658bcc9736cSJeff Kirsher 	/* Read 16 VLAN entries from device's VLAN table. */
2659bcc9736cSJeff Kirsher 	for (entry = 0; entry < VLAN_TABLE_ENTRIES; entry++) {
2660bcc9736cSJeff Kirsher 		sw_r_vlan_table(hw, entry,
2661bcc9736cSJeff Kirsher 			&sw->vlan_table[entry].vid,
2662bcc9736cSJeff Kirsher 			&sw->vlan_table[entry].fid,
2663bcc9736cSJeff Kirsher 			&sw->vlan_table[entry].member);
2664bcc9736cSJeff Kirsher 	}
2665bcc9736cSJeff Kirsher 
2666bcc9736cSJeff Kirsher 	for (port = 0; port < TOTAL_PORT_NUM; port++) {
2667bcc9736cSJeff Kirsher 		port_get_def_vid(hw, port, &sw->port_cfg[port].vid);
2668bcc9736cSJeff Kirsher 		sw->port_cfg[port].member = PORT_MASK;
2669bcc9736cSJeff Kirsher 	}
2670bcc9736cSJeff Kirsher }
2671bcc9736cSJeff Kirsher 
2672bcc9736cSJeff Kirsher /**
2673bcc9736cSJeff Kirsher  * sw_cfg_port_base_vlan - configure port-based VLAN membership
2674bcc9736cSJeff Kirsher  * @hw: 	The hardware instance.
2675bcc9736cSJeff Kirsher  * @port:	The port index.
2676bcc9736cSJeff Kirsher  * @member:	The port-based VLAN membership.
2677bcc9736cSJeff Kirsher  *
2678bcc9736cSJeff Kirsher  * This routine configures the port-based VLAN membership of the port.
2679bcc9736cSJeff Kirsher  */
2680bcc9736cSJeff Kirsher static void sw_cfg_port_base_vlan(struct ksz_hw *hw, int port, u8 member)
2681bcc9736cSJeff Kirsher {
2682bcc9736cSJeff Kirsher 	u32 addr;
2683bcc9736cSJeff Kirsher 	u8 data;
2684bcc9736cSJeff Kirsher 
2685bcc9736cSJeff Kirsher 	PORT_CTRL_ADDR(port, addr);
2686bcc9736cSJeff Kirsher 	addr += KS8842_PORT_CTRL_2_OFFSET;
2687bcc9736cSJeff Kirsher 
2688bcc9736cSJeff Kirsher 	data = readb(hw->io + addr);
2689bcc9736cSJeff Kirsher 	data &= ~PORT_VLAN_MEMBERSHIP;
2690bcc9736cSJeff Kirsher 	data |= (member & PORT_MASK);
2691bcc9736cSJeff Kirsher 	writeb(data, hw->io + addr);
2692bcc9736cSJeff Kirsher 
2693bcc9736cSJeff Kirsher 	hw->ksz_switch->port_cfg[port].member = member;
2694bcc9736cSJeff Kirsher }
2695bcc9736cSJeff Kirsher 
2696bcc9736cSJeff Kirsher /**
2697bcc9736cSJeff Kirsher  * sw_get_addr - get the switch MAC address.
2698bcc9736cSJeff Kirsher  * @hw: 	The hardware instance.
2699bcc9736cSJeff Kirsher  * @mac_addr:	Buffer to store the MAC address.
2700bcc9736cSJeff Kirsher  *
2701bcc9736cSJeff Kirsher  * This function retrieves the MAC address of the switch.
2702bcc9736cSJeff Kirsher  */
2703bcc9736cSJeff Kirsher static inline void sw_get_addr(struct ksz_hw *hw, u8 *mac_addr)
2704bcc9736cSJeff Kirsher {
2705bcc9736cSJeff Kirsher 	int i;
2706bcc9736cSJeff Kirsher 
2707bcc9736cSJeff Kirsher 	for (i = 0; i < 6; i += 2) {
2708bcc9736cSJeff Kirsher 		mac_addr[i] = readb(hw->io + KS8842_MAC_ADDR_0_OFFSET + i);
2709bcc9736cSJeff Kirsher 		mac_addr[1 + i] = readb(hw->io + KS8842_MAC_ADDR_1_OFFSET + i);
2710bcc9736cSJeff Kirsher 	}
2711bcc9736cSJeff Kirsher }
2712bcc9736cSJeff Kirsher 
2713bcc9736cSJeff Kirsher /**
2714bcc9736cSJeff Kirsher  * sw_set_addr - configure switch MAC address
2715bcc9736cSJeff Kirsher  * @hw: 	The hardware instance.
2716bcc9736cSJeff Kirsher  * @mac_addr:	The MAC address.
2717bcc9736cSJeff Kirsher  *
2718bcc9736cSJeff Kirsher  * This function configures the MAC address of the switch.
2719bcc9736cSJeff Kirsher  */
2720bcc9736cSJeff Kirsher static void sw_set_addr(struct ksz_hw *hw, u8 *mac_addr)
2721bcc9736cSJeff Kirsher {
2722bcc9736cSJeff Kirsher 	int i;
2723bcc9736cSJeff Kirsher 
2724bcc9736cSJeff Kirsher 	for (i = 0; i < 6; i += 2) {
2725bcc9736cSJeff Kirsher 		writeb(mac_addr[i], hw->io + KS8842_MAC_ADDR_0_OFFSET + i);
2726bcc9736cSJeff Kirsher 		writeb(mac_addr[1 + i], hw->io + KS8842_MAC_ADDR_1_OFFSET + i);
2727bcc9736cSJeff Kirsher 	}
2728bcc9736cSJeff Kirsher }
2729bcc9736cSJeff Kirsher 
2730bcc9736cSJeff Kirsher /**
2731bcc9736cSJeff Kirsher  * sw_set_global_ctrl - set switch global control
2732bcc9736cSJeff Kirsher  * @hw: 	The hardware instance.
2733bcc9736cSJeff Kirsher  *
2734bcc9736cSJeff Kirsher  * This routine sets the global control of the switch function.
2735bcc9736cSJeff Kirsher  */
2736bcc9736cSJeff Kirsher static void sw_set_global_ctrl(struct ksz_hw *hw)
2737bcc9736cSJeff Kirsher {
2738bcc9736cSJeff Kirsher 	u16 data;
2739bcc9736cSJeff Kirsher 
2740bcc9736cSJeff Kirsher 	/* Enable switch MII flow control. */
2741bcc9736cSJeff Kirsher 	data = readw(hw->io + KS8842_SWITCH_CTRL_3_OFFSET);
2742bcc9736cSJeff Kirsher 	data |= SWITCH_FLOW_CTRL;
2743bcc9736cSJeff Kirsher 	writew(data, hw->io + KS8842_SWITCH_CTRL_3_OFFSET);
2744bcc9736cSJeff Kirsher 
2745bcc9736cSJeff Kirsher 	data = readw(hw->io + KS8842_SWITCH_CTRL_1_OFFSET);
2746bcc9736cSJeff Kirsher 
2747bcc9736cSJeff Kirsher 	/* Enable aggressive back off algorithm in half duplex mode. */
2748bcc9736cSJeff Kirsher 	data |= SWITCH_AGGR_BACKOFF;
2749bcc9736cSJeff Kirsher 
2750bcc9736cSJeff Kirsher 	/* Enable automatic fast aging when link changed detected. */
2751bcc9736cSJeff Kirsher 	data |= SWITCH_AGING_ENABLE;
2752bcc9736cSJeff Kirsher 	data |= SWITCH_LINK_AUTO_AGING;
2753bcc9736cSJeff Kirsher 
2754bcc9736cSJeff Kirsher 	if (hw->overrides & FAST_AGING)
2755bcc9736cSJeff Kirsher 		data |= SWITCH_FAST_AGING;
2756bcc9736cSJeff Kirsher 	else
2757bcc9736cSJeff Kirsher 		data &= ~SWITCH_FAST_AGING;
2758bcc9736cSJeff Kirsher 	writew(data, hw->io + KS8842_SWITCH_CTRL_1_OFFSET);
2759bcc9736cSJeff Kirsher 
2760bcc9736cSJeff Kirsher 	data = readw(hw->io + KS8842_SWITCH_CTRL_2_OFFSET);
2761bcc9736cSJeff Kirsher 
2762bcc9736cSJeff Kirsher 	/* Enable no excessive collision drop. */
2763bcc9736cSJeff Kirsher 	data |= NO_EXC_COLLISION_DROP;
2764bcc9736cSJeff Kirsher 	writew(data, hw->io + KS8842_SWITCH_CTRL_2_OFFSET);
2765bcc9736cSJeff Kirsher }
2766bcc9736cSJeff Kirsher 
2767bcc9736cSJeff Kirsher enum {
2768bcc9736cSJeff Kirsher 	STP_STATE_DISABLED = 0,
2769bcc9736cSJeff Kirsher 	STP_STATE_LISTENING,
2770bcc9736cSJeff Kirsher 	STP_STATE_LEARNING,
2771bcc9736cSJeff Kirsher 	STP_STATE_FORWARDING,
2772bcc9736cSJeff Kirsher 	STP_STATE_BLOCKED,
2773bcc9736cSJeff Kirsher 	STP_STATE_SIMPLE
2774bcc9736cSJeff Kirsher };
2775bcc9736cSJeff Kirsher 
2776bcc9736cSJeff Kirsher /**
2777bcc9736cSJeff Kirsher  * port_set_stp_state - configure port spanning tree state
2778bcc9736cSJeff Kirsher  * @hw: 	The hardware instance.
2779bcc9736cSJeff Kirsher  * @port:	The port index.
2780bcc9736cSJeff Kirsher  * @state:	The spanning tree state.
2781bcc9736cSJeff Kirsher  *
2782bcc9736cSJeff Kirsher  * This routine configures the spanning tree state of the port.
2783bcc9736cSJeff Kirsher  */
2784bcc9736cSJeff Kirsher static void port_set_stp_state(struct ksz_hw *hw, int port, int state)
2785bcc9736cSJeff Kirsher {
2786bcc9736cSJeff Kirsher 	u16 data;
2787bcc9736cSJeff Kirsher 
2788bcc9736cSJeff Kirsher 	port_r16(hw, port, KS8842_PORT_CTRL_2_OFFSET, &data);
2789bcc9736cSJeff Kirsher 	switch (state) {
2790bcc9736cSJeff Kirsher 	case STP_STATE_DISABLED:
2791bcc9736cSJeff Kirsher 		data &= ~(PORT_TX_ENABLE | PORT_RX_ENABLE);
2792bcc9736cSJeff Kirsher 		data |= PORT_LEARN_DISABLE;
2793bcc9736cSJeff Kirsher 		break;
2794bcc9736cSJeff Kirsher 	case STP_STATE_LISTENING:
2795bcc9736cSJeff Kirsher /*
2796bcc9736cSJeff Kirsher  * No need to turn on transmit because of port direct mode.
2797bcc9736cSJeff Kirsher  * Turning on receive is required if static MAC table is not setup.
2798bcc9736cSJeff Kirsher  */
2799bcc9736cSJeff Kirsher 		data &= ~PORT_TX_ENABLE;
2800bcc9736cSJeff Kirsher 		data |= PORT_RX_ENABLE;
2801bcc9736cSJeff Kirsher 		data |= PORT_LEARN_DISABLE;
2802bcc9736cSJeff Kirsher 		break;
2803bcc9736cSJeff Kirsher 	case STP_STATE_LEARNING:
2804bcc9736cSJeff Kirsher 		data &= ~PORT_TX_ENABLE;
2805bcc9736cSJeff Kirsher 		data |= PORT_RX_ENABLE;
2806bcc9736cSJeff Kirsher 		data &= ~PORT_LEARN_DISABLE;
2807bcc9736cSJeff Kirsher 		break;
2808bcc9736cSJeff Kirsher 	case STP_STATE_FORWARDING:
2809bcc9736cSJeff Kirsher 		data |= (PORT_TX_ENABLE | PORT_RX_ENABLE);
2810bcc9736cSJeff Kirsher 		data &= ~PORT_LEARN_DISABLE;
2811bcc9736cSJeff Kirsher 		break;
2812bcc9736cSJeff Kirsher 	case STP_STATE_BLOCKED:
2813bcc9736cSJeff Kirsher /*
2814bcc9736cSJeff Kirsher  * Need to setup static MAC table with override to keep receiving BPDU
2815bcc9736cSJeff Kirsher  * messages.  See sw_init_stp routine.
2816bcc9736cSJeff Kirsher  */
2817bcc9736cSJeff Kirsher 		data &= ~(PORT_TX_ENABLE | PORT_RX_ENABLE);
2818bcc9736cSJeff Kirsher 		data |= PORT_LEARN_DISABLE;
2819bcc9736cSJeff Kirsher 		break;
2820bcc9736cSJeff Kirsher 	case STP_STATE_SIMPLE:
2821bcc9736cSJeff Kirsher 		data |= (PORT_TX_ENABLE | PORT_RX_ENABLE);
2822bcc9736cSJeff Kirsher 		data |= PORT_LEARN_DISABLE;
2823bcc9736cSJeff Kirsher 		break;
2824bcc9736cSJeff Kirsher 	}
2825bcc9736cSJeff Kirsher 	port_w16(hw, port, KS8842_PORT_CTRL_2_OFFSET, data);
2826bcc9736cSJeff Kirsher 	hw->ksz_switch->port_cfg[port].stp_state = state;
2827bcc9736cSJeff Kirsher }
2828bcc9736cSJeff Kirsher 
2829bcc9736cSJeff Kirsher #define STP_ENTRY			0
2830bcc9736cSJeff Kirsher #define BROADCAST_ENTRY			1
2831bcc9736cSJeff Kirsher #define BRIDGE_ADDR_ENTRY		2
2832bcc9736cSJeff Kirsher #define IPV6_ADDR_ENTRY			3
2833bcc9736cSJeff Kirsher 
2834bcc9736cSJeff Kirsher /**
2835bcc9736cSJeff Kirsher  * sw_clr_sta_mac_table - clear static MAC table
2836bcc9736cSJeff Kirsher  * @hw: 	The hardware instance.
2837bcc9736cSJeff Kirsher  *
2838bcc9736cSJeff Kirsher  * This routine clears the static MAC table.
2839bcc9736cSJeff Kirsher  */
2840bcc9736cSJeff Kirsher static void sw_clr_sta_mac_table(struct ksz_hw *hw)
2841bcc9736cSJeff Kirsher {
2842bcc9736cSJeff Kirsher 	struct ksz_mac_table *entry;
2843bcc9736cSJeff Kirsher 	int i;
2844bcc9736cSJeff Kirsher 
2845bcc9736cSJeff Kirsher 	for (i = 0; i < STATIC_MAC_TABLE_ENTRIES; i++) {
2846bcc9736cSJeff Kirsher 		entry = &hw->ksz_switch->mac_table[i];
2847bcc9736cSJeff Kirsher 		sw_w_sta_mac_table(hw, i,
2848bcc9736cSJeff Kirsher 			entry->mac_addr, entry->ports,
2849bcc9736cSJeff Kirsher 			entry->override, 0,
2850bcc9736cSJeff Kirsher 			entry->use_fid, entry->fid);
2851bcc9736cSJeff Kirsher 	}
2852bcc9736cSJeff Kirsher }
2853bcc9736cSJeff Kirsher 
2854bcc9736cSJeff Kirsher /**
2855bcc9736cSJeff Kirsher  * sw_init_stp - initialize switch spanning tree support
2856bcc9736cSJeff Kirsher  * @hw: 	The hardware instance.
2857bcc9736cSJeff Kirsher  *
2858bcc9736cSJeff Kirsher  * This routine initializes the spanning tree support of the switch.
2859bcc9736cSJeff Kirsher  */
2860bcc9736cSJeff Kirsher static void sw_init_stp(struct ksz_hw *hw)
2861bcc9736cSJeff Kirsher {
2862bcc9736cSJeff Kirsher 	struct ksz_mac_table *entry;
2863bcc9736cSJeff Kirsher 
2864bcc9736cSJeff Kirsher 	entry = &hw->ksz_switch->mac_table[STP_ENTRY];
2865bcc9736cSJeff Kirsher 	entry->mac_addr[0] = 0x01;
2866bcc9736cSJeff Kirsher 	entry->mac_addr[1] = 0x80;
2867bcc9736cSJeff Kirsher 	entry->mac_addr[2] = 0xC2;
2868bcc9736cSJeff Kirsher 	entry->mac_addr[3] = 0x00;
2869bcc9736cSJeff Kirsher 	entry->mac_addr[4] = 0x00;
2870bcc9736cSJeff Kirsher 	entry->mac_addr[5] = 0x00;
2871bcc9736cSJeff Kirsher 	entry->ports = HOST_MASK;
2872bcc9736cSJeff Kirsher 	entry->override = 1;
2873bcc9736cSJeff Kirsher 	entry->valid = 1;
2874bcc9736cSJeff Kirsher 	sw_w_sta_mac_table(hw, STP_ENTRY,
2875bcc9736cSJeff Kirsher 		entry->mac_addr, entry->ports,
2876bcc9736cSJeff Kirsher 		entry->override, entry->valid,
2877bcc9736cSJeff Kirsher 		entry->use_fid, entry->fid);
2878bcc9736cSJeff Kirsher }
2879bcc9736cSJeff Kirsher 
2880bcc9736cSJeff Kirsher /**
2881bcc9736cSJeff Kirsher  * sw_block_addr - block certain packets from the host port
2882bcc9736cSJeff Kirsher  * @hw: 	The hardware instance.
2883bcc9736cSJeff Kirsher  *
2884bcc9736cSJeff Kirsher  * This routine blocks certain packets from reaching to the host port.
2885bcc9736cSJeff Kirsher  */
2886bcc9736cSJeff Kirsher static void sw_block_addr(struct ksz_hw *hw)
2887bcc9736cSJeff Kirsher {
2888bcc9736cSJeff Kirsher 	struct ksz_mac_table *entry;
2889bcc9736cSJeff Kirsher 	int i;
2890bcc9736cSJeff Kirsher 
2891bcc9736cSJeff Kirsher 	for (i = BROADCAST_ENTRY; i <= IPV6_ADDR_ENTRY; i++) {
2892bcc9736cSJeff Kirsher 		entry = &hw->ksz_switch->mac_table[i];
2893bcc9736cSJeff Kirsher 		entry->valid = 0;
2894bcc9736cSJeff Kirsher 		sw_w_sta_mac_table(hw, i,
2895bcc9736cSJeff Kirsher 			entry->mac_addr, entry->ports,
2896bcc9736cSJeff Kirsher 			entry->override, entry->valid,
2897bcc9736cSJeff Kirsher 			entry->use_fid, entry->fid);
2898bcc9736cSJeff Kirsher 	}
2899bcc9736cSJeff Kirsher }
2900bcc9736cSJeff Kirsher 
2901bcc9736cSJeff Kirsher #define PHY_LINK_SUPPORT		\
2902bcc9736cSJeff Kirsher 	(PHY_AUTO_NEG_ASYM_PAUSE |	\
2903bcc9736cSJeff Kirsher 	PHY_AUTO_NEG_SYM_PAUSE |	\
2904bcc9736cSJeff Kirsher 	PHY_AUTO_NEG_100BT4 |		\
2905bcc9736cSJeff Kirsher 	PHY_AUTO_NEG_100BTX_FD |	\
2906bcc9736cSJeff Kirsher 	PHY_AUTO_NEG_100BTX |		\
2907bcc9736cSJeff Kirsher 	PHY_AUTO_NEG_10BT_FD |		\
2908bcc9736cSJeff Kirsher 	PHY_AUTO_NEG_10BT)
2909bcc9736cSJeff Kirsher 
2910bcc9736cSJeff Kirsher static inline void hw_r_phy_ctrl(struct ksz_hw *hw, int phy, u16 *data)
2911bcc9736cSJeff Kirsher {
2912bcc9736cSJeff Kirsher 	*data = readw(hw->io + phy + KS884X_PHY_CTRL_OFFSET);
2913bcc9736cSJeff Kirsher }
2914bcc9736cSJeff Kirsher 
2915bcc9736cSJeff Kirsher static inline void hw_w_phy_ctrl(struct ksz_hw *hw, int phy, u16 data)
2916bcc9736cSJeff Kirsher {
2917bcc9736cSJeff Kirsher 	writew(data, hw->io + phy + KS884X_PHY_CTRL_OFFSET);
2918bcc9736cSJeff Kirsher }
2919bcc9736cSJeff Kirsher 
2920bcc9736cSJeff Kirsher static inline void hw_r_phy_link_stat(struct ksz_hw *hw, int phy, u16 *data)
2921bcc9736cSJeff Kirsher {
2922bcc9736cSJeff Kirsher 	*data = readw(hw->io + phy + KS884X_PHY_STATUS_OFFSET);
2923bcc9736cSJeff Kirsher }
2924bcc9736cSJeff Kirsher 
2925bcc9736cSJeff Kirsher static inline void hw_r_phy_auto_neg(struct ksz_hw *hw, int phy, u16 *data)
2926bcc9736cSJeff Kirsher {
2927bcc9736cSJeff Kirsher 	*data = readw(hw->io + phy + KS884X_PHY_AUTO_NEG_OFFSET);
2928bcc9736cSJeff Kirsher }
2929bcc9736cSJeff Kirsher 
2930bcc9736cSJeff Kirsher static inline void hw_w_phy_auto_neg(struct ksz_hw *hw, int phy, u16 data)
2931bcc9736cSJeff Kirsher {
2932bcc9736cSJeff Kirsher 	writew(data, hw->io + phy + KS884X_PHY_AUTO_NEG_OFFSET);
2933bcc9736cSJeff Kirsher }
2934bcc9736cSJeff Kirsher 
2935bcc9736cSJeff Kirsher static inline void hw_r_phy_rem_cap(struct ksz_hw *hw, int phy, u16 *data)
2936bcc9736cSJeff Kirsher {
2937bcc9736cSJeff Kirsher 	*data = readw(hw->io + phy + KS884X_PHY_REMOTE_CAP_OFFSET);
2938bcc9736cSJeff Kirsher }
2939bcc9736cSJeff Kirsher 
2940bcc9736cSJeff Kirsher static inline void hw_r_phy_crossover(struct ksz_hw *hw, int phy, u16 *data)
2941bcc9736cSJeff Kirsher {
2942bcc9736cSJeff Kirsher 	*data = readw(hw->io + phy + KS884X_PHY_CTRL_OFFSET);
2943bcc9736cSJeff Kirsher }
2944bcc9736cSJeff Kirsher 
2945bcc9736cSJeff Kirsher static inline void hw_w_phy_crossover(struct ksz_hw *hw, int phy, u16 data)
2946bcc9736cSJeff Kirsher {
2947bcc9736cSJeff Kirsher 	writew(data, hw->io + phy + KS884X_PHY_CTRL_OFFSET);
2948bcc9736cSJeff Kirsher }
2949bcc9736cSJeff Kirsher 
2950bcc9736cSJeff Kirsher static inline void hw_r_phy_polarity(struct ksz_hw *hw, int phy, u16 *data)
2951bcc9736cSJeff Kirsher {
2952bcc9736cSJeff Kirsher 	*data = readw(hw->io + phy + KS884X_PHY_PHY_CTRL_OFFSET);
2953bcc9736cSJeff Kirsher }
2954bcc9736cSJeff Kirsher 
2955bcc9736cSJeff Kirsher static inline void hw_w_phy_polarity(struct ksz_hw *hw, int phy, u16 data)
2956bcc9736cSJeff Kirsher {
2957bcc9736cSJeff Kirsher 	writew(data, hw->io + phy + KS884X_PHY_PHY_CTRL_OFFSET);
2958bcc9736cSJeff Kirsher }
2959bcc9736cSJeff Kirsher 
2960bcc9736cSJeff Kirsher static inline void hw_r_phy_link_md(struct ksz_hw *hw, int phy, u16 *data)
2961bcc9736cSJeff Kirsher {
2962bcc9736cSJeff Kirsher 	*data = readw(hw->io + phy + KS884X_PHY_LINK_MD_OFFSET);
2963bcc9736cSJeff Kirsher }
2964bcc9736cSJeff Kirsher 
2965bcc9736cSJeff Kirsher static inline void hw_w_phy_link_md(struct ksz_hw *hw, int phy, u16 data)
2966bcc9736cSJeff Kirsher {
2967bcc9736cSJeff Kirsher 	writew(data, hw->io + phy + KS884X_PHY_LINK_MD_OFFSET);
2968bcc9736cSJeff Kirsher }
2969bcc9736cSJeff Kirsher 
2970bcc9736cSJeff Kirsher /**
2971bcc9736cSJeff Kirsher  * hw_r_phy - read data from PHY register
2972bcc9736cSJeff Kirsher  * @hw: 	The hardware instance.
2973bcc9736cSJeff Kirsher  * @port:	Port to read.
2974bcc9736cSJeff Kirsher  * @reg:	PHY register to read.
2975bcc9736cSJeff Kirsher  * @val:	Buffer to store the read data.
2976bcc9736cSJeff Kirsher  *
2977bcc9736cSJeff Kirsher  * This routine reads data from the PHY register.
2978bcc9736cSJeff Kirsher  */
2979bcc9736cSJeff Kirsher static void hw_r_phy(struct ksz_hw *hw, int port, u16 reg, u16 *val)
2980bcc9736cSJeff Kirsher {
2981bcc9736cSJeff Kirsher 	int phy;
2982bcc9736cSJeff Kirsher 
2983bcc9736cSJeff Kirsher 	phy = KS884X_PHY_1_CTRL_OFFSET + port * PHY_CTRL_INTERVAL + reg;
2984bcc9736cSJeff Kirsher 	*val = readw(hw->io + phy);
2985bcc9736cSJeff Kirsher }
2986bcc9736cSJeff Kirsher 
2987bcc9736cSJeff Kirsher /**
2988bcc9736cSJeff Kirsher  * port_w_phy - write data to PHY register
2989bcc9736cSJeff Kirsher  * @hw: 	The hardware instance.
2990bcc9736cSJeff Kirsher  * @port:	Port to write.
2991bcc9736cSJeff Kirsher  * @reg:	PHY register to write.
2992bcc9736cSJeff Kirsher  * @val:	Word data to write.
2993bcc9736cSJeff Kirsher  *
2994bcc9736cSJeff Kirsher  * This routine writes data to the PHY register.
2995bcc9736cSJeff Kirsher  */
2996bcc9736cSJeff Kirsher static void hw_w_phy(struct ksz_hw *hw, int port, u16 reg, u16 val)
2997bcc9736cSJeff Kirsher {
2998bcc9736cSJeff Kirsher 	int phy;
2999bcc9736cSJeff Kirsher 
3000bcc9736cSJeff Kirsher 	phy = KS884X_PHY_1_CTRL_OFFSET + port * PHY_CTRL_INTERVAL + reg;
3001bcc9736cSJeff Kirsher 	writew(val, hw->io + phy);
3002bcc9736cSJeff Kirsher }
3003bcc9736cSJeff Kirsher 
3004bcc9736cSJeff Kirsher /*
3005bcc9736cSJeff Kirsher  * EEPROM access functions
3006bcc9736cSJeff Kirsher  */
3007bcc9736cSJeff Kirsher 
3008bcc9736cSJeff Kirsher #define AT93C_CODE			0
3009bcc9736cSJeff Kirsher #define AT93C_WR_OFF			0x00
3010bcc9736cSJeff Kirsher #define AT93C_WR_ALL			0x10
3011bcc9736cSJeff Kirsher #define AT93C_ER_ALL			0x20
3012bcc9736cSJeff Kirsher #define AT93C_WR_ON			0x30
3013bcc9736cSJeff Kirsher 
3014bcc9736cSJeff Kirsher #define AT93C_WRITE			1
3015bcc9736cSJeff Kirsher #define AT93C_READ			2
3016bcc9736cSJeff Kirsher #define AT93C_ERASE			3
3017bcc9736cSJeff Kirsher 
3018bcc9736cSJeff Kirsher #define EEPROM_DELAY			4
3019bcc9736cSJeff Kirsher 
3020bcc9736cSJeff Kirsher static inline void drop_gpio(struct ksz_hw *hw, u8 gpio)
3021bcc9736cSJeff Kirsher {
3022bcc9736cSJeff Kirsher 	u16 data;
3023bcc9736cSJeff Kirsher 
3024bcc9736cSJeff Kirsher 	data = readw(hw->io + KS884X_EEPROM_CTRL_OFFSET);
3025bcc9736cSJeff Kirsher 	data &= ~gpio;
3026bcc9736cSJeff Kirsher 	writew(data, hw->io + KS884X_EEPROM_CTRL_OFFSET);
3027bcc9736cSJeff Kirsher }
3028bcc9736cSJeff Kirsher 
3029bcc9736cSJeff Kirsher static inline void raise_gpio(struct ksz_hw *hw, u8 gpio)
3030bcc9736cSJeff Kirsher {
3031bcc9736cSJeff Kirsher 	u16 data;
3032bcc9736cSJeff Kirsher 
3033bcc9736cSJeff Kirsher 	data = readw(hw->io + KS884X_EEPROM_CTRL_OFFSET);
3034bcc9736cSJeff Kirsher 	data |= gpio;
3035bcc9736cSJeff Kirsher 	writew(data, hw->io + KS884X_EEPROM_CTRL_OFFSET);
3036bcc9736cSJeff Kirsher }
3037bcc9736cSJeff Kirsher 
3038bcc9736cSJeff Kirsher static inline u8 state_gpio(struct ksz_hw *hw, u8 gpio)
3039bcc9736cSJeff Kirsher {
3040bcc9736cSJeff Kirsher 	u16 data;
3041bcc9736cSJeff Kirsher 
3042bcc9736cSJeff Kirsher 	data = readw(hw->io + KS884X_EEPROM_CTRL_OFFSET);
3043bcc9736cSJeff Kirsher 	return (u8)(data & gpio);
3044bcc9736cSJeff Kirsher }
3045bcc9736cSJeff Kirsher 
3046bcc9736cSJeff Kirsher static void eeprom_clk(struct ksz_hw *hw)
3047bcc9736cSJeff Kirsher {
3048bcc9736cSJeff Kirsher 	raise_gpio(hw, EEPROM_SERIAL_CLOCK);
3049bcc9736cSJeff Kirsher 	udelay(EEPROM_DELAY);
3050bcc9736cSJeff Kirsher 	drop_gpio(hw, EEPROM_SERIAL_CLOCK);
3051bcc9736cSJeff Kirsher 	udelay(EEPROM_DELAY);
3052bcc9736cSJeff Kirsher }
3053bcc9736cSJeff Kirsher 
3054bcc9736cSJeff Kirsher static u16 spi_r(struct ksz_hw *hw)
3055bcc9736cSJeff Kirsher {
3056bcc9736cSJeff Kirsher 	int i;
3057bcc9736cSJeff Kirsher 	u16 temp = 0;
3058bcc9736cSJeff Kirsher 
3059bcc9736cSJeff Kirsher 	for (i = 15; i >= 0; i--) {
3060bcc9736cSJeff Kirsher 		raise_gpio(hw, EEPROM_SERIAL_CLOCK);
3061bcc9736cSJeff Kirsher 		udelay(EEPROM_DELAY);
3062bcc9736cSJeff Kirsher 
3063bcc9736cSJeff Kirsher 		temp |= (state_gpio(hw, EEPROM_DATA_IN)) ? 1 << i : 0;
3064bcc9736cSJeff Kirsher 
3065bcc9736cSJeff Kirsher 		drop_gpio(hw, EEPROM_SERIAL_CLOCK);
3066bcc9736cSJeff Kirsher 		udelay(EEPROM_DELAY);
3067bcc9736cSJeff Kirsher 	}
3068bcc9736cSJeff Kirsher 	return temp;
3069bcc9736cSJeff Kirsher }
3070bcc9736cSJeff Kirsher 
3071bcc9736cSJeff Kirsher static void spi_w(struct ksz_hw *hw, u16 data)
3072bcc9736cSJeff Kirsher {
3073bcc9736cSJeff Kirsher 	int i;
3074bcc9736cSJeff Kirsher 
3075bcc9736cSJeff Kirsher 	for (i = 15; i >= 0; i--) {
3076bcc9736cSJeff Kirsher 		(data & (0x01 << i)) ? raise_gpio(hw, EEPROM_DATA_OUT) :
3077bcc9736cSJeff Kirsher 			drop_gpio(hw, EEPROM_DATA_OUT);
3078bcc9736cSJeff Kirsher 		eeprom_clk(hw);
3079bcc9736cSJeff Kirsher 	}
3080bcc9736cSJeff Kirsher }
3081bcc9736cSJeff Kirsher 
3082bcc9736cSJeff Kirsher static void spi_reg(struct ksz_hw *hw, u8 data, u8 reg)
3083bcc9736cSJeff Kirsher {
3084bcc9736cSJeff Kirsher 	int i;
3085bcc9736cSJeff Kirsher 
3086bcc9736cSJeff Kirsher 	/* Initial start bit */
3087bcc9736cSJeff Kirsher 	raise_gpio(hw, EEPROM_DATA_OUT);
3088bcc9736cSJeff Kirsher 	eeprom_clk(hw);
3089bcc9736cSJeff Kirsher 
3090bcc9736cSJeff Kirsher 	/* AT93C operation */
3091bcc9736cSJeff Kirsher 	for (i = 1; i >= 0; i--) {
3092bcc9736cSJeff Kirsher 		(data & (0x01 << i)) ? raise_gpio(hw, EEPROM_DATA_OUT) :
3093bcc9736cSJeff Kirsher 			drop_gpio(hw, EEPROM_DATA_OUT);
3094bcc9736cSJeff Kirsher 		eeprom_clk(hw);
3095bcc9736cSJeff Kirsher 	}
3096bcc9736cSJeff Kirsher 
3097bcc9736cSJeff Kirsher 	/* Address location */
3098bcc9736cSJeff Kirsher 	for (i = 5; i >= 0; i--) {
3099bcc9736cSJeff Kirsher 		(reg & (0x01 << i)) ? raise_gpio(hw, EEPROM_DATA_OUT) :
3100bcc9736cSJeff Kirsher 			drop_gpio(hw, EEPROM_DATA_OUT);
3101bcc9736cSJeff Kirsher 		eeprom_clk(hw);
3102bcc9736cSJeff Kirsher 	}
3103bcc9736cSJeff Kirsher }
3104bcc9736cSJeff Kirsher 
3105bcc9736cSJeff Kirsher #define EEPROM_DATA_RESERVED		0
3106bcc9736cSJeff Kirsher #define EEPROM_DATA_MAC_ADDR_0		1
3107bcc9736cSJeff Kirsher #define EEPROM_DATA_MAC_ADDR_1		2
3108bcc9736cSJeff Kirsher #define EEPROM_DATA_MAC_ADDR_2		3
3109bcc9736cSJeff Kirsher #define EEPROM_DATA_SUBSYS_ID		4
3110bcc9736cSJeff Kirsher #define EEPROM_DATA_SUBSYS_VEN_ID	5
3111bcc9736cSJeff Kirsher #define EEPROM_DATA_PM_CAP		6
3112bcc9736cSJeff Kirsher 
3113bcc9736cSJeff Kirsher /* User defined EEPROM data */
3114bcc9736cSJeff Kirsher #define EEPROM_DATA_OTHER_MAC_ADDR	9
3115bcc9736cSJeff Kirsher 
3116bcc9736cSJeff Kirsher /**
3117bcc9736cSJeff Kirsher  * eeprom_read - read from AT93C46 EEPROM
3118bcc9736cSJeff Kirsher  * @hw: 	The hardware instance.
3119bcc9736cSJeff Kirsher  * @reg:	The register offset.
3120bcc9736cSJeff Kirsher  *
3121bcc9736cSJeff Kirsher  * This function reads a word from the AT93C46 EEPROM.
3122bcc9736cSJeff Kirsher  *
3123bcc9736cSJeff Kirsher  * Return the data value.
3124bcc9736cSJeff Kirsher  */
3125bcc9736cSJeff Kirsher static u16 eeprom_read(struct ksz_hw *hw, u8 reg)
3126bcc9736cSJeff Kirsher {
3127bcc9736cSJeff Kirsher 	u16 data;
3128bcc9736cSJeff Kirsher 
3129bcc9736cSJeff Kirsher 	raise_gpio(hw, EEPROM_ACCESS_ENABLE | EEPROM_CHIP_SELECT);
3130bcc9736cSJeff Kirsher 
3131bcc9736cSJeff Kirsher 	spi_reg(hw, AT93C_READ, reg);
3132bcc9736cSJeff Kirsher 	data = spi_r(hw);
3133bcc9736cSJeff Kirsher 
3134bcc9736cSJeff Kirsher 	drop_gpio(hw, EEPROM_ACCESS_ENABLE | EEPROM_CHIP_SELECT);
3135bcc9736cSJeff Kirsher 
3136bcc9736cSJeff Kirsher 	return data;
3137bcc9736cSJeff Kirsher }
3138bcc9736cSJeff Kirsher 
3139bcc9736cSJeff Kirsher /**
3140bcc9736cSJeff Kirsher  * eeprom_write - write to AT93C46 EEPROM
3141bcc9736cSJeff Kirsher  * @hw: 	The hardware instance.
3142bcc9736cSJeff Kirsher  * @reg:	The register offset.
3143bcc9736cSJeff Kirsher  * @data:	The data value.
3144bcc9736cSJeff Kirsher  *
3145bcc9736cSJeff Kirsher  * This procedure writes a word to the AT93C46 EEPROM.
3146bcc9736cSJeff Kirsher  */
3147bcc9736cSJeff Kirsher static void eeprom_write(struct ksz_hw *hw, u8 reg, u16 data)
3148bcc9736cSJeff Kirsher {
3149bcc9736cSJeff Kirsher 	int timeout;
3150bcc9736cSJeff Kirsher 
3151bcc9736cSJeff Kirsher 	raise_gpio(hw, EEPROM_ACCESS_ENABLE | EEPROM_CHIP_SELECT);
3152bcc9736cSJeff Kirsher 
3153bcc9736cSJeff Kirsher 	/* Enable write. */
3154bcc9736cSJeff Kirsher 	spi_reg(hw, AT93C_CODE, AT93C_WR_ON);
3155bcc9736cSJeff Kirsher 	drop_gpio(hw, EEPROM_CHIP_SELECT);
3156bcc9736cSJeff Kirsher 	udelay(1);
3157bcc9736cSJeff Kirsher 
3158bcc9736cSJeff Kirsher 	/* Erase the register. */
3159bcc9736cSJeff Kirsher 	raise_gpio(hw, EEPROM_CHIP_SELECT);
3160bcc9736cSJeff Kirsher 	spi_reg(hw, AT93C_ERASE, reg);
3161bcc9736cSJeff Kirsher 	drop_gpio(hw, EEPROM_CHIP_SELECT);
3162bcc9736cSJeff Kirsher 	udelay(1);
3163bcc9736cSJeff Kirsher 
3164bcc9736cSJeff Kirsher 	/* Check operation complete. */
3165bcc9736cSJeff Kirsher 	raise_gpio(hw, EEPROM_CHIP_SELECT);
3166bcc9736cSJeff Kirsher 	timeout = 8;
3167bcc9736cSJeff Kirsher 	mdelay(2);
3168bcc9736cSJeff Kirsher 	do {
3169bcc9736cSJeff Kirsher 		mdelay(1);
3170bcc9736cSJeff Kirsher 	} while (!state_gpio(hw, EEPROM_DATA_IN) && --timeout);
3171bcc9736cSJeff Kirsher 	drop_gpio(hw, EEPROM_CHIP_SELECT);
3172bcc9736cSJeff Kirsher 	udelay(1);
3173bcc9736cSJeff Kirsher 
3174bcc9736cSJeff Kirsher 	/* Write the register. */
3175bcc9736cSJeff Kirsher 	raise_gpio(hw, EEPROM_CHIP_SELECT);
3176bcc9736cSJeff Kirsher 	spi_reg(hw, AT93C_WRITE, reg);
3177bcc9736cSJeff Kirsher 	spi_w(hw, data);
3178bcc9736cSJeff Kirsher 	drop_gpio(hw, EEPROM_CHIP_SELECT);
3179bcc9736cSJeff Kirsher 	udelay(1);
3180bcc9736cSJeff Kirsher 
3181bcc9736cSJeff Kirsher 	/* Check operation complete. */
3182bcc9736cSJeff Kirsher 	raise_gpio(hw, EEPROM_CHIP_SELECT);
3183bcc9736cSJeff Kirsher 	timeout = 8;
3184bcc9736cSJeff Kirsher 	mdelay(2);
3185bcc9736cSJeff Kirsher 	do {
3186bcc9736cSJeff Kirsher 		mdelay(1);
3187bcc9736cSJeff Kirsher 	} while (!state_gpio(hw, EEPROM_DATA_IN) && --timeout);
3188bcc9736cSJeff Kirsher 	drop_gpio(hw, EEPROM_CHIP_SELECT);
3189bcc9736cSJeff Kirsher 	udelay(1);
3190bcc9736cSJeff Kirsher 
3191bcc9736cSJeff Kirsher 	/* Disable write. */
3192bcc9736cSJeff Kirsher 	raise_gpio(hw, EEPROM_CHIP_SELECT);
3193bcc9736cSJeff Kirsher 	spi_reg(hw, AT93C_CODE, AT93C_WR_OFF);
3194bcc9736cSJeff Kirsher 
3195bcc9736cSJeff Kirsher 	drop_gpio(hw, EEPROM_ACCESS_ENABLE | EEPROM_CHIP_SELECT);
3196bcc9736cSJeff Kirsher }
3197bcc9736cSJeff Kirsher 
3198bcc9736cSJeff Kirsher /*
3199bcc9736cSJeff Kirsher  * Link detection routines
3200bcc9736cSJeff Kirsher  */
3201bcc9736cSJeff Kirsher 
3202bcc9736cSJeff Kirsher static u16 advertised_flow_ctrl(struct ksz_port *port, u16 ctrl)
3203bcc9736cSJeff Kirsher {
3204bcc9736cSJeff Kirsher 	ctrl &= ~PORT_AUTO_NEG_SYM_PAUSE;
3205bcc9736cSJeff Kirsher 	switch (port->flow_ctrl) {
3206bcc9736cSJeff Kirsher 	case PHY_FLOW_CTRL:
3207bcc9736cSJeff Kirsher 		ctrl |= PORT_AUTO_NEG_SYM_PAUSE;
3208bcc9736cSJeff Kirsher 		break;
3209bcc9736cSJeff Kirsher 	/* Not supported. */
3210bcc9736cSJeff Kirsher 	case PHY_TX_ONLY:
3211bcc9736cSJeff Kirsher 	case PHY_RX_ONLY:
3212bcc9736cSJeff Kirsher 	default:
3213bcc9736cSJeff Kirsher 		break;
3214bcc9736cSJeff Kirsher 	}
3215bcc9736cSJeff Kirsher 	return ctrl;
3216bcc9736cSJeff Kirsher }
3217bcc9736cSJeff Kirsher 
3218bcc9736cSJeff Kirsher static void set_flow_ctrl(struct ksz_hw *hw, int rx, int tx)
3219bcc9736cSJeff Kirsher {
3220bcc9736cSJeff Kirsher 	u32 rx_cfg;
3221bcc9736cSJeff Kirsher 	u32 tx_cfg;
3222bcc9736cSJeff Kirsher 
3223bcc9736cSJeff Kirsher 	rx_cfg = hw->rx_cfg;
3224bcc9736cSJeff Kirsher 	tx_cfg = hw->tx_cfg;
3225bcc9736cSJeff Kirsher 	if (rx)
3226bcc9736cSJeff Kirsher 		hw->rx_cfg |= DMA_RX_FLOW_ENABLE;
3227bcc9736cSJeff Kirsher 	else
3228bcc9736cSJeff Kirsher 		hw->rx_cfg &= ~DMA_RX_FLOW_ENABLE;
3229bcc9736cSJeff Kirsher 	if (tx)
3230bcc9736cSJeff Kirsher 		hw->tx_cfg |= DMA_TX_FLOW_ENABLE;
3231bcc9736cSJeff Kirsher 	else
3232bcc9736cSJeff Kirsher 		hw->tx_cfg &= ~DMA_TX_FLOW_ENABLE;
3233bcc9736cSJeff Kirsher 	if (hw->enabled) {
3234bcc9736cSJeff Kirsher 		if (rx_cfg != hw->rx_cfg)
3235bcc9736cSJeff Kirsher 			writel(hw->rx_cfg, hw->io + KS_DMA_RX_CTRL);
3236bcc9736cSJeff Kirsher 		if (tx_cfg != hw->tx_cfg)
3237bcc9736cSJeff Kirsher 			writel(hw->tx_cfg, hw->io + KS_DMA_TX_CTRL);
3238bcc9736cSJeff Kirsher 	}
3239bcc9736cSJeff Kirsher }
3240bcc9736cSJeff Kirsher 
3241bcc9736cSJeff Kirsher static void determine_flow_ctrl(struct ksz_hw *hw, struct ksz_port *port,
3242bcc9736cSJeff Kirsher 	u16 local, u16 remote)
3243bcc9736cSJeff Kirsher {
3244bcc9736cSJeff Kirsher 	int rx;
3245bcc9736cSJeff Kirsher 	int tx;
3246bcc9736cSJeff Kirsher 
3247bcc9736cSJeff Kirsher 	if (hw->overrides & PAUSE_FLOW_CTRL)
3248bcc9736cSJeff Kirsher 		return;
3249bcc9736cSJeff Kirsher 
3250bcc9736cSJeff Kirsher 	rx = tx = 0;
3251bcc9736cSJeff Kirsher 	if (port->force_link)
3252bcc9736cSJeff Kirsher 		rx = tx = 1;
3253bcc9736cSJeff Kirsher 	if (remote & PHY_AUTO_NEG_SYM_PAUSE) {
3254bcc9736cSJeff Kirsher 		if (local & PHY_AUTO_NEG_SYM_PAUSE) {
3255bcc9736cSJeff Kirsher 			rx = tx = 1;
3256bcc9736cSJeff Kirsher 		} else if ((remote & PHY_AUTO_NEG_ASYM_PAUSE) &&
3257bcc9736cSJeff Kirsher 				(local & PHY_AUTO_NEG_PAUSE) ==
3258bcc9736cSJeff Kirsher 				PHY_AUTO_NEG_ASYM_PAUSE) {
3259bcc9736cSJeff Kirsher 			tx = 1;
3260bcc9736cSJeff Kirsher 		}
3261bcc9736cSJeff Kirsher 	} else if (remote & PHY_AUTO_NEG_ASYM_PAUSE) {
3262bcc9736cSJeff Kirsher 		if ((local & PHY_AUTO_NEG_PAUSE) == PHY_AUTO_NEG_PAUSE)
3263bcc9736cSJeff Kirsher 			rx = 1;
3264bcc9736cSJeff Kirsher 	}
3265bcc9736cSJeff Kirsher 	if (!hw->ksz_switch)
3266bcc9736cSJeff Kirsher 		set_flow_ctrl(hw, rx, tx);
3267bcc9736cSJeff Kirsher }
3268bcc9736cSJeff Kirsher 
3269bcc9736cSJeff Kirsher static inline void port_cfg_change(struct ksz_hw *hw, struct ksz_port *port,
3270bcc9736cSJeff Kirsher 	struct ksz_port_info *info, u16 link_status)
3271bcc9736cSJeff Kirsher {
3272bcc9736cSJeff Kirsher 	if ((hw->features & HALF_DUPLEX_SIGNAL_BUG) &&
3273bcc9736cSJeff Kirsher 			!(hw->overrides & PAUSE_FLOW_CTRL)) {
3274bcc9736cSJeff Kirsher 		u32 cfg = hw->tx_cfg;
3275bcc9736cSJeff Kirsher 
3276bcc9736cSJeff Kirsher 		/* Disable flow control in the half duplex mode. */
3277bcc9736cSJeff Kirsher 		if (1 == info->duplex)
3278bcc9736cSJeff Kirsher 			hw->tx_cfg &= ~DMA_TX_FLOW_ENABLE;
3279bcc9736cSJeff Kirsher 		if (hw->enabled && cfg != hw->tx_cfg)
3280bcc9736cSJeff Kirsher 			writel(hw->tx_cfg, hw->io + KS_DMA_TX_CTRL);
3281bcc9736cSJeff Kirsher 	}
3282bcc9736cSJeff Kirsher }
3283bcc9736cSJeff Kirsher 
3284bcc9736cSJeff Kirsher /**
3285bcc9736cSJeff Kirsher  * port_get_link_speed - get current link status
3286bcc9736cSJeff Kirsher  * @port: 	The port instance.
3287bcc9736cSJeff Kirsher  *
3288bcc9736cSJeff Kirsher  * This routine reads PHY registers to determine the current link status of the
3289bcc9736cSJeff Kirsher  * switch ports.
3290bcc9736cSJeff Kirsher  */
3291bcc9736cSJeff Kirsher static void port_get_link_speed(struct ksz_port *port)
3292bcc9736cSJeff Kirsher {
3293bcc9736cSJeff Kirsher 	uint interrupt;
3294bcc9736cSJeff Kirsher 	struct ksz_port_info *info;
3295bcc9736cSJeff Kirsher 	struct ksz_port_info *linked = NULL;
3296bcc9736cSJeff Kirsher 	struct ksz_hw *hw = port->hw;
3297bcc9736cSJeff Kirsher 	u16 data;
3298bcc9736cSJeff Kirsher 	u16 status;
3299bcc9736cSJeff Kirsher 	u8 local;
3300bcc9736cSJeff Kirsher 	u8 remote;
3301bcc9736cSJeff Kirsher 	int i;
3302bcc9736cSJeff Kirsher 	int p;
3303bcc9736cSJeff Kirsher 	int change = 0;
3304bcc9736cSJeff Kirsher 
3305bcc9736cSJeff Kirsher 	interrupt = hw_block_intr(hw);
3306bcc9736cSJeff Kirsher 
3307bcc9736cSJeff Kirsher 	for (i = 0, p = port->first_port; i < port->port_cnt; i++, p++) {
3308bcc9736cSJeff Kirsher 		info = &hw->port_info[p];
3309bcc9736cSJeff Kirsher 		port_r16(hw, p, KS884X_PORT_CTRL_4_OFFSET, &data);
3310bcc9736cSJeff Kirsher 		port_r16(hw, p, KS884X_PORT_STATUS_OFFSET, &status);
3311bcc9736cSJeff Kirsher 
3312bcc9736cSJeff Kirsher 		/*
3313bcc9736cSJeff Kirsher 		 * Link status is changing all the time even when there is no
3314bcc9736cSJeff Kirsher 		 * cable connection!
3315bcc9736cSJeff Kirsher 		 */
3316bcc9736cSJeff Kirsher 		remote = status & (PORT_AUTO_NEG_COMPLETE |
3317bcc9736cSJeff Kirsher 			PORT_STATUS_LINK_GOOD);
3318bcc9736cSJeff Kirsher 		local = (u8) data;
3319bcc9736cSJeff Kirsher 
3320bcc9736cSJeff Kirsher 		/* No change to status. */
3321bcc9736cSJeff Kirsher 		if (local == info->advertised && remote == info->partner)
3322bcc9736cSJeff Kirsher 			continue;
3323bcc9736cSJeff Kirsher 
3324bcc9736cSJeff Kirsher 		info->advertised = local;
3325bcc9736cSJeff Kirsher 		info->partner = remote;
3326bcc9736cSJeff Kirsher 		if (status & PORT_STATUS_LINK_GOOD) {
3327bcc9736cSJeff Kirsher 
3328bcc9736cSJeff Kirsher 			/* Remember the first linked port. */
3329bcc9736cSJeff Kirsher 			if (!linked)
3330bcc9736cSJeff Kirsher 				linked = info;
3331bcc9736cSJeff Kirsher 
3332bcc9736cSJeff Kirsher 			info->tx_rate = 10 * TX_RATE_UNIT;
3333bcc9736cSJeff Kirsher 			if (status & PORT_STATUS_SPEED_100MBIT)
3334bcc9736cSJeff Kirsher 				info->tx_rate = 100 * TX_RATE_UNIT;
3335bcc9736cSJeff Kirsher 
3336bcc9736cSJeff Kirsher 			info->duplex = 1;
3337bcc9736cSJeff Kirsher 			if (status & PORT_STATUS_FULL_DUPLEX)
3338bcc9736cSJeff Kirsher 				info->duplex = 2;
3339bcc9736cSJeff Kirsher 
3340bcc9736cSJeff Kirsher 			if (media_connected != info->state) {
3341bcc9736cSJeff Kirsher 				hw_r_phy(hw, p, KS884X_PHY_AUTO_NEG_OFFSET,
3342bcc9736cSJeff Kirsher 					&data);
3343bcc9736cSJeff Kirsher 				hw_r_phy(hw, p, KS884X_PHY_REMOTE_CAP_OFFSET,
3344bcc9736cSJeff Kirsher 					&status);
3345bcc9736cSJeff Kirsher 				determine_flow_ctrl(hw, port, data, status);
3346bcc9736cSJeff Kirsher 				if (hw->ksz_switch) {
3347bcc9736cSJeff Kirsher 					port_cfg_back_pressure(hw, p,
3348bcc9736cSJeff Kirsher 						(1 == info->duplex));
3349bcc9736cSJeff Kirsher 				}
3350bcc9736cSJeff Kirsher 				change |= 1 << i;
3351bcc9736cSJeff Kirsher 				port_cfg_change(hw, port, info, status);
3352bcc9736cSJeff Kirsher 			}
3353bcc9736cSJeff Kirsher 			info->state = media_connected;
3354bcc9736cSJeff Kirsher 		} else {
3355bcc9736cSJeff Kirsher 			if (media_disconnected != info->state) {
3356bcc9736cSJeff Kirsher 				change |= 1 << i;
3357bcc9736cSJeff Kirsher 
3358bcc9736cSJeff Kirsher 				/* Indicate the link just goes down. */
3359bcc9736cSJeff Kirsher 				hw->port_mib[p].link_down = 1;
3360bcc9736cSJeff Kirsher 			}
3361bcc9736cSJeff Kirsher 			info->state = media_disconnected;
3362bcc9736cSJeff Kirsher 		}
3363bcc9736cSJeff Kirsher 		hw->port_mib[p].state = (u8) info->state;
3364bcc9736cSJeff Kirsher 	}
3365bcc9736cSJeff Kirsher 
3366bcc9736cSJeff Kirsher 	if (linked && media_disconnected == port->linked->state)
3367bcc9736cSJeff Kirsher 		port->linked = linked;
3368bcc9736cSJeff Kirsher 
3369bcc9736cSJeff Kirsher 	hw_restore_intr(hw, interrupt);
3370bcc9736cSJeff Kirsher }
3371bcc9736cSJeff Kirsher 
3372bcc9736cSJeff Kirsher #define PHY_RESET_TIMEOUT		10
3373bcc9736cSJeff Kirsher 
3374bcc9736cSJeff Kirsher /**
3375bcc9736cSJeff Kirsher  * port_set_link_speed - set port speed
3376bcc9736cSJeff Kirsher  * @port: 	The port instance.
3377bcc9736cSJeff Kirsher  *
3378bcc9736cSJeff Kirsher  * This routine sets the link speed of the switch ports.
3379bcc9736cSJeff Kirsher  */
3380bcc9736cSJeff Kirsher static void port_set_link_speed(struct ksz_port *port)
3381bcc9736cSJeff Kirsher {
3382bcc9736cSJeff Kirsher 	struct ksz_port_info *info;
3383bcc9736cSJeff Kirsher 	struct ksz_hw *hw = port->hw;
3384bcc9736cSJeff Kirsher 	u16 data;
3385bcc9736cSJeff Kirsher 	u16 cfg;
3386bcc9736cSJeff Kirsher 	u8 status;
3387bcc9736cSJeff Kirsher 	int i;
3388bcc9736cSJeff Kirsher 	int p;
3389bcc9736cSJeff Kirsher 
3390bcc9736cSJeff Kirsher 	for (i = 0, p = port->first_port; i < port->port_cnt; i++, p++) {
3391bcc9736cSJeff Kirsher 		info = &hw->port_info[p];
3392bcc9736cSJeff Kirsher 
3393bcc9736cSJeff Kirsher 		port_r16(hw, p, KS884X_PORT_CTRL_4_OFFSET, &data);
3394bcc9736cSJeff Kirsher 		port_r8(hw, p, KS884X_PORT_STATUS_OFFSET, &status);
3395bcc9736cSJeff Kirsher 
3396bcc9736cSJeff Kirsher 		cfg = 0;
3397bcc9736cSJeff Kirsher 		if (status & PORT_STATUS_LINK_GOOD)
3398bcc9736cSJeff Kirsher 			cfg = data;
3399bcc9736cSJeff Kirsher 
3400bcc9736cSJeff Kirsher 		data |= PORT_AUTO_NEG_ENABLE;
3401bcc9736cSJeff Kirsher 		data = advertised_flow_ctrl(port, data);
3402bcc9736cSJeff Kirsher 
3403bcc9736cSJeff Kirsher 		data |= PORT_AUTO_NEG_100BTX_FD | PORT_AUTO_NEG_100BTX |
3404bcc9736cSJeff Kirsher 			PORT_AUTO_NEG_10BT_FD | PORT_AUTO_NEG_10BT;
3405bcc9736cSJeff Kirsher 
3406bcc9736cSJeff Kirsher 		/* Check if manual configuration is specified by the user. */
3407bcc9736cSJeff Kirsher 		if (port->speed || port->duplex) {
3408bcc9736cSJeff Kirsher 			if (10 == port->speed)
3409bcc9736cSJeff Kirsher 				data &= ~(PORT_AUTO_NEG_100BTX_FD |
3410bcc9736cSJeff Kirsher 					PORT_AUTO_NEG_100BTX);
3411bcc9736cSJeff Kirsher 			else if (100 == port->speed)
3412bcc9736cSJeff Kirsher 				data &= ~(PORT_AUTO_NEG_10BT_FD |
3413bcc9736cSJeff Kirsher 					PORT_AUTO_NEG_10BT);
3414bcc9736cSJeff Kirsher 			if (1 == port->duplex)
3415bcc9736cSJeff Kirsher 				data &= ~(PORT_AUTO_NEG_100BTX_FD |
3416bcc9736cSJeff Kirsher 					PORT_AUTO_NEG_10BT_FD);
3417bcc9736cSJeff Kirsher 			else if (2 == port->duplex)
3418bcc9736cSJeff Kirsher 				data &= ~(PORT_AUTO_NEG_100BTX |
3419bcc9736cSJeff Kirsher 					PORT_AUTO_NEG_10BT);
3420bcc9736cSJeff Kirsher 		}
3421bcc9736cSJeff Kirsher 		if (data != cfg) {
3422bcc9736cSJeff Kirsher 			data |= PORT_AUTO_NEG_RESTART;
3423bcc9736cSJeff Kirsher 			port_w16(hw, p, KS884X_PORT_CTRL_4_OFFSET, data);
3424bcc9736cSJeff Kirsher 		}
3425bcc9736cSJeff Kirsher 	}
3426bcc9736cSJeff Kirsher }
3427bcc9736cSJeff Kirsher 
3428bcc9736cSJeff Kirsher /**
3429bcc9736cSJeff Kirsher  * port_force_link_speed - force port speed
3430bcc9736cSJeff Kirsher  * @port: 	The port instance.
3431bcc9736cSJeff Kirsher  *
3432bcc9736cSJeff Kirsher  * This routine forces the link speed of the switch ports.
3433bcc9736cSJeff Kirsher  */
3434bcc9736cSJeff Kirsher static void port_force_link_speed(struct ksz_port *port)
3435bcc9736cSJeff Kirsher {
3436bcc9736cSJeff Kirsher 	struct ksz_hw *hw = port->hw;
3437bcc9736cSJeff Kirsher 	u16 data;
3438bcc9736cSJeff Kirsher 	int i;
3439bcc9736cSJeff Kirsher 	int phy;
3440bcc9736cSJeff Kirsher 	int p;
3441bcc9736cSJeff Kirsher 
3442bcc9736cSJeff Kirsher 	for (i = 0, p = port->first_port; i < port->port_cnt; i++, p++) {
3443bcc9736cSJeff Kirsher 		phy = KS884X_PHY_1_CTRL_OFFSET + p * PHY_CTRL_INTERVAL;
3444bcc9736cSJeff Kirsher 		hw_r_phy_ctrl(hw, phy, &data);
3445bcc9736cSJeff Kirsher 
3446bcc9736cSJeff Kirsher 		data &= ~PHY_AUTO_NEG_ENABLE;
3447bcc9736cSJeff Kirsher 
3448bcc9736cSJeff Kirsher 		if (10 == port->speed)
3449bcc9736cSJeff Kirsher 			data &= ~PHY_SPEED_100MBIT;
3450bcc9736cSJeff Kirsher 		else if (100 == port->speed)
3451bcc9736cSJeff Kirsher 			data |= PHY_SPEED_100MBIT;
3452bcc9736cSJeff Kirsher 		if (1 == port->duplex)
3453bcc9736cSJeff Kirsher 			data &= ~PHY_FULL_DUPLEX;
3454bcc9736cSJeff Kirsher 		else if (2 == port->duplex)
3455bcc9736cSJeff Kirsher 			data |= PHY_FULL_DUPLEX;
3456bcc9736cSJeff Kirsher 		hw_w_phy_ctrl(hw, phy, data);
3457bcc9736cSJeff Kirsher 	}
3458bcc9736cSJeff Kirsher }
3459bcc9736cSJeff Kirsher 
3460bcc9736cSJeff Kirsher static void port_set_power_saving(struct ksz_port *port, int enable)
3461bcc9736cSJeff Kirsher {
3462bcc9736cSJeff Kirsher 	struct ksz_hw *hw = port->hw;
3463bcc9736cSJeff Kirsher 	int i;
3464bcc9736cSJeff Kirsher 	int p;
3465bcc9736cSJeff Kirsher 
3466bcc9736cSJeff Kirsher 	for (i = 0, p = port->first_port; i < port->port_cnt; i++, p++)
3467bcc9736cSJeff Kirsher 		port_cfg(hw, p,
3468bcc9736cSJeff Kirsher 			KS884X_PORT_CTRL_4_OFFSET, PORT_POWER_DOWN, enable);
3469bcc9736cSJeff Kirsher }
3470bcc9736cSJeff Kirsher 
3471bcc9736cSJeff Kirsher /*
3472bcc9736cSJeff Kirsher  * KSZ8841 power management functions
3473bcc9736cSJeff Kirsher  */
3474bcc9736cSJeff Kirsher 
3475bcc9736cSJeff Kirsher /**
3476bcc9736cSJeff Kirsher  * hw_chk_wol_pme_status - check PMEN pin
3477bcc9736cSJeff Kirsher  * @hw: 	The hardware instance.
3478bcc9736cSJeff Kirsher  *
3479bcc9736cSJeff Kirsher  * This function is used to check PMEN pin is asserted.
3480bcc9736cSJeff Kirsher  *
3481bcc9736cSJeff Kirsher  * Return 1 if PMEN pin is asserted; otherwise, 0.
3482bcc9736cSJeff Kirsher  */
3483bcc9736cSJeff Kirsher static int hw_chk_wol_pme_status(struct ksz_hw *hw)
3484bcc9736cSJeff Kirsher {
3485bcc9736cSJeff Kirsher 	struct dev_info *hw_priv = container_of(hw, struct dev_info, hw);
3486bcc9736cSJeff Kirsher 	struct pci_dev *pdev = hw_priv->pdev;
3487bcc9736cSJeff Kirsher 	u16 data;
3488bcc9736cSJeff Kirsher 
3489bcc9736cSJeff Kirsher 	if (!pdev->pm_cap)
3490bcc9736cSJeff Kirsher 		return 0;
3491bcc9736cSJeff Kirsher 	pci_read_config_word(pdev, pdev->pm_cap + PCI_PM_CTRL, &data);
3492bcc9736cSJeff Kirsher 	return (data & PCI_PM_CTRL_PME_STATUS) == PCI_PM_CTRL_PME_STATUS;
3493bcc9736cSJeff Kirsher }
3494bcc9736cSJeff Kirsher 
3495bcc9736cSJeff Kirsher /**
3496bcc9736cSJeff Kirsher  * hw_clr_wol_pme_status - clear PMEN pin
3497bcc9736cSJeff Kirsher  * @hw: 	The hardware instance.
3498bcc9736cSJeff Kirsher  *
3499bcc9736cSJeff Kirsher  * This routine is used to clear PME_Status to deassert PMEN pin.
3500bcc9736cSJeff Kirsher  */
3501bcc9736cSJeff Kirsher static void hw_clr_wol_pme_status(struct ksz_hw *hw)
3502bcc9736cSJeff Kirsher {
3503bcc9736cSJeff Kirsher 	struct dev_info *hw_priv = container_of(hw, struct dev_info, hw);
3504bcc9736cSJeff Kirsher 	struct pci_dev *pdev = hw_priv->pdev;
3505bcc9736cSJeff Kirsher 	u16 data;
3506bcc9736cSJeff Kirsher 
3507bcc9736cSJeff Kirsher 	if (!pdev->pm_cap)
3508bcc9736cSJeff Kirsher 		return;
3509bcc9736cSJeff Kirsher 
3510bcc9736cSJeff Kirsher 	/* Clear PME_Status to deassert PMEN pin. */
3511bcc9736cSJeff Kirsher 	pci_read_config_word(pdev, pdev->pm_cap + PCI_PM_CTRL, &data);
3512bcc9736cSJeff Kirsher 	data |= PCI_PM_CTRL_PME_STATUS;
3513bcc9736cSJeff Kirsher 	pci_write_config_word(pdev, pdev->pm_cap + PCI_PM_CTRL, data);
3514bcc9736cSJeff Kirsher }
3515bcc9736cSJeff Kirsher 
3516bcc9736cSJeff Kirsher /**
3517bcc9736cSJeff Kirsher  * hw_cfg_wol_pme - enable or disable Wake-on-LAN
3518bcc9736cSJeff Kirsher  * @hw: 	The hardware instance.
3519bcc9736cSJeff Kirsher  * @set:	The flag indicating whether to enable or disable.
3520bcc9736cSJeff Kirsher  *
3521bcc9736cSJeff Kirsher  * This routine is used to enable or disable Wake-on-LAN.
3522bcc9736cSJeff Kirsher  */
3523bcc9736cSJeff Kirsher static void hw_cfg_wol_pme(struct ksz_hw *hw, int set)
3524bcc9736cSJeff Kirsher {
3525bcc9736cSJeff Kirsher 	struct dev_info *hw_priv = container_of(hw, struct dev_info, hw);
3526bcc9736cSJeff Kirsher 	struct pci_dev *pdev = hw_priv->pdev;
3527bcc9736cSJeff Kirsher 	u16 data;
3528bcc9736cSJeff Kirsher 
3529bcc9736cSJeff Kirsher 	if (!pdev->pm_cap)
3530bcc9736cSJeff Kirsher 		return;
3531bcc9736cSJeff Kirsher 	pci_read_config_word(pdev, pdev->pm_cap + PCI_PM_CTRL, &data);
3532bcc9736cSJeff Kirsher 	data &= ~PCI_PM_CTRL_STATE_MASK;
3533bcc9736cSJeff Kirsher 	if (set)
3534bcc9736cSJeff Kirsher 		data |= PCI_PM_CTRL_PME_ENABLE | PCI_D3hot;
3535bcc9736cSJeff Kirsher 	else
3536bcc9736cSJeff Kirsher 		data &= ~PCI_PM_CTRL_PME_ENABLE;
3537bcc9736cSJeff Kirsher 	pci_write_config_word(pdev, pdev->pm_cap + PCI_PM_CTRL, data);
3538bcc9736cSJeff Kirsher }
3539bcc9736cSJeff Kirsher 
3540bcc9736cSJeff Kirsher /**
3541bcc9736cSJeff Kirsher  * hw_cfg_wol - configure Wake-on-LAN features
3542bcc9736cSJeff Kirsher  * @hw: 	The hardware instance.
3543bcc9736cSJeff Kirsher  * @frame:	The pattern frame bit.
3544bcc9736cSJeff Kirsher  * @set:	The flag indicating whether to enable or disable.
3545bcc9736cSJeff Kirsher  *
3546bcc9736cSJeff Kirsher  * This routine is used to enable or disable certain Wake-on-LAN features.
3547bcc9736cSJeff Kirsher  */
3548bcc9736cSJeff Kirsher static void hw_cfg_wol(struct ksz_hw *hw, u16 frame, int set)
3549bcc9736cSJeff Kirsher {
3550bcc9736cSJeff Kirsher 	u16 data;
3551bcc9736cSJeff Kirsher 
3552bcc9736cSJeff Kirsher 	data = readw(hw->io + KS8841_WOL_CTRL_OFFSET);
3553bcc9736cSJeff Kirsher 	if (set)
3554bcc9736cSJeff Kirsher 		data |= frame;
3555bcc9736cSJeff Kirsher 	else
3556bcc9736cSJeff Kirsher 		data &= ~frame;
3557bcc9736cSJeff Kirsher 	writew(data, hw->io + KS8841_WOL_CTRL_OFFSET);
3558bcc9736cSJeff Kirsher }
3559bcc9736cSJeff Kirsher 
3560bcc9736cSJeff Kirsher /**
3561bcc9736cSJeff Kirsher  * hw_set_wol_frame - program Wake-on-LAN pattern
3562bcc9736cSJeff Kirsher  * @hw: 	The hardware instance.
3563bcc9736cSJeff Kirsher  * @i:		The frame index.
3564bcc9736cSJeff Kirsher  * @mask_size:	The size of the mask.
3565bcc9736cSJeff Kirsher  * @mask:	Mask to ignore certain bytes in the pattern.
3566bcc9736cSJeff Kirsher  * @frame_size:	The size of the frame.
3567bcc9736cSJeff Kirsher  * @pattern:	The frame data.
3568bcc9736cSJeff Kirsher  *
3569bcc9736cSJeff Kirsher  * This routine is used to program Wake-on-LAN pattern.
3570bcc9736cSJeff Kirsher  */
3571bcc9736cSJeff Kirsher static void hw_set_wol_frame(struct ksz_hw *hw, int i, uint mask_size,
3572bcc9736cSJeff Kirsher 	const u8 *mask, uint frame_size, const u8 *pattern)
3573bcc9736cSJeff Kirsher {
3574bcc9736cSJeff Kirsher 	int bits;
3575bcc9736cSJeff Kirsher 	int from;
3576bcc9736cSJeff Kirsher 	int len;
3577bcc9736cSJeff Kirsher 	int to;
3578bcc9736cSJeff Kirsher 	u32 crc;
3579bcc9736cSJeff Kirsher 	u8 data[64];
3580bcc9736cSJeff Kirsher 	u8 val = 0;
3581bcc9736cSJeff Kirsher 
3582bcc9736cSJeff Kirsher 	if (frame_size > mask_size * 8)
3583bcc9736cSJeff Kirsher 		frame_size = mask_size * 8;
3584bcc9736cSJeff Kirsher 	if (frame_size > 64)
3585bcc9736cSJeff Kirsher 		frame_size = 64;
3586bcc9736cSJeff Kirsher 
3587bcc9736cSJeff Kirsher 	i *= 0x10;
3588bcc9736cSJeff Kirsher 	writel(0, hw->io + KS8841_WOL_FRAME_BYTE0_OFFSET + i);
3589bcc9736cSJeff Kirsher 	writel(0, hw->io + KS8841_WOL_FRAME_BYTE2_OFFSET + i);
3590bcc9736cSJeff Kirsher 
3591bcc9736cSJeff Kirsher 	bits = len = from = to = 0;
3592bcc9736cSJeff Kirsher 	do {
3593bcc9736cSJeff Kirsher 		if (bits) {
3594bcc9736cSJeff Kirsher 			if ((val & 1))
3595bcc9736cSJeff Kirsher 				data[to++] = pattern[from];
3596bcc9736cSJeff Kirsher 			val >>= 1;
3597bcc9736cSJeff Kirsher 			++from;
3598bcc9736cSJeff Kirsher 			--bits;
3599bcc9736cSJeff Kirsher 		} else {
3600bcc9736cSJeff Kirsher 			val = mask[len];
3601bcc9736cSJeff Kirsher 			writeb(val, hw->io + KS8841_WOL_FRAME_BYTE0_OFFSET + i
3602bcc9736cSJeff Kirsher 				+ len);
3603bcc9736cSJeff Kirsher 			++len;
3604bcc9736cSJeff Kirsher 			if (val)
3605bcc9736cSJeff Kirsher 				bits = 8;
3606bcc9736cSJeff Kirsher 			else
3607bcc9736cSJeff Kirsher 				from += 8;
3608bcc9736cSJeff Kirsher 		}
3609bcc9736cSJeff Kirsher 	} while (from < (int) frame_size);
3610bcc9736cSJeff Kirsher 	if (val) {
3611bcc9736cSJeff Kirsher 		bits = mask[len - 1];
3612bcc9736cSJeff Kirsher 		val <<= (from % 8);
3613bcc9736cSJeff Kirsher 		bits &= ~val;
3614bcc9736cSJeff Kirsher 		writeb(bits, hw->io + KS8841_WOL_FRAME_BYTE0_OFFSET + i + len -
3615bcc9736cSJeff Kirsher 			1);
3616bcc9736cSJeff Kirsher 	}
3617bcc9736cSJeff Kirsher 	crc = ether_crc(to, data);
3618bcc9736cSJeff Kirsher 	writel(crc, hw->io + KS8841_WOL_FRAME_CRC_OFFSET + i);
3619bcc9736cSJeff Kirsher }
3620bcc9736cSJeff Kirsher 
3621bcc9736cSJeff Kirsher /**
3622bcc9736cSJeff Kirsher  * hw_add_wol_arp - add ARP pattern
3623bcc9736cSJeff Kirsher  * @hw: 	The hardware instance.
3624bcc9736cSJeff Kirsher  * @ip_addr:	The IPv4 address assigned to the device.
3625bcc9736cSJeff Kirsher  *
3626bcc9736cSJeff Kirsher  * This routine is used to add ARP pattern for waking up the host.
3627bcc9736cSJeff Kirsher  */
3628bcc9736cSJeff Kirsher static void hw_add_wol_arp(struct ksz_hw *hw, const u8 *ip_addr)
3629bcc9736cSJeff Kirsher {
3630bcc9736cSJeff Kirsher 	static const u8 mask[6] = { 0x3F, 0xF0, 0x3F, 0x00, 0xC0, 0x03 };
3631bcc9736cSJeff Kirsher 	u8 pattern[42] = {
3632bcc9736cSJeff Kirsher 		0xFF, 0xFF, 0xFF, 0xFF, 0xFF, 0xFF,
3633bcc9736cSJeff Kirsher 		0x00, 0x00, 0x00, 0x00, 0x00, 0x00,
3634bcc9736cSJeff Kirsher 		0x08, 0x06,
3635bcc9736cSJeff Kirsher 		0x00, 0x01, 0x08, 0x00, 0x06, 0x04, 0x00, 0x01,
3636bcc9736cSJeff Kirsher 		0x00, 0x00, 0x00, 0x00, 0x00, 0x00,
3637bcc9736cSJeff Kirsher 		0x00, 0x00, 0x00, 0x00,
3638bcc9736cSJeff Kirsher 		0x00, 0x00, 0x00, 0x00, 0x00, 0x00,
3639bcc9736cSJeff Kirsher 		0x00, 0x00, 0x00, 0x00 };
3640bcc9736cSJeff Kirsher 
3641bcc9736cSJeff Kirsher 	memcpy(&pattern[38], ip_addr, 4);
3642bcc9736cSJeff Kirsher 	hw_set_wol_frame(hw, 3, 6, mask, 42, pattern);
3643bcc9736cSJeff Kirsher }
3644bcc9736cSJeff Kirsher 
3645bcc9736cSJeff Kirsher /**
3646bcc9736cSJeff Kirsher  * hw_add_wol_bcast - add broadcast pattern
3647bcc9736cSJeff Kirsher  * @hw: 	The hardware instance.
3648bcc9736cSJeff Kirsher  *
3649bcc9736cSJeff Kirsher  * This routine is used to add broadcast pattern for waking up the host.
3650bcc9736cSJeff Kirsher  */
3651bcc9736cSJeff Kirsher static void hw_add_wol_bcast(struct ksz_hw *hw)
3652bcc9736cSJeff Kirsher {
3653bcc9736cSJeff Kirsher 	static const u8 mask[] = { 0x3F };
3654bcc9736cSJeff Kirsher 	static const u8 pattern[] = { 0xFF, 0xFF, 0xFF, 0xFF, 0xFF, 0xFF };
3655bcc9736cSJeff Kirsher 
36566a3c910cSJoe Perches 	hw_set_wol_frame(hw, 2, 1, mask, ETH_ALEN, pattern);
3657bcc9736cSJeff Kirsher }
3658bcc9736cSJeff Kirsher 
3659bcc9736cSJeff Kirsher /**
3660bcc9736cSJeff Kirsher  * hw_add_wol_mcast - add multicast pattern
3661bcc9736cSJeff Kirsher  * @hw: 	The hardware instance.
3662bcc9736cSJeff Kirsher  *
3663bcc9736cSJeff Kirsher  * This routine is used to add multicast pattern for waking up the host.
3664bcc9736cSJeff Kirsher  *
3665bcc9736cSJeff Kirsher  * It is assumed the multicast packet is the ICMPv6 neighbor solicitation used
3666bcc9736cSJeff Kirsher  * by IPv6 ping command.  Note that multicast packets are filtred through the
3667bcc9736cSJeff Kirsher  * multicast hash table, so not all multicast packets can wake up the host.
3668bcc9736cSJeff Kirsher  */
3669bcc9736cSJeff Kirsher static void hw_add_wol_mcast(struct ksz_hw *hw)
3670bcc9736cSJeff Kirsher {
3671bcc9736cSJeff Kirsher 	static const u8 mask[] = { 0x3F };
3672bcc9736cSJeff Kirsher 	u8 pattern[] = { 0x33, 0x33, 0xFF, 0x00, 0x00, 0x00 };
3673bcc9736cSJeff Kirsher 
3674bcc9736cSJeff Kirsher 	memcpy(&pattern[3], &hw->override_addr[3], 3);
3675bcc9736cSJeff Kirsher 	hw_set_wol_frame(hw, 1, 1, mask, 6, pattern);
3676bcc9736cSJeff Kirsher }
3677bcc9736cSJeff Kirsher 
3678bcc9736cSJeff Kirsher /**
3679bcc9736cSJeff Kirsher  * hw_add_wol_ucast - add unicast pattern
3680bcc9736cSJeff Kirsher  * @hw: 	The hardware instance.
3681bcc9736cSJeff Kirsher  *
3682bcc9736cSJeff Kirsher  * This routine is used to add unicast pattern to wakeup the host.
3683bcc9736cSJeff Kirsher  *
3684bcc9736cSJeff Kirsher  * It is assumed the unicast packet is directed to the device, as the hardware
3685bcc9736cSJeff Kirsher  * can only receive them in normal case.
3686bcc9736cSJeff Kirsher  */
3687bcc9736cSJeff Kirsher static void hw_add_wol_ucast(struct ksz_hw *hw)
3688bcc9736cSJeff Kirsher {
3689bcc9736cSJeff Kirsher 	static const u8 mask[] = { 0x3F };
3690bcc9736cSJeff Kirsher 
36916a3c910cSJoe Perches 	hw_set_wol_frame(hw, 0, 1, mask, ETH_ALEN, hw->override_addr);
3692bcc9736cSJeff Kirsher }
3693bcc9736cSJeff Kirsher 
3694bcc9736cSJeff Kirsher /**
3695bcc9736cSJeff Kirsher  * hw_enable_wol - enable Wake-on-LAN
3696bcc9736cSJeff Kirsher  * @hw: 	The hardware instance.
3697bcc9736cSJeff Kirsher  * @wol_enable:	The Wake-on-LAN settings.
3698bcc9736cSJeff Kirsher  * @net_addr:	The IPv4 address assigned to the device.
3699bcc9736cSJeff Kirsher  *
3700bcc9736cSJeff Kirsher  * This routine is used to enable Wake-on-LAN depending on driver settings.
3701bcc9736cSJeff Kirsher  */
3702bcc9736cSJeff Kirsher static void hw_enable_wol(struct ksz_hw *hw, u32 wol_enable, const u8 *net_addr)
3703bcc9736cSJeff Kirsher {
3704bcc9736cSJeff Kirsher 	hw_cfg_wol(hw, KS8841_WOL_MAGIC_ENABLE, (wol_enable & WAKE_MAGIC));
3705bcc9736cSJeff Kirsher 	hw_cfg_wol(hw, KS8841_WOL_FRAME0_ENABLE, (wol_enable & WAKE_UCAST));
3706bcc9736cSJeff Kirsher 	hw_add_wol_ucast(hw);
3707bcc9736cSJeff Kirsher 	hw_cfg_wol(hw, KS8841_WOL_FRAME1_ENABLE, (wol_enable & WAKE_MCAST));
3708bcc9736cSJeff Kirsher 	hw_add_wol_mcast(hw);
3709bcc9736cSJeff Kirsher 	hw_cfg_wol(hw, KS8841_WOL_FRAME2_ENABLE, (wol_enable & WAKE_BCAST));
3710bcc9736cSJeff Kirsher 	hw_cfg_wol(hw, KS8841_WOL_FRAME3_ENABLE, (wol_enable & WAKE_ARP));
3711bcc9736cSJeff Kirsher 	hw_add_wol_arp(hw, net_addr);
3712bcc9736cSJeff Kirsher }
3713bcc9736cSJeff Kirsher 
3714bcc9736cSJeff Kirsher /**
3715bcc9736cSJeff Kirsher  * hw_init - check driver is correct for the hardware
3716bcc9736cSJeff Kirsher  * @hw: 	The hardware instance.
3717bcc9736cSJeff Kirsher  *
3718bcc9736cSJeff Kirsher  * This function checks the hardware is correct for this driver and sets the
3719bcc9736cSJeff Kirsher  * hardware up for proper initialization.
3720bcc9736cSJeff Kirsher  *
3721bcc9736cSJeff Kirsher  * Return number of ports or 0 if not right.
3722bcc9736cSJeff Kirsher  */
3723bcc9736cSJeff Kirsher static int hw_init(struct ksz_hw *hw)
3724bcc9736cSJeff Kirsher {
3725bcc9736cSJeff Kirsher 	int rc = 0;
3726bcc9736cSJeff Kirsher 	u16 data;
3727bcc9736cSJeff Kirsher 	u16 revision;
3728bcc9736cSJeff Kirsher 
3729bcc9736cSJeff Kirsher 	/* Set bus speed to 125MHz. */
3730bcc9736cSJeff Kirsher 	writew(BUS_SPEED_125_MHZ, hw->io + KS884X_BUS_CTRL_OFFSET);
3731bcc9736cSJeff Kirsher 
3732bcc9736cSJeff Kirsher 	/* Check KSZ884x chip ID. */
3733bcc9736cSJeff Kirsher 	data = readw(hw->io + KS884X_CHIP_ID_OFFSET);
3734bcc9736cSJeff Kirsher 
3735bcc9736cSJeff Kirsher 	revision = (data & KS884X_REVISION_MASK) >> KS884X_REVISION_SHIFT;
3736bcc9736cSJeff Kirsher 	data &= KS884X_CHIP_ID_MASK_41;
3737bcc9736cSJeff Kirsher 	if (REG_CHIP_ID_41 == data)
3738bcc9736cSJeff Kirsher 		rc = 1;
3739bcc9736cSJeff Kirsher 	else if (REG_CHIP_ID_42 == data)
3740bcc9736cSJeff Kirsher 		rc = 2;
3741bcc9736cSJeff Kirsher 	else
3742bcc9736cSJeff Kirsher 		return 0;
3743bcc9736cSJeff Kirsher 
3744bcc9736cSJeff Kirsher 	/* Setup hardware features or bug workarounds. */
3745bcc9736cSJeff Kirsher 	if (revision <= 1) {
3746bcc9736cSJeff Kirsher 		hw->features |= SMALL_PACKET_TX_BUG;
3747bcc9736cSJeff Kirsher 		if (1 == rc)
3748bcc9736cSJeff Kirsher 			hw->features |= HALF_DUPLEX_SIGNAL_BUG;
3749bcc9736cSJeff Kirsher 	}
3750bcc9736cSJeff Kirsher 	return rc;
3751bcc9736cSJeff Kirsher }
3752bcc9736cSJeff Kirsher 
3753bcc9736cSJeff Kirsher /**
3754bcc9736cSJeff Kirsher  * hw_reset - reset the hardware
3755bcc9736cSJeff Kirsher  * @hw: 	The hardware instance.
3756bcc9736cSJeff Kirsher  *
3757bcc9736cSJeff Kirsher  * This routine resets the hardware.
3758bcc9736cSJeff Kirsher  */
3759bcc9736cSJeff Kirsher static void hw_reset(struct ksz_hw *hw)
3760bcc9736cSJeff Kirsher {
3761bcc9736cSJeff Kirsher 	writew(GLOBAL_SOFTWARE_RESET, hw->io + KS884X_GLOBAL_CTRL_OFFSET);
3762bcc9736cSJeff Kirsher 
3763bcc9736cSJeff Kirsher 	/* Wait for device to reset. */
3764bcc9736cSJeff Kirsher 	mdelay(10);
3765bcc9736cSJeff Kirsher 
3766bcc9736cSJeff Kirsher 	/* Write 0 to clear device reset. */
3767bcc9736cSJeff Kirsher 	writew(0, hw->io + KS884X_GLOBAL_CTRL_OFFSET);
3768bcc9736cSJeff Kirsher }
3769bcc9736cSJeff Kirsher 
3770bcc9736cSJeff Kirsher /**
3771bcc9736cSJeff Kirsher  * hw_setup - setup the hardware
3772bcc9736cSJeff Kirsher  * @hw: 	The hardware instance.
3773bcc9736cSJeff Kirsher  *
3774bcc9736cSJeff Kirsher  * This routine setup the hardware for proper operation.
3775bcc9736cSJeff Kirsher  */
3776bcc9736cSJeff Kirsher static void hw_setup(struct ksz_hw *hw)
3777bcc9736cSJeff Kirsher {
3778bcc9736cSJeff Kirsher #if SET_DEFAULT_LED
3779bcc9736cSJeff Kirsher 	u16 data;
3780bcc9736cSJeff Kirsher 
3781bcc9736cSJeff Kirsher 	/* Change default LED mode. */
3782bcc9736cSJeff Kirsher 	data = readw(hw->io + KS8842_SWITCH_CTRL_5_OFFSET);
3783bcc9736cSJeff Kirsher 	data &= ~LED_MODE;
3784bcc9736cSJeff Kirsher 	data |= SET_DEFAULT_LED;
3785bcc9736cSJeff Kirsher 	writew(data, hw->io + KS8842_SWITCH_CTRL_5_OFFSET);
3786bcc9736cSJeff Kirsher #endif
3787bcc9736cSJeff Kirsher 
3788bcc9736cSJeff Kirsher 	/* Setup transmit control. */
3789bcc9736cSJeff Kirsher 	hw->tx_cfg = (DMA_TX_PAD_ENABLE | DMA_TX_CRC_ENABLE |
3790bcc9736cSJeff Kirsher 		(DMA_BURST_DEFAULT << DMA_BURST_SHIFT) | DMA_TX_ENABLE);
3791bcc9736cSJeff Kirsher 
3792bcc9736cSJeff Kirsher 	/* Setup receive control. */
3793bcc9736cSJeff Kirsher 	hw->rx_cfg = (DMA_RX_BROADCAST | DMA_RX_UNICAST |
3794bcc9736cSJeff Kirsher 		(DMA_BURST_DEFAULT << DMA_BURST_SHIFT) | DMA_RX_ENABLE);
3795bcc9736cSJeff Kirsher 	hw->rx_cfg |= KS884X_DMA_RX_MULTICAST;
3796bcc9736cSJeff Kirsher 
3797bcc9736cSJeff Kirsher 	/* Hardware cannot handle UDP packet in IP fragments. */
3798bcc9736cSJeff Kirsher 	hw->rx_cfg |= (DMA_RX_CSUM_TCP | DMA_RX_CSUM_IP);
3799bcc9736cSJeff Kirsher 
3800bcc9736cSJeff Kirsher 	if (hw->all_multi)
3801bcc9736cSJeff Kirsher 		hw->rx_cfg |= DMA_RX_ALL_MULTICAST;
3802bcc9736cSJeff Kirsher 	if (hw->promiscuous)
3803bcc9736cSJeff Kirsher 		hw->rx_cfg |= DMA_RX_PROMISCUOUS;
3804bcc9736cSJeff Kirsher }
3805bcc9736cSJeff Kirsher 
3806bcc9736cSJeff Kirsher /**
3807bcc9736cSJeff Kirsher  * hw_setup_intr - setup interrupt mask
3808bcc9736cSJeff Kirsher  * @hw: 	The hardware instance.
3809bcc9736cSJeff Kirsher  *
3810bcc9736cSJeff Kirsher  * This routine setup the interrupt mask for proper operation.
3811bcc9736cSJeff Kirsher  */
3812bcc9736cSJeff Kirsher static void hw_setup_intr(struct ksz_hw *hw)
3813bcc9736cSJeff Kirsher {
3814bcc9736cSJeff Kirsher 	hw->intr_mask = KS884X_INT_MASK | KS884X_INT_RX_OVERRUN;
3815bcc9736cSJeff Kirsher }
3816bcc9736cSJeff Kirsher 
3817bcc9736cSJeff Kirsher static void ksz_check_desc_num(struct ksz_desc_info *info)
3818bcc9736cSJeff Kirsher {
3819bcc9736cSJeff Kirsher #define MIN_DESC_SHIFT  2
3820bcc9736cSJeff Kirsher 
3821bcc9736cSJeff Kirsher 	int alloc = info->alloc;
3822bcc9736cSJeff Kirsher 	int shift;
3823bcc9736cSJeff Kirsher 
3824bcc9736cSJeff Kirsher 	shift = 0;
3825bcc9736cSJeff Kirsher 	while (!(alloc & 1)) {
3826bcc9736cSJeff Kirsher 		shift++;
3827bcc9736cSJeff Kirsher 		alloc >>= 1;
3828bcc9736cSJeff Kirsher 	}
3829bcc9736cSJeff Kirsher 	if (alloc != 1 || shift < MIN_DESC_SHIFT) {
3830bcc9736cSJeff Kirsher 		pr_alert("Hardware descriptor numbers not right!\n");
3831bcc9736cSJeff Kirsher 		while (alloc) {
3832bcc9736cSJeff Kirsher 			shift++;
3833bcc9736cSJeff Kirsher 			alloc >>= 1;
3834bcc9736cSJeff Kirsher 		}
3835bcc9736cSJeff Kirsher 		if (shift < MIN_DESC_SHIFT)
3836bcc9736cSJeff Kirsher 			shift = MIN_DESC_SHIFT;
3837bcc9736cSJeff Kirsher 		alloc = 1 << shift;
3838bcc9736cSJeff Kirsher 		info->alloc = alloc;
3839bcc9736cSJeff Kirsher 	}
3840bcc9736cSJeff Kirsher 	info->mask = info->alloc - 1;
3841bcc9736cSJeff Kirsher }
3842bcc9736cSJeff Kirsher 
3843bcc9736cSJeff Kirsher static void hw_init_desc(struct ksz_desc_info *desc_info, int transmit)
3844bcc9736cSJeff Kirsher {
3845bcc9736cSJeff Kirsher 	int i;
3846bcc9736cSJeff Kirsher 	u32 phys = desc_info->ring_phys;
3847bcc9736cSJeff Kirsher 	struct ksz_hw_desc *desc = desc_info->ring_virt;
3848bcc9736cSJeff Kirsher 	struct ksz_desc *cur = desc_info->ring;
3849bcc9736cSJeff Kirsher 	struct ksz_desc *previous = NULL;
3850bcc9736cSJeff Kirsher 
3851bcc9736cSJeff Kirsher 	for (i = 0; i < desc_info->alloc; i++) {
3852bcc9736cSJeff Kirsher 		cur->phw = desc++;
3853bcc9736cSJeff Kirsher 		phys += desc_info->size;
3854bcc9736cSJeff Kirsher 		previous = cur++;
3855bcc9736cSJeff Kirsher 		previous->phw->next = cpu_to_le32(phys);
3856bcc9736cSJeff Kirsher 	}
3857bcc9736cSJeff Kirsher 	previous->phw->next = cpu_to_le32(desc_info->ring_phys);
3858bcc9736cSJeff Kirsher 	previous->sw.buf.rx.end_of_ring = 1;
3859bcc9736cSJeff Kirsher 	previous->phw->buf.data = cpu_to_le32(previous->sw.buf.data);
3860bcc9736cSJeff Kirsher 
3861bcc9736cSJeff Kirsher 	desc_info->avail = desc_info->alloc;
3862bcc9736cSJeff Kirsher 	desc_info->last = desc_info->next = 0;
3863bcc9736cSJeff Kirsher 
3864bcc9736cSJeff Kirsher 	desc_info->cur = desc_info->ring;
3865bcc9736cSJeff Kirsher }
3866bcc9736cSJeff Kirsher 
3867bcc9736cSJeff Kirsher /**
3868bcc9736cSJeff Kirsher  * hw_set_desc_base - set descriptor base addresses
3869bcc9736cSJeff Kirsher  * @hw: 	The hardware instance.
3870bcc9736cSJeff Kirsher  * @tx_addr:	The transmit descriptor base.
3871bcc9736cSJeff Kirsher  * @rx_addr:	The receive descriptor base.
3872bcc9736cSJeff Kirsher  *
3873bcc9736cSJeff Kirsher  * This routine programs the descriptor base addresses after reset.
3874bcc9736cSJeff Kirsher  */
3875bcc9736cSJeff Kirsher static void hw_set_desc_base(struct ksz_hw *hw, u32 tx_addr, u32 rx_addr)
3876bcc9736cSJeff Kirsher {
3877bcc9736cSJeff Kirsher 	/* Set base address of Tx/Rx descriptors. */
3878bcc9736cSJeff Kirsher 	writel(tx_addr, hw->io + KS_DMA_TX_ADDR);
3879bcc9736cSJeff Kirsher 	writel(rx_addr, hw->io + KS_DMA_RX_ADDR);
3880bcc9736cSJeff Kirsher }
3881bcc9736cSJeff Kirsher 
3882bcc9736cSJeff Kirsher static void hw_reset_pkts(struct ksz_desc_info *info)
3883bcc9736cSJeff Kirsher {
3884bcc9736cSJeff Kirsher 	info->cur = info->ring;
3885bcc9736cSJeff Kirsher 	info->avail = info->alloc;
3886bcc9736cSJeff Kirsher 	info->last = info->next = 0;
3887bcc9736cSJeff Kirsher }
3888bcc9736cSJeff Kirsher 
3889bcc9736cSJeff Kirsher static inline void hw_resume_rx(struct ksz_hw *hw)
3890bcc9736cSJeff Kirsher {
3891bcc9736cSJeff Kirsher 	writel(DMA_START, hw->io + KS_DMA_RX_START);
3892bcc9736cSJeff Kirsher }
3893bcc9736cSJeff Kirsher 
3894bcc9736cSJeff Kirsher /**
3895bcc9736cSJeff Kirsher  * hw_start_rx - start receiving
3896bcc9736cSJeff Kirsher  * @hw: 	The hardware instance.
3897bcc9736cSJeff Kirsher  *
3898bcc9736cSJeff Kirsher  * This routine starts the receive function of the hardware.
3899bcc9736cSJeff Kirsher  */
3900bcc9736cSJeff Kirsher static void hw_start_rx(struct ksz_hw *hw)
3901bcc9736cSJeff Kirsher {
3902bcc9736cSJeff Kirsher 	writel(hw->rx_cfg, hw->io + KS_DMA_RX_CTRL);
3903bcc9736cSJeff Kirsher 
3904bcc9736cSJeff Kirsher 	/* Notify when the receive stops. */
3905bcc9736cSJeff Kirsher 	hw->intr_mask |= KS884X_INT_RX_STOPPED;
3906bcc9736cSJeff Kirsher 
3907bcc9736cSJeff Kirsher 	writel(DMA_START, hw->io + KS_DMA_RX_START);
3908bcc9736cSJeff Kirsher 	hw_ack_intr(hw, KS884X_INT_RX_STOPPED);
3909bcc9736cSJeff Kirsher 	hw->rx_stop++;
3910bcc9736cSJeff Kirsher 
3911bcc9736cSJeff Kirsher 	/* Variable overflows. */
3912bcc9736cSJeff Kirsher 	if (0 == hw->rx_stop)
3913bcc9736cSJeff Kirsher 		hw->rx_stop = 2;
3914bcc9736cSJeff Kirsher }
3915bcc9736cSJeff Kirsher 
3916bcc9736cSJeff Kirsher /*
3917bcc9736cSJeff Kirsher  * hw_stop_rx - stop receiving
3918bcc9736cSJeff Kirsher  * @hw: 	The hardware instance.
3919bcc9736cSJeff Kirsher  *
3920bcc9736cSJeff Kirsher  * This routine stops the receive function of the hardware.
3921bcc9736cSJeff Kirsher  */
3922bcc9736cSJeff Kirsher static void hw_stop_rx(struct ksz_hw *hw)
3923bcc9736cSJeff Kirsher {
3924bcc9736cSJeff Kirsher 	hw->rx_stop = 0;
3925bcc9736cSJeff Kirsher 	hw_turn_off_intr(hw, KS884X_INT_RX_STOPPED);
3926bcc9736cSJeff Kirsher 	writel((hw->rx_cfg & ~DMA_RX_ENABLE), hw->io + KS_DMA_RX_CTRL);
3927bcc9736cSJeff Kirsher }
3928bcc9736cSJeff Kirsher 
3929bcc9736cSJeff Kirsher /**
3930bcc9736cSJeff Kirsher  * hw_start_tx - start transmitting
3931bcc9736cSJeff Kirsher  * @hw: 	The hardware instance.
3932bcc9736cSJeff Kirsher  *
3933bcc9736cSJeff Kirsher  * This routine starts the transmit function of the hardware.
3934bcc9736cSJeff Kirsher  */
3935bcc9736cSJeff Kirsher static void hw_start_tx(struct ksz_hw *hw)
3936bcc9736cSJeff Kirsher {
3937bcc9736cSJeff Kirsher 	writel(hw->tx_cfg, hw->io + KS_DMA_TX_CTRL);
3938bcc9736cSJeff Kirsher }
3939bcc9736cSJeff Kirsher 
3940bcc9736cSJeff Kirsher /**
3941bcc9736cSJeff Kirsher  * hw_stop_tx - stop transmitting
3942bcc9736cSJeff Kirsher  * @hw: 	The hardware instance.
3943bcc9736cSJeff Kirsher  *
3944bcc9736cSJeff Kirsher  * This routine stops the transmit function of the hardware.
3945bcc9736cSJeff Kirsher  */
3946bcc9736cSJeff Kirsher static void hw_stop_tx(struct ksz_hw *hw)
3947bcc9736cSJeff Kirsher {
3948bcc9736cSJeff Kirsher 	writel((hw->tx_cfg & ~DMA_TX_ENABLE), hw->io + KS_DMA_TX_CTRL);
3949bcc9736cSJeff Kirsher }
3950bcc9736cSJeff Kirsher 
3951bcc9736cSJeff Kirsher /**
3952bcc9736cSJeff Kirsher  * hw_disable - disable hardware
3953bcc9736cSJeff Kirsher  * @hw: 	The hardware instance.
3954bcc9736cSJeff Kirsher  *
3955bcc9736cSJeff Kirsher  * This routine disables the hardware.
3956bcc9736cSJeff Kirsher  */
3957bcc9736cSJeff Kirsher static void hw_disable(struct ksz_hw *hw)
3958bcc9736cSJeff Kirsher {
3959bcc9736cSJeff Kirsher 	hw_stop_rx(hw);
3960bcc9736cSJeff Kirsher 	hw_stop_tx(hw);
3961bcc9736cSJeff Kirsher 	hw->enabled = 0;
3962bcc9736cSJeff Kirsher }
3963bcc9736cSJeff Kirsher 
3964bcc9736cSJeff Kirsher /**
3965bcc9736cSJeff Kirsher  * hw_enable - enable hardware
3966bcc9736cSJeff Kirsher  * @hw: 	The hardware instance.
3967bcc9736cSJeff Kirsher  *
3968bcc9736cSJeff Kirsher  * This routine enables the hardware.
3969bcc9736cSJeff Kirsher  */
3970bcc9736cSJeff Kirsher static void hw_enable(struct ksz_hw *hw)
3971bcc9736cSJeff Kirsher {
3972bcc9736cSJeff Kirsher 	hw_start_tx(hw);
3973bcc9736cSJeff Kirsher 	hw_start_rx(hw);
3974bcc9736cSJeff Kirsher 	hw->enabled = 1;
3975bcc9736cSJeff Kirsher }
3976bcc9736cSJeff Kirsher 
3977bcc9736cSJeff Kirsher /**
3978bcc9736cSJeff Kirsher  * hw_alloc_pkt - allocate enough descriptors for transmission
3979bcc9736cSJeff Kirsher  * @hw: 	The hardware instance.
3980bcc9736cSJeff Kirsher  * @length:	The length of the packet.
3981bcc9736cSJeff Kirsher  * @physical:	Number of descriptors required.
3982bcc9736cSJeff Kirsher  *
3983bcc9736cSJeff Kirsher  * This function allocates descriptors for transmission.
3984bcc9736cSJeff Kirsher  *
3985bcc9736cSJeff Kirsher  * Return 0 if not successful; 1 for buffer copy; or number of descriptors.
3986bcc9736cSJeff Kirsher  */
3987bcc9736cSJeff Kirsher static int hw_alloc_pkt(struct ksz_hw *hw, int length, int physical)
3988bcc9736cSJeff Kirsher {
3989bcc9736cSJeff Kirsher 	/* Always leave one descriptor free. */
3990bcc9736cSJeff Kirsher 	if (hw->tx_desc_info.avail <= 1)
3991bcc9736cSJeff Kirsher 		return 0;
3992bcc9736cSJeff Kirsher 
3993bcc9736cSJeff Kirsher 	/* Allocate a descriptor for transmission and mark it current. */
3994bcc9736cSJeff Kirsher 	get_tx_pkt(&hw->tx_desc_info, &hw->tx_desc_info.cur);
3995bcc9736cSJeff Kirsher 	hw->tx_desc_info.cur->sw.buf.tx.first_seg = 1;
3996bcc9736cSJeff Kirsher 
3997bcc9736cSJeff Kirsher 	/* Keep track of number of transmit descriptors used so far. */
3998bcc9736cSJeff Kirsher 	++hw->tx_int_cnt;
3999bcc9736cSJeff Kirsher 	hw->tx_size += length;
4000bcc9736cSJeff Kirsher 
4001bcc9736cSJeff Kirsher 	/* Cannot hold on too much data. */
4002bcc9736cSJeff Kirsher 	if (hw->tx_size >= MAX_TX_HELD_SIZE)
4003bcc9736cSJeff Kirsher 		hw->tx_int_cnt = hw->tx_int_mask + 1;
4004bcc9736cSJeff Kirsher 
4005bcc9736cSJeff Kirsher 	if (physical > hw->tx_desc_info.avail)
4006bcc9736cSJeff Kirsher 		return 1;
4007bcc9736cSJeff Kirsher 
4008bcc9736cSJeff Kirsher 	return hw->tx_desc_info.avail;
4009bcc9736cSJeff Kirsher }
4010bcc9736cSJeff Kirsher 
4011bcc9736cSJeff Kirsher /**
4012bcc9736cSJeff Kirsher  * hw_send_pkt - mark packet for transmission
4013bcc9736cSJeff Kirsher  * @hw: 	The hardware instance.
4014bcc9736cSJeff Kirsher  *
4015bcc9736cSJeff Kirsher  * This routine marks the packet for transmission in PCI version.
4016bcc9736cSJeff Kirsher  */
4017bcc9736cSJeff Kirsher static void hw_send_pkt(struct ksz_hw *hw)
4018bcc9736cSJeff Kirsher {
4019bcc9736cSJeff Kirsher 	struct ksz_desc *cur = hw->tx_desc_info.cur;
4020bcc9736cSJeff Kirsher 
4021bcc9736cSJeff Kirsher 	cur->sw.buf.tx.last_seg = 1;
4022bcc9736cSJeff Kirsher 
4023bcc9736cSJeff Kirsher 	/* Interrupt only after specified number of descriptors used. */
4024bcc9736cSJeff Kirsher 	if (hw->tx_int_cnt > hw->tx_int_mask) {
4025bcc9736cSJeff Kirsher 		cur->sw.buf.tx.intr = 1;
4026bcc9736cSJeff Kirsher 		hw->tx_int_cnt = 0;
4027bcc9736cSJeff Kirsher 		hw->tx_size = 0;
4028bcc9736cSJeff Kirsher 	}
4029bcc9736cSJeff Kirsher 
4030bcc9736cSJeff Kirsher 	/* KSZ8842 supports port directed transmission. */
4031bcc9736cSJeff Kirsher 	cur->sw.buf.tx.dest_port = hw->dst_ports;
4032bcc9736cSJeff Kirsher 
4033bcc9736cSJeff Kirsher 	release_desc(cur);
4034bcc9736cSJeff Kirsher 
4035bcc9736cSJeff Kirsher 	writel(0, hw->io + KS_DMA_TX_START);
4036bcc9736cSJeff Kirsher }
4037bcc9736cSJeff Kirsher 
4038bcc9736cSJeff Kirsher static int empty_addr(u8 *addr)
4039bcc9736cSJeff Kirsher {
4040bcc9736cSJeff Kirsher 	u32 *addr1 = (u32 *) addr;
4041bcc9736cSJeff Kirsher 	u16 *addr2 = (u16 *) &addr[4];
4042bcc9736cSJeff Kirsher 
4043bcc9736cSJeff Kirsher 	return 0 == *addr1 && 0 == *addr2;
4044bcc9736cSJeff Kirsher }
4045bcc9736cSJeff Kirsher 
4046bcc9736cSJeff Kirsher /**
4047bcc9736cSJeff Kirsher  * hw_set_addr - set MAC address
4048bcc9736cSJeff Kirsher  * @hw: 	The hardware instance.
4049bcc9736cSJeff Kirsher  *
4050bcc9736cSJeff Kirsher  * This routine programs the MAC address of the hardware when the address is
4051bcc9736cSJeff Kirsher  * overrided.
4052bcc9736cSJeff Kirsher  */
4053bcc9736cSJeff Kirsher static void hw_set_addr(struct ksz_hw *hw)
4054bcc9736cSJeff Kirsher {
4055bcc9736cSJeff Kirsher 	int i;
4056bcc9736cSJeff Kirsher 
40576a3c910cSJoe Perches 	for (i = 0; i < ETH_ALEN; i++)
4058bcc9736cSJeff Kirsher 		writeb(hw->override_addr[MAC_ADDR_ORDER(i)],
4059bcc9736cSJeff Kirsher 			hw->io + KS884X_ADDR_0_OFFSET + i);
4060bcc9736cSJeff Kirsher 
4061bcc9736cSJeff Kirsher 	sw_set_addr(hw, hw->override_addr);
4062bcc9736cSJeff Kirsher }
4063bcc9736cSJeff Kirsher 
4064bcc9736cSJeff Kirsher /**
4065bcc9736cSJeff Kirsher  * hw_read_addr - read MAC address
4066bcc9736cSJeff Kirsher  * @hw: 	The hardware instance.
4067bcc9736cSJeff Kirsher  *
4068bcc9736cSJeff Kirsher  * This routine retrieves the MAC address of the hardware.
4069bcc9736cSJeff Kirsher  */
4070bcc9736cSJeff Kirsher static void hw_read_addr(struct ksz_hw *hw)
4071bcc9736cSJeff Kirsher {
4072bcc9736cSJeff Kirsher 	int i;
4073bcc9736cSJeff Kirsher 
40746a3c910cSJoe Perches 	for (i = 0; i < ETH_ALEN; i++)
4075bcc9736cSJeff Kirsher 		hw->perm_addr[MAC_ADDR_ORDER(i)] = readb(hw->io +
4076bcc9736cSJeff Kirsher 			KS884X_ADDR_0_OFFSET + i);
4077bcc9736cSJeff Kirsher 
4078bcc9736cSJeff Kirsher 	if (!hw->mac_override) {
40796a3c910cSJoe Perches 		memcpy(hw->override_addr, hw->perm_addr, ETH_ALEN);
4080bcc9736cSJeff Kirsher 		if (empty_addr(hw->override_addr)) {
40816a3c910cSJoe Perches 			memcpy(hw->perm_addr, DEFAULT_MAC_ADDRESS, ETH_ALEN);
4082bcc9736cSJeff Kirsher 			memcpy(hw->override_addr, DEFAULT_MAC_ADDRESS,
40836a3c910cSJoe Perches 			       ETH_ALEN);
4084bcc9736cSJeff Kirsher 			hw->override_addr[5] += hw->id;
4085bcc9736cSJeff Kirsher 			hw_set_addr(hw);
4086bcc9736cSJeff Kirsher 		}
4087bcc9736cSJeff Kirsher 	}
4088bcc9736cSJeff Kirsher }
4089bcc9736cSJeff Kirsher 
4090bcc9736cSJeff Kirsher static void hw_ena_add_addr(struct ksz_hw *hw, int index, u8 *mac_addr)
4091bcc9736cSJeff Kirsher {
4092bcc9736cSJeff Kirsher 	int i;
4093bcc9736cSJeff Kirsher 	u32 mac_addr_lo;
4094bcc9736cSJeff Kirsher 	u32 mac_addr_hi;
4095bcc9736cSJeff Kirsher 
4096bcc9736cSJeff Kirsher 	mac_addr_hi = 0;
4097bcc9736cSJeff Kirsher 	for (i = 0; i < 2; i++) {
4098bcc9736cSJeff Kirsher 		mac_addr_hi <<= 8;
4099bcc9736cSJeff Kirsher 		mac_addr_hi |= mac_addr[i];
4100bcc9736cSJeff Kirsher 	}
4101bcc9736cSJeff Kirsher 	mac_addr_hi |= ADD_ADDR_ENABLE;
4102bcc9736cSJeff Kirsher 	mac_addr_lo = 0;
4103bcc9736cSJeff Kirsher 	for (i = 2; i < 6; i++) {
4104bcc9736cSJeff Kirsher 		mac_addr_lo <<= 8;
4105bcc9736cSJeff Kirsher 		mac_addr_lo |= mac_addr[i];
4106bcc9736cSJeff Kirsher 	}
4107bcc9736cSJeff Kirsher 	index *= ADD_ADDR_INCR;
4108bcc9736cSJeff Kirsher 
4109bcc9736cSJeff Kirsher 	writel(mac_addr_lo, hw->io + index + KS_ADD_ADDR_0_LO);
4110bcc9736cSJeff Kirsher 	writel(mac_addr_hi, hw->io + index + KS_ADD_ADDR_0_HI);
4111bcc9736cSJeff Kirsher }
4112bcc9736cSJeff Kirsher 
4113bcc9736cSJeff Kirsher static void hw_set_add_addr(struct ksz_hw *hw)
4114bcc9736cSJeff Kirsher {
4115bcc9736cSJeff Kirsher 	int i;
4116bcc9736cSJeff Kirsher 
4117bcc9736cSJeff Kirsher 	for (i = 0; i < ADDITIONAL_ENTRIES; i++) {
4118bcc9736cSJeff Kirsher 		if (empty_addr(hw->address[i]))
4119bcc9736cSJeff Kirsher 			writel(0, hw->io + ADD_ADDR_INCR * i +
4120bcc9736cSJeff Kirsher 				KS_ADD_ADDR_0_HI);
4121bcc9736cSJeff Kirsher 		else
4122bcc9736cSJeff Kirsher 			hw_ena_add_addr(hw, i, hw->address[i]);
4123bcc9736cSJeff Kirsher 	}
4124bcc9736cSJeff Kirsher }
4125bcc9736cSJeff Kirsher 
4126bcc9736cSJeff Kirsher static int hw_add_addr(struct ksz_hw *hw, u8 *mac_addr)
4127bcc9736cSJeff Kirsher {
4128bcc9736cSJeff Kirsher 	int i;
4129bcc9736cSJeff Kirsher 	int j = ADDITIONAL_ENTRIES;
4130bcc9736cSJeff Kirsher 
41316a3c910cSJoe Perches 	if (!memcmp(hw->override_addr, mac_addr, ETH_ALEN))
4132bcc9736cSJeff Kirsher 		return 0;
4133bcc9736cSJeff Kirsher 	for (i = 0; i < hw->addr_list_size; i++) {
41346a3c910cSJoe Perches 		if (!memcmp(hw->address[i], mac_addr, ETH_ALEN))
4135bcc9736cSJeff Kirsher 			return 0;
4136bcc9736cSJeff Kirsher 		if (ADDITIONAL_ENTRIES == j && empty_addr(hw->address[i]))
4137bcc9736cSJeff Kirsher 			j = i;
4138bcc9736cSJeff Kirsher 	}
4139bcc9736cSJeff Kirsher 	if (j < ADDITIONAL_ENTRIES) {
41406a3c910cSJoe Perches 		memcpy(hw->address[j], mac_addr, ETH_ALEN);
4141bcc9736cSJeff Kirsher 		hw_ena_add_addr(hw, j, hw->address[j]);
4142bcc9736cSJeff Kirsher 		return 0;
4143bcc9736cSJeff Kirsher 	}
4144bcc9736cSJeff Kirsher 	return -1;
4145bcc9736cSJeff Kirsher }
4146bcc9736cSJeff Kirsher 
4147bcc9736cSJeff Kirsher static int hw_del_addr(struct ksz_hw *hw, u8 *mac_addr)
4148bcc9736cSJeff Kirsher {
4149bcc9736cSJeff Kirsher 	int i;
4150bcc9736cSJeff Kirsher 
4151bcc9736cSJeff Kirsher 	for (i = 0; i < hw->addr_list_size; i++) {
41526a3c910cSJoe Perches 		if (!memcmp(hw->address[i], mac_addr, ETH_ALEN)) {
41536a3c910cSJoe Perches 			memset(hw->address[i], 0, ETH_ALEN);
4154bcc9736cSJeff Kirsher 			writel(0, hw->io + ADD_ADDR_INCR * i +
4155bcc9736cSJeff Kirsher 				KS_ADD_ADDR_0_HI);
4156bcc9736cSJeff Kirsher 			return 0;
4157bcc9736cSJeff Kirsher 		}
4158bcc9736cSJeff Kirsher 	}
4159bcc9736cSJeff Kirsher 	return -1;
4160bcc9736cSJeff Kirsher }
4161bcc9736cSJeff Kirsher 
4162bcc9736cSJeff Kirsher /**
4163bcc9736cSJeff Kirsher  * hw_clr_multicast - clear multicast addresses
4164bcc9736cSJeff Kirsher  * @hw: 	The hardware instance.
4165bcc9736cSJeff Kirsher  *
4166bcc9736cSJeff Kirsher  * This routine removes all multicast addresses set in the hardware.
4167bcc9736cSJeff Kirsher  */
4168bcc9736cSJeff Kirsher static void hw_clr_multicast(struct ksz_hw *hw)
4169bcc9736cSJeff Kirsher {
4170bcc9736cSJeff Kirsher 	int i;
4171bcc9736cSJeff Kirsher 
4172bcc9736cSJeff Kirsher 	for (i = 0; i < HW_MULTICAST_SIZE; i++) {
4173bcc9736cSJeff Kirsher 		hw->multi_bits[i] = 0;
4174bcc9736cSJeff Kirsher 
4175bcc9736cSJeff Kirsher 		writeb(0, hw->io + KS884X_MULTICAST_0_OFFSET + i);
4176bcc9736cSJeff Kirsher 	}
4177bcc9736cSJeff Kirsher }
4178bcc9736cSJeff Kirsher 
4179bcc9736cSJeff Kirsher /**
4180bcc9736cSJeff Kirsher  * hw_set_grp_addr - set multicast addresses
4181bcc9736cSJeff Kirsher  * @hw: 	The hardware instance.
4182bcc9736cSJeff Kirsher  *
4183bcc9736cSJeff Kirsher  * This routine programs multicast addresses for the hardware to accept those
4184bcc9736cSJeff Kirsher  * addresses.
4185bcc9736cSJeff Kirsher  */
4186bcc9736cSJeff Kirsher static void hw_set_grp_addr(struct ksz_hw *hw)
4187bcc9736cSJeff Kirsher {
4188bcc9736cSJeff Kirsher 	int i;
4189bcc9736cSJeff Kirsher 	int index;
4190bcc9736cSJeff Kirsher 	int position;
4191bcc9736cSJeff Kirsher 	int value;
4192bcc9736cSJeff Kirsher 
4193bcc9736cSJeff Kirsher 	memset(hw->multi_bits, 0, sizeof(u8) * HW_MULTICAST_SIZE);
4194bcc9736cSJeff Kirsher 
4195bcc9736cSJeff Kirsher 	for (i = 0; i < hw->multi_list_size; i++) {
4196bcc9736cSJeff Kirsher 		position = (ether_crc(6, hw->multi_list[i]) >> 26) & 0x3f;
4197bcc9736cSJeff Kirsher 		index = position >> 3;
4198bcc9736cSJeff Kirsher 		value = 1 << (position & 7);
4199bcc9736cSJeff Kirsher 		hw->multi_bits[index] |= (u8) value;
4200bcc9736cSJeff Kirsher 	}
4201bcc9736cSJeff Kirsher 
4202bcc9736cSJeff Kirsher 	for (i = 0; i < HW_MULTICAST_SIZE; i++)
4203bcc9736cSJeff Kirsher 		writeb(hw->multi_bits[i], hw->io + KS884X_MULTICAST_0_OFFSET +
4204bcc9736cSJeff Kirsher 			i);
4205bcc9736cSJeff Kirsher }
4206bcc9736cSJeff Kirsher 
4207bcc9736cSJeff Kirsher /**
4208bcc9736cSJeff Kirsher  * hw_set_multicast - enable or disable all multicast receiving
4209bcc9736cSJeff Kirsher  * @hw: 	The hardware instance.
4210bcc9736cSJeff Kirsher  * @multicast:	To turn on or off the all multicast feature.
4211bcc9736cSJeff Kirsher  *
4212bcc9736cSJeff Kirsher  * This routine enables/disables the hardware to accept all multicast packets.
4213bcc9736cSJeff Kirsher  */
4214bcc9736cSJeff Kirsher static void hw_set_multicast(struct ksz_hw *hw, u8 multicast)
4215bcc9736cSJeff Kirsher {
4216bcc9736cSJeff Kirsher 	/* Stop receiving for reconfiguration. */
4217bcc9736cSJeff Kirsher 	hw_stop_rx(hw);
4218bcc9736cSJeff Kirsher 
4219bcc9736cSJeff Kirsher 	if (multicast)
4220bcc9736cSJeff Kirsher 		hw->rx_cfg |= DMA_RX_ALL_MULTICAST;
4221bcc9736cSJeff Kirsher 	else
4222bcc9736cSJeff Kirsher 		hw->rx_cfg &= ~DMA_RX_ALL_MULTICAST;
4223bcc9736cSJeff Kirsher 
4224bcc9736cSJeff Kirsher 	if (hw->enabled)
4225bcc9736cSJeff Kirsher 		hw_start_rx(hw);
4226bcc9736cSJeff Kirsher }
4227bcc9736cSJeff Kirsher 
4228bcc9736cSJeff Kirsher /**
4229bcc9736cSJeff Kirsher  * hw_set_promiscuous - enable or disable promiscuous receiving
4230bcc9736cSJeff Kirsher  * @hw: 	The hardware instance.
4231bcc9736cSJeff Kirsher  * @prom:	To turn on or off the promiscuous feature.
4232bcc9736cSJeff Kirsher  *
4233bcc9736cSJeff Kirsher  * This routine enables/disables the hardware to accept all packets.
4234bcc9736cSJeff Kirsher  */
4235bcc9736cSJeff Kirsher static void hw_set_promiscuous(struct ksz_hw *hw, u8 prom)
4236bcc9736cSJeff Kirsher {
4237bcc9736cSJeff Kirsher 	/* Stop receiving for reconfiguration. */
4238bcc9736cSJeff Kirsher 	hw_stop_rx(hw);
4239bcc9736cSJeff Kirsher 
4240bcc9736cSJeff Kirsher 	if (prom)
4241bcc9736cSJeff Kirsher 		hw->rx_cfg |= DMA_RX_PROMISCUOUS;
4242bcc9736cSJeff Kirsher 	else
4243bcc9736cSJeff Kirsher 		hw->rx_cfg &= ~DMA_RX_PROMISCUOUS;
4244bcc9736cSJeff Kirsher 
4245bcc9736cSJeff Kirsher 	if (hw->enabled)
4246bcc9736cSJeff Kirsher 		hw_start_rx(hw);
4247bcc9736cSJeff Kirsher }
4248bcc9736cSJeff Kirsher 
4249bcc9736cSJeff Kirsher /**
4250bcc9736cSJeff Kirsher  * sw_enable - enable the switch
4251bcc9736cSJeff Kirsher  * @hw: 	The hardware instance.
4252bcc9736cSJeff Kirsher  * @enable:	The flag to enable or disable the switch
4253bcc9736cSJeff Kirsher  *
4254bcc9736cSJeff Kirsher  * This routine is used to enable/disable the switch in KSZ8842.
4255bcc9736cSJeff Kirsher  */
4256bcc9736cSJeff Kirsher static void sw_enable(struct ksz_hw *hw, int enable)
4257bcc9736cSJeff Kirsher {
4258bcc9736cSJeff Kirsher 	int port;
4259bcc9736cSJeff Kirsher 
4260bcc9736cSJeff Kirsher 	for (port = 0; port < SWITCH_PORT_NUM; port++) {
4261bcc9736cSJeff Kirsher 		if (hw->dev_count > 1) {
4262bcc9736cSJeff Kirsher 			/* Set port-base vlan membership with host port. */
4263bcc9736cSJeff Kirsher 			sw_cfg_port_base_vlan(hw, port,
4264bcc9736cSJeff Kirsher 				HOST_MASK | (1 << port));
4265bcc9736cSJeff Kirsher 			port_set_stp_state(hw, port, STP_STATE_DISABLED);
4266bcc9736cSJeff Kirsher 		} else {
4267bcc9736cSJeff Kirsher 			sw_cfg_port_base_vlan(hw, port, PORT_MASK);
4268bcc9736cSJeff Kirsher 			port_set_stp_state(hw, port, STP_STATE_FORWARDING);
4269bcc9736cSJeff Kirsher 		}
4270bcc9736cSJeff Kirsher 	}
4271bcc9736cSJeff Kirsher 	if (hw->dev_count > 1)
4272bcc9736cSJeff Kirsher 		port_set_stp_state(hw, SWITCH_PORT_NUM, STP_STATE_SIMPLE);
4273bcc9736cSJeff Kirsher 	else
4274bcc9736cSJeff Kirsher 		port_set_stp_state(hw, SWITCH_PORT_NUM, STP_STATE_FORWARDING);
4275bcc9736cSJeff Kirsher 
4276bcc9736cSJeff Kirsher 	if (enable)
4277bcc9736cSJeff Kirsher 		enable = KS8842_START;
4278bcc9736cSJeff Kirsher 	writew(enable, hw->io + KS884X_CHIP_ID_OFFSET);
4279bcc9736cSJeff Kirsher }
4280bcc9736cSJeff Kirsher 
4281bcc9736cSJeff Kirsher /**
4282bcc9736cSJeff Kirsher  * sw_setup - setup the switch
4283bcc9736cSJeff Kirsher  * @hw: 	The hardware instance.
4284bcc9736cSJeff Kirsher  *
4285bcc9736cSJeff Kirsher  * This routine setup the hardware switch engine for default operation.
4286bcc9736cSJeff Kirsher  */
4287bcc9736cSJeff Kirsher static void sw_setup(struct ksz_hw *hw)
4288bcc9736cSJeff Kirsher {
4289bcc9736cSJeff Kirsher 	int port;
4290bcc9736cSJeff Kirsher 
4291bcc9736cSJeff Kirsher 	sw_set_global_ctrl(hw);
4292bcc9736cSJeff Kirsher 
4293bcc9736cSJeff Kirsher 	/* Enable switch broadcast storm protection at 10% percent rate. */
4294bcc9736cSJeff Kirsher 	sw_init_broad_storm(hw);
4295bcc9736cSJeff Kirsher 	hw_cfg_broad_storm(hw, BROADCAST_STORM_PROTECTION_RATE);
4296bcc9736cSJeff Kirsher 	for (port = 0; port < SWITCH_PORT_NUM; port++)
4297bcc9736cSJeff Kirsher 		sw_ena_broad_storm(hw, port);
4298bcc9736cSJeff Kirsher 
4299bcc9736cSJeff Kirsher 	sw_init_prio(hw);
4300bcc9736cSJeff Kirsher 
4301bcc9736cSJeff Kirsher 	sw_init_mirror(hw);
4302bcc9736cSJeff Kirsher 
4303bcc9736cSJeff Kirsher 	sw_init_prio_rate(hw);
4304bcc9736cSJeff Kirsher 
4305bcc9736cSJeff Kirsher 	sw_init_vlan(hw);
4306bcc9736cSJeff Kirsher 
4307bcc9736cSJeff Kirsher 	if (hw->features & STP_SUPPORT)
4308bcc9736cSJeff Kirsher 		sw_init_stp(hw);
4309bcc9736cSJeff Kirsher 	if (!sw_chk(hw, KS8842_SWITCH_CTRL_1_OFFSET,
4310bcc9736cSJeff Kirsher 			SWITCH_TX_FLOW_CTRL | SWITCH_RX_FLOW_CTRL))
4311bcc9736cSJeff Kirsher 		hw->overrides |= PAUSE_FLOW_CTRL;
4312bcc9736cSJeff Kirsher 	sw_enable(hw, 1);
4313bcc9736cSJeff Kirsher }
4314bcc9736cSJeff Kirsher 
4315bcc9736cSJeff Kirsher /**
4316bcc9736cSJeff Kirsher  * ksz_start_timer - start kernel timer
4317bcc9736cSJeff Kirsher  * @info:	Kernel timer information.
4318bcc9736cSJeff Kirsher  * @time:	The time tick.
4319bcc9736cSJeff Kirsher  *
4320bcc9736cSJeff Kirsher  * This routine starts the kernel timer after the specified time tick.
4321bcc9736cSJeff Kirsher  */
4322bcc9736cSJeff Kirsher static void ksz_start_timer(struct ksz_timer_info *info, int time)
4323bcc9736cSJeff Kirsher {
4324bcc9736cSJeff Kirsher 	info->cnt = 0;
4325bcc9736cSJeff Kirsher 	info->timer.expires = jiffies + time;
4326bcc9736cSJeff Kirsher 	add_timer(&info->timer);
4327bcc9736cSJeff Kirsher 
4328bcc9736cSJeff Kirsher 	/* infinity */
4329bcc9736cSJeff Kirsher 	info->max = -1;
4330bcc9736cSJeff Kirsher }
4331bcc9736cSJeff Kirsher 
4332bcc9736cSJeff Kirsher /**
4333bcc9736cSJeff Kirsher  * ksz_stop_timer - stop kernel timer
4334bcc9736cSJeff Kirsher  * @info:	Kernel timer information.
4335bcc9736cSJeff Kirsher  *
4336bcc9736cSJeff Kirsher  * This routine stops the kernel timer.
4337bcc9736cSJeff Kirsher  */
4338bcc9736cSJeff Kirsher static void ksz_stop_timer(struct ksz_timer_info *info)
4339bcc9736cSJeff Kirsher {
4340bcc9736cSJeff Kirsher 	if (info->max) {
4341bcc9736cSJeff Kirsher 		info->max = 0;
4342bcc9736cSJeff Kirsher 		del_timer_sync(&info->timer);
4343bcc9736cSJeff Kirsher 	}
4344bcc9736cSJeff Kirsher }
4345bcc9736cSJeff Kirsher 
4346bcc9736cSJeff Kirsher static void ksz_init_timer(struct ksz_timer_info *info, int period,
4347bcc9736cSJeff Kirsher 	void (*function)(unsigned long), void *data)
4348bcc9736cSJeff Kirsher {
4349bcc9736cSJeff Kirsher 	info->max = 0;
4350bcc9736cSJeff Kirsher 	info->period = period;
4351bcc9736cSJeff Kirsher 	init_timer(&info->timer);
4352bcc9736cSJeff Kirsher 	info->timer.function = function;
4353bcc9736cSJeff Kirsher 	info->timer.data = (unsigned long) data;
4354bcc9736cSJeff Kirsher }
4355bcc9736cSJeff Kirsher 
4356bcc9736cSJeff Kirsher static void ksz_update_timer(struct ksz_timer_info *info)
4357bcc9736cSJeff Kirsher {
4358bcc9736cSJeff Kirsher 	++info->cnt;
4359bcc9736cSJeff Kirsher 	if (info->max > 0) {
4360bcc9736cSJeff Kirsher 		if (info->cnt < info->max) {
4361bcc9736cSJeff Kirsher 			info->timer.expires = jiffies + info->period;
4362bcc9736cSJeff Kirsher 			add_timer(&info->timer);
4363bcc9736cSJeff Kirsher 		} else
4364bcc9736cSJeff Kirsher 			info->max = 0;
4365bcc9736cSJeff Kirsher 	} else if (info->max < 0) {
4366bcc9736cSJeff Kirsher 		info->timer.expires = jiffies + info->period;
4367bcc9736cSJeff Kirsher 		add_timer(&info->timer);
4368bcc9736cSJeff Kirsher 	}
4369bcc9736cSJeff Kirsher }
4370bcc9736cSJeff Kirsher 
4371bcc9736cSJeff Kirsher /**
4372bcc9736cSJeff Kirsher  * ksz_alloc_soft_desc - allocate software descriptors
4373bcc9736cSJeff Kirsher  * @desc_info:	Descriptor information structure.
4374bcc9736cSJeff Kirsher  * @transmit:	Indication that descriptors are for transmit.
4375bcc9736cSJeff Kirsher  *
4376bcc9736cSJeff Kirsher  * This local function allocates software descriptors for manipulation in
4377bcc9736cSJeff Kirsher  * memory.
4378bcc9736cSJeff Kirsher  *
4379bcc9736cSJeff Kirsher  * Return 0 if successful.
4380bcc9736cSJeff Kirsher  */
4381bcc9736cSJeff Kirsher static int ksz_alloc_soft_desc(struct ksz_desc_info *desc_info, int transmit)
4382bcc9736cSJeff Kirsher {
4383cb508701SThomas Meyer 	desc_info->ring = kzalloc(sizeof(struct ksz_desc) * desc_info->alloc,
4384bcc9736cSJeff Kirsher 				  GFP_KERNEL);
4385bcc9736cSJeff Kirsher 	if (!desc_info->ring)
4386bcc9736cSJeff Kirsher 		return 1;
4387bcc9736cSJeff Kirsher 	hw_init_desc(desc_info, transmit);
4388bcc9736cSJeff Kirsher 	return 0;
4389bcc9736cSJeff Kirsher }
4390bcc9736cSJeff Kirsher 
4391bcc9736cSJeff Kirsher /**
4392bcc9736cSJeff Kirsher  * ksz_alloc_desc - allocate hardware descriptors
4393bcc9736cSJeff Kirsher  * @adapter:	Adapter information structure.
4394bcc9736cSJeff Kirsher  *
4395bcc9736cSJeff Kirsher  * This local function allocates hardware descriptors for receiving and
4396bcc9736cSJeff Kirsher  * transmitting.
4397bcc9736cSJeff Kirsher  *
4398bcc9736cSJeff Kirsher  * Return 0 if successful.
4399bcc9736cSJeff Kirsher  */
4400bcc9736cSJeff Kirsher static int ksz_alloc_desc(struct dev_info *adapter)
4401bcc9736cSJeff Kirsher {
4402bcc9736cSJeff Kirsher 	struct ksz_hw *hw = &adapter->hw;
4403bcc9736cSJeff Kirsher 	int offset;
4404bcc9736cSJeff Kirsher 
4405bcc9736cSJeff Kirsher 	/* Allocate memory for RX & TX descriptors. */
4406bcc9736cSJeff Kirsher 	adapter->desc_pool.alloc_size =
4407bcc9736cSJeff Kirsher 		hw->rx_desc_info.size * hw->rx_desc_info.alloc +
4408bcc9736cSJeff Kirsher 		hw->tx_desc_info.size * hw->tx_desc_info.alloc +
4409bcc9736cSJeff Kirsher 		DESC_ALIGNMENT;
4410bcc9736cSJeff Kirsher 
4411bcc9736cSJeff Kirsher 	adapter->desc_pool.alloc_virt =
4412bcc9736cSJeff Kirsher 		pci_alloc_consistent(
4413bcc9736cSJeff Kirsher 			adapter->pdev, adapter->desc_pool.alloc_size,
4414bcc9736cSJeff Kirsher 			&adapter->desc_pool.dma_addr);
4415bcc9736cSJeff Kirsher 	if (adapter->desc_pool.alloc_virt == NULL) {
4416bcc9736cSJeff Kirsher 		adapter->desc_pool.alloc_size = 0;
4417bcc9736cSJeff Kirsher 		return 1;
4418bcc9736cSJeff Kirsher 	}
4419bcc9736cSJeff Kirsher 	memset(adapter->desc_pool.alloc_virt, 0, adapter->desc_pool.alloc_size);
4420bcc9736cSJeff Kirsher 
4421bcc9736cSJeff Kirsher 	/* Align to the next cache line boundary. */
4422bcc9736cSJeff Kirsher 	offset = (((ulong) adapter->desc_pool.alloc_virt % DESC_ALIGNMENT) ?
4423bcc9736cSJeff Kirsher 		(DESC_ALIGNMENT -
4424bcc9736cSJeff Kirsher 		((ulong) adapter->desc_pool.alloc_virt % DESC_ALIGNMENT)) : 0);
4425bcc9736cSJeff Kirsher 	adapter->desc_pool.virt = adapter->desc_pool.alloc_virt + offset;
4426bcc9736cSJeff Kirsher 	adapter->desc_pool.phys = adapter->desc_pool.dma_addr + offset;
4427bcc9736cSJeff Kirsher 
4428bcc9736cSJeff Kirsher 	/* Allocate receive/transmit descriptors. */
4429bcc9736cSJeff Kirsher 	hw->rx_desc_info.ring_virt = (struct ksz_hw_desc *)
4430bcc9736cSJeff Kirsher 		adapter->desc_pool.virt;
4431bcc9736cSJeff Kirsher 	hw->rx_desc_info.ring_phys = adapter->desc_pool.phys;
4432bcc9736cSJeff Kirsher 	offset = hw->rx_desc_info.alloc * hw->rx_desc_info.size;
4433bcc9736cSJeff Kirsher 	hw->tx_desc_info.ring_virt = (struct ksz_hw_desc *)
4434bcc9736cSJeff Kirsher 		(adapter->desc_pool.virt + offset);
4435bcc9736cSJeff Kirsher 	hw->tx_desc_info.ring_phys = adapter->desc_pool.phys + offset;
4436bcc9736cSJeff Kirsher 
4437bcc9736cSJeff Kirsher 	if (ksz_alloc_soft_desc(&hw->rx_desc_info, 0))
4438bcc9736cSJeff Kirsher 		return 1;
4439bcc9736cSJeff Kirsher 	if (ksz_alloc_soft_desc(&hw->tx_desc_info, 1))
4440bcc9736cSJeff Kirsher 		return 1;
4441bcc9736cSJeff Kirsher 
4442bcc9736cSJeff Kirsher 	return 0;
4443bcc9736cSJeff Kirsher }
4444bcc9736cSJeff Kirsher 
4445bcc9736cSJeff Kirsher /**
4446bcc9736cSJeff Kirsher  * free_dma_buf - release DMA buffer resources
4447bcc9736cSJeff Kirsher  * @adapter:	Adapter information structure.
4448bcc9736cSJeff Kirsher  *
4449bcc9736cSJeff Kirsher  * This routine is just a helper function to release the DMA buffer resources.
4450bcc9736cSJeff Kirsher  */
4451bcc9736cSJeff Kirsher static void free_dma_buf(struct dev_info *adapter, struct ksz_dma_buf *dma_buf,
4452bcc9736cSJeff Kirsher 	int direction)
4453bcc9736cSJeff Kirsher {
4454bcc9736cSJeff Kirsher 	pci_unmap_single(adapter->pdev, dma_buf->dma, dma_buf->len, direction);
4455bcc9736cSJeff Kirsher 	dev_kfree_skb(dma_buf->skb);
4456bcc9736cSJeff Kirsher 	dma_buf->skb = NULL;
4457bcc9736cSJeff Kirsher 	dma_buf->dma = 0;
4458bcc9736cSJeff Kirsher }
4459bcc9736cSJeff Kirsher 
4460bcc9736cSJeff Kirsher /**
4461bcc9736cSJeff Kirsher  * ksz_init_rx_buffers - initialize receive descriptors
4462bcc9736cSJeff Kirsher  * @adapter:	Adapter information structure.
4463bcc9736cSJeff Kirsher  *
4464bcc9736cSJeff Kirsher  * This routine initializes DMA buffers for receiving.
4465bcc9736cSJeff Kirsher  */
4466bcc9736cSJeff Kirsher static void ksz_init_rx_buffers(struct dev_info *adapter)
4467bcc9736cSJeff Kirsher {
4468bcc9736cSJeff Kirsher 	int i;
4469bcc9736cSJeff Kirsher 	struct ksz_desc *desc;
4470bcc9736cSJeff Kirsher 	struct ksz_dma_buf *dma_buf;
4471bcc9736cSJeff Kirsher 	struct ksz_hw *hw = &adapter->hw;
4472bcc9736cSJeff Kirsher 	struct ksz_desc_info *info = &hw->rx_desc_info;
4473bcc9736cSJeff Kirsher 
4474bcc9736cSJeff Kirsher 	for (i = 0; i < hw->rx_desc_info.alloc; i++) {
4475bcc9736cSJeff Kirsher 		get_rx_pkt(info, &desc);
4476bcc9736cSJeff Kirsher 
4477bcc9736cSJeff Kirsher 		dma_buf = DMA_BUFFER(desc);
4478bcc9736cSJeff Kirsher 		if (dma_buf->skb && dma_buf->len != adapter->mtu)
4479bcc9736cSJeff Kirsher 			free_dma_buf(adapter, dma_buf, PCI_DMA_FROMDEVICE);
4480bcc9736cSJeff Kirsher 		dma_buf->len = adapter->mtu;
4481bcc9736cSJeff Kirsher 		if (!dma_buf->skb)
4482bcc9736cSJeff Kirsher 			dma_buf->skb = alloc_skb(dma_buf->len, GFP_ATOMIC);
4483bcc9736cSJeff Kirsher 		if (dma_buf->skb && !dma_buf->dma) {
4484bcc9736cSJeff Kirsher 			dma_buf->skb->dev = adapter->dev;
4485bcc9736cSJeff Kirsher 			dma_buf->dma = pci_map_single(
4486bcc9736cSJeff Kirsher 				adapter->pdev,
4487bcc9736cSJeff Kirsher 				skb_tail_pointer(dma_buf->skb),
4488bcc9736cSJeff Kirsher 				dma_buf->len,
4489bcc9736cSJeff Kirsher 				PCI_DMA_FROMDEVICE);
4490bcc9736cSJeff Kirsher 		}
4491bcc9736cSJeff Kirsher 
4492bcc9736cSJeff Kirsher 		/* Set descriptor. */
4493bcc9736cSJeff Kirsher 		set_rx_buf(desc, dma_buf->dma);
4494bcc9736cSJeff Kirsher 		set_rx_len(desc, dma_buf->len);
4495bcc9736cSJeff Kirsher 		release_desc(desc);
4496bcc9736cSJeff Kirsher 	}
4497bcc9736cSJeff Kirsher }
4498bcc9736cSJeff Kirsher 
4499bcc9736cSJeff Kirsher /**
4500bcc9736cSJeff Kirsher  * ksz_alloc_mem - allocate memory for hardware descriptors
4501bcc9736cSJeff Kirsher  * @adapter:	Adapter information structure.
4502bcc9736cSJeff Kirsher  *
4503bcc9736cSJeff Kirsher  * This function allocates memory for use by hardware descriptors for receiving
4504bcc9736cSJeff Kirsher  * and transmitting.
4505bcc9736cSJeff Kirsher  *
4506bcc9736cSJeff Kirsher  * Return 0 if successful.
4507bcc9736cSJeff Kirsher  */
4508bcc9736cSJeff Kirsher static int ksz_alloc_mem(struct dev_info *adapter)
4509bcc9736cSJeff Kirsher {
4510bcc9736cSJeff Kirsher 	struct ksz_hw *hw = &adapter->hw;
4511bcc9736cSJeff Kirsher 
4512bcc9736cSJeff Kirsher 	/* Determine the number of receive and transmit descriptors. */
4513bcc9736cSJeff Kirsher 	hw->rx_desc_info.alloc = NUM_OF_RX_DESC;
4514bcc9736cSJeff Kirsher 	hw->tx_desc_info.alloc = NUM_OF_TX_DESC;
4515bcc9736cSJeff Kirsher 
4516bcc9736cSJeff Kirsher 	/* Determine how many descriptors to skip transmit interrupt. */
4517bcc9736cSJeff Kirsher 	hw->tx_int_cnt = 0;
4518bcc9736cSJeff Kirsher 	hw->tx_int_mask = NUM_OF_TX_DESC / 4;
4519bcc9736cSJeff Kirsher 	if (hw->tx_int_mask > 8)
4520bcc9736cSJeff Kirsher 		hw->tx_int_mask = 8;
4521bcc9736cSJeff Kirsher 	while (hw->tx_int_mask) {
4522bcc9736cSJeff Kirsher 		hw->tx_int_cnt++;
4523bcc9736cSJeff Kirsher 		hw->tx_int_mask >>= 1;
4524bcc9736cSJeff Kirsher 	}
4525bcc9736cSJeff Kirsher 	if (hw->tx_int_cnt) {
4526bcc9736cSJeff Kirsher 		hw->tx_int_mask = (1 << (hw->tx_int_cnt - 1)) - 1;
4527bcc9736cSJeff Kirsher 		hw->tx_int_cnt = 0;
4528bcc9736cSJeff Kirsher 	}
4529bcc9736cSJeff Kirsher 
4530bcc9736cSJeff Kirsher 	/* Determine the descriptor size. */
4531bcc9736cSJeff Kirsher 	hw->rx_desc_info.size =
4532bcc9736cSJeff Kirsher 		(((sizeof(struct ksz_hw_desc) + DESC_ALIGNMENT - 1) /
4533bcc9736cSJeff Kirsher 		DESC_ALIGNMENT) * DESC_ALIGNMENT);
4534bcc9736cSJeff Kirsher 	hw->tx_desc_info.size =
4535bcc9736cSJeff Kirsher 		(((sizeof(struct ksz_hw_desc) + DESC_ALIGNMENT - 1) /
4536bcc9736cSJeff Kirsher 		DESC_ALIGNMENT) * DESC_ALIGNMENT);
4537bcc9736cSJeff Kirsher 	if (hw->rx_desc_info.size != sizeof(struct ksz_hw_desc))
4538bcc9736cSJeff Kirsher 		pr_alert("Hardware descriptor size not right!\n");
4539bcc9736cSJeff Kirsher 	ksz_check_desc_num(&hw->rx_desc_info);
4540bcc9736cSJeff Kirsher 	ksz_check_desc_num(&hw->tx_desc_info);
4541bcc9736cSJeff Kirsher 
4542bcc9736cSJeff Kirsher 	/* Allocate descriptors. */
4543bcc9736cSJeff Kirsher 	if (ksz_alloc_desc(adapter))
4544bcc9736cSJeff Kirsher 		return 1;
4545bcc9736cSJeff Kirsher 
4546bcc9736cSJeff Kirsher 	return 0;
4547bcc9736cSJeff Kirsher }
4548bcc9736cSJeff Kirsher 
4549bcc9736cSJeff Kirsher /**
4550bcc9736cSJeff Kirsher  * ksz_free_desc - free software and hardware descriptors
4551bcc9736cSJeff Kirsher  * @adapter:	Adapter information structure.
4552bcc9736cSJeff Kirsher  *
4553bcc9736cSJeff Kirsher  * This local routine frees the software and hardware descriptors allocated by
4554bcc9736cSJeff Kirsher  * ksz_alloc_desc().
4555bcc9736cSJeff Kirsher  */
4556bcc9736cSJeff Kirsher static void ksz_free_desc(struct dev_info *adapter)
4557bcc9736cSJeff Kirsher {
4558bcc9736cSJeff Kirsher 	struct ksz_hw *hw = &adapter->hw;
4559bcc9736cSJeff Kirsher 
4560bcc9736cSJeff Kirsher 	/* Reset descriptor. */
4561bcc9736cSJeff Kirsher 	hw->rx_desc_info.ring_virt = NULL;
4562bcc9736cSJeff Kirsher 	hw->tx_desc_info.ring_virt = NULL;
4563bcc9736cSJeff Kirsher 	hw->rx_desc_info.ring_phys = 0;
4564bcc9736cSJeff Kirsher 	hw->tx_desc_info.ring_phys = 0;
4565bcc9736cSJeff Kirsher 
4566bcc9736cSJeff Kirsher 	/* Free memory. */
4567bcc9736cSJeff Kirsher 	if (adapter->desc_pool.alloc_virt)
4568bcc9736cSJeff Kirsher 		pci_free_consistent(
4569bcc9736cSJeff Kirsher 			adapter->pdev,
4570bcc9736cSJeff Kirsher 			adapter->desc_pool.alloc_size,
4571bcc9736cSJeff Kirsher 			adapter->desc_pool.alloc_virt,
4572bcc9736cSJeff Kirsher 			adapter->desc_pool.dma_addr);
4573bcc9736cSJeff Kirsher 
4574bcc9736cSJeff Kirsher 	/* Reset resource pool. */
4575bcc9736cSJeff Kirsher 	adapter->desc_pool.alloc_size = 0;
4576bcc9736cSJeff Kirsher 	adapter->desc_pool.alloc_virt = NULL;
4577bcc9736cSJeff Kirsher 
4578bcc9736cSJeff Kirsher 	kfree(hw->rx_desc_info.ring);
4579bcc9736cSJeff Kirsher 	hw->rx_desc_info.ring = NULL;
4580bcc9736cSJeff Kirsher 	kfree(hw->tx_desc_info.ring);
4581bcc9736cSJeff Kirsher 	hw->tx_desc_info.ring = NULL;
4582bcc9736cSJeff Kirsher }
4583bcc9736cSJeff Kirsher 
4584bcc9736cSJeff Kirsher /**
4585bcc9736cSJeff Kirsher  * ksz_free_buffers - free buffers used in the descriptors
4586bcc9736cSJeff Kirsher  * @adapter:	Adapter information structure.
4587bcc9736cSJeff Kirsher  * @desc_info:	Descriptor information structure.
4588bcc9736cSJeff Kirsher  *
4589bcc9736cSJeff Kirsher  * This local routine frees buffers used in the DMA buffers.
4590bcc9736cSJeff Kirsher  */
4591bcc9736cSJeff Kirsher static void ksz_free_buffers(struct dev_info *adapter,
4592bcc9736cSJeff Kirsher 	struct ksz_desc_info *desc_info, int direction)
4593bcc9736cSJeff Kirsher {
4594bcc9736cSJeff Kirsher 	int i;
4595bcc9736cSJeff Kirsher 	struct ksz_dma_buf *dma_buf;
4596bcc9736cSJeff Kirsher 	struct ksz_desc *desc = desc_info->ring;
4597bcc9736cSJeff Kirsher 
4598bcc9736cSJeff Kirsher 	for (i = 0; i < desc_info->alloc; i++) {
4599bcc9736cSJeff Kirsher 		dma_buf = DMA_BUFFER(desc);
4600bcc9736cSJeff Kirsher 		if (dma_buf->skb)
4601bcc9736cSJeff Kirsher 			free_dma_buf(adapter, dma_buf, direction);
4602bcc9736cSJeff Kirsher 		desc++;
4603bcc9736cSJeff Kirsher 	}
4604bcc9736cSJeff Kirsher }
4605bcc9736cSJeff Kirsher 
4606bcc9736cSJeff Kirsher /**
4607bcc9736cSJeff Kirsher  * ksz_free_mem - free all resources used by descriptors
4608bcc9736cSJeff Kirsher  * @adapter:	Adapter information structure.
4609bcc9736cSJeff Kirsher  *
4610bcc9736cSJeff Kirsher  * This local routine frees all the resources allocated by ksz_alloc_mem().
4611bcc9736cSJeff Kirsher  */
4612bcc9736cSJeff Kirsher static void ksz_free_mem(struct dev_info *adapter)
4613bcc9736cSJeff Kirsher {
4614bcc9736cSJeff Kirsher 	/* Free transmit buffers. */
4615bcc9736cSJeff Kirsher 	ksz_free_buffers(adapter, &adapter->hw.tx_desc_info,
4616bcc9736cSJeff Kirsher 		PCI_DMA_TODEVICE);
4617bcc9736cSJeff Kirsher 
4618bcc9736cSJeff Kirsher 	/* Free receive buffers. */
4619bcc9736cSJeff Kirsher 	ksz_free_buffers(adapter, &adapter->hw.rx_desc_info,
4620bcc9736cSJeff Kirsher 		PCI_DMA_FROMDEVICE);
4621bcc9736cSJeff Kirsher 
4622bcc9736cSJeff Kirsher 	/* Free descriptors. */
4623bcc9736cSJeff Kirsher 	ksz_free_desc(adapter);
4624bcc9736cSJeff Kirsher }
4625bcc9736cSJeff Kirsher 
4626bcc9736cSJeff Kirsher static void get_mib_counters(struct ksz_hw *hw, int first, int cnt,
4627bcc9736cSJeff Kirsher 	u64 *counter)
4628bcc9736cSJeff Kirsher {
4629bcc9736cSJeff Kirsher 	int i;
4630bcc9736cSJeff Kirsher 	int mib;
4631bcc9736cSJeff Kirsher 	int port;
4632bcc9736cSJeff Kirsher 	struct ksz_port_mib *port_mib;
4633bcc9736cSJeff Kirsher 
4634bcc9736cSJeff Kirsher 	memset(counter, 0, sizeof(u64) * TOTAL_PORT_COUNTER_NUM);
4635bcc9736cSJeff Kirsher 	for (i = 0, port = first; i < cnt; i++, port++) {
4636bcc9736cSJeff Kirsher 		port_mib = &hw->port_mib[port];
4637bcc9736cSJeff Kirsher 		for (mib = port_mib->mib_start; mib < hw->mib_cnt; mib++)
4638bcc9736cSJeff Kirsher 			counter[mib] += port_mib->counter[mib];
4639bcc9736cSJeff Kirsher 	}
4640bcc9736cSJeff Kirsher }
4641bcc9736cSJeff Kirsher 
4642bcc9736cSJeff Kirsher /**
4643bcc9736cSJeff Kirsher  * send_packet - send packet
4644bcc9736cSJeff Kirsher  * @skb:	Socket buffer.
4645bcc9736cSJeff Kirsher  * @dev:	Network device.
4646bcc9736cSJeff Kirsher  *
4647bcc9736cSJeff Kirsher  * This routine is used to send a packet out to the network.
4648bcc9736cSJeff Kirsher  */
4649bcc9736cSJeff Kirsher static void send_packet(struct sk_buff *skb, struct net_device *dev)
4650bcc9736cSJeff Kirsher {
4651bcc9736cSJeff Kirsher 	struct ksz_desc *desc;
4652bcc9736cSJeff Kirsher 	struct ksz_desc *first;
4653bcc9736cSJeff Kirsher 	struct dev_priv *priv = netdev_priv(dev);
4654bcc9736cSJeff Kirsher 	struct dev_info *hw_priv = priv->adapter;
4655bcc9736cSJeff Kirsher 	struct ksz_hw *hw = &hw_priv->hw;
4656bcc9736cSJeff Kirsher 	struct ksz_desc_info *info = &hw->tx_desc_info;
4657bcc9736cSJeff Kirsher 	struct ksz_dma_buf *dma_buf;
4658bcc9736cSJeff Kirsher 	int len;
4659bcc9736cSJeff Kirsher 	int last_frag = skb_shinfo(skb)->nr_frags;
4660bcc9736cSJeff Kirsher 
4661bcc9736cSJeff Kirsher 	/*
4662bcc9736cSJeff Kirsher 	 * KSZ8842 with multiple device interfaces needs to be told which port
4663bcc9736cSJeff Kirsher 	 * to send.
4664bcc9736cSJeff Kirsher 	 */
4665bcc9736cSJeff Kirsher 	if (hw->dev_count > 1)
4666bcc9736cSJeff Kirsher 		hw->dst_ports = 1 << priv->port.first_port;
4667bcc9736cSJeff Kirsher 
4668bcc9736cSJeff Kirsher 	/* Hardware will pad the length to 60. */
4669bcc9736cSJeff Kirsher 	len = skb->len;
4670bcc9736cSJeff Kirsher 
4671bcc9736cSJeff Kirsher 	/* Remember the very first descriptor. */
4672bcc9736cSJeff Kirsher 	first = info->cur;
4673bcc9736cSJeff Kirsher 	desc = first;
4674bcc9736cSJeff Kirsher 
4675bcc9736cSJeff Kirsher 	dma_buf = DMA_BUFFER(desc);
4676bcc9736cSJeff Kirsher 	if (last_frag) {
4677bcc9736cSJeff Kirsher 		int frag;
4678bcc9736cSJeff Kirsher 		skb_frag_t *this_frag;
4679bcc9736cSJeff Kirsher 
4680bcc9736cSJeff Kirsher 		dma_buf->len = skb_headlen(skb);
4681bcc9736cSJeff Kirsher 
4682bcc9736cSJeff Kirsher 		dma_buf->dma = pci_map_single(
4683bcc9736cSJeff Kirsher 			hw_priv->pdev, skb->data, dma_buf->len,
4684bcc9736cSJeff Kirsher 			PCI_DMA_TODEVICE);
4685bcc9736cSJeff Kirsher 		set_tx_buf(desc, dma_buf->dma);
4686bcc9736cSJeff Kirsher 		set_tx_len(desc, dma_buf->len);
4687bcc9736cSJeff Kirsher 
4688bcc9736cSJeff Kirsher 		frag = 0;
4689bcc9736cSJeff Kirsher 		do {
4690bcc9736cSJeff Kirsher 			this_frag = &skb_shinfo(skb)->frags[frag];
4691bcc9736cSJeff Kirsher 
4692bcc9736cSJeff Kirsher 			/* Get a new descriptor. */
4693bcc9736cSJeff Kirsher 			get_tx_pkt(info, &desc);
4694bcc9736cSJeff Kirsher 
4695bcc9736cSJeff Kirsher 			/* Keep track of descriptors used so far. */
4696bcc9736cSJeff Kirsher 			++hw->tx_int_cnt;
4697bcc9736cSJeff Kirsher 
4698bcc9736cSJeff Kirsher 			dma_buf = DMA_BUFFER(desc);
46999e903e08SEric Dumazet 			dma_buf->len = skb_frag_size(this_frag);
4700bcc9736cSJeff Kirsher 
4701bcc9736cSJeff Kirsher 			dma_buf->dma = pci_map_single(
4702bcc9736cSJeff Kirsher 				hw_priv->pdev,
4703787343adSIan Campbell 				skb_frag_address(this_frag),
4704bcc9736cSJeff Kirsher 				dma_buf->len,
4705bcc9736cSJeff Kirsher 				PCI_DMA_TODEVICE);
4706bcc9736cSJeff Kirsher 			set_tx_buf(desc, dma_buf->dma);
4707bcc9736cSJeff Kirsher 			set_tx_len(desc, dma_buf->len);
4708bcc9736cSJeff Kirsher 
4709bcc9736cSJeff Kirsher 			frag++;
4710bcc9736cSJeff Kirsher 			if (frag == last_frag)
4711bcc9736cSJeff Kirsher 				break;
4712bcc9736cSJeff Kirsher 
4713bcc9736cSJeff Kirsher 			/* Do not release the last descriptor here. */
4714bcc9736cSJeff Kirsher 			release_desc(desc);
4715bcc9736cSJeff Kirsher 		} while (1);
4716bcc9736cSJeff Kirsher 
4717bcc9736cSJeff Kirsher 		/* current points to the last descriptor. */
4718bcc9736cSJeff Kirsher 		info->cur = desc;
4719bcc9736cSJeff Kirsher 
4720bcc9736cSJeff Kirsher 		/* Release the first descriptor. */
4721bcc9736cSJeff Kirsher 		release_desc(first);
4722bcc9736cSJeff Kirsher 	} else {
4723bcc9736cSJeff Kirsher 		dma_buf->len = len;
4724bcc9736cSJeff Kirsher 
4725bcc9736cSJeff Kirsher 		dma_buf->dma = pci_map_single(
4726bcc9736cSJeff Kirsher 			hw_priv->pdev, skb->data, dma_buf->len,
4727bcc9736cSJeff Kirsher 			PCI_DMA_TODEVICE);
4728bcc9736cSJeff Kirsher 		set_tx_buf(desc, dma_buf->dma);
4729bcc9736cSJeff Kirsher 		set_tx_len(desc, dma_buf->len);
4730bcc9736cSJeff Kirsher 	}
4731bcc9736cSJeff Kirsher 
4732bcc9736cSJeff Kirsher 	if (skb->ip_summed == CHECKSUM_PARTIAL) {
4733bcc9736cSJeff Kirsher 		(desc)->sw.buf.tx.csum_gen_tcp = 1;
4734bcc9736cSJeff Kirsher 		(desc)->sw.buf.tx.csum_gen_udp = 1;
4735bcc9736cSJeff Kirsher 	}
4736bcc9736cSJeff Kirsher 
4737bcc9736cSJeff Kirsher 	/*
4738bcc9736cSJeff Kirsher 	 * The last descriptor holds the packet so that it can be returned to
4739bcc9736cSJeff Kirsher 	 * network subsystem after all descriptors are transmitted.
4740bcc9736cSJeff Kirsher 	 */
4741bcc9736cSJeff Kirsher 	dma_buf->skb = skb;
4742bcc9736cSJeff Kirsher 
4743bcc9736cSJeff Kirsher 	hw_send_pkt(hw);
4744bcc9736cSJeff Kirsher 
4745bcc9736cSJeff Kirsher 	/* Update transmit statistics. */
4746bcc9736cSJeff Kirsher 	dev->stats.tx_packets++;
4747bcc9736cSJeff Kirsher 	dev->stats.tx_bytes += len;
4748bcc9736cSJeff Kirsher }
4749bcc9736cSJeff Kirsher 
4750bcc9736cSJeff Kirsher /**
4751bcc9736cSJeff Kirsher  * transmit_cleanup - clean up transmit descriptors
4752bcc9736cSJeff Kirsher  * @dev:	Network device.
4753bcc9736cSJeff Kirsher  *
4754bcc9736cSJeff Kirsher  * This routine is called to clean up the transmitted buffers.
4755bcc9736cSJeff Kirsher  */
4756bcc9736cSJeff Kirsher static void transmit_cleanup(struct dev_info *hw_priv, int normal)
4757bcc9736cSJeff Kirsher {
4758bcc9736cSJeff Kirsher 	int last;
4759bcc9736cSJeff Kirsher 	union desc_stat status;
4760bcc9736cSJeff Kirsher 	struct ksz_hw *hw = &hw_priv->hw;
4761bcc9736cSJeff Kirsher 	struct ksz_desc_info *info = &hw->tx_desc_info;
4762bcc9736cSJeff Kirsher 	struct ksz_desc *desc;
4763bcc9736cSJeff Kirsher 	struct ksz_dma_buf *dma_buf;
4764bcc9736cSJeff Kirsher 	struct net_device *dev = NULL;
4765bcc9736cSJeff Kirsher 
4766bcc9736cSJeff Kirsher 	spin_lock(&hw_priv->hwlock);
4767bcc9736cSJeff Kirsher 	last = info->last;
4768bcc9736cSJeff Kirsher 
4769bcc9736cSJeff Kirsher 	while (info->avail < info->alloc) {
4770bcc9736cSJeff Kirsher 		/* Get next descriptor which is not hardware owned. */
4771bcc9736cSJeff Kirsher 		desc = &info->ring[last];
4772bcc9736cSJeff Kirsher 		status.data = le32_to_cpu(desc->phw->ctrl.data);
4773bcc9736cSJeff Kirsher 		if (status.tx.hw_owned) {
4774bcc9736cSJeff Kirsher 			if (normal)
4775bcc9736cSJeff Kirsher 				break;
4776bcc9736cSJeff Kirsher 			else
4777bcc9736cSJeff Kirsher 				reset_desc(desc, status);
4778bcc9736cSJeff Kirsher 		}
4779bcc9736cSJeff Kirsher 
4780bcc9736cSJeff Kirsher 		dma_buf = DMA_BUFFER(desc);
4781bcc9736cSJeff Kirsher 		pci_unmap_single(
4782bcc9736cSJeff Kirsher 			hw_priv->pdev, dma_buf->dma, dma_buf->len,
4783bcc9736cSJeff Kirsher 			PCI_DMA_TODEVICE);
4784bcc9736cSJeff Kirsher 
4785bcc9736cSJeff Kirsher 		/* This descriptor contains the last buffer in the packet. */
4786bcc9736cSJeff Kirsher 		if (dma_buf->skb) {
4787bcc9736cSJeff Kirsher 			dev = dma_buf->skb->dev;
4788bcc9736cSJeff Kirsher 
4789bcc9736cSJeff Kirsher 			/* Release the packet back to network subsystem. */
4790bcc9736cSJeff Kirsher 			dev_kfree_skb_irq(dma_buf->skb);
4791bcc9736cSJeff Kirsher 			dma_buf->skb = NULL;
4792bcc9736cSJeff Kirsher 		}
4793bcc9736cSJeff Kirsher 
4794bcc9736cSJeff Kirsher 		/* Free the transmitted descriptor. */
4795bcc9736cSJeff Kirsher 		last++;
4796bcc9736cSJeff Kirsher 		last &= info->mask;
4797bcc9736cSJeff Kirsher 		info->avail++;
4798bcc9736cSJeff Kirsher 	}
4799bcc9736cSJeff Kirsher 	info->last = last;
4800bcc9736cSJeff Kirsher 	spin_unlock(&hw_priv->hwlock);
4801bcc9736cSJeff Kirsher 
4802bcc9736cSJeff Kirsher 	/* Notify the network subsystem that the packet has been sent. */
4803bcc9736cSJeff Kirsher 	if (dev)
4804bcc9736cSJeff Kirsher 		dev->trans_start = jiffies;
4805bcc9736cSJeff Kirsher }
4806bcc9736cSJeff Kirsher 
4807bcc9736cSJeff Kirsher /**
4808bcc9736cSJeff Kirsher  * transmit_done - transmit done processing
4809bcc9736cSJeff Kirsher  * @dev:	Network device.
4810bcc9736cSJeff Kirsher  *
4811bcc9736cSJeff Kirsher  * This routine is called when the transmit interrupt is triggered, indicating
4812bcc9736cSJeff Kirsher  * either a packet is sent successfully or there are transmit errors.
4813bcc9736cSJeff Kirsher  */
4814bcc9736cSJeff Kirsher static void tx_done(struct dev_info *hw_priv)
4815bcc9736cSJeff Kirsher {
4816bcc9736cSJeff Kirsher 	struct ksz_hw *hw = &hw_priv->hw;
4817bcc9736cSJeff Kirsher 	int port;
4818bcc9736cSJeff Kirsher 
4819bcc9736cSJeff Kirsher 	transmit_cleanup(hw_priv, 1);
4820bcc9736cSJeff Kirsher 
4821bcc9736cSJeff Kirsher 	for (port = 0; port < hw->dev_count; port++) {
4822bcc9736cSJeff Kirsher 		struct net_device *dev = hw->port_info[port].pdev;
4823bcc9736cSJeff Kirsher 
4824bcc9736cSJeff Kirsher 		if (netif_running(dev) && netif_queue_stopped(dev))
4825bcc9736cSJeff Kirsher 			netif_wake_queue(dev);
4826bcc9736cSJeff Kirsher 	}
4827bcc9736cSJeff Kirsher }
4828bcc9736cSJeff Kirsher 
4829bcc9736cSJeff Kirsher static inline void copy_old_skb(struct sk_buff *old, struct sk_buff *skb)
4830bcc9736cSJeff Kirsher {
4831bcc9736cSJeff Kirsher 	skb->dev = old->dev;
4832bcc9736cSJeff Kirsher 	skb->protocol = old->protocol;
4833bcc9736cSJeff Kirsher 	skb->ip_summed = old->ip_summed;
4834bcc9736cSJeff Kirsher 	skb->csum = old->csum;
4835bcc9736cSJeff Kirsher 	skb_set_network_header(skb, ETH_HLEN);
4836bcc9736cSJeff Kirsher 
4837bcc9736cSJeff Kirsher 	dev_kfree_skb(old);
4838bcc9736cSJeff Kirsher }
4839bcc9736cSJeff Kirsher 
4840bcc9736cSJeff Kirsher /**
4841bcc9736cSJeff Kirsher  * netdev_tx - send out packet
4842bcc9736cSJeff Kirsher  * @skb:	Socket buffer.
4843bcc9736cSJeff Kirsher  * @dev:	Network device.
4844bcc9736cSJeff Kirsher  *
4845bcc9736cSJeff Kirsher  * This function is used by the upper network layer to send out a packet.
4846bcc9736cSJeff Kirsher  *
4847bcc9736cSJeff Kirsher  * Return 0 if successful; otherwise an error code indicating failure.
4848bcc9736cSJeff Kirsher  */
4849bcc9736cSJeff Kirsher static netdev_tx_t netdev_tx(struct sk_buff *skb, struct net_device *dev)
4850bcc9736cSJeff Kirsher {
4851bcc9736cSJeff Kirsher 	struct dev_priv *priv = netdev_priv(dev);
4852bcc9736cSJeff Kirsher 	struct dev_info *hw_priv = priv->adapter;
4853bcc9736cSJeff Kirsher 	struct ksz_hw *hw = &hw_priv->hw;
4854bcc9736cSJeff Kirsher 	int left;
4855bcc9736cSJeff Kirsher 	int num = 1;
4856bcc9736cSJeff Kirsher 	int rc = 0;
4857bcc9736cSJeff Kirsher 
4858bcc9736cSJeff Kirsher 	if (hw->features & SMALL_PACKET_TX_BUG) {
4859bcc9736cSJeff Kirsher 		struct sk_buff *org_skb = skb;
4860bcc9736cSJeff Kirsher 
4861bcc9736cSJeff Kirsher 		if (skb->len <= 48) {
4862bcc9736cSJeff Kirsher 			if (skb_end_pointer(skb) - skb->data >= 50) {
4863bcc9736cSJeff Kirsher 				memset(&skb->data[skb->len], 0, 50 - skb->len);
4864bcc9736cSJeff Kirsher 				skb->len = 50;
4865bcc9736cSJeff Kirsher 			} else {
4866bcc9736cSJeff Kirsher 				skb = dev_alloc_skb(50);
4867bcc9736cSJeff Kirsher 				if (!skb)
4868bcc9736cSJeff Kirsher 					return NETDEV_TX_BUSY;
4869bcc9736cSJeff Kirsher 				memcpy(skb->data, org_skb->data, org_skb->len);
4870bcc9736cSJeff Kirsher 				memset(&skb->data[org_skb->len], 0,
4871bcc9736cSJeff Kirsher 					50 - org_skb->len);
4872bcc9736cSJeff Kirsher 				skb->len = 50;
4873bcc9736cSJeff Kirsher 				copy_old_skb(org_skb, skb);
4874bcc9736cSJeff Kirsher 			}
4875bcc9736cSJeff Kirsher 		}
4876bcc9736cSJeff Kirsher 	}
4877bcc9736cSJeff Kirsher 
4878bcc9736cSJeff Kirsher 	spin_lock_irq(&hw_priv->hwlock);
4879bcc9736cSJeff Kirsher 
4880bcc9736cSJeff Kirsher 	num = skb_shinfo(skb)->nr_frags + 1;
4881bcc9736cSJeff Kirsher 	left = hw_alloc_pkt(hw, skb->len, num);
4882bcc9736cSJeff Kirsher 	if (left) {
4883bcc9736cSJeff Kirsher 		if (left < num ||
4884bcc9736cSJeff Kirsher 				((CHECKSUM_PARTIAL == skb->ip_summed) &&
4885bcc9736cSJeff Kirsher 				(ETH_P_IPV6 == htons(skb->protocol)))) {
4886bcc9736cSJeff Kirsher 			struct sk_buff *org_skb = skb;
4887bcc9736cSJeff Kirsher 
4888bcc9736cSJeff Kirsher 			skb = dev_alloc_skb(org_skb->len);
4889bcc9736cSJeff Kirsher 			if (!skb) {
4890bcc9736cSJeff Kirsher 				rc = NETDEV_TX_BUSY;
4891bcc9736cSJeff Kirsher 				goto unlock;
4892bcc9736cSJeff Kirsher 			}
4893bcc9736cSJeff Kirsher 			skb_copy_and_csum_dev(org_skb, skb->data);
4894bcc9736cSJeff Kirsher 			org_skb->ip_summed = CHECKSUM_NONE;
4895bcc9736cSJeff Kirsher 			skb->len = org_skb->len;
4896bcc9736cSJeff Kirsher 			copy_old_skb(org_skb, skb);
4897bcc9736cSJeff Kirsher 		}
4898bcc9736cSJeff Kirsher 		send_packet(skb, dev);
4899bcc9736cSJeff Kirsher 		if (left <= num)
4900bcc9736cSJeff Kirsher 			netif_stop_queue(dev);
4901bcc9736cSJeff Kirsher 	} else {
4902bcc9736cSJeff Kirsher 		/* Stop the transmit queue until packet is allocated. */
4903bcc9736cSJeff Kirsher 		netif_stop_queue(dev);
4904bcc9736cSJeff Kirsher 		rc = NETDEV_TX_BUSY;
4905bcc9736cSJeff Kirsher 	}
4906bcc9736cSJeff Kirsher unlock:
4907bcc9736cSJeff Kirsher 	spin_unlock_irq(&hw_priv->hwlock);
4908bcc9736cSJeff Kirsher 
4909bcc9736cSJeff Kirsher 	return rc;
4910bcc9736cSJeff Kirsher }
4911bcc9736cSJeff Kirsher 
4912bcc9736cSJeff Kirsher /**
4913bcc9736cSJeff Kirsher  * netdev_tx_timeout - transmit timeout processing
4914bcc9736cSJeff Kirsher  * @dev:	Network device.
4915bcc9736cSJeff Kirsher  *
4916bcc9736cSJeff Kirsher  * This routine is called when the transmit timer expires.  That indicates the
4917bcc9736cSJeff Kirsher  * hardware is not running correctly because transmit interrupts are not
4918bcc9736cSJeff Kirsher  * triggered to free up resources so that the transmit routine can continue
4919bcc9736cSJeff Kirsher  * sending out packets.  The hardware is reset to correct the problem.
4920bcc9736cSJeff Kirsher  */
4921bcc9736cSJeff Kirsher static void netdev_tx_timeout(struct net_device *dev)
4922bcc9736cSJeff Kirsher {
4923bcc9736cSJeff Kirsher 	static unsigned long last_reset;
4924bcc9736cSJeff Kirsher 
4925bcc9736cSJeff Kirsher 	struct dev_priv *priv = netdev_priv(dev);
4926bcc9736cSJeff Kirsher 	struct dev_info *hw_priv = priv->adapter;
4927bcc9736cSJeff Kirsher 	struct ksz_hw *hw = &hw_priv->hw;
4928bcc9736cSJeff Kirsher 	int port;
4929bcc9736cSJeff Kirsher 
4930bcc9736cSJeff Kirsher 	if (hw->dev_count > 1) {
4931bcc9736cSJeff Kirsher 		/*
4932bcc9736cSJeff Kirsher 		 * Only reset the hardware if time between calls is long
4933bcc9736cSJeff Kirsher 		 * enough.
4934bcc9736cSJeff Kirsher 		 */
4935bcc9736cSJeff Kirsher 		if (jiffies - last_reset <= dev->watchdog_timeo)
4936bcc9736cSJeff Kirsher 			hw_priv = NULL;
4937bcc9736cSJeff Kirsher 	}
4938bcc9736cSJeff Kirsher 
4939bcc9736cSJeff Kirsher 	last_reset = jiffies;
4940bcc9736cSJeff Kirsher 	if (hw_priv) {
4941bcc9736cSJeff Kirsher 		hw_dis_intr(hw);
4942bcc9736cSJeff Kirsher 		hw_disable(hw);
4943bcc9736cSJeff Kirsher 
4944bcc9736cSJeff Kirsher 		transmit_cleanup(hw_priv, 0);
4945bcc9736cSJeff Kirsher 		hw_reset_pkts(&hw->rx_desc_info);
4946bcc9736cSJeff Kirsher 		hw_reset_pkts(&hw->tx_desc_info);
4947bcc9736cSJeff Kirsher 		ksz_init_rx_buffers(hw_priv);
4948bcc9736cSJeff Kirsher 
4949bcc9736cSJeff Kirsher 		hw_reset(hw);
4950bcc9736cSJeff Kirsher 
4951bcc9736cSJeff Kirsher 		hw_set_desc_base(hw,
4952bcc9736cSJeff Kirsher 			hw->tx_desc_info.ring_phys,
4953bcc9736cSJeff Kirsher 			hw->rx_desc_info.ring_phys);
4954bcc9736cSJeff Kirsher 		hw_set_addr(hw);
4955bcc9736cSJeff Kirsher 		if (hw->all_multi)
4956bcc9736cSJeff Kirsher 			hw_set_multicast(hw, hw->all_multi);
4957bcc9736cSJeff Kirsher 		else if (hw->multi_list_size)
4958bcc9736cSJeff Kirsher 			hw_set_grp_addr(hw);
4959bcc9736cSJeff Kirsher 
4960bcc9736cSJeff Kirsher 		if (hw->dev_count > 1) {
4961bcc9736cSJeff Kirsher 			hw_set_add_addr(hw);
4962bcc9736cSJeff Kirsher 			for (port = 0; port < SWITCH_PORT_NUM; port++) {
4963bcc9736cSJeff Kirsher 				struct net_device *port_dev;
4964bcc9736cSJeff Kirsher 
4965bcc9736cSJeff Kirsher 				port_set_stp_state(hw, port,
4966bcc9736cSJeff Kirsher 					STP_STATE_DISABLED);
4967bcc9736cSJeff Kirsher 
4968bcc9736cSJeff Kirsher 				port_dev = hw->port_info[port].pdev;
4969bcc9736cSJeff Kirsher 				if (netif_running(port_dev))
4970bcc9736cSJeff Kirsher 					port_set_stp_state(hw, port,
4971bcc9736cSJeff Kirsher 						STP_STATE_SIMPLE);
4972bcc9736cSJeff Kirsher 			}
4973bcc9736cSJeff Kirsher 		}
4974bcc9736cSJeff Kirsher 
4975bcc9736cSJeff Kirsher 		hw_enable(hw);
4976bcc9736cSJeff Kirsher 		hw_ena_intr(hw);
4977bcc9736cSJeff Kirsher 	}
4978bcc9736cSJeff Kirsher 
4979bcc9736cSJeff Kirsher 	dev->trans_start = jiffies;
4980bcc9736cSJeff Kirsher 	netif_wake_queue(dev);
4981bcc9736cSJeff Kirsher }
4982bcc9736cSJeff Kirsher 
4983bcc9736cSJeff Kirsher static inline void csum_verified(struct sk_buff *skb)
4984bcc9736cSJeff Kirsher {
4985bcc9736cSJeff Kirsher 	unsigned short protocol;
4986bcc9736cSJeff Kirsher 	struct iphdr *iph;
4987bcc9736cSJeff Kirsher 
4988bcc9736cSJeff Kirsher 	protocol = skb->protocol;
4989bcc9736cSJeff Kirsher 	skb_reset_network_header(skb);
4990bcc9736cSJeff Kirsher 	iph = (struct iphdr *) skb_network_header(skb);
4991bcc9736cSJeff Kirsher 	if (protocol == htons(ETH_P_8021Q)) {
4992bcc9736cSJeff Kirsher 		protocol = iph->tot_len;
4993bcc9736cSJeff Kirsher 		skb_set_network_header(skb, VLAN_HLEN);
4994bcc9736cSJeff Kirsher 		iph = (struct iphdr *) skb_network_header(skb);
4995bcc9736cSJeff Kirsher 	}
4996bcc9736cSJeff Kirsher 	if (protocol == htons(ETH_P_IP)) {
4997bcc9736cSJeff Kirsher 		if (iph->protocol == IPPROTO_TCP)
4998bcc9736cSJeff Kirsher 			skb->ip_summed = CHECKSUM_UNNECESSARY;
4999bcc9736cSJeff Kirsher 	}
5000bcc9736cSJeff Kirsher }
5001bcc9736cSJeff Kirsher 
5002bcc9736cSJeff Kirsher static inline int rx_proc(struct net_device *dev, struct ksz_hw* hw,
5003bcc9736cSJeff Kirsher 	struct ksz_desc *desc, union desc_stat status)
5004bcc9736cSJeff Kirsher {
5005bcc9736cSJeff Kirsher 	int packet_len;
5006bcc9736cSJeff Kirsher 	struct dev_priv *priv = netdev_priv(dev);
5007bcc9736cSJeff Kirsher 	struct dev_info *hw_priv = priv->adapter;
5008bcc9736cSJeff Kirsher 	struct ksz_dma_buf *dma_buf;
5009bcc9736cSJeff Kirsher 	struct sk_buff *skb;
5010bcc9736cSJeff Kirsher 	int rx_status;
5011bcc9736cSJeff Kirsher 
5012bcc9736cSJeff Kirsher 	/* Received length includes 4-byte CRC. */
5013bcc9736cSJeff Kirsher 	packet_len = status.rx.frame_len - 4;
5014bcc9736cSJeff Kirsher 
5015bcc9736cSJeff Kirsher 	dma_buf = DMA_BUFFER(desc);
5016bcc9736cSJeff Kirsher 	pci_dma_sync_single_for_cpu(
5017bcc9736cSJeff Kirsher 		hw_priv->pdev, dma_buf->dma, packet_len + 4,
5018bcc9736cSJeff Kirsher 		PCI_DMA_FROMDEVICE);
5019bcc9736cSJeff Kirsher 
5020bcc9736cSJeff Kirsher 	do {
5021bcc9736cSJeff Kirsher 		/* skb->data != skb->head */
5022bcc9736cSJeff Kirsher 		skb = dev_alloc_skb(packet_len + 2);
5023bcc9736cSJeff Kirsher 		if (!skb) {
5024bcc9736cSJeff Kirsher 			dev->stats.rx_dropped++;
5025bcc9736cSJeff Kirsher 			return -ENOMEM;
5026bcc9736cSJeff Kirsher 		}
5027bcc9736cSJeff Kirsher 
5028bcc9736cSJeff Kirsher 		/*
5029bcc9736cSJeff Kirsher 		 * Align socket buffer in 4-byte boundary for better
5030bcc9736cSJeff Kirsher 		 * performance.
5031bcc9736cSJeff Kirsher 		 */
5032bcc9736cSJeff Kirsher 		skb_reserve(skb, 2);
5033bcc9736cSJeff Kirsher 
5034bcc9736cSJeff Kirsher 		memcpy(skb_put(skb, packet_len),
5035bcc9736cSJeff Kirsher 			dma_buf->skb->data, packet_len);
5036bcc9736cSJeff Kirsher 	} while (0);
5037bcc9736cSJeff Kirsher 
5038bcc9736cSJeff Kirsher 	skb->protocol = eth_type_trans(skb, dev);
5039bcc9736cSJeff Kirsher 
5040bcc9736cSJeff Kirsher 	if (hw->rx_cfg & (DMA_RX_CSUM_UDP | DMA_RX_CSUM_TCP))
5041bcc9736cSJeff Kirsher 		csum_verified(skb);
5042bcc9736cSJeff Kirsher 
5043bcc9736cSJeff Kirsher 	/* Update receive statistics. */
5044bcc9736cSJeff Kirsher 	dev->stats.rx_packets++;
5045bcc9736cSJeff Kirsher 	dev->stats.rx_bytes += packet_len;
5046bcc9736cSJeff Kirsher 
5047bcc9736cSJeff Kirsher 	/* Notify upper layer for received packet. */
5048bcc9736cSJeff Kirsher 	rx_status = netif_rx(skb);
5049bcc9736cSJeff Kirsher 
5050bcc9736cSJeff Kirsher 	return 0;
5051bcc9736cSJeff Kirsher }
5052bcc9736cSJeff Kirsher 
5053bcc9736cSJeff Kirsher static int dev_rcv_packets(struct dev_info *hw_priv)
5054bcc9736cSJeff Kirsher {
5055bcc9736cSJeff Kirsher 	int next;
5056bcc9736cSJeff Kirsher 	union desc_stat status;
5057bcc9736cSJeff Kirsher 	struct ksz_hw *hw = &hw_priv->hw;
5058bcc9736cSJeff Kirsher 	struct net_device *dev = hw->port_info[0].pdev;
5059bcc9736cSJeff Kirsher 	struct ksz_desc_info *info = &hw->rx_desc_info;
5060bcc9736cSJeff Kirsher 	int left = info->alloc;
5061bcc9736cSJeff Kirsher 	struct ksz_desc *desc;
5062bcc9736cSJeff Kirsher 	int received = 0;
5063bcc9736cSJeff Kirsher 
5064bcc9736cSJeff Kirsher 	next = info->next;
5065bcc9736cSJeff Kirsher 	while (left--) {
5066bcc9736cSJeff Kirsher 		/* Get next descriptor which is not hardware owned. */
5067bcc9736cSJeff Kirsher 		desc = &info->ring[next];
5068bcc9736cSJeff Kirsher 		status.data = le32_to_cpu(desc->phw->ctrl.data);
5069bcc9736cSJeff Kirsher 		if (status.rx.hw_owned)
5070bcc9736cSJeff Kirsher 			break;
5071bcc9736cSJeff Kirsher 
5072bcc9736cSJeff Kirsher 		/* Status valid only when last descriptor bit is set. */
5073bcc9736cSJeff Kirsher 		if (status.rx.last_desc && status.rx.first_desc) {
5074bcc9736cSJeff Kirsher 			if (rx_proc(dev, hw, desc, status))
5075bcc9736cSJeff Kirsher 				goto release_packet;
5076bcc9736cSJeff Kirsher 			received++;
5077bcc9736cSJeff Kirsher 		}
5078bcc9736cSJeff Kirsher 
5079bcc9736cSJeff Kirsher release_packet:
5080bcc9736cSJeff Kirsher 		release_desc(desc);
5081bcc9736cSJeff Kirsher 		next++;
5082bcc9736cSJeff Kirsher 		next &= info->mask;
5083bcc9736cSJeff Kirsher 	}
5084bcc9736cSJeff Kirsher 	info->next = next;
5085bcc9736cSJeff Kirsher 
5086bcc9736cSJeff Kirsher 	return received;
5087bcc9736cSJeff Kirsher }
5088bcc9736cSJeff Kirsher 
5089bcc9736cSJeff Kirsher static int port_rcv_packets(struct dev_info *hw_priv)
5090bcc9736cSJeff Kirsher {
5091bcc9736cSJeff Kirsher 	int next;
5092bcc9736cSJeff Kirsher 	union desc_stat status;
5093bcc9736cSJeff Kirsher 	struct ksz_hw *hw = &hw_priv->hw;
5094bcc9736cSJeff Kirsher 	struct net_device *dev = hw->port_info[0].pdev;
5095bcc9736cSJeff Kirsher 	struct ksz_desc_info *info = &hw->rx_desc_info;
5096bcc9736cSJeff Kirsher 	int left = info->alloc;
5097bcc9736cSJeff Kirsher 	struct ksz_desc *desc;
5098bcc9736cSJeff Kirsher 	int received = 0;
5099bcc9736cSJeff Kirsher 
5100bcc9736cSJeff Kirsher 	next = info->next;
5101bcc9736cSJeff Kirsher 	while (left--) {
5102bcc9736cSJeff Kirsher 		/* Get next descriptor which is not hardware owned. */
5103bcc9736cSJeff Kirsher 		desc = &info->ring[next];
5104bcc9736cSJeff Kirsher 		status.data = le32_to_cpu(desc->phw->ctrl.data);
5105bcc9736cSJeff Kirsher 		if (status.rx.hw_owned)
5106bcc9736cSJeff Kirsher 			break;
5107bcc9736cSJeff Kirsher 
5108bcc9736cSJeff Kirsher 		if (hw->dev_count > 1) {
5109bcc9736cSJeff Kirsher 			/* Get received port number. */
5110bcc9736cSJeff Kirsher 			int p = HW_TO_DEV_PORT(status.rx.src_port);
5111bcc9736cSJeff Kirsher 
5112bcc9736cSJeff Kirsher 			dev = hw->port_info[p].pdev;
5113bcc9736cSJeff Kirsher 			if (!netif_running(dev))
5114bcc9736cSJeff Kirsher 				goto release_packet;
5115bcc9736cSJeff Kirsher 		}
5116bcc9736cSJeff Kirsher 
5117bcc9736cSJeff Kirsher 		/* Status valid only when last descriptor bit is set. */
5118bcc9736cSJeff Kirsher 		if (status.rx.last_desc && status.rx.first_desc) {
5119bcc9736cSJeff Kirsher 			if (rx_proc(dev, hw, desc, status))
5120bcc9736cSJeff Kirsher 				goto release_packet;
5121bcc9736cSJeff Kirsher 			received++;
5122bcc9736cSJeff Kirsher 		}
5123bcc9736cSJeff Kirsher 
5124bcc9736cSJeff Kirsher release_packet:
5125bcc9736cSJeff Kirsher 		release_desc(desc);
5126bcc9736cSJeff Kirsher 		next++;
5127bcc9736cSJeff Kirsher 		next &= info->mask;
5128bcc9736cSJeff Kirsher 	}
5129bcc9736cSJeff Kirsher 	info->next = next;
5130bcc9736cSJeff Kirsher 
5131bcc9736cSJeff Kirsher 	return received;
5132bcc9736cSJeff Kirsher }
5133bcc9736cSJeff Kirsher 
5134bcc9736cSJeff Kirsher static int dev_rcv_special(struct dev_info *hw_priv)
5135bcc9736cSJeff Kirsher {
5136bcc9736cSJeff Kirsher 	int next;
5137bcc9736cSJeff Kirsher 	union desc_stat status;
5138bcc9736cSJeff Kirsher 	struct ksz_hw *hw = &hw_priv->hw;
5139bcc9736cSJeff Kirsher 	struct net_device *dev = hw->port_info[0].pdev;
5140bcc9736cSJeff Kirsher 	struct ksz_desc_info *info = &hw->rx_desc_info;
5141bcc9736cSJeff Kirsher 	int left = info->alloc;
5142bcc9736cSJeff Kirsher 	struct ksz_desc *desc;
5143bcc9736cSJeff Kirsher 	int received = 0;
5144bcc9736cSJeff Kirsher 
5145bcc9736cSJeff Kirsher 	next = info->next;
5146bcc9736cSJeff Kirsher 	while (left--) {
5147bcc9736cSJeff Kirsher 		/* Get next descriptor which is not hardware owned. */
5148bcc9736cSJeff Kirsher 		desc = &info->ring[next];
5149bcc9736cSJeff Kirsher 		status.data = le32_to_cpu(desc->phw->ctrl.data);
5150bcc9736cSJeff Kirsher 		if (status.rx.hw_owned)
5151bcc9736cSJeff Kirsher 			break;
5152bcc9736cSJeff Kirsher 
5153bcc9736cSJeff Kirsher 		if (hw->dev_count > 1) {
5154bcc9736cSJeff Kirsher 			/* Get received port number. */
5155bcc9736cSJeff Kirsher 			int p = HW_TO_DEV_PORT(status.rx.src_port);
5156bcc9736cSJeff Kirsher 
5157bcc9736cSJeff Kirsher 			dev = hw->port_info[p].pdev;
5158bcc9736cSJeff Kirsher 			if (!netif_running(dev))
5159bcc9736cSJeff Kirsher 				goto release_packet;
5160bcc9736cSJeff Kirsher 		}
5161bcc9736cSJeff Kirsher 
5162bcc9736cSJeff Kirsher 		/* Status valid only when last descriptor bit is set. */
5163bcc9736cSJeff Kirsher 		if (status.rx.last_desc && status.rx.first_desc) {
5164bcc9736cSJeff Kirsher 			/*
5165bcc9736cSJeff Kirsher 			 * Receive without error.  With receive errors
5166bcc9736cSJeff Kirsher 			 * disabled, packets with receive errors will be
5167bcc9736cSJeff Kirsher 			 * dropped, so no need to check the error bit.
5168bcc9736cSJeff Kirsher 			 */
5169bcc9736cSJeff Kirsher 			if (!status.rx.error || (status.data &
5170bcc9736cSJeff Kirsher 					KS_DESC_RX_ERROR_COND) ==
5171bcc9736cSJeff Kirsher 					KS_DESC_RX_ERROR_TOO_LONG) {
5172bcc9736cSJeff Kirsher 				if (rx_proc(dev, hw, desc, status))
5173bcc9736cSJeff Kirsher 					goto release_packet;
5174bcc9736cSJeff Kirsher 				received++;
5175bcc9736cSJeff Kirsher 			} else {
5176bcc9736cSJeff Kirsher 				struct dev_priv *priv = netdev_priv(dev);
5177bcc9736cSJeff Kirsher 
5178bcc9736cSJeff Kirsher 				/* Update receive error statistics. */
5179bcc9736cSJeff Kirsher 				priv->port.counter[OID_COUNTER_RCV_ERROR]++;
5180bcc9736cSJeff Kirsher 			}
5181bcc9736cSJeff Kirsher 		}
5182bcc9736cSJeff Kirsher 
5183bcc9736cSJeff Kirsher release_packet:
5184bcc9736cSJeff Kirsher 		release_desc(desc);
5185bcc9736cSJeff Kirsher 		next++;
5186bcc9736cSJeff Kirsher 		next &= info->mask;
5187bcc9736cSJeff Kirsher 	}
5188bcc9736cSJeff Kirsher 	info->next = next;
5189bcc9736cSJeff Kirsher 
5190bcc9736cSJeff Kirsher 	return received;
5191bcc9736cSJeff Kirsher }
5192bcc9736cSJeff Kirsher 
5193bcc9736cSJeff Kirsher static void rx_proc_task(unsigned long data)
5194bcc9736cSJeff Kirsher {
5195bcc9736cSJeff Kirsher 	struct dev_info *hw_priv = (struct dev_info *) data;
5196bcc9736cSJeff Kirsher 	struct ksz_hw *hw = &hw_priv->hw;
5197bcc9736cSJeff Kirsher 
5198bcc9736cSJeff Kirsher 	if (!hw->enabled)
5199bcc9736cSJeff Kirsher 		return;
5200bcc9736cSJeff Kirsher 	if (unlikely(!hw_priv->dev_rcv(hw_priv))) {
5201bcc9736cSJeff Kirsher 
5202bcc9736cSJeff Kirsher 		/* In case receive process is suspended because of overrun. */
5203bcc9736cSJeff Kirsher 		hw_resume_rx(hw);
5204bcc9736cSJeff Kirsher 
5205bcc9736cSJeff Kirsher 		/* tasklets are interruptible. */
5206bcc9736cSJeff Kirsher 		spin_lock_irq(&hw_priv->hwlock);
5207bcc9736cSJeff Kirsher 		hw_turn_on_intr(hw, KS884X_INT_RX_MASK);
5208bcc9736cSJeff Kirsher 		spin_unlock_irq(&hw_priv->hwlock);
5209bcc9736cSJeff Kirsher 	} else {
5210bcc9736cSJeff Kirsher 		hw_ack_intr(hw, KS884X_INT_RX);
5211bcc9736cSJeff Kirsher 		tasklet_schedule(&hw_priv->rx_tasklet);
5212bcc9736cSJeff Kirsher 	}
5213bcc9736cSJeff Kirsher }
5214bcc9736cSJeff Kirsher 
5215bcc9736cSJeff Kirsher static void tx_proc_task(unsigned long data)
5216bcc9736cSJeff Kirsher {
5217bcc9736cSJeff Kirsher 	struct dev_info *hw_priv = (struct dev_info *) data;
5218bcc9736cSJeff Kirsher 	struct ksz_hw *hw = &hw_priv->hw;
5219bcc9736cSJeff Kirsher 
5220bcc9736cSJeff Kirsher 	hw_ack_intr(hw, KS884X_INT_TX_MASK);
5221bcc9736cSJeff Kirsher 
5222bcc9736cSJeff Kirsher 	tx_done(hw_priv);
5223bcc9736cSJeff Kirsher 
5224bcc9736cSJeff Kirsher 	/* tasklets are interruptible. */
5225bcc9736cSJeff Kirsher 	spin_lock_irq(&hw_priv->hwlock);
5226bcc9736cSJeff Kirsher 	hw_turn_on_intr(hw, KS884X_INT_TX);
5227bcc9736cSJeff Kirsher 	spin_unlock_irq(&hw_priv->hwlock);
5228bcc9736cSJeff Kirsher }
5229bcc9736cSJeff Kirsher 
5230bcc9736cSJeff Kirsher static inline void handle_rx_stop(struct ksz_hw *hw)
5231bcc9736cSJeff Kirsher {
5232bcc9736cSJeff Kirsher 	/* Receive just has been stopped. */
5233bcc9736cSJeff Kirsher 	if (0 == hw->rx_stop)
5234bcc9736cSJeff Kirsher 		hw->intr_mask &= ~KS884X_INT_RX_STOPPED;
5235bcc9736cSJeff Kirsher 	else if (hw->rx_stop > 1) {
5236bcc9736cSJeff Kirsher 		if (hw->enabled && (hw->rx_cfg & DMA_RX_ENABLE)) {
5237bcc9736cSJeff Kirsher 			hw_start_rx(hw);
5238bcc9736cSJeff Kirsher 		} else {
5239bcc9736cSJeff Kirsher 			hw->intr_mask &= ~KS884X_INT_RX_STOPPED;
5240bcc9736cSJeff Kirsher 			hw->rx_stop = 0;
5241bcc9736cSJeff Kirsher 		}
5242bcc9736cSJeff Kirsher 	} else
5243bcc9736cSJeff Kirsher 		/* Receive just has been started. */
5244bcc9736cSJeff Kirsher 		hw->rx_stop++;
5245bcc9736cSJeff Kirsher }
5246bcc9736cSJeff Kirsher 
5247bcc9736cSJeff Kirsher /**
5248bcc9736cSJeff Kirsher  * netdev_intr - interrupt handling
5249bcc9736cSJeff Kirsher  * @irq:	Interrupt number.
5250bcc9736cSJeff Kirsher  * @dev_id:	Network device.
5251bcc9736cSJeff Kirsher  *
5252bcc9736cSJeff Kirsher  * This function is called by upper network layer to signal interrupt.
5253bcc9736cSJeff Kirsher  *
5254bcc9736cSJeff Kirsher  * Return IRQ_HANDLED if interrupt is handled.
5255bcc9736cSJeff Kirsher  */
5256bcc9736cSJeff Kirsher static irqreturn_t netdev_intr(int irq, void *dev_id)
5257bcc9736cSJeff Kirsher {
5258bcc9736cSJeff Kirsher 	uint int_enable = 0;
5259bcc9736cSJeff Kirsher 	struct net_device *dev = (struct net_device *) dev_id;
5260bcc9736cSJeff Kirsher 	struct dev_priv *priv = netdev_priv(dev);
5261bcc9736cSJeff Kirsher 	struct dev_info *hw_priv = priv->adapter;
5262bcc9736cSJeff Kirsher 	struct ksz_hw *hw = &hw_priv->hw;
5263bcc9736cSJeff Kirsher 
5264bcc9736cSJeff Kirsher 	hw_read_intr(hw, &int_enable);
5265bcc9736cSJeff Kirsher 
5266bcc9736cSJeff Kirsher 	/* Not our interrupt! */
5267bcc9736cSJeff Kirsher 	if (!int_enable)
5268bcc9736cSJeff Kirsher 		return IRQ_NONE;
5269bcc9736cSJeff Kirsher 
5270bcc9736cSJeff Kirsher 	do {
5271bcc9736cSJeff Kirsher 		hw_ack_intr(hw, int_enable);
5272bcc9736cSJeff Kirsher 		int_enable &= hw->intr_mask;
5273bcc9736cSJeff Kirsher 
5274bcc9736cSJeff Kirsher 		if (unlikely(int_enable & KS884X_INT_TX_MASK)) {
5275bcc9736cSJeff Kirsher 			hw_dis_intr_bit(hw, KS884X_INT_TX_MASK);
5276bcc9736cSJeff Kirsher 			tasklet_schedule(&hw_priv->tx_tasklet);
5277bcc9736cSJeff Kirsher 		}
5278bcc9736cSJeff Kirsher 
5279bcc9736cSJeff Kirsher 		if (likely(int_enable & KS884X_INT_RX)) {
5280bcc9736cSJeff Kirsher 			hw_dis_intr_bit(hw, KS884X_INT_RX);
5281bcc9736cSJeff Kirsher 			tasklet_schedule(&hw_priv->rx_tasklet);
5282bcc9736cSJeff Kirsher 		}
5283bcc9736cSJeff Kirsher 
5284bcc9736cSJeff Kirsher 		if (unlikely(int_enable & KS884X_INT_RX_OVERRUN)) {
5285bcc9736cSJeff Kirsher 			dev->stats.rx_fifo_errors++;
5286bcc9736cSJeff Kirsher 			hw_resume_rx(hw);
5287bcc9736cSJeff Kirsher 		}
5288bcc9736cSJeff Kirsher 
5289bcc9736cSJeff Kirsher 		if (unlikely(int_enable & KS884X_INT_PHY)) {
5290bcc9736cSJeff Kirsher 			struct ksz_port *port = &priv->port;
5291bcc9736cSJeff Kirsher 
5292bcc9736cSJeff Kirsher 			hw->features |= LINK_INT_WORKING;
5293bcc9736cSJeff Kirsher 			port_get_link_speed(port);
5294bcc9736cSJeff Kirsher 		}
5295bcc9736cSJeff Kirsher 
5296bcc9736cSJeff Kirsher 		if (unlikely(int_enable & KS884X_INT_RX_STOPPED)) {
5297bcc9736cSJeff Kirsher 			handle_rx_stop(hw);
5298bcc9736cSJeff Kirsher 			break;
5299bcc9736cSJeff Kirsher 		}
5300bcc9736cSJeff Kirsher 
5301bcc9736cSJeff Kirsher 		if (unlikely(int_enable & KS884X_INT_TX_STOPPED)) {
5302bcc9736cSJeff Kirsher 			u32 data;
5303bcc9736cSJeff Kirsher 
5304bcc9736cSJeff Kirsher 			hw->intr_mask &= ~KS884X_INT_TX_STOPPED;
5305bcc9736cSJeff Kirsher 			pr_info("Tx stopped\n");
5306bcc9736cSJeff Kirsher 			data = readl(hw->io + KS_DMA_TX_CTRL);
5307bcc9736cSJeff Kirsher 			if (!(data & DMA_TX_ENABLE))
5308bcc9736cSJeff Kirsher 				pr_info("Tx disabled\n");
5309bcc9736cSJeff Kirsher 			break;
5310bcc9736cSJeff Kirsher 		}
5311bcc9736cSJeff Kirsher 	} while (0);
5312bcc9736cSJeff Kirsher 
5313bcc9736cSJeff Kirsher 	hw_ena_intr(hw);
5314bcc9736cSJeff Kirsher 
5315bcc9736cSJeff Kirsher 	return IRQ_HANDLED;
5316bcc9736cSJeff Kirsher }
5317bcc9736cSJeff Kirsher 
5318bcc9736cSJeff Kirsher /*
5319bcc9736cSJeff Kirsher  * Linux network device functions
5320bcc9736cSJeff Kirsher  */
5321bcc9736cSJeff Kirsher 
5322bcc9736cSJeff Kirsher static unsigned long next_jiffies;
5323bcc9736cSJeff Kirsher 
5324bcc9736cSJeff Kirsher #ifdef CONFIG_NET_POLL_CONTROLLER
5325bcc9736cSJeff Kirsher static void netdev_netpoll(struct net_device *dev)
5326bcc9736cSJeff Kirsher {
5327bcc9736cSJeff Kirsher 	struct dev_priv *priv = netdev_priv(dev);
5328bcc9736cSJeff Kirsher 	struct dev_info *hw_priv = priv->adapter;
5329bcc9736cSJeff Kirsher 
5330bcc9736cSJeff Kirsher 	hw_dis_intr(&hw_priv->hw);
5331bcc9736cSJeff Kirsher 	netdev_intr(dev->irq, dev);
5332bcc9736cSJeff Kirsher }
5333bcc9736cSJeff Kirsher #endif
5334bcc9736cSJeff Kirsher 
5335bcc9736cSJeff Kirsher static void bridge_change(struct ksz_hw *hw)
5336bcc9736cSJeff Kirsher {
5337bcc9736cSJeff Kirsher 	int port;
5338bcc9736cSJeff Kirsher 	u8  member;
5339bcc9736cSJeff Kirsher 	struct ksz_switch *sw = hw->ksz_switch;
5340bcc9736cSJeff Kirsher 
5341bcc9736cSJeff Kirsher 	/* No ports in forwarding state. */
5342bcc9736cSJeff Kirsher 	if (!sw->member) {
5343bcc9736cSJeff Kirsher 		port_set_stp_state(hw, SWITCH_PORT_NUM, STP_STATE_SIMPLE);
5344bcc9736cSJeff Kirsher 		sw_block_addr(hw);
5345bcc9736cSJeff Kirsher 	}
5346bcc9736cSJeff Kirsher 	for (port = 0; port < SWITCH_PORT_NUM; port++) {
5347bcc9736cSJeff Kirsher 		if (STP_STATE_FORWARDING == sw->port_cfg[port].stp_state)
5348bcc9736cSJeff Kirsher 			member = HOST_MASK | sw->member;
5349bcc9736cSJeff Kirsher 		else
5350bcc9736cSJeff Kirsher 			member = HOST_MASK | (1 << port);
5351bcc9736cSJeff Kirsher 		if (member != sw->port_cfg[port].member)
5352bcc9736cSJeff Kirsher 			sw_cfg_port_base_vlan(hw, port, member);
5353bcc9736cSJeff Kirsher 	}
5354bcc9736cSJeff Kirsher }
5355bcc9736cSJeff Kirsher 
5356bcc9736cSJeff Kirsher /**
5357bcc9736cSJeff Kirsher  * netdev_close - close network device
5358bcc9736cSJeff Kirsher  * @dev:	Network device.
5359bcc9736cSJeff Kirsher  *
5360bcc9736cSJeff Kirsher  * This function process the close operation of network device.  This is caused
5361bcc9736cSJeff Kirsher  * by the user command "ifconfig ethX down."
5362bcc9736cSJeff Kirsher  *
5363bcc9736cSJeff Kirsher  * Return 0 if successful; otherwise an error code indicating failure.
5364bcc9736cSJeff Kirsher  */
5365bcc9736cSJeff Kirsher static int netdev_close(struct net_device *dev)
5366bcc9736cSJeff Kirsher {
5367bcc9736cSJeff Kirsher 	struct dev_priv *priv = netdev_priv(dev);
5368bcc9736cSJeff Kirsher 	struct dev_info *hw_priv = priv->adapter;
5369bcc9736cSJeff Kirsher 	struct ksz_port *port = &priv->port;
5370bcc9736cSJeff Kirsher 	struct ksz_hw *hw = &hw_priv->hw;
5371bcc9736cSJeff Kirsher 	int pi;
5372bcc9736cSJeff Kirsher 
5373bcc9736cSJeff Kirsher 	netif_stop_queue(dev);
5374bcc9736cSJeff Kirsher 
5375bcc9736cSJeff Kirsher 	ksz_stop_timer(&priv->monitor_timer_info);
5376bcc9736cSJeff Kirsher 
5377bcc9736cSJeff Kirsher 	/* Need to shut the port manually in multiple device interfaces mode. */
5378bcc9736cSJeff Kirsher 	if (hw->dev_count > 1) {
5379bcc9736cSJeff Kirsher 		port_set_stp_state(hw, port->first_port, STP_STATE_DISABLED);
5380bcc9736cSJeff Kirsher 
5381bcc9736cSJeff Kirsher 		/* Port is closed.  Need to change bridge setting. */
5382bcc9736cSJeff Kirsher 		if (hw->features & STP_SUPPORT) {
5383bcc9736cSJeff Kirsher 			pi = 1 << port->first_port;
5384bcc9736cSJeff Kirsher 			if (hw->ksz_switch->member & pi) {
5385bcc9736cSJeff Kirsher 				hw->ksz_switch->member &= ~pi;
5386bcc9736cSJeff Kirsher 				bridge_change(hw);
5387bcc9736cSJeff Kirsher 			}
5388bcc9736cSJeff Kirsher 		}
5389bcc9736cSJeff Kirsher 	}
5390bcc9736cSJeff Kirsher 	if (port->first_port > 0)
5391bcc9736cSJeff Kirsher 		hw_del_addr(hw, dev->dev_addr);
5392bcc9736cSJeff Kirsher 	if (!hw_priv->wol_enable)
5393bcc9736cSJeff Kirsher 		port_set_power_saving(port, true);
5394bcc9736cSJeff Kirsher 
5395bcc9736cSJeff Kirsher 	if (priv->multicast)
5396bcc9736cSJeff Kirsher 		--hw->all_multi;
5397bcc9736cSJeff Kirsher 	if (priv->promiscuous)
5398bcc9736cSJeff Kirsher 		--hw->promiscuous;
5399bcc9736cSJeff Kirsher 
5400bcc9736cSJeff Kirsher 	hw_priv->opened--;
5401bcc9736cSJeff Kirsher 	if (!(hw_priv->opened)) {
5402bcc9736cSJeff Kirsher 		ksz_stop_timer(&hw_priv->mib_timer_info);
5403bcc9736cSJeff Kirsher 		flush_work(&hw_priv->mib_read);
5404bcc9736cSJeff Kirsher 
5405bcc9736cSJeff Kirsher 		hw_dis_intr(hw);
5406bcc9736cSJeff Kirsher 		hw_disable(hw);
5407bcc9736cSJeff Kirsher 		hw_clr_multicast(hw);
5408bcc9736cSJeff Kirsher 
5409bcc9736cSJeff Kirsher 		/* Delay for receive task to stop scheduling itself. */
5410bcc9736cSJeff Kirsher 		msleep(2000 / HZ);
5411bcc9736cSJeff Kirsher 
5412bcc9736cSJeff Kirsher 		tasklet_disable(&hw_priv->rx_tasklet);
5413bcc9736cSJeff Kirsher 		tasklet_disable(&hw_priv->tx_tasklet);
5414bcc9736cSJeff Kirsher 		free_irq(dev->irq, hw_priv->dev);
5415bcc9736cSJeff Kirsher 
5416bcc9736cSJeff Kirsher 		transmit_cleanup(hw_priv, 0);
5417bcc9736cSJeff Kirsher 		hw_reset_pkts(&hw->rx_desc_info);
5418bcc9736cSJeff Kirsher 		hw_reset_pkts(&hw->tx_desc_info);
5419bcc9736cSJeff Kirsher 
5420bcc9736cSJeff Kirsher 		/* Clean out static MAC table when the switch is shutdown. */
5421bcc9736cSJeff Kirsher 		if (hw->features & STP_SUPPORT)
5422bcc9736cSJeff Kirsher 			sw_clr_sta_mac_table(hw);
5423bcc9736cSJeff Kirsher 	}
5424bcc9736cSJeff Kirsher 
5425bcc9736cSJeff Kirsher 	return 0;
5426bcc9736cSJeff Kirsher }
5427bcc9736cSJeff Kirsher 
5428bcc9736cSJeff Kirsher static void hw_cfg_huge_frame(struct dev_info *hw_priv, struct ksz_hw *hw)
5429bcc9736cSJeff Kirsher {
5430bcc9736cSJeff Kirsher 	if (hw->ksz_switch) {
5431bcc9736cSJeff Kirsher 		u32 data;
5432bcc9736cSJeff Kirsher 
5433bcc9736cSJeff Kirsher 		data = readw(hw->io + KS8842_SWITCH_CTRL_2_OFFSET);
5434bcc9736cSJeff Kirsher 		if (hw->features & RX_HUGE_FRAME)
5435bcc9736cSJeff Kirsher 			data |= SWITCH_HUGE_PACKET;
5436bcc9736cSJeff Kirsher 		else
5437bcc9736cSJeff Kirsher 			data &= ~SWITCH_HUGE_PACKET;
5438bcc9736cSJeff Kirsher 		writew(data, hw->io + KS8842_SWITCH_CTRL_2_OFFSET);
5439bcc9736cSJeff Kirsher 	}
5440bcc9736cSJeff Kirsher 	if (hw->features & RX_HUGE_FRAME) {
5441bcc9736cSJeff Kirsher 		hw->rx_cfg |= DMA_RX_ERROR;
5442bcc9736cSJeff Kirsher 		hw_priv->dev_rcv = dev_rcv_special;
5443bcc9736cSJeff Kirsher 	} else {
5444bcc9736cSJeff Kirsher 		hw->rx_cfg &= ~DMA_RX_ERROR;
5445bcc9736cSJeff Kirsher 		if (hw->dev_count > 1)
5446bcc9736cSJeff Kirsher 			hw_priv->dev_rcv = port_rcv_packets;
5447bcc9736cSJeff Kirsher 		else
5448bcc9736cSJeff Kirsher 			hw_priv->dev_rcv = dev_rcv_packets;
5449bcc9736cSJeff Kirsher 	}
5450bcc9736cSJeff Kirsher }
5451bcc9736cSJeff Kirsher 
5452bcc9736cSJeff Kirsher static int prepare_hardware(struct net_device *dev)
5453bcc9736cSJeff Kirsher {
5454bcc9736cSJeff Kirsher 	struct dev_priv *priv = netdev_priv(dev);
5455bcc9736cSJeff Kirsher 	struct dev_info *hw_priv = priv->adapter;
5456bcc9736cSJeff Kirsher 	struct ksz_hw *hw = &hw_priv->hw;
5457bcc9736cSJeff Kirsher 	int rc = 0;
5458bcc9736cSJeff Kirsher 
5459bcc9736cSJeff Kirsher 	/* Remember the network device that requests interrupts. */
5460bcc9736cSJeff Kirsher 	hw_priv->dev = dev;
5461bcc9736cSJeff Kirsher 	rc = request_irq(dev->irq, netdev_intr, IRQF_SHARED, dev->name, dev);
5462bcc9736cSJeff Kirsher 	if (rc)
5463bcc9736cSJeff Kirsher 		return rc;
5464bcc9736cSJeff Kirsher 	tasklet_enable(&hw_priv->rx_tasklet);
5465bcc9736cSJeff Kirsher 	tasklet_enable(&hw_priv->tx_tasklet);
5466bcc9736cSJeff Kirsher 
5467bcc9736cSJeff Kirsher 	hw->promiscuous = 0;
5468bcc9736cSJeff Kirsher 	hw->all_multi = 0;
5469bcc9736cSJeff Kirsher 	hw->multi_list_size = 0;
5470bcc9736cSJeff Kirsher 
5471bcc9736cSJeff Kirsher 	hw_reset(hw);
5472bcc9736cSJeff Kirsher 
5473bcc9736cSJeff Kirsher 	hw_set_desc_base(hw,
5474bcc9736cSJeff Kirsher 		hw->tx_desc_info.ring_phys, hw->rx_desc_info.ring_phys);
5475bcc9736cSJeff Kirsher 	hw_set_addr(hw);
5476bcc9736cSJeff Kirsher 	hw_cfg_huge_frame(hw_priv, hw);
5477bcc9736cSJeff Kirsher 	ksz_init_rx_buffers(hw_priv);
5478bcc9736cSJeff Kirsher 	return 0;
5479bcc9736cSJeff Kirsher }
5480bcc9736cSJeff Kirsher 
5481bcc9736cSJeff Kirsher static void set_media_state(struct net_device *dev, int media_state)
5482bcc9736cSJeff Kirsher {
5483bcc9736cSJeff Kirsher 	struct dev_priv *priv = netdev_priv(dev);
5484bcc9736cSJeff Kirsher 
5485bcc9736cSJeff Kirsher 	if (media_state == priv->media_state)
5486bcc9736cSJeff Kirsher 		netif_carrier_on(dev);
5487bcc9736cSJeff Kirsher 	else
5488bcc9736cSJeff Kirsher 		netif_carrier_off(dev);
5489bcc9736cSJeff Kirsher 	netif_info(priv, link, dev, "link %s\n",
5490bcc9736cSJeff Kirsher 		   media_state == priv->media_state ? "on" : "off");
5491bcc9736cSJeff Kirsher }
5492bcc9736cSJeff Kirsher 
5493bcc9736cSJeff Kirsher /**
5494bcc9736cSJeff Kirsher  * netdev_open - open network device
5495bcc9736cSJeff Kirsher  * @dev:	Network device.
5496bcc9736cSJeff Kirsher  *
5497bcc9736cSJeff Kirsher  * This function process the open operation of network device.  This is caused
5498bcc9736cSJeff Kirsher  * by the user command "ifconfig ethX up."
5499bcc9736cSJeff Kirsher  *
5500bcc9736cSJeff Kirsher  * Return 0 if successful; otherwise an error code indicating failure.
5501bcc9736cSJeff Kirsher  */
5502bcc9736cSJeff Kirsher static int netdev_open(struct net_device *dev)
5503bcc9736cSJeff Kirsher {
5504bcc9736cSJeff Kirsher 	struct dev_priv *priv = netdev_priv(dev);
5505bcc9736cSJeff Kirsher 	struct dev_info *hw_priv = priv->adapter;
5506bcc9736cSJeff Kirsher 	struct ksz_hw *hw = &hw_priv->hw;
5507bcc9736cSJeff Kirsher 	struct ksz_port *port = &priv->port;
5508bcc9736cSJeff Kirsher 	int i;
5509bcc9736cSJeff Kirsher 	int p;
5510bcc9736cSJeff Kirsher 	int rc = 0;
5511bcc9736cSJeff Kirsher 
5512bcc9736cSJeff Kirsher 	priv->multicast = 0;
5513bcc9736cSJeff Kirsher 	priv->promiscuous = 0;
5514bcc9736cSJeff Kirsher 
5515bcc9736cSJeff Kirsher 	/* Reset device statistics. */
5516bcc9736cSJeff Kirsher 	memset(&dev->stats, 0, sizeof(struct net_device_stats));
5517bcc9736cSJeff Kirsher 	memset((void *) port->counter, 0,
5518bcc9736cSJeff Kirsher 		(sizeof(u64) * OID_COUNTER_LAST));
5519bcc9736cSJeff Kirsher 
5520bcc9736cSJeff Kirsher 	if (!(hw_priv->opened)) {
5521bcc9736cSJeff Kirsher 		rc = prepare_hardware(dev);
5522bcc9736cSJeff Kirsher 		if (rc)
5523bcc9736cSJeff Kirsher 			return rc;
5524bcc9736cSJeff Kirsher 		for (i = 0; i < hw->mib_port_cnt; i++) {
5525bcc9736cSJeff Kirsher 			if (next_jiffies < jiffies)
5526bcc9736cSJeff Kirsher 				next_jiffies = jiffies + HZ * 2;
5527bcc9736cSJeff Kirsher 			else
5528bcc9736cSJeff Kirsher 				next_jiffies += HZ * 1;
5529bcc9736cSJeff Kirsher 			hw_priv->counter[i].time = next_jiffies;
5530bcc9736cSJeff Kirsher 			hw->port_mib[i].state = media_disconnected;
5531bcc9736cSJeff Kirsher 			port_init_cnt(hw, i);
5532bcc9736cSJeff Kirsher 		}
5533bcc9736cSJeff Kirsher 		if (hw->ksz_switch)
5534bcc9736cSJeff Kirsher 			hw->port_mib[HOST_PORT].state = media_connected;
5535bcc9736cSJeff Kirsher 		else {
5536bcc9736cSJeff Kirsher 			hw_add_wol_bcast(hw);
5537bcc9736cSJeff Kirsher 			hw_cfg_wol_pme(hw, 0);
5538bcc9736cSJeff Kirsher 			hw_clr_wol_pme_status(&hw_priv->hw);
5539bcc9736cSJeff Kirsher 		}
5540bcc9736cSJeff Kirsher 	}
5541bcc9736cSJeff Kirsher 	port_set_power_saving(port, false);
5542bcc9736cSJeff Kirsher 
5543bcc9736cSJeff Kirsher 	for (i = 0, p = port->first_port; i < port->port_cnt; i++, p++) {
5544bcc9736cSJeff Kirsher 		/*
5545bcc9736cSJeff Kirsher 		 * Initialize to invalid value so that link detection
5546bcc9736cSJeff Kirsher 		 * is done.
5547bcc9736cSJeff Kirsher 		 */
5548bcc9736cSJeff Kirsher 		hw->port_info[p].partner = 0xFF;
5549bcc9736cSJeff Kirsher 		hw->port_info[p].state = media_disconnected;
5550bcc9736cSJeff Kirsher 	}
5551bcc9736cSJeff Kirsher 
5552bcc9736cSJeff Kirsher 	/* Need to open the port in multiple device interfaces mode. */
5553bcc9736cSJeff Kirsher 	if (hw->dev_count > 1) {
5554bcc9736cSJeff Kirsher 		port_set_stp_state(hw, port->first_port, STP_STATE_SIMPLE);
5555bcc9736cSJeff Kirsher 		if (port->first_port > 0)
5556bcc9736cSJeff Kirsher 			hw_add_addr(hw, dev->dev_addr);
5557bcc9736cSJeff Kirsher 	}
5558bcc9736cSJeff Kirsher 
5559bcc9736cSJeff Kirsher 	port_get_link_speed(port);
5560bcc9736cSJeff Kirsher 	if (port->force_link)
5561bcc9736cSJeff Kirsher 		port_force_link_speed(port);
5562bcc9736cSJeff Kirsher 	else
5563bcc9736cSJeff Kirsher 		port_set_link_speed(port);
5564bcc9736cSJeff Kirsher 
5565bcc9736cSJeff Kirsher 	if (!(hw_priv->opened)) {
5566bcc9736cSJeff Kirsher 		hw_setup_intr(hw);
5567bcc9736cSJeff Kirsher 		hw_enable(hw);
5568bcc9736cSJeff Kirsher 		hw_ena_intr(hw);
5569bcc9736cSJeff Kirsher 
5570bcc9736cSJeff Kirsher 		if (hw->mib_port_cnt)
5571bcc9736cSJeff Kirsher 			ksz_start_timer(&hw_priv->mib_timer_info,
5572bcc9736cSJeff Kirsher 				hw_priv->mib_timer_info.period);
5573bcc9736cSJeff Kirsher 	}
5574bcc9736cSJeff Kirsher 
5575bcc9736cSJeff Kirsher 	hw_priv->opened++;
5576bcc9736cSJeff Kirsher 
5577bcc9736cSJeff Kirsher 	ksz_start_timer(&priv->monitor_timer_info,
5578bcc9736cSJeff Kirsher 		priv->monitor_timer_info.period);
5579bcc9736cSJeff Kirsher 
5580bcc9736cSJeff Kirsher 	priv->media_state = port->linked->state;
5581bcc9736cSJeff Kirsher 
5582bcc9736cSJeff Kirsher 	set_media_state(dev, media_connected);
5583bcc9736cSJeff Kirsher 	netif_start_queue(dev);
5584bcc9736cSJeff Kirsher 
5585bcc9736cSJeff Kirsher 	return 0;
5586bcc9736cSJeff Kirsher }
5587bcc9736cSJeff Kirsher 
5588bcc9736cSJeff Kirsher /* RX errors = rx_errors */
5589bcc9736cSJeff Kirsher /* RX dropped = rx_dropped */
5590bcc9736cSJeff Kirsher /* RX overruns = rx_fifo_errors */
5591bcc9736cSJeff Kirsher /* RX frame = rx_crc_errors + rx_frame_errors + rx_length_errors */
5592bcc9736cSJeff Kirsher /* TX errors = tx_errors */
5593bcc9736cSJeff Kirsher /* TX dropped = tx_dropped */
5594bcc9736cSJeff Kirsher /* TX overruns = tx_fifo_errors */
5595bcc9736cSJeff Kirsher /* TX carrier = tx_aborted_errors + tx_carrier_errors + tx_window_errors */
5596bcc9736cSJeff Kirsher /* collisions = collisions */
5597bcc9736cSJeff Kirsher 
5598bcc9736cSJeff Kirsher /**
5599bcc9736cSJeff Kirsher  * netdev_query_statistics - query network device statistics
5600bcc9736cSJeff Kirsher  * @dev:	Network device.
5601bcc9736cSJeff Kirsher  *
5602bcc9736cSJeff Kirsher  * This function returns the statistics of the network device.  The device
5603bcc9736cSJeff Kirsher  * needs not be opened.
5604bcc9736cSJeff Kirsher  *
5605bcc9736cSJeff Kirsher  * Return network device statistics.
5606bcc9736cSJeff Kirsher  */
5607bcc9736cSJeff Kirsher static struct net_device_stats *netdev_query_statistics(struct net_device *dev)
5608bcc9736cSJeff Kirsher {
5609bcc9736cSJeff Kirsher 	struct dev_priv *priv = netdev_priv(dev);
5610bcc9736cSJeff Kirsher 	struct ksz_port *port = &priv->port;
5611bcc9736cSJeff Kirsher 	struct ksz_hw *hw = &priv->adapter->hw;
5612bcc9736cSJeff Kirsher 	struct ksz_port_mib *mib;
5613bcc9736cSJeff Kirsher 	int i;
5614bcc9736cSJeff Kirsher 	int p;
5615bcc9736cSJeff Kirsher 
5616bcc9736cSJeff Kirsher 	dev->stats.rx_errors = port->counter[OID_COUNTER_RCV_ERROR];
5617bcc9736cSJeff Kirsher 	dev->stats.tx_errors = port->counter[OID_COUNTER_XMIT_ERROR];
5618bcc9736cSJeff Kirsher 
5619bcc9736cSJeff Kirsher 	/* Reset to zero to add count later. */
5620bcc9736cSJeff Kirsher 	dev->stats.multicast = 0;
5621bcc9736cSJeff Kirsher 	dev->stats.collisions = 0;
5622bcc9736cSJeff Kirsher 	dev->stats.rx_length_errors = 0;
5623bcc9736cSJeff Kirsher 	dev->stats.rx_crc_errors = 0;
5624bcc9736cSJeff Kirsher 	dev->stats.rx_frame_errors = 0;
5625bcc9736cSJeff Kirsher 	dev->stats.tx_window_errors = 0;
5626bcc9736cSJeff Kirsher 
5627bcc9736cSJeff Kirsher 	for (i = 0, p = port->first_port; i < port->mib_port_cnt; i++, p++) {
5628bcc9736cSJeff Kirsher 		mib = &hw->port_mib[p];
5629bcc9736cSJeff Kirsher 
5630bcc9736cSJeff Kirsher 		dev->stats.multicast += (unsigned long)
5631bcc9736cSJeff Kirsher 			mib->counter[MIB_COUNTER_RX_MULTICAST];
5632bcc9736cSJeff Kirsher 
5633bcc9736cSJeff Kirsher 		dev->stats.collisions += (unsigned long)
5634bcc9736cSJeff Kirsher 			mib->counter[MIB_COUNTER_TX_TOTAL_COLLISION];
5635bcc9736cSJeff Kirsher 
5636bcc9736cSJeff Kirsher 		dev->stats.rx_length_errors += (unsigned long)(
5637bcc9736cSJeff Kirsher 			mib->counter[MIB_COUNTER_RX_UNDERSIZE] +
5638bcc9736cSJeff Kirsher 			mib->counter[MIB_COUNTER_RX_FRAGMENT] +
5639bcc9736cSJeff Kirsher 			mib->counter[MIB_COUNTER_RX_OVERSIZE] +
5640bcc9736cSJeff Kirsher 			mib->counter[MIB_COUNTER_RX_JABBER]);
5641bcc9736cSJeff Kirsher 		dev->stats.rx_crc_errors += (unsigned long)
5642bcc9736cSJeff Kirsher 			mib->counter[MIB_COUNTER_RX_CRC_ERR];
5643bcc9736cSJeff Kirsher 		dev->stats.rx_frame_errors += (unsigned long)(
5644bcc9736cSJeff Kirsher 			mib->counter[MIB_COUNTER_RX_ALIGNMENT_ERR] +
5645bcc9736cSJeff Kirsher 			mib->counter[MIB_COUNTER_RX_SYMBOL_ERR]);
5646bcc9736cSJeff Kirsher 
5647bcc9736cSJeff Kirsher 		dev->stats.tx_window_errors += (unsigned long)
5648bcc9736cSJeff Kirsher 			mib->counter[MIB_COUNTER_TX_LATE_COLLISION];
5649bcc9736cSJeff Kirsher 	}
5650bcc9736cSJeff Kirsher 
5651bcc9736cSJeff Kirsher 	return &dev->stats;
5652bcc9736cSJeff Kirsher }
5653bcc9736cSJeff Kirsher 
5654bcc9736cSJeff Kirsher /**
5655bcc9736cSJeff Kirsher  * netdev_set_mac_address - set network device MAC address
5656bcc9736cSJeff Kirsher  * @dev:	Network device.
5657bcc9736cSJeff Kirsher  * @addr:	Buffer of MAC address.
5658bcc9736cSJeff Kirsher  *
5659bcc9736cSJeff Kirsher  * This function is used to set the MAC address of the network device.
5660bcc9736cSJeff Kirsher  *
5661bcc9736cSJeff Kirsher  * Return 0 to indicate success.
5662bcc9736cSJeff Kirsher  */
5663bcc9736cSJeff Kirsher static int netdev_set_mac_address(struct net_device *dev, void *addr)
5664bcc9736cSJeff Kirsher {
5665bcc9736cSJeff Kirsher 	struct dev_priv *priv = netdev_priv(dev);
5666bcc9736cSJeff Kirsher 	struct dev_info *hw_priv = priv->adapter;
5667bcc9736cSJeff Kirsher 	struct ksz_hw *hw = &hw_priv->hw;
5668bcc9736cSJeff Kirsher 	struct sockaddr *mac = addr;
5669bcc9736cSJeff Kirsher 	uint interrupt;
5670bcc9736cSJeff Kirsher 
5671bcc9736cSJeff Kirsher 	if (priv->port.first_port > 0)
5672bcc9736cSJeff Kirsher 		hw_del_addr(hw, dev->dev_addr);
5673bcc9736cSJeff Kirsher 	else {
5674bcc9736cSJeff Kirsher 		hw->mac_override = 1;
56756a3c910cSJoe Perches 		memcpy(hw->override_addr, mac->sa_data, ETH_ALEN);
5676bcc9736cSJeff Kirsher 	}
5677bcc9736cSJeff Kirsher 
5678bcc9736cSJeff Kirsher 	memcpy(dev->dev_addr, mac->sa_data, MAX_ADDR_LEN);
5679bcc9736cSJeff Kirsher 
5680bcc9736cSJeff Kirsher 	interrupt = hw_block_intr(hw);
5681bcc9736cSJeff Kirsher 
5682bcc9736cSJeff Kirsher 	if (priv->port.first_port > 0)
5683bcc9736cSJeff Kirsher 		hw_add_addr(hw, dev->dev_addr);
5684bcc9736cSJeff Kirsher 	else
5685bcc9736cSJeff Kirsher 		hw_set_addr(hw);
5686bcc9736cSJeff Kirsher 	hw_restore_intr(hw, interrupt);
5687bcc9736cSJeff Kirsher 
5688bcc9736cSJeff Kirsher 	return 0;
5689bcc9736cSJeff Kirsher }
5690bcc9736cSJeff Kirsher 
5691bcc9736cSJeff Kirsher static void dev_set_promiscuous(struct net_device *dev, struct dev_priv *priv,
5692bcc9736cSJeff Kirsher 	struct ksz_hw *hw, int promiscuous)
5693bcc9736cSJeff Kirsher {
5694bcc9736cSJeff Kirsher 	if (promiscuous != priv->promiscuous) {
5695bcc9736cSJeff Kirsher 		u8 prev_state = hw->promiscuous;
5696bcc9736cSJeff Kirsher 
5697bcc9736cSJeff Kirsher 		if (promiscuous)
5698bcc9736cSJeff Kirsher 			++hw->promiscuous;
5699bcc9736cSJeff Kirsher 		else
5700bcc9736cSJeff Kirsher 			--hw->promiscuous;
5701bcc9736cSJeff Kirsher 		priv->promiscuous = promiscuous;
5702bcc9736cSJeff Kirsher 
5703bcc9736cSJeff Kirsher 		/* Turn on/off promiscuous mode. */
5704bcc9736cSJeff Kirsher 		if (hw->promiscuous <= 1 && prev_state <= 1)
5705bcc9736cSJeff Kirsher 			hw_set_promiscuous(hw, hw->promiscuous);
5706bcc9736cSJeff Kirsher 
5707bcc9736cSJeff Kirsher 		/*
5708bcc9736cSJeff Kirsher 		 * Port is not in promiscuous mode, meaning it is released
5709bcc9736cSJeff Kirsher 		 * from the bridge.
5710bcc9736cSJeff Kirsher 		 */
5711bcc9736cSJeff Kirsher 		if ((hw->features & STP_SUPPORT) && !promiscuous &&
5712bcc9736cSJeff Kirsher 		    (dev->priv_flags & IFF_BRIDGE_PORT)) {
5713bcc9736cSJeff Kirsher 			struct ksz_switch *sw = hw->ksz_switch;
5714bcc9736cSJeff Kirsher 			int port = priv->port.first_port;
5715bcc9736cSJeff Kirsher 
5716bcc9736cSJeff Kirsher 			port_set_stp_state(hw, port, STP_STATE_DISABLED);
5717bcc9736cSJeff Kirsher 			port = 1 << port;
5718bcc9736cSJeff Kirsher 			if (sw->member & port) {
5719bcc9736cSJeff Kirsher 				sw->member &= ~port;
5720bcc9736cSJeff Kirsher 				bridge_change(hw);
5721bcc9736cSJeff Kirsher 			}
5722bcc9736cSJeff Kirsher 		}
5723bcc9736cSJeff Kirsher 	}
5724bcc9736cSJeff Kirsher }
5725bcc9736cSJeff Kirsher 
5726bcc9736cSJeff Kirsher static void dev_set_multicast(struct dev_priv *priv, struct ksz_hw *hw,
5727bcc9736cSJeff Kirsher 	int multicast)
5728bcc9736cSJeff Kirsher {
5729bcc9736cSJeff Kirsher 	if (multicast != priv->multicast) {
5730bcc9736cSJeff Kirsher 		u8 all_multi = hw->all_multi;
5731bcc9736cSJeff Kirsher 
5732bcc9736cSJeff Kirsher 		if (multicast)
5733bcc9736cSJeff Kirsher 			++hw->all_multi;
5734bcc9736cSJeff Kirsher 		else
5735bcc9736cSJeff Kirsher 			--hw->all_multi;
5736bcc9736cSJeff Kirsher 		priv->multicast = multicast;
5737bcc9736cSJeff Kirsher 
5738bcc9736cSJeff Kirsher 		/* Turn on/off all multicast mode. */
5739bcc9736cSJeff Kirsher 		if (hw->all_multi <= 1 && all_multi <= 1)
5740bcc9736cSJeff Kirsher 			hw_set_multicast(hw, hw->all_multi);
5741bcc9736cSJeff Kirsher 	}
5742bcc9736cSJeff Kirsher }
5743bcc9736cSJeff Kirsher 
5744bcc9736cSJeff Kirsher /**
5745bcc9736cSJeff Kirsher  * netdev_set_rx_mode
5746bcc9736cSJeff Kirsher  * @dev:	Network device.
5747bcc9736cSJeff Kirsher  *
5748bcc9736cSJeff Kirsher  * This routine is used to set multicast addresses or put the network device
5749bcc9736cSJeff Kirsher  * into promiscuous mode.
5750bcc9736cSJeff Kirsher  */
5751bcc9736cSJeff Kirsher static void netdev_set_rx_mode(struct net_device *dev)
5752bcc9736cSJeff Kirsher {
5753bcc9736cSJeff Kirsher 	struct dev_priv *priv = netdev_priv(dev);
5754bcc9736cSJeff Kirsher 	struct dev_info *hw_priv = priv->adapter;
5755bcc9736cSJeff Kirsher 	struct ksz_hw *hw = &hw_priv->hw;
5756bcc9736cSJeff Kirsher 	struct netdev_hw_addr *ha;
5757bcc9736cSJeff Kirsher 	int multicast = (dev->flags & IFF_ALLMULTI);
5758bcc9736cSJeff Kirsher 
5759bcc9736cSJeff Kirsher 	dev_set_promiscuous(dev, priv, hw, (dev->flags & IFF_PROMISC));
5760bcc9736cSJeff Kirsher 
5761bcc9736cSJeff Kirsher 	if (hw_priv->hw.dev_count > 1)
5762bcc9736cSJeff Kirsher 		multicast |= (dev->flags & IFF_MULTICAST);
5763bcc9736cSJeff Kirsher 	dev_set_multicast(priv, hw, multicast);
5764bcc9736cSJeff Kirsher 
5765bcc9736cSJeff Kirsher 	/* Cannot use different hashes in multiple device interfaces mode. */
5766bcc9736cSJeff Kirsher 	if (hw_priv->hw.dev_count > 1)
5767bcc9736cSJeff Kirsher 		return;
5768bcc9736cSJeff Kirsher 
5769bcc9736cSJeff Kirsher 	if ((dev->flags & IFF_MULTICAST) && !netdev_mc_empty(dev)) {
5770bcc9736cSJeff Kirsher 		int i = 0;
5771bcc9736cSJeff Kirsher 
5772bcc9736cSJeff Kirsher 		/* List too big to support so turn on all multicast mode. */
5773bcc9736cSJeff Kirsher 		if (netdev_mc_count(dev) > MAX_MULTICAST_LIST) {
5774bcc9736cSJeff Kirsher 			if (MAX_MULTICAST_LIST != hw->multi_list_size) {
5775bcc9736cSJeff Kirsher 				hw->multi_list_size = MAX_MULTICAST_LIST;
5776bcc9736cSJeff Kirsher 				++hw->all_multi;
5777bcc9736cSJeff Kirsher 				hw_set_multicast(hw, hw->all_multi);
5778bcc9736cSJeff Kirsher 			}
5779bcc9736cSJeff Kirsher 			return;
5780bcc9736cSJeff Kirsher 		}
5781bcc9736cSJeff Kirsher 
5782bcc9736cSJeff Kirsher 		netdev_for_each_mc_addr(ha, dev) {
5783bcc9736cSJeff Kirsher 			if (i >= MAX_MULTICAST_LIST)
5784bcc9736cSJeff Kirsher 				break;
57856a3c910cSJoe Perches 			memcpy(hw->multi_list[i++], ha->addr, ETH_ALEN);
5786bcc9736cSJeff Kirsher 		}
5787bcc9736cSJeff Kirsher 		hw->multi_list_size = (u8) i;
5788bcc9736cSJeff Kirsher 		hw_set_grp_addr(hw);
5789bcc9736cSJeff Kirsher 	} else {
5790bcc9736cSJeff Kirsher 		if (MAX_MULTICAST_LIST == hw->multi_list_size) {
5791bcc9736cSJeff Kirsher 			--hw->all_multi;
5792bcc9736cSJeff Kirsher 			hw_set_multicast(hw, hw->all_multi);
5793bcc9736cSJeff Kirsher 		}
5794bcc9736cSJeff Kirsher 		hw->multi_list_size = 0;
5795bcc9736cSJeff Kirsher 		hw_clr_multicast(hw);
5796bcc9736cSJeff Kirsher 	}
5797bcc9736cSJeff Kirsher }
5798bcc9736cSJeff Kirsher 
5799bcc9736cSJeff Kirsher static int netdev_change_mtu(struct net_device *dev, int new_mtu)
5800bcc9736cSJeff Kirsher {
5801bcc9736cSJeff Kirsher 	struct dev_priv *priv = netdev_priv(dev);
5802bcc9736cSJeff Kirsher 	struct dev_info *hw_priv = priv->adapter;
5803bcc9736cSJeff Kirsher 	struct ksz_hw *hw = &hw_priv->hw;
5804bcc9736cSJeff Kirsher 	int hw_mtu;
5805bcc9736cSJeff Kirsher 
5806bcc9736cSJeff Kirsher 	if (netif_running(dev))
5807bcc9736cSJeff Kirsher 		return -EBUSY;
5808bcc9736cSJeff Kirsher 
5809bcc9736cSJeff Kirsher 	/* Cannot use different MTU in multiple device interfaces mode. */
5810bcc9736cSJeff Kirsher 	if (hw->dev_count > 1)
5811bcc9736cSJeff Kirsher 		if (dev != hw_priv->dev)
5812bcc9736cSJeff Kirsher 			return 0;
5813bcc9736cSJeff Kirsher 	if (new_mtu < 60)
5814bcc9736cSJeff Kirsher 		return -EINVAL;
5815bcc9736cSJeff Kirsher 
5816bcc9736cSJeff Kirsher 	if (dev->mtu != new_mtu) {
5817bcc9736cSJeff Kirsher 		hw_mtu = new_mtu + ETHERNET_HEADER_SIZE + 4;
5818bcc9736cSJeff Kirsher 		if (hw_mtu > MAX_RX_BUF_SIZE)
5819bcc9736cSJeff Kirsher 			return -EINVAL;
5820bcc9736cSJeff Kirsher 		if (hw_mtu > REGULAR_RX_BUF_SIZE) {
5821bcc9736cSJeff Kirsher 			hw->features |= RX_HUGE_FRAME;
5822bcc9736cSJeff Kirsher 			hw_mtu = MAX_RX_BUF_SIZE;
5823bcc9736cSJeff Kirsher 		} else {
5824bcc9736cSJeff Kirsher 			hw->features &= ~RX_HUGE_FRAME;
5825bcc9736cSJeff Kirsher 			hw_mtu = REGULAR_RX_BUF_SIZE;
5826bcc9736cSJeff Kirsher 		}
5827bcc9736cSJeff Kirsher 		hw_mtu = (hw_mtu + 3) & ~3;
5828bcc9736cSJeff Kirsher 		hw_priv->mtu = hw_mtu;
5829bcc9736cSJeff Kirsher 		dev->mtu = new_mtu;
5830bcc9736cSJeff Kirsher 	}
5831bcc9736cSJeff Kirsher 	return 0;
5832bcc9736cSJeff Kirsher }
5833bcc9736cSJeff Kirsher 
5834bcc9736cSJeff Kirsher /**
5835bcc9736cSJeff Kirsher  * netdev_ioctl - I/O control processing
5836bcc9736cSJeff Kirsher  * @dev:	Network device.
5837bcc9736cSJeff Kirsher  * @ifr:	Interface request structure.
5838bcc9736cSJeff Kirsher  * @cmd:	I/O control code.
5839bcc9736cSJeff Kirsher  *
5840bcc9736cSJeff Kirsher  * This function is used to process I/O control calls.
5841bcc9736cSJeff Kirsher  *
5842bcc9736cSJeff Kirsher  * Return 0 to indicate success.
5843bcc9736cSJeff Kirsher  */
5844bcc9736cSJeff Kirsher static int netdev_ioctl(struct net_device *dev, struct ifreq *ifr, int cmd)
5845bcc9736cSJeff Kirsher {
5846bcc9736cSJeff Kirsher 	struct dev_priv *priv = netdev_priv(dev);
5847bcc9736cSJeff Kirsher 	struct dev_info *hw_priv = priv->adapter;
5848bcc9736cSJeff Kirsher 	struct ksz_hw *hw = &hw_priv->hw;
5849bcc9736cSJeff Kirsher 	struct ksz_port *port = &priv->port;
5850bcc9736cSJeff Kirsher 	int rc;
5851bcc9736cSJeff Kirsher 	int result = 0;
5852bcc9736cSJeff Kirsher 	struct mii_ioctl_data *data = if_mii(ifr);
5853bcc9736cSJeff Kirsher 
5854bcc9736cSJeff Kirsher 	if (down_interruptible(&priv->proc_sem))
5855bcc9736cSJeff Kirsher 		return -ERESTARTSYS;
5856bcc9736cSJeff Kirsher 
5857bcc9736cSJeff Kirsher 	/* assume success */
5858bcc9736cSJeff Kirsher 	rc = 0;
5859bcc9736cSJeff Kirsher 	switch (cmd) {
5860bcc9736cSJeff Kirsher 	/* Get address of MII PHY in use. */
5861bcc9736cSJeff Kirsher 	case SIOCGMIIPHY:
5862bcc9736cSJeff Kirsher 		data->phy_id = priv->id;
5863bcc9736cSJeff Kirsher 
5864bcc9736cSJeff Kirsher 		/* Fallthrough... */
5865bcc9736cSJeff Kirsher 
5866bcc9736cSJeff Kirsher 	/* Read MII PHY register. */
5867bcc9736cSJeff Kirsher 	case SIOCGMIIREG:
5868bcc9736cSJeff Kirsher 		if (data->phy_id != priv->id || data->reg_num >= 6)
5869bcc9736cSJeff Kirsher 			result = -EIO;
5870bcc9736cSJeff Kirsher 		else
5871bcc9736cSJeff Kirsher 			hw_r_phy(hw, port->linked->port_id, data->reg_num,
5872bcc9736cSJeff Kirsher 				&data->val_out);
5873bcc9736cSJeff Kirsher 		break;
5874bcc9736cSJeff Kirsher 
5875bcc9736cSJeff Kirsher 	/* Write MII PHY register. */
5876bcc9736cSJeff Kirsher 	case SIOCSMIIREG:
5877bcc9736cSJeff Kirsher 		if (!capable(CAP_NET_ADMIN))
5878bcc9736cSJeff Kirsher 			result = -EPERM;
5879bcc9736cSJeff Kirsher 		else if (data->phy_id != priv->id || data->reg_num >= 6)
5880bcc9736cSJeff Kirsher 			result = -EIO;
5881bcc9736cSJeff Kirsher 		else
5882bcc9736cSJeff Kirsher 			hw_w_phy(hw, port->linked->port_id, data->reg_num,
5883bcc9736cSJeff Kirsher 				data->val_in);
5884bcc9736cSJeff Kirsher 		break;
5885bcc9736cSJeff Kirsher 
5886bcc9736cSJeff Kirsher 	default:
5887bcc9736cSJeff Kirsher 		result = -EOPNOTSUPP;
5888bcc9736cSJeff Kirsher 	}
5889bcc9736cSJeff Kirsher 
5890bcc9736cSJeff Kirsher 	up(&priv->proc_sem);
5891bcc9736cSJeff Kirsher 
5892bcc9736cSJeff Kirsher 	return result;
5893bcc9736cSJeff Kirsher }
5894bcc9736cSJeff Kirsher 
5895bcc9736cSJeff Kirsher /*
5896bcc9736cSJeff Kirsher  * MII support
5897bcc9736cSJeff Kirsher  */
5898bcc9736cSJeff Kirsher 
5899bcc9736cSJeff Kirsher /**
5900bcc9736cSJeff Kirsher  * mdio_read - read PHY register
5901bcc9736cSJeff Kirsher  * @dev:	Network device.
5902bcc9736cSJeff Kirsher  * @phy_id:	The PHY id.
5903bcc9736cSJeff Kirsher  * @reg_num:	The register number.
5904bcc9736cSJeff Kirsher  *
5905bcc9736cSJeff Kirsher  * This function returns the PHY register value.
5906bcc9736cSJeff Kirsher  *
5907bcc9736cSJeff Kirsher  * Return the register value.
5908bcc9736cSJeff Kirsher  */
5909bcc9736cSJeff Kirsher static int mdio_read(struct net_device *dev, int phy_id, int reg_num)
5910bcc9736cSJeff Kirsher {
5911bcc9736cSJeff Kirsher 	struct dev_priv *priv = netdev_priv(dev);
5912bcc9736cSJeff Kirsher 	struct ksz_port *port = &priv->port;
5913bcc9736cSJeff Kirsher 	struct ksz_hw *hw = port->hw;
5914bcc9736cSJeff Kirsher 	u16 val_out;
5915bcc9736cSJeff Kirsher 
5916bcc9736cSJeff Kirsher 	hw_r_phy(hw, port->linked->port_id, reg_num << 1, &val_out);
5917bcc9736cSJeff Kirsher 	return val_out;
5918bcc9736cSJeff Kirsher }
5919bcc9736cSJeff Kirsher 
5920bcc9736cSJeff Kirsher /**
5921bcc9736cSJeff Kirsher  * mdio_write - set PHY register
5922bcc9736cSJeff Kirsher  * @dev:	Network device.
5923bcc9736cSJeff Kirsher  * @phy_id:	The PHY id.
5924bcc9736cSJeff Kirsher  * @reg_num:	The register number.
5925bcc9736cSJeff Kirsher  * @val:	The register value.
5926bcc9736cSJeff Kirsher  *
5927bcc9736cSJeff Kirsher  * This procedure sets the PHY register value.
5928bcc9736cSJeff Kirsher  */
5929bcc9736cSJeff Kirsher static void mdio_write(struct net_device *dev, int phy_id, int reg_num, int val)
5930bcc9736cSJeff Kirsher {
5931bcc9736cSJeff Kirsher 	struct dev_priv *priv = netdev_priv(dev);
5932bcc9736cSJeff Kirsher 	struct ksz_port *port = &priv->port;
5933bcc9736cSJeff Kirsher 	struct ksz_hw *hw = port->hw;
5934bcc9736cSJeff Kirsher 	int i;
5935bcc9736cSJeff Kirsher 	int pi;
5936bcc9736cSJeff Kirsher 
5937bcc9736cSJeff Kirsher 	for (i = 0, pi = port->first_port; i < port->port_cnt; i++, pi++)
5938bcc9736cSJeff Kirsher 		hw_w_phy(hw, pi, reg_num << 1, val);
5939bcc9736cSJeff Kirsher }
5940bcc9736cSJeff Kirsher 
5941bcc9736cSJeff Kirsher /*
5942bcc9736cSJeff Kirsher  * ethtool support
5943bcc9736cSJeff Kirsher  */
5944bcc9736cSJeff Kirsher 
5945bcc9736cSJeff Kirsher #define EEPROM_SIZE			0x40
5946bcc9736cSJeff Kirsher 
5947bcc9736cSJeff Kirsher static u16 eeprom_data[EEPROM_SIZE] = { 0 };
5948bcc9736cSJeff Kirsher 
5949bcc9736cSJeff Kirsher #define ADVERTISED_ALL			\
5950bcc9736cSJeff Kirsher 	(ADVERTISED_10baseT_Half |	\
5951bcc9736cSJeff Kirsher 	ADVERTISED_10baseT_Full |	\
5952bcc9736cSJeff Kirsher 	ADVERTISED_100baseT_Half |	\
5953bcc9736cSJeff Kirsher 	ADVERTISED_100baseT_Full)
5954bcc9736cSJeff Kirsher 
5955bcc9736cSJeff Kirsher /* These functions use the MII functions in mii.c. */
5956bcc9736cSJeff Kirsher 
5957bcc9736cSJeff Kirsher /**
5958bcc9736cSJeff Kirsher  * netdev_get_settings - get network device settings
5959bcc9736cSJeff Kirsher  * @dev:	Network device.
5960bcc9736cSJeff Kirsher  * @cmd:	Ethtool command.
5961bcc9736cSJeff Kirsher  *
5962bcc9736cSJeff Kirsher  * This function queries the PHY and returns its state in the ethtool command.
5963bcc9736cSJeff Kirsher  *
5964bcc9736cSJeff Kirsher  * Return 0 if successful; otherwise an error code.
5965bcc9736cSJeff Kirsher  */
5966bcc9736cSJeff Kirsher static int netdev_get_settings(struct net_device *dev, struct ethtool_cmd *cmd)
5967bcc9736cSJeff Kirsher {
5968bcc9736cSJeff Kirsher 	struct dev_priv *priv = netdev_priv(dev);
5969bcc9736cSJeff Kirsher 	struct dev_info *hw_priv = priv->adapter;
5970bcc9736cSJeff Kirsher 
5971bcc9736cSJeff Kirsher 	mutex_lock(&hw_priv->lock);
5972bcc9736cSJeff Kirsher 	mii_ethtool_gset(&priv->mii_if, cmd);
5973bcc9736cSJeff Kirsher 	cmd->advertising |= SUPPORTED_TP;
5974bcc9736cSJeff Kirsher 	mutex_unlock(&hw_priv->lock);
5975bcc9736cSJeff Kirsher 
5976bcc9736cSJeff Kirsher 	/* Save advertised settings for workaround in next function. */
5977bcc9736cSJeff Kirsher 	priv->advertising = cmd->advertising;
5978bcc9736cSJeff Kirsher 	return 0;
5979bcc9736cSJeff Kirsher }
5980bcc9736cSJeff Kirsher 
5981bcc9736cSJeff Kirsher /**
5982bcc9736cSJeff Kirsher  * netdev_set_settings - set network device settings
5983bcc9736cSJeff Kirsher  * @dev:	Network device.
5984bcc9736cSJeff Kirsher  * @cmd:	Ethtool command.
5985bcc9736cSJeff Kirsher  *
5986bcc9736cSJeff Kirsher  * This function sets the PHY according to the ethtool command.
5987bcc9736cSJeff Kirsher  *
5988bcc9736cSJeff Kirsher  * Return 0 if successful; otherwise an error code.
5989bcc9736cSJeff Kirsher  */
5990bcc9736cSJeff Kirsher static int netdev_set_settings(struct net_device *dev, struct ethtool_cmd *cmd)
5991bcc9736cSJeff Kirsher {
5992bcc9736cSJeff Kirsher 	struct dev_priv *priv = netdev_priv(dev);
5993bcc9736cSJeff Kirsher 	struct dev_info *hw_priv = priv->adapter;
5994bcc9736cSJeff Kirsher 	struct ksz_port *port = &priv->port;
5995bcc9736cSJeff Kirsher 	u32 speed = ethtool_cmd_speed(cmd);
5996bcc9736cSJeff Kirsher 	int rc;
5997bcc9736cSJeff Kirsher 
5998bcc9736cSJeff Kirsher 	/*
5999bcc9736cSJeff Kirsher 	 * ethtool utility does not change advertised setting if auto
6000bcc9736cSJeff Kirsher 	 * negotiation is not specified explicitly.
6001bcc9736cSJeff Kirsher 	 */
6002bcc9736cSJeff Kirsher 	if (cmd->autoneg && priv->advertising == cmd->advertising) {
6003bcc9736cSJeff Kirsher 		cmd->advertising |= ADVERTISED_ALL;
6004bcc9736cSJeff Kirsher 		if (10 == speed)
6005bcc9736cSJeff Kirsher 			cmd->advertising &=
6006bcc9736cSJeff Kirsher 				~(ADVERTISED_100baseT_Full |
6007bcc9736cSJeff Kirsher 				ADVERTISED_100baseT_Half);
6008bcc9736cSJeff Kirsher 		else if (100 == speed)
6009bcc9736cSJeff Kirsher 			cmd->advertising &=
6010bcc9736cSJeff Kirsher 				~(ADVERTISED_10baseT_Full |
6011bcc9736cSJeff Kirsher 				ADVERTISED_10baseT_Half);
6012bcc9736cSJeff Kirsher 		if (0 == cmd->duplex)
6013bcc9736cSJeff Kirsher 			cmd->advertising &=
6014bcc9736cSJeff Kirsher 				~(ADVERTISED_100baseT_Full |
6015bcc9736cSJeff Kirsher 				ADVERTISED_10baseT_Full);
6016bcc9736cSJeff Kirsher 		else if (1 == cmd->duplex)
6017bcc9736cSJeff Kirsher 			cmd->advertising &=
6018bcc9736cSJeff Kirsher 				~(ADVERTISED_100baseT_Half |
6019bcc9736cSJeff Kirsher 				ADVERTISED_10baseT_Half);
6020bcc9736cSJeff Kirsher 	}
6021bcc9736cSJeff Kirsher 	mutex_lock(&hw_priv->lock);
6022bcc9736cSJeff Kirsher 	if (cmd->autoneg &&
6023bcc9736cSJeff Kirsher 			(cmd->advertising & ADVERTISED_ALL) ==
6024bcc9736cSJeff Kirsher 			ADVERTISED_ALL) {
6025bcc9736cSJeff Kirsher 		port->duplex = 0;
6026bcc9736cSJeff Kirsher 		port->speed = 0;
6027bcc9736cSJeff Kirsher 		port->force_link = 0;
6028bcc9736cSJeff Kirsher 	} else {
6029bcc9736cSJeff Kirsher 		port->duplex = cmd->duplex + 1;
6030bcc9736cSJeff Kirsher 		if (1000 != speed)
6031bcc9736cSJeff Kirsher 			port->speed = speed;
6032bcc9736cSJeff Kirsher 		if (cmd->autoneg)
6033bcc9736cSJeff Kirsher 			port->force_link = 0;
6034bcc9736cSJeff Kirsher 		else
6035bcc9736cSJeff Kirsher 			port->force_link = 1;
6036bcc9736cSJeff Kirsher 	}
6037bcc9736cSJeff Kirsher 	rc = mii_ethtool_sset(&priv->mii_if, cmd);
6038bcc9736cSJeff Kirsher 	mutex_unlock(&hw_priv->lock);
6039bcc9736cSJeff Kirsher 	return rc;
6040bcc9736cSJeff Kirsher }
6041bcc9736cSJeff Kirsher 
6042bcc9736cSJeff Kirsher /**
6043bcc9736cSJeff Kirsher  * netdev_nway_reset - restart auto-negotiation
6044bcc9736cSJeff Kirsher  * @dev:	Network device.
6045bcc9736cSJeff Kirsher  *
6046bcc9736cSJeff Kirsher  * This function restarts the PHY for auto-negotiation.
6047bcc9736cSJeff Kirsher  *
6048bcc9736cSJeff Kirsher  * Return 0 if successful; otherwise an error code.
6049bcc9736cSJeff Kirsher  */
6050bcc9736cSJeff Kirsher static int netdev_nway_reset(struct net_device *dev)
6051bcc9736cSJeff Kirsher {
6052bcc9736cSJeff Kirsher 	struct dev_priv *priv = netdev_priv(dev);
6053bcc9736cSJeff Kirsher 	struct dev_info *hw_priv = priv->adapter;
6054bcc9736cSJeff Kirsher 	int rc;
6055bcc9736cSJeff Kirsher 
6056bcc9736cSJeff Kirsher 	mutex_lock(&hw_priv->lock);
6057bcc9736cSJeff Kirsher 	rc = mii_nway_restart(&priv->mii_if);
6058bcc9736cSJeff Kirsher 	mutex_unlock(&hw_priv->lock);
6059bcc9736cSJeff Kirsher 	return rc;
6060bcc9736cSJeff Kirsher }
6061bcc9736cSJeff Kirsher 
6062bcc9736cSJeff Kirsher /**
6063bcc9736cSJeff Kirsher  * netdev_get_link - get network device link status
6064bcc9736cSJeff Kirsher  * @dev:	Network device.
6065bcc9736cSJeff Kirsher  *
6066bcc9736cSJeff Kirsher  * This function gets the link status from the PHY.
6067bcc9736cSJeff Kirsher  *
6068bcc9736cSJeff Kirsher  * Return true if PHY is linked and false otherwise.
6069bcc9736cSJeff Kirsher  */
6070bcc9736cSJeff Kirsher static u32 netdev_get_link(struct net_device *dev)
6071bcc9736cSJeff Kirsher {
6072bcc9736cSJeff Kirsher 	struct dev_priv *priv = netdev_priv(dev);
6073bcc9736cSJeff Kirsher 	int rc;
6074bcc9736cSJeff Kirsher 
6075bcc9736cSJeff Kirsher 	rc = mii_link_ok(&priv->mii_if);
6076bcc9736cSJeff Kirsher 	return rc;
6077bcc9736cSJeff Kirsher }
6078bcc9736cSJeff Kirsher 
6079bcc9736cSJeff Kirsher /**
6080bcc9736cSJeff Kirsher  * netdev_get_drvinfo - get network driver information
6081bcc9736cSJeff Kirsher  * @dev:	Network device.
6082bcc9736cSJeff Kirsher  * @info:	Ethtool driver info data structure.
6083bcc9736cSJeff Kirsher  *
6084bcc9736cSJeff Kirsher  * This procedure returns the driver information.
6085bcc9736cSJeff Kirsher  */
6086bcc9736cSJeff Kirsher static void netdev_get_drvinfo(struct net_device *dev,
6087bcc9736cSJeff Kirsher 	struct ethtool_drvinfo *info)
6088bcc9736cSJeff Kirsher {
6089bcc9736cSJeff Kirsher 	struct dev_priv *priv = netdev_priv(dev);
6090bcc9736cSJeff Kirsher 	struct dev_info *hw_priv = priv->adapter;
6091bcc9736cSJeff Kirsher 
609223020ab3SRick Jones 	strlcpy(info->driver, DRV_NAME, sizeof(info->driver));
609323020ab3SRick Jones 	strlcpy(info->version, DRV_VERSION, sizeof(info->version));
609423020ab3SRick Jones 	strlcpy(info->bus_info, pci_name(hw_priv->pdev),
609523020ab3SRick Jones 		sizeof(info->bus_info));
6096bcc9736cSJeff Kirsher }
6097bcc9736cSJeff Kirsher 
6098bcc9736cSJeff Kirsher /**
6099bcc9736cSJeff Kirsher  * netdev_get_regs_len - get length of register dump
6100bcc9736cSJeff Kirsher  * @dev:	Network device.
6101bcc9736cSJeff Kirsher  *
6102bcc9736cSJeff Kirsher  * This function returns the length of the register dump.
6103bcc9736cSJeff Kirsher  *
6104bcc9736cSJeff Kirsher  * Return length of the register dump.
6105bcc9736cSJeff Kirsher  */
6106bcc9736cSJeff Kirsher static struct hw_regs {
6107bcc9736cSJeff Kirsher 	int start;
6108bcc9736cSJeff Kirsher 	int end;
6109bcc9736cSJeff Kirsher } hw_regs_range[] = {
6110bcc9736cSJeff Kirsher 	{ KS_DMA_TX_CTRL,	KS884X_INTERRUPTS_STATUS },
6111bcc9736cSJeff Kirsher 	{ KS_ADD_ADDR_0_LO,	KS_ADD_ADDR_F_HI },
6112bcc9736cSJeff Kirsher 	{ KS884X_ADDR_0_OFFSET,	KS8841_WOL_FRAME_BYTE2_OFFSET },
6113bcc9736cSJeff Kirsher 	{ KS884X_SIDER_P,	KS8842_SGCR7_P },
6114bcc9736cSJeff Kirsher 	{ KS8842_MACAR1_P,	KS8842_TOSR8_P },
6115bcc9736cSJeff Kirsher 	{ KS884X_P1MBCR_P,	KS8842_P3ERCR_P },
6116bcc9736cSJeff Kirsher 	{ 0, 0 }
6117bcc9736cSJeff Kirsher };
6118bcc9736cSJeff Kirsher 
6119bcc9736cSJeff Kirsher static int netdev_get_regs_len(struct net_device *dev)
6120bcc9736cSJeff Kirsher {
6121bcc9736cSJeff Kirsher 	struct hw_regs *range = hw_regs_range;
6122bcc9736cSJeff Kirsher 	int regs_len = 0x10 * sizeof(u32);
6123bcc9736cSJeff Kirsher 
6124bcc9736cSJeff Kirsher 	while (range->end > range->start) {
6125bcc9736cSJeff Kirsher 		regs_len += (range->end - range->start + 3) / 4 * 4;
6126bcc9736cSJeff Kirsher 		range++;
6127bcc9736cSJeff Kirsher 	}
6128bcc9736cSJeff Kirsher 	return regs_len;
6129bcc9736cSJeff Kirsher }
6130bcc9736cSJeff Kirsher 
6131bcc9736cSJeff Kirsher /**
6132bcc9736cSJeff Kirsher  * netdev_get_regs - get register dump
6133bcc9736cSJeff Kirsher  * @dev:	Network device.
6134bcc9736cSJeff Kirsher  * @regs:	Ethtool registers data structure.
6135bcc9736cSJeff Kirsher  * @ptr:	Buffer to store the register values.
6136bcc9736cSJeff Kirsher  *
6137bcc9736cSJeff Kirsher  * This procedure dumps the register values in the provided buffer.
6138bcc9736cSJeff Kirsher  */
6139bcc9736cSJeff Kirsher static void netdev_get_regs(struct net_device *dev, struct ethtool_regs *regs,
6140bcc9736cSJeff Kirsher 	void *ptr)
6141bcc9736cSJeff Kirsher {
6142bcc9736cSJeff Kirsher 	struct dev_priv *priv = netdev_priv(dev);
6143bcc9736cSJeff Kirsher 	struct dev_info *hw_priv = priv->adapter;
6144bcc9736cSJeff Kirsher 	struct ksz_hw *hw = &hw_priv->hw;
6145bcc9736cSJeff Kirsher 	int *buf = (int *) ptr;
6146bcc9736cSJeff Kirsher 	struct hw_regs *range = hw_regs_range;
6147bcc9736cSJeff Kirsher 	int len;
6148bcc9736cSJeff Kirsher 
6149bcc9736cSJeff Kirsher 	mutex_lock(&hw_priv->lock);
6150bcc9736cSJeff Kirsher 	regs->version = 0;
6151bcc9736cSJeff Kirsher 	for (len = 0; len < 0x40; len += 4) {
6152bcc9736cSJeff Kirsher 		pci_read_config_dword(hw_priv->pdev, len, buf);
6153bcc9736cSJeff Kirsher 		buf++;
6154bcc9736cSJeff Kirsher 	}
6155bcc9736cSJeff Kirsher 	while (range->end > range->start) {
6156bcc9736cSJeff Kirsher 		for (len = range->start; len < range->end; len += 4) {
6157bcc9736cSJeff Kirsher 			*buf = readl(hw->io + len);
6158bcc9736cSJeff Kirsher 			buf++;
6159bcc9736cSJeff Kirsher 		}
6160bcc9736cSJeff Kirsher 		range++;
6161bcc9736cSJeff Kirsher 	}
6162bcc9736cSJeff Kirsher 	mutex_unlock(&hw_priv->lock);
6163bcc9736cSJeff Kirsher }
6164bcc9736cSJeff Kirsher 
6165bcc9736cSJeff Kirsher #define WOL_SUPPORT			\
6166bcc9736cSJeff Kirsher 	(WAKE_PHY | WAKE_MAGIC |	\
6167bcc9736cSJeff Kirsher 	WAKE_UCAST | WAKE_MCAST |	\
6168bcc9736cSJeff Kirsher 	WAKE_BCAST | WAKE_ARP)
6169bcc9736cSJeff Kirsher 
6170bcc9736cSJeff Kirsher /**
6171bcc9736cSJeff Kirsher  * netdev_get_wol - get Wake-on-LAN support
6172bcc9736cSJeff Kirsher  * @dev:	Network device.
6173bcc9736cSJeff Kirsher  * @wol:	Ethtool Wake-on-LAN data structure.
6174bcc9736cSJeff Kirsher  *
6175bcc9736cSJeff Kirsher  * This procedure returns Wake-on-LAN support.
6176bcc9736cSJeff Kirsher  */
6177bcc9736cSJeff Kirsher static void netdev_get_wol(struct net_device *dev,
6178bcc9736cSJeff Kirsher 	struct ethtool_wolinfo *wol)
6179bcc9736cSJeff Kirsher {
6180bcc9736cSJeff Kirsher 	struct dev_priv *priv = netdev_priv(dev);
6181bcc9736cSJeff Kirsher 	struct dev_info *hw_priv = priv->adapter;
6182bcc9736cSJeff Kirsher 
6183bcc9736cSJeff Kirsher 	wol->supported = hw_priv->wol_support;
6184bcc9736cSJeff Kirsher 	wol->wolopts = hw_priv->wol_enable;
6185bcc9736cSJeff Kirsher 	memset(&wol->sopass, 0, sizeof(wol->sopass));
6186bcc9736cSJeff Kirsher }
6187bcc9736cSJeff Kirsher 
6188bcc9736cSJeff Kirsher /**
6189bcc9736cSJeff Kirsher  * netdev_set_wol - set Wake-on-LAN support
6190bcc9736cSJeff Kirsher  * @dev:	Network device.
6191bcc9736cSJeff Kirsher  * @wol:	Ethtool Wake-on-LAN data structure.
6192bcc9736cSJeff Kirsher  *
6193bcc9736cSJeff Kirsher  * This function sets Wake-on-LAN support.
6194bcc9736cSJeff Kirsher  *
6195bcc9736cSJeff Kirsher  * Return 0 if successful; otherwise an error code.
6196bcc9736cSJeff Kirsher  */
6197bcc9736cSJeff Kirsher static int netdev_set_wol(struct net_device *dev,
6198bcc9736cSJeff Kirsher 	struct ethtool_wolinfo *wol)
6199bcc9736cSJeff Kirsher {
6200bcc9736cSJeff Kirsher 	struct dev_priv *priv = netdev_priv(dev);
6201bcc9736cSJeff Kirsher 	struct dev_info *hw_priv = priv->adapter;
6202bcc9736cSJeff Kirsher 
6203bcc9736cSJeff Kirsher 	/* Need to find a way to retrieve the device IP address. */
6204bcc9736cSJeff Kirsher 	static const u8 net_addr[] = { 192, 168, 1, 1 };
6205bcc9736cSJeff Kirsher 
6206bcc9736cSJeff Kirsher 	if (wol->wolopts & ~hw_priv->wol_support)
6207bcc9736cSJeff Kirsher 		return -EINVAL;
6208bcc9736cSJeff Kirsher 
6209bcc9736cSJeff Kirsher 	hw_priv->wol_enable = wol->wolopts;
6210bcc9736cSJeff Kirsher 
6211bcc9736cSJeff Kirsher 	/* Link wakeup cannot really be disabled. */
6212bcc9736cSJeff Kirsher 	if (wol->wolopts)
6213bcc9736cSJeff Kirsher 		hw_priv->wol_enable |= WAKE_PHY;
6214bcc9736cSJeff Kirsher 	hw_enable_wol(&hw_priv->hw, hw_priv->wol_enable, net_addr);
6215bcc9736cSJeff Kirsher 	return 0;
6216bcc9736cSJeff Kirsher }
6217bcc9736cSJeff Kirsher 
6218bcc9736cSJeff Kirsher /**
6219bcc9736cSJeff Kirsher  * netdev_get_msglevel - get debug message level
6220bcc9736cSJeff Kirsher  * @dev:	Network device.
6221bcc9736cSJeff Kirsher  *
6222bcc9736cSJeff Kirsher  * This function returns current debug message level.
6223bcc9736cSJeff Kirsher  *
6224bcc9736cSJeff Kirsher  * Return current debug message flags.
6225bcc9736cSJeff Kirsher  */
6226bcc9736cSJeff Kirsher static u32 netdev_get_msglevel(struct net_device *dev)
6227bcc9736cSJeff Kirsher {
6228bcc9736cSJeff Kirsher 	struct dev_priv *priv = netdev_priv(dev);
6229bcc9736cSJeff Kirsher 
6230bcc9736cSJeff Kirsher 	return priv->msg_enable;
6231bcc9736cSJeff Kirsher }
6232bcc9736cSJeff Kirsher 
6233bcc9736cSJeff Kirsher /**
6234bcc9736cSJeff Kirsher  * netdev_set_msglevel - set debug message level
6235bcc9736cSJeff Kirsher  * @dev:	Network device.
6236bcc9736cSJeff Kirsher  * @value:	Debug message flags.
6237bcc9736cSJeff Kirsher  *
6238bcc9736cSJeff Kirsher  * This procedure sets debug message level.
6239bcc9736cSJeff Kirsher  */
6240bcc9736cSJeff Kirsher static void netdev_set_msglevel(struct net_device *dev, u32 value)
6241bcc9736cSJeff Kirsher {
6242bcc9736cSJeff Kirsher 	struct dev_priv *priv = netdev_priv(dev);
6243bcc9736cSJeff Kirsher 
6244bcc9736cSJeff Kirsher 	priv->msg_enable = value;
6245bcc9736cSJeff Kirsher }
6246bcc9736cSJeff Kirsher 
6247bcc9736cSJeff Kirsher /**
6248bcc9736cSJeff Kirsher  * netdev_get_eeprom_len - get EEPROM length
6249bcc9736cSJeff Kirsher  * @dev:	Network device.
6250bcc9736cSJeff Kirsher  *
6251bcc9736cSJeff Kirsher  * This function returns the length of the EEPROM.
6252bcc9736cSJeff Kirsher  *
6253bcc9736cSJeff Kirsher  * Return length of the EEPROM.
6254bcc9736cSJeff Kirsher  */
6255bcc9736cSJeff Kirsher static int netdev_get_eeprom_len(struct net_device *dev)
6256bcc9736cSJeff Kirsher {
6257bcc9736cSJeff Kirsher 	return EEPROM_SIZE * 2;
6258bcc9736cSJeff Kirsher }
6259bcc9736cSJeff Kirsher 
6260bcc9736cSJeff Kirsher /**
6261bcc9736cSJeff Kirsher  * netdev_get_eeprom - get EEPROM data
6262bcc9736cSJeff Kirsher  * @dev:	Network device.
6263bcc9736cSJeff Kirsher  * @eeprom:	Ethtool EEPROM data structure.
6264bcc9736cSJeff Kirsher  * @data:	Buffer to store the EEPROM data.
6265bcc9736cSJeff Kirsher  *
6266bcc9736cSJeff Kirsher  * This function dumps the EEPROM data in the provided buffer.
6267bcc9736cSJeff Kirsher  *
6268bcc9736cSJeff Kirsher  * Return 0 if successful; otherwise an error code.
6269bcc9736cSJeff Kirsher  */
6270bcc9736cSJeff Kirsher #define EEPROM_MAGIC			0x10A18842
6271bcc9736cSJeff Kirsher 
6272bcc9736cSJeff Kirsher static int netdev_get_eeprom(struct net_device *dev,
6273bcc9736cSJeff Kirsher 	struct ethtool_eeprom *eeprom, u8 *data)
6274bcc9736cSJeff Kirsher {
6275bcc9736cSJeff Kirsher 	struct dev_priv *priv = netdev_priv(dev);
6276bcc9736cSJeff Kirsher 	struct dev_info *hw_priv = priv->adapter;
6277bcc9736cSJeff Kirsher 	u8 *eeprom_byte = (u8 *) eeprom_data;
6278bcc9736cSJeff Kirsher 	int i;
6279bcc9736cSJeff Kirsher 	int len;
6280bcc9736cSJeff Kirsher 
6281bcc9736cSJeff Kirsher 	len = (eeprom->offset + eeprom->len + 1) / 2;
6282bcc9736cSJeff Kirsher 	for (i = eeprom->offset / 2; i < len; i++)
6283bcc9736cSJeff Kirsher 		eeprom_data[i] = eeprom_read(&hw_priv->hw, i);
6284bcc9736cSJeff Kirsher 	eeprom->magic = EEPROM_MAGIC;
6285bcc9736cSJeff Kirsher 	memcpy(data, &eeprom_byte[eeprom->offset], eeprom->len);
6286bcc9736cSJeff Kirsher 
6287bcc9736cSJeff Kirsher 	return 0;
6288bcc9736cSJeff Kirsher }
6289bcc9736cSJeff Kirsher 
6290bcc9736cSJeff Kirsher /**
6291bcc9736cSJeff Kirsher  * netdev_set_eeprom - write EEPROM data
6292bcc9736cSJeff Kirsher  * @dev:	Network device.
6293bcc9736cSJeff Kirsher  * @eeprom:	Ethtool EEPROM data structure.
6294bcc9736cSJeff Kirsher  * @data:	Data buffer.
6295bcc9736cSJeff Kirsher  *
6296bcc9736cSJeff Kirsher  * This function modifies the EEPROM data one byte at a time.
6297bcc9736cSJeff Kirsher  *
6298bcc9736cSJeff Kirsher  * Return 0 if successful; otherwise an error code.
6299bcc9736cSJeff Kirsher  */
6300bcc9736cSJeff Kirsher static int netdev_set_eeprom(struct net_device *dev,
6301bcc9736cSJeff Kirsher 	struct ethtool_eeprom *eeprom, u8 *data)
6302bcc9736cSJeff Kirsher {
6303bcc9736cSJeff Kirsher 	struct dev_priv *priv = netdev_priv(dev);
6304bcc9736cSJeff Kirsher 	struct dev_info *hw_priv = priv->adapter;
6305bcc9736cSJeff Kirsher 	u16 eeprom_word[EEPROM_SIZE];
6306bcc9736cSJeff Kirsher 	u8 *eeprom_byte = (u8 *) eeprom_word;
6307bcc9736cSJeff Kirsher 	int i;
6308bcc9736cSJeff Kirsher 	int len;
6309bcc9736cSJeff Kirsher 
6310bcc9736cSJeff Kirsher 	if (eeprom->magic != EEPROM_MAGIC)
6311bcc9736cSJeff Kirsher 		return -EINVAL;
6312bcc9736cSJeff Kirsher 
6313bcc9736cSJeff Kirsher 	len = (eeprom->offset + eeprom->len + 1) / 2;
6314bcc9736cSJeff Kirsher 	for (i = eeprom->offset / 2; i < len; i++)
6315bcc9736cSJeff Kirsher 		eeprom_data[i] = eeprom_read(&hw_priv->hw, i);
6316bcc9736cSJeff Kirsher 	memcpy(eeprom_word, eeprom_data, EEPROM_SIZE * 2);
6317bcc9736cSJeff Kirsher 	memcpy(&eeprom_byte[eeprom->offset], data, eeprom->len);
6318bcc9736cSJeff Kirsher 	for (i = 0; i < EEPROM_SIZE; i++)
6319bcc9736cSJeff Kirsher 		if (eeprom_word[i] != eeprom_data[i]) {
6320bcc9736cSJeff Kirsher 			eeprom_data[i] = eeprom_word[i];
6321bcc9736cSJeff Kirsher 			eeprom_write(&hw_priv->hw, i, eeprom_data[i]);
6322bcc9736cSJeff Kirsher 	}
6323bcc9736cSJeff Kirsher 
6324bcc9736cSJeff Kirsher 	return 0;
6325bcc9736cSJeff Kirsher }
6326bcc9736cSJeff Kirsher 
6327bcc9736cSJeff Kirsher /**
6328bcc9736cSJeff Kirsher  * netdev_get_pauseparam - get flow control parameters
6329bcc9736cSJeff Kirsher  * @dev:	Network device.
6330bcc9736cSJeff Kirsher  * @pause:	Ethtool PAUSE settings data structure.
6331bcc9736cSJeff Kirsher  *
6332bcc9736cSJeff Kirsher  * This procedure returns the PAUSE control flow settings.
6333bcc9736cSJeff Kirsher  */
6334bcc9736cSJeff Kirsher static void netdev_get_pauseparam(struct net_device *dev,
6335bcc9736cSJeff Kirsher 	struct ethtool_pauseparam *pause)
6336bcc9736cSJeff Kirsher {
6337bcc9736cSJeff Kirsher 	struct dev_priv *priv = netdev_priv(dev);
6338bcc9736cSJeff Kirsher 	struct dev_info *hw_priv = priv->adapter;
6339bcc9736cSJeff Kirsher 	struct ksz_hw *hw = &hw_priv->hw;
6340bcc9736cSJeff Kirsher 
6341bcc9736cSJeff Kirsher 	pause->autoneg = (hw->overrides & PAUSE_FLOW_CTRL) ? 0 : 1;
6342bcc9736cSJeff Kirsher 	if (!hw->ksz_switch) {
6343bcc9736cSJeff Kirsher 		pause->rx_pause =
6344bcc9736cSJeff Kirsher 			(hw->rx_cfg & DMA_RX_FLOW_ENABLE) ? 1 : 0;
6345bcc9736cSJeff Kirsher 		pause->tx_pause =
6346bcc9736cSJeff Kirsher 			(hw->tx_cfg & DMA_TX_FLOW_ENABLE) ? 1 : 0;
6347bcc9736cSJeff Kirsher 	} else {
6348bcc9736cSJeff Kirsher 		pause->rx_pause =
6349bcc9736cSJeff Kirsher 			(sw_chk(hw, KS8842_SWITCH_CTRL_1_OFFSET,
6350bcc9736cSJeff Kirsher 				SWITCH_RX_FLOW_CTRL)) ? 1 : 0;
6351bcc9736cSJeff Kirsher 		pause->tx_pause =
6352bcc9736cSJeff Kirsher 			(sw_chk(hw, KS8842_SWITCH_CTRL_1_OFFSET,
6353bcc9736cSJeff Kirsher 				SWITCH_TX_FLOW_CTRL)) ? 1 : 0;
6354bcc9736cSJeff Kirsher 	}
6355bcc9736cSJeff Kirsher }
6356bcc9736cSJeff Kirsher 
6357bcc9736cSJeff Kirsher /**
6358bcc9736cSJeff Kirsher  * netdev_set_pauseparam - set flow control parameters
6359bcc9736cSJeff Kirsher  * @dev:	Network device.
6360bcc9736cSJeff Kirsher  * @pause:	Ethtool PAUSE settings data structure.
6361bcc9736cSJeff Kirsher  *
6362bcc9736cSJeff Kirsher  * This function sets the PAUSE control flow settings.
6363bcc9736cSJeff Kirsher  * Not implemented yet.
6364bcc9736cSJeff Kirsher  *
6365bcc9736cSJeff Kirsher  * Return 0 if successful; otherwise an error code.
6366bcc9736cSJeff Kirsher  */
6367bcc9736cSJeff Kirsher static int netdev_set_pauseparam(struct net_device *dev,
6368bcc9736cSJeff Kirsher 	struct ethtool_pauseparam *pause)
6369bcc9736cSJeff Kirsher {
6370bcc9736cSJeff Kirsher 	struct dev_priv *priv = netdev_priv(dev);
6371bcc9736cSJeff Kirsher 	struct dev_info *hw_priv = priv->adapter;
6372bcc9736cSJeff Kirsher 	struct ksz_hw *hw = &hw_priv->hw;
6373bcc9736cSJeff Kirsher 	struct ksz_port *port = &priv->port;
6374bcc9736cSJeff Kirsher 
6375bcc9736cSJeff Kirsher 	mutex_lock(&hw_priv->lock);
6376bcc9736cSJeff Kirsher 	if (pause->autoneg) {
6377bcc9736cSJeff Kirsher 		if (!pause->rx_pause && !pause->tx_pause)
6378bcc9736cSJeff Kirsher 			port->flow_ctrl = PHY_NO_FLOW_CTRL;
6379bcc9736cSJeff Kirsher 		else
6380bcc9736cSJeff Kirsher 			port->flow_ctrl = PHY_FLOW_CTRL;
6381bcc9736cSJeff Kirsher 		hw->overrides &= ~PAUSE_FLOW_CTRL;
6382bcc9736cSJeff Kirsher 		port->force_link = 0;
6383bcc9736cSJeff Kirsher 		if (hw->ksz_switch) {
6384bcc9736cSJeff Kirsher 			sw_cfg(hw, KS8842_SWITCH_CTRL_1_OFFSET,
6385bcc9736cSJeff Kirsher 				SWITCH_RX_FLOW_CTRL, 1);
6386bcc9736cSJeff Kirsher 			sw_cfg(hw, KS8842_SWITCH_CTRL_1_OFFSET,
6387bcc9736cSJeff Kirsher 				SWITCH_TX_FLOW_CTRL, 1);
6388bcc9736cSJeff Kirsher 		}
6389bcc9736cSJeff Kirsher 		port_set_link_speed(port);
6390bcc9736cSJeff Kirsher 	} else {
6391bcc9736cSJeff Kirsher 		hw->overrides |= PAUSE_FLOW_CTRL;
6392bcc9736cSJeff Kirsher 		if (hw->ksz_switch) {
6393bcc9736cSJeff Kirsher 			sw_cfg(hw, KS8842_SWITCH_CTRL_1_OFFSET,
6394bcc9736cSJeff Kirsher 				SWITCH_RX_FLOW_CTRL, pause->rx_pause);
6395bcc9736cSJeff Kirsher 			sw_cfg(hw, KS8842_SWITCH_CTRL_1_OFFSET,
6396bcc9736cSJeff Kirsher 				SWITCH_TX_FLOW_CTRL, pause->tx_pause);
6397bcc9736cSJeff Kirsher 		} else
6398bcc9736cSJeff Kirsher 			set_flow_ctrl(hw, pause->rx_pause, pause->tx_pause);
6399bcc9736cSJeff Kirsher 	}
6400bcc9736cSJeff Kirsher 	mutex_unlock(&hw_priv->lock);
6401bcc9736cSJeff Kirsher 
6402bcc9736cSJeff Kirsher 	return 0;
6403bcc9736cSJeff Kirsher }
6404bcc9736cSJeff Kirsher 
6405bcc9736cSJeff Kirsher /**
6406bcc9736cSJeff Kirsher  * netdev_get_ringparam - get tx/rx ring parameters
6407bcc9736cSJeff Kirsher  * @dev:	Network device.
6408bcc9736cSJeff Kirsher  * @pause:	Ethtool RING settings data structure.
6409bcc9736cSJeff Kirsher  *
6410bcc9736cSJeff Kirsher  * This procedure returns the TX/RX ring settings.
6411bcc9736cSJeff Kirsher  */
6412bcc9736cSJeff Kirsher static void netdev_get_ringparam(struct net_device *dev,
6413bcc9736cSJeff Kirsher 	struct ethtool_ringparam *ring)
6414bcc9736cSJeff Kirsher {
6415bcc9736cSJeff Kirsher 	struct dev_priv *priv = netdev_priv(dev);
6416bcc9736cSJeff Kirsher 	struct dev_info *hw_priv = priv->adapter;
6417bcc9736cSJeff Kirsher 	struct ksz_hw *hw = &hw_priv->hw;
6418bcc9736cSJeff Kirsher 
6419bcc9736cSJeff Kirsher 	ring->tx_max_pending = (1 << 9);
6420bcc9736cSJeff Kirsher 	ring->tx_pending = hw->tx_desc_info.alloc;
6421bcc9736cSJeff Kirsher 	ring->rx_max_pending = (1 << 9);
6422bcc9736cSJeff Kirsher 	ring->rx_pending = hw->rx_desc_info.alloc;
6423bcc9736cSJeff Kirsher }
6424bcc9736cSJeff Kirsher 
6425bcc9736cSJeff Kirsher #define STATS_LEN			(TOTAL_PORT_COUNTER_NUM)
6426bcc9736cSJeff Kirsher 
6427bcc9736cSJeff Kirsher static struct {
6428bcc9736cSJeff Kirsher 	char string[ETH_GSTRING_LEN];
6429bcc9736cSJeff Kirsher } ethtool_stats_keys[STATS_LEN] = {
6430bcc9736cSJeff Kirsher 	{ "rx_lo_priority_octets" },
6431bcc9736cSJeff Kirsher 	{ "rx_hi_priority_octets" },
6432bcc9736cSJeff Kirsher 	{ "rx_undersize_packets" },
6433bcc9736cSJeff Kirsher 	{ "rx_fragments" },
6434bcc9736cSJeff Kirsher 	{ "rx_oversize_packets" },
6435bcc9736cSJeff Kirsher 	{ "rx_jabbers" },
6436bcc9736cSJeff Kirsher 	{ "rx_symbol_errors" },
6437bcc9736cSJeff Kirsher 	{ "rx_crc_errors" },
6438bcc9736cSJeff Kirsher 	{ "rx_align_errors" },
6439bcc9736cSJeff Kirsher 	{ "rx_mac_ctrl_packets" },
6440bcc9736cSJeff Kirsher 	{ "rx_pause_packets" },
6441bcc9736cSJeff Kirsher 	{ "rx_bcast_packets" },
6442bcc9736cSJeff Kirsher 	{ "rx_mcast_packets" },
6443bcc9736cSJeff Kirsher 	{ "rx_ucast_packets" },
6444bcc9736cSJeff Kirsher 	{ "rx_64_or_less_octet_packets" },
6445bcc9736cSJeff Kirsher 	{ "rx_65_to_127_octet_packets" },
6446bcc9736cSJeff Kirsher 	{ "rx_128_to_255_octet_packets" },
6447bcc9736cSJeff Kirsher 	{ "rx_256_to_511_octet_packets" },
6448bcc9736cSJeff Kirsher 	{ "rx_512_to_1023_octet_packets" },
6449bcc9736cSJeff Kirsher 	{ "rx_1024_to_1522_octet_packets" },
6450bcc9736cSJeff Kirsher 
6451bcc9736cSJeff Kirsher 	{ "tx_lo_priority_octets" },
6452bcc9736cSJeff Kirsher 	{ "tx_hi_priority_octets" },
6453bcc9736cSJeff Kirsher 	{ "tx_late_collisions" },
6454bcc9736cSJeff Kirsher 	{ "tx_pause_packets" },
6455bcc9736cSJeff Kirsher 	{ "tx_bcast_packets" },
6456bcc9736cSJeff Kirsher 	{ "tx_mcast_packets" },
6457bcc9736cSJeff Kirsher 	{ "tx_ucast_packets" },
6458bcc9736cSJeff Kirsher 	{ "tx_deferred" },
6459bcc9736cSJeff Kirsher 	{ "tx_total_collisions" },
6460bcc9736cSJeff Kirsher 	{ "tx_excessive_collisions" },
6461bcc9736cSJeff Kirsher 	{ "tx_single_collisions" },
6462bcc9736cSJeff Kirsher 	{ "tx_mult_collisions" },
6463bcc9736cSJeff Kirsher 
6464bcc9736cSJeff Kirsher 	{ "rx_discards" },
6465bcc9736cSJeff Kirsher 	{ "tx_discards" },
6466bcc9736cSJeff Kirsher };
6467bcc9736cSJeff Kirsher 
6468bcc9736cSJeff Kirsher /**
6469bcc9736cSJeff Kirsher  * netdev_get_strings - get statistics identity strings
6470bcc9736cSJeff Kirsher  * @dev:	Network device.
6471bcc9736cSJeff Kirsher  * @stringset:	String set identifier.
6472bcc9736cSJeff Kirsher  * @buf:	Buffer to store the strings.
6473bcc9736cSJeff Kirsher  *
6474bcc9736cSJeff Kirsher  * This procedure returns the strings used to identify the statistics.
6475bcc9736cSJeff Kirsher  */
6476bcc9736cSJeff Kirsher static void netdev_get_strings(struct net_device *dev, u32 stringset, u8 *buf)
6477bcc9736cSJeff Kirsher {
6478bcc9736cSJeff Kirsher 	struct dev_priv *priv = netdev_priv(dev);
6479bcc9736cSJeff Kirsher 	struct dev_info *hw_priv = priv->adapter;
6480bcc9736cSJeff Kirsher 	struct ksz_hw *hw = &hw_priv->hw;
6481bcc9736cSJeff Kirsher 
6482bcc9736cSJeff Kirsher 	if (ETH_SS_STATS == stringset)
6483bcc9736cSJeff Kirsher 		memcpy(buf, &ethtool_stats_keys,
6484bcc9736cSJeff Kirsher 			ETH_GSTRING_LEN * hw->mib_cnt);
6485bcc9736cSJeff Kirsher }
6486bcc9736cSJeff Kirsher 
6487bcc9736cSJeff Kirsher /**
6488bcc9736cSJeff Kirsher  * netdev_get_sset_count - get statistics size
6489bcc9736cSJeff Kirsher  * @dev:	Network device.
6490bcc9736cSJeff Kirsher  * @sset:	The statistics set number.
6491bcc9736cSJeff Kirsher  *
6492bcc9736cSJeff Kirsher  * This function returns the size of the statistics to be reported.
6493bcc9736cSJeff Kirsher  *
6494bcc9736cSJeff Kirsher  * Return size of the statistics to be reported.
6495bcc9736cSJeff Kirsher  */
6496bcc9736cSJeff Kirsher static int netdev_get_sset_count(struct net_device *dev, int sset)
6497bcc9736cSJeff Kirsher {
6498bcc9736cSJeff Kirsher 	struct dev_priv *priv = netdev_priv(dev);
6499bcc9736cSJeff Kirsher 	struct dev_info *hw_priv = priv->adapter;
6500bcc9736cSJeff Kirsher 	struct ksz_hw *hw = &hw_priv->hw;
6501bcc9736cSJeff Kirsher 
6502bcc9736cSJeff Kirsher 	switch (sset) {
6503bcc9736cSJeff Kirsher 	case ETH_SS_STATS:
6504bcc9736cSJeff Kirsher 		return hw->mib_cnt;
6505bcc9736cSJeff Kirsher 	default:
6506bcc9736cSJeff Kirsher 		return -EOPNOTSUPP;
6507bcc9736cSJeff Kirsher 	}
6508bcc9736cSJeff Kirsher }
6509bcc9736cSJeff Kirsher 
6510bcc9736cSJeff Kirsher /**
6511bcc9736cSJeff Kirsher  * netdev_get_ethtool_stats - get network device statistics
6512bcc9736cSJeff Kirsher  * @dev:	Network device.
6513bcc9736cSJeff Kirsher  * @stats:	Ethtool statistics data structure.
6514bcc9736cSJeff Kirsher  * @data:	Buffer to store the statistics.
6515bcc9736cSJeff Kirsher  *
6516bcc9736cSJeff Kirsher  * This procedure returns the statistics.
6517bcc9736cSJeff Kirsher  */
6518bcc9736cSJeff Kirsher static void netdev_get_ethtool_stats(struct net_device *dev,
6519bcc9736cSJeff Kirsher 	struct ethtool_stats *stats, u64 *data)
6520bcc9736cSJeff Kirsher {
6521bcc9736cSJeff Kirsher 	struct dev_priv *priv = netdev_priv(dev);
6522bcc9736cSJeff Kirsher 	struct dev_info *hw_priv = priv->adapter;
6523bcc9736cSJeff Kirsher 	struct ksz_hw *hw = &hw_priv->hw;
6524bcc9736cSJeff Kirsher 	struct ksz_port *port = &priv->port;
6525bcc9736cSJeff Kirsher 	int n_stats = stats->n_stats;
6526bcc9736cSJeff Kirsher 	int i;
6527bcc9736cSJeff Kirsher 	int n;
6528bcc9736cSJeff Kirsher 	int p;
6529bcc9736cSJeff Kirsher 	int rc;
6530bcc9736cSJeff Kirsher 	u64 counter[TOTAL_PORT_COUNTER_NUM];
6531bcc9736cSJeff Kirsher 
6532bcc9736cSJeff Kirsher 	mutex_lock(&hw_priv->lock);
6533bcc9736cSJeff Kirsher 	n = SWITCH_PORT_NUM;
6534bcc9736cSJeff Kirsher 	for (i = 0, p = port->first_port; i < port->mib_port_cnt; i++, p++) {
6535bcc9736cSJeff Kirsher 		if (media_connected == hw->port_mib[p].state) {
6536bcc9736cSJeff Kirsher 			hw_priv->counter[p].read = 1;
6537bcc9736cSJeff Kirsher 
6538bcc9736cSJeff Kirsher 			/* Remember first port that requests read. */
6539bcc9736cSJeff Kirsher 			if (n == SWITCH_PORT_NUM)
6540bcc9736cSJeff Kirsher 				n = p;
6541bcc9736cSJeff Kirsher 		}
6542bcc9736cSJeff Kirsher 	}
6543bcc9736cSJeff Kirsher 	mutex_unlock(&hw_priv->lock);
6544bcc9736cSJeff Kirsher 
6545bcc9736cSJeff Kirsher 	if (n < SWITCH_PORT_NUM)
6546bcc9736cSJeff Kirsher 		schedule_work(&hw_priv->mib_read);
6547bcc9736cSJeff Kirsher 
6548bcc9736cSJeff Kirsher 	if (1 == port->mib_port_cnt && n < SWITCH_PORT_NUM) {
6549bcc9736cSJeff Kirsher 		p = n;
6550bcc9736cSJeff Kirsher 		rc = wait_event_interruptible_timeout(
6551bcc9736cSJeff Kirsher 			hw_priv->counter[p].counter,
6552bcc9736cSJeff Kirsher 			2 == hw_priv->counter[p].read,
6553bcc9736cSJeff Kirsher 			HZ * 1);
6554bcc9736cSJeff Kirsher 	} else
6555bcc9736cSJeff Kirsher 		for (i = 0, p = n; i < port->mib_port_cnt - n; i++, p++) {
6556bcc9736cSJeff Kirsher 			if (0 == i) {
6557bcc9736cSJeff Kirsher 				rc = wait_event_interruptible_timeout(
6558bcc9736cSJeff Kirsher 					hw_priv->counter[p].counter,
6559bcc9736cSJeff Kirsher 					2 == hw_priv->counter[p].read,
6560bcc9736cSJeff Kirsher 					HZ * 2);
6561bcc9736cSJeff Kirsher 			} else if (hw->port_mib[p].cnt_ptr) {
6562bcc9736cSJeff Kirsher 				rc = wait_event_interruptible_timeout(
6563bcc9736cSJeff Kirsher 					hw_priv->counter[p].counter,
6564bcc9736cSJeff Kirsher 					2 == hw_priv->counter[p].read,
6565bcc9736cSJeff Kirsher 					HZ * 1);
6566bcc9736cSJeff Kirsher 			}
6567bcc9736cSJeff Kirsher 		}
6568bcc9736cSJeff Kirsher 
6569bcc9736cSJeff Kirsher 	get_mib_counters(hw, port->first_port, port->mib_port_cnt, counter);
6570bcc9736cSJeff Kirsher 	n = hw->mib_cnt;
6571bcc9736cSJeff Kirsher 	if (n > n_stats)
6572bcc9736cSJeff Kirsher 		n = n_stats;
6573bcc9736cSJeff Kirsher 	n_stats -= n;
6574bcc9736cSJeff Kirsher 	for (i = 0; i < n; i++)
6575bcc9736cSJeff Kirsher 		*data++ = counter[i];
6576bcc9736cSJeff Kirsher }
6577bcc9736cSJeff Kirsher 
6578bcc9736cSJeff Kirsher /**
6579bcc9736cSJeff Kirsher  * netdev_set_features - set receive checksum support
6580bcc9736cSJeff Kirsher  * @dev:	Network device.
6581bcc9736cSJeff Kirsher  * @features:	New device features (offloads).
6582bcc9736cSJeff Kirsher  *
6583bcc9736cSJeff Kirsher  * This function sets receive checksum support setting.
6584bcc9736cSJeff Kirsher  *
6585bcc9736cSJeff Kirsher  * Return 0 if successful; otherwise an error code.
6586bcc9736cSJeff Kirsher  */
6587c8f44affSMichał Mirosław static int netdev_set_features(struct net_device *dev,
6588c8f44affSMichał Mirosław 	netdev_features_t features)
6589bcc9736cSJeff Kirsher {
6590bcc9736cSJeff Kirsher 	struct dev_priv *priv = netdev_priv(dev);
6591bcc9736cSJeff Kirsher 	struct dev_info *hw_priv = priv->adapter;
6592bcc9736cSJeff Kirsher 	struct ksz_hw *hw = &hw_priv->hw;
6593bcc9736cSJeff Kirsher 
6594bcc9736cSJeff Kirsher 	mutex_lock(&hw_priv->lock);
6595bcc9736cSJeff Kirsher 
6596bcc9736cSJeff Kirsher 	/* see note in hw_setup() */
6597bcc9736cSJeff Kirsher 	if (features & NETIF_F_RXCSUM)
6598bcc9736cSJeff Kirsher 		hw->rx_cfg |= DMA_RX_CSUM_TCP | DMA_RX_CSUM_IP;
6599bcc9736cSJeff Kirsher 	else
6600bcc9736cSJeff Kirsher 		hw->rx_cfg &= ~(DMA_RX_CSUM_TCP | DMA_RX_CSUM_IP);
6601bcc9736cSJeff Kirsher 
6602bcc9736cSJeff Kirsher 	if (hw->enabled)
6603bcc9736cSJeff Kirsher 		writel(hw->rx_cfg, hw->io + KS_DMA_RX_CTRL);
6604bcc9736cSJeff Kirsher 
6605bcc9736cSJeff Kirsher 	mutex_unlock(&hw_priv->lock);
6606bcc9736cSJeff Kirsher 
6607bcc9736cSJeff Kirsher 	return 0;
6608bcc9736cSJeff Kirsher }
6609bcc9736cSJeff Kirsher 
66109b07be4bSstephen hemminger static const struct ethtool_ops netdev_ethtool_ops = {
6611bcc9736cSJeff Kirsher 	.get_settings		= netdev_get_settings,
6612bcc9736cSJeff Kirsher 	.set_settings		= netdev_set_settings,
6613bcc9736cSJeff Kirsher 	.nway_reset		= netdev_nway_reset,
6614bcc9736cSJeff Kirsher 	.get_link		= netdev_get_link,
6615bcc9736cSJeff Kirsher 	.get_drvinfo		= netdev_get_drvinfo,
6616bcc9736cSJeff Kirsher 	.get_regs_len		= netdev_get_regs_len,
6617bcc9736cSJeff Kirsher 	.get_regs		= netdev_get_regs,
6618bcc9736cSJeff Kirsher 	.get_wol		= netdev_get_wol,
6619bcc9736cSJeff Kirsher 	.set_wol		= netdev_set_wol,
6620bcc9736cSJeff Kirsher 	.get_msglevel		= netdev_get_msglevel,
6621bcc9736cSJeff Kirsher 	.set_msglevel		= netdev_set_msglevel,
6622bcc9736cSJeff Kirsher 	.get_eeprom_len		= netdev_get_eeprom_len,
6623bcc9736cSJeff Kirsher 	.get_eeprom		= netdev_get_eeprom,
6624bcc9736cSJeff Kirsher 	.set_eeprom		= netdev_set_eeprom,
6625bcc9736cSJeff Kirsher 	.get_pauseparam		= netdev_get_pauseparam,
6626bcc9736cSJeff Kirsher 	.set_pauseparam		= netdev_set_pauseparam,
6627bcc9736cSJeff Kirsher 	.get_ringparam		= netdev_get_ringparam,
6628bcc9736cSJeff Kirsher 	.get_strings		= netdev_get_strings,
6629bcc9736cSJeff Kirsher 	.get_sset_count		= netdev_get_sset_count,
6630bcc9736cSJeff Kirsher 	.get_ethtool_stats	= netdev_get_ethtool_stats,
6631bcc9736cSJeff Kirsher };
6632bcc9736cSJeff Kirsher 
6633bcc9736cSJeff Kirsher /*
6634bcc9736cSJeff Kirsher  * Hardware monitoring
6635bcc9736cSJeff Kirsher  */
6636bcc9736cSJeff Kirsher 
6637bcc9736cSJeff Kirsher static void update_link(struct net_device *dev, struct dev_priv *priv,
6638bcc9736cSJeff Kirsher 	struct ksz_port *port)
6639bcc9736cSJeff Kirsher {
6640bcc9736cSJeff Kirsher 	if (priv->media_state != port->linked->state) {
6641bcc9736cSJeff Kirsher 		priv->media_state = port->linked->state;
6642bcc9736cSJeff Kirsher 		if (netif_running(dev))
6643bcc9736cSJeff Kirsher 			set_media_state(dev, media_connected);
6644bcc9736cSJeff Kirsher 	}
6645bcc9736cSJeff Kirsher }
6646bcc9736cSJeff Kirsher 
6647bcc9736cSJeff Kirsher static void mib_read_work(struct work_struct *work)
6648bcc9736cSJeff Kirsher {
6649bcc9736cSJeff Kirsher 	struct dev_info *hw_priv =
6650bcc9736cSJeff Kirsher 		container_of(work, struct dev_info, mib_read);
6651bcc9736cSJeff Kirsher 	struct ksz_hw *hw = &hw_priv->hw;
6652bcc9736cSJeff Kirsher 	struct ksz_port_mib *mib;
6653bcc9736cSJeff Kirsher 	int i;
6654bcc9736cSJeff Kirsher 
6655bcc9736cSJeff Kirsher 	next_jiffies = jiffies;
6656bcc9736cSJeff Kirsher 	for (i = 0; i < hw->mib_port_cnt; i++) {
6657bcc9736cSJeff Kirsher 		mib = &hw->port_mib[i];
6658bcc9736cSJeff Kirsher 
6659bcc9736cSJeff Kirsher 		/* Reading MIB counters or requested to read. */
6660bcc9736cSJeff Kirsher 		if (mib->cnt_ptr || 1 == hw_priv->counter[i].read) {
6661bcc9736cSJeff Kirsher 
6662bcc9736cSJeff Kirsher 			/* Need to process receive interrupt. */
6663bcc9736cSJeff Kirsher 			if (port_r_cnt(hw, i))
6664bcc9736cSJeff Kirsher 				break;
6665bcc9736cSJeff Kirsher 			hw_priv->counter[i].read = 0;
6666bcc9736cSJeff Kirsher 
6667bcc9736cSJeff Kirsher 			/* Finish reading counters. */
6668bcc9736cSJeff Kirsher 			if (0 == mib->cnt_ptr) {
6669bcc9736cSJeff Kirsher 				hw_priv->counter[i].read = 2;
6670bcc9736cSJeff Kirsher 				wake_up_interruptible(
6671bcc9736cSJeff Kirsher 					&hw_priv->counter[i].counter);
6672bcc9736cSJeff Kirsher 			}
6673bcc9736cSJeff Kirsher 		} else if (jiffies >= hw_priv->counter[i].time) {
6674bcc9736cSJeff Kirsher 			/* Only read MIB counters when the port is connected. */
6675bcc9736cSJeff Kirsher 			if (media_connected == mib->state)
6676bcc9736cSJeff Kirsher 				hw_priv->counter[i].read = 1;
6677bcc9736cSJeff Kirsher 			next_jiffies += HZ * 1 * hw->mib_port_cnt;
6678bcc9736cSJeff Kirsher 			hw_priv->counter[i].time = next_jiffies;
6679bcc9736cSJeff Kirsher 
6680bcc9736cSJeff Kirsher 		/* Port is just disconnected. */
6681bcc9736cSJeff Kirsher 		} else if (mib->link_down) {
6682bcc9736cSJeff Kirsher 			mib->link_down = 0;
6683bcc9736cSJeff Kirsher 
6684bcc9736cSJeff Kirsher 			/* Read counters one last time after link is lost. */
6685bcc9736cSJeff Kirsher 			hw_priv->counter[i].read = 1;
6686bcc9736cSJeff Kirsher 		}
6687bcc9736cSJeff Kirsher 	}
6688bcc9736cSJeff Kirsher }
6689bcc9736cSJeff Kirsher 
6690bcc9736cSJeff Kirsher static void mib_monitor(unsigned long ptr)
6691bcc9736cSJeff Kirsher {
6692bcc9736cSJeff Kirsher 	struct dev_info *hw_priv = (struct dev_info *) ptr;
6693bcc9736cSJeff Kirsher 
6694bcc9736cSJeff Kirsher 	mib_read_work(&hw_priv->mib_read);
6695bcc9736cSJeff Kirsher 
6696bcc9736cSJeff Kirsher 	/* This is used to verify Wake-on-LAN is working. */
6697bcc9736cSJeff Kirsher 	if (hw_priv->pme_wait) {
6698bcc9736cSJeff Kirsher 		if (hw_priv->pme_wait <= jiffies) {
6699bcc9736cSJeff Kirsher 			hw_clr_wol_pme_status(&hw_priv->hw);
6700bcc9736cSJeff Kirsher 			hw_priv->pme_wait = 0;
6701bcc9736cSJeff Kirsher 		}
6702bcc9736cSJeff Kirsher 	} else if (hw_chk_wol_pme_status(&hw_priv->hw)) {
6703bcc9736cSJeff Kirsher 
6704bcc9736cSJeff Kirsher 		/* PME is asserted.  Wait 2 seconds to clear it. */
6705bcc9736cSJeff Kirsher 		hw_priv->pme_wait = jiffies + HZ * 2;
6706bcc9736cSJeff Kirsher 	}
6707bcc9736cSJeff Kirsher 
6708bcc9736cSJeff Kirsher 	ksz_update_timer(&hw_priv->mib_timer_info);
6709bcc9736cSJeff Kirsher }
6710bcc9736cSJeff Kirsher 
6711bcc9736cSJeff Kirsher /**
6712bcc9736cSJeff Kirsher  * dev_monitor - periodic monitoring
6713bcc9736cSJeff Kirsher  * @ptr:	Network device pointer.
6714bcc9736cSJeff Kirsher  *
6715bcc9736cSJeff Kirsher  * This routine is run in a kernel timer to monitor the network device.
6716bcc9736cSJeff Kirsher  */
6717bcc9736cSJeff Kirsher static void dev_monitor(unsigned long ptr)
6718bcc9736cSJeff Kirsher {
6719bcc9736cSJeff Kirsher 	struct net_device *dev = (struct net_device *) ptr;
6720bcc9736cSJeff Kirsher 	struct dev_priv *priv = netdev_priv(dev);
6721bcc9736cSJeff Kirsher 	struct dev_info *hw_priv = priv->adapter;
6722bcc9736cSJeff Kirsher 	struct ksz_hw *hw = &hw_priv->hw;
6723bcc9736cSJeff Kirsher 	struct ksz_port *port = &priv->port;
6724bcc9736cSJeff Kirsher 
6725bcc9736cSJeff Kirsher 	if (!(hw->features & LINK_INT_WORKING))
6726bcc9736cSJeff Kirsher 		port_get_link_speed(port);
6727bcc9736cSJeff Kirsher 	update_link(dev, priv, port);
6728bcc9736cSJeff Kirsher 
6729bcc9736cSJeff Kirsher 	ksz_update_timer(&priv->monitor_timer_info);
6730bcc9736cSJeff Kirsher }
6731bcc9736cSJeff Kirsher 
6732bcc9736cSJeff Kirsher /*
6733bcc9736cSJeff Kirsher  * Linux network device interface functions
6734bcc9736cSJeff Kirsher  */
6735bcc9736cSJeff Kirsher 
6736bcc9736cSJeff Kirsher /* Driver exported variables */
6737bcc9736cSJeff Kirsher 
6738bcc9736cSJeff Kirsher static int msg_enable;
6739bcc9736cSJeff Kirsher 
6740bcc9736cSJeff Kirsher static char *macaddr = ":";
6741bcc9736cSJeff Kirsher static char *mac1addr = ":";
6742bcc9736cSJeff Kirsher 
6743bcc9736cSJeff Kirsher /*
6744bcc9736cSJeff Kirsher  * This enables multiple network device mode for KSZ8842, which contains a
6745bcc9736cSJeff Kirsher  * switch with two physical ports.  Some users like to take control of the
6746bcc9736cSJeff Kirsher  * ports for running Spanning Tree Protocol.  The driver will create an
6747bcc9736cSJeff Kirsher  * additional eth? device for the other port.
6748bcc9736cSJeff Kirsher  *
6749bcc9736cSJeff Kirsher  * Some limitations are the network devices cannot have different MTU and
6750bcc9736cSJeff Kirsher  * multicast hash tables.
6751bcc9736cSJeff Kirsher  */
6752bcc9736cSJeff Kirsher static int multi_dev;
6753bcc9736cSJeff Kirsher 
6754bcc9736cSJeff Kirsher /*
6755bcc9736cSJeff Kirsher  * As most users select multiple network device mode to use Spanning Tree
6756bcc9736cSJeff Kirsher  * Protocol, this enables a feature in which most unicast and multicast packets
6757bcc9736cSJeff Kirsher  * are forwarded inside the switch and not passed to the host.  Only packets
6758bcc9736cSJeff Kirsher  * that need the host's attention are passed to it.  This prevents the host
6759bcc9736cSJeff Kirsher  * wasting CPU time to examine each and every incoming packets and do the
6760bcc9736cSJeff Kirsher  * forwarding itself.
6761bcc9736cSJeff Kirsher  *
6762bcc9736cSJeff Kirsher  * As the hack requires the private bridge header, the driver cannot compile
6763bcc9736cSJeff Kirsher  * with just the kernel headers.
6764bcc9736cSJeff Kirsher  *
6765bcc9736cSJeff Kirsher  * Enabling STP support also turns on multiple network device mode.
6766bcc9736cSJeff Kirsher  */
6767bcc9736cSJeff Kirsher static int stp;
6768bcc9736cSJeff Kirsher 
6769bcc9736cSJeff Kirsher /*
6770bcc9736cSJeff Kirsher  * This enables fast aging in the KSZ8842 switch.  Not sure what situation
6771bcc9736cSJeff Kirsher  * needs that.  However, fast aging is used to flush the dynamic MAC table when
6772bcc9736cSJeff Kirsher  * STP suport is enabled.
6773bcc9736cSJeff Kirsher  */
6774bcc9736cSJeff Kirsher static int fast_aging;
6775bcc9736cSJeff Kirsher 
6776bcc9736cSJeff Kirsher /**
6777bcc9736cSJeff Kirsher  * netdev_init - initialize network device.
6778bcc9736cSJeff Kirsher  * @dev:	Network device.
6779bcc9736cSJeff Kirsher  *
6780bcc9736cSJeff Kirsher  * This function initializes the network device.
6781bcc9736cSJeff Kirsher  *
6782bcc9736cSJeff Kirsher  * Return 0 if successful; otherwise an error code indicating failure.
6783bcc9736cSJeff Kirsher  */
6784bcc9736cSJeff Kirsher static int __init netdev_init(struct net_device *dev)
6785bcc9736cSJeff Kirsher {
6786bcc9736cSJeff Kirsher 	struct dev_priv *priv = netdev_priv(dev);
6787bcc9736cSJeff Kirsher 
6788bcc9736cSJeff Kirsher 	/* 500 ms timeout */
6789bcc9736cSJeff Kirsher 	ksz_init_timer(&priv->monitor_timer_info, 500 * HZ / 1000,
6790bcc9736cSJeff Kirsher 		dev_monitor, dev);
6791bcc9736cSJeff Kirsher 
6792bcc9736cSJeff Kirsher 	/* 500 ms timeout */
6793bcc9736cSJeff Kirsher 	dev->watchdog_timeo = HZ / 2;
6794bcc9736cSJeff Kirsher 
6795bcc9736cSJeff Kirsher 	dev->hw_features = NETIF_F_IP_CSUM | NETIF_F_SG | NETIF_F_RXCSUM;
6796bcc9736cSJeff Kirsher 
6797bcc9736cSJeff Kirsher 	/*
6798bcc9736cSJeff Kirsher 	 * Hardware does not really support IPv6 checksum generation, but
6799bcc9736cSJeff Kirsher 	 * driver actually runs faster with this on.
6800bcc9736cSJeff Kirsher 	 */
6801bcc9736cSJeff Kirsher 	dev->hw_features |= NETIF_F_IPV6_CSUM;
6802bcc9736cSJeff Kirsher 
6803bcc9736cSJeff Kirsher 	dev->features |= dev->hw_features;
6804bcc9736cSJeff Kirsher 
6805bcc9736cSJeff Kirsher 	sema_init(&priv->proc_sem, 1);
6806bcc9736cSJeff Kirsher 
6807bcc9736cSJeff Kirsher 	priv->mii_if.phy_id_mask = 0x1;
6808bcc9736cSJeff Kirsher 	priv->mii_if.reg_num_mask = 0x7;
6809bcc9736cSJeff Kirsher 	priv->mii_if.dev = dev;
6810bcc9736cSJeff Kirsher 	priv->mii_if.mdio_read = mdio_read;
6811bcc9736cSJeff Kirsher 	priv->mii_if.mdio_write = mdio_write;
6812bcc9736cSJeff Kirsher 	priv->mii_if.phy_id = priv->port.first_port + 1;
6813bcc9736cSJeff Kirsher 
6814bcc9736cSJeff Kirsher 	priv->msg_enable = netif_msg_init(msg_enable,
6815bcc9736cSJeff Kirsher 		(NETIF_MSG_DRV | NETIF_MSG_PROBE | NETIF_MSG_LINK));
6816bcc9736cSJeff Kirsher 
6817bcc9736cSJeff Kirsher 	return 0;
6818bcc9736cSJeff Kirsher }
6819bcc9736cSJeff Kirsher 
6820bcc9736cSJeff Kirsher static const struct net_device_ops netdev_ops = {
6821bcc9736cSJeff Kirsher 	.ndo_init		= netdev_init,
6822bcc9736cSJeff Kirsher 	.ndo_open		= netdev_open,
6823bcc9736cSJeff Kirsher 	.ndo_stop		= netdev_close,
6824bcc9736cSJeff Kirsher 	.ndo_get_stats		= netdev_query_statistics,
6825bcc9736cSJeff Kirsher 	.ndo_start_xmit		= netdev_tx,
6826bcc9736cSJeff Kirsher 	.ndo_tx_timeout		= netdev_tx_timeout,
6827bcc9736cSJeff Kirsher 	.ndo_change_mtu		= netdev_change_mtu,
6828bcc9736cSJeff Kirsher 	.ndo_set_features	= netdev_set_features,
6829bcc9736cSJeff Kirsher 	.ndo_set_mac_address	= netdev_set_mac_address,
6830bcc9736cSJeff Kirsher 	.ndo_validate_addr	= eth_validate_addr,
6831bcc9736cSJeff Kirsher 	.ndo_do_ioctl		= netdev_ioctl,
6832bcc9736cSJeff Kirsher 	.ndo_set_rx_mode	= netdev_set_rx_mode,
6833bcc9736cSJeff Kirsher #ifdef CONFIG_NET_POLL_CONTROLLER
6834bcc9736cSJeff Kirsher 	.ndo_poll_controller	= netdev_netpoll,
6835bcc9736cSJeff Kirsher #endif
6836bcc9736cSJeff Kirsher };
6837bcc9736cSJeff Kirsher 
6838bcc9736cSJeff Kirsher static void netdev_free(struct net_device *dev)
6839bcc9736cSJeff Kirsher {
6840bcc9736cSJeff Kirsher 	if (dev->watchdog_timeo)
6841bcc9736cSJeff Kirsher 		unregister_netdev(dev);
6842bcc9736cSJeff Kirsher 
6843bcc9736cSJeff Kirsher 	free_netdev(dev);
6844bcc9736cSJeff Kirsher }
6845bcc9736cSJeff Kirsher 
6846bcc9736cSJeff Kirsher struct platform_info {
6847bcc9736cSJeff Kirsher 	struct dev_info dev_info;
6848bcc9736cSJeff Kirsher 	struct net_device *netdev[SWITCH_PORT_NUM];
6849bcc9736cSJeff Kirsher };
6850bcc9736cSJeff Kirsher 
6851bcc9736cSJeff Kirsher static int net_device_present;
6852bcc9736cSJeff Kirsher 
6853bcc9736cSJeff Kirsher static void get_mac_addr(struct dev_info *hw_priv, u8 *macaddr, int port)
6854bcc9736cSJeff Kirsher {
6855bcc9736cSJeff Kirsher 	int i;
6856bcc9736cSJeff Kirsher 	int j;
6857bcc9736cSJeff Kirsher 	int got_num;
6858bcc9736cSJeff Kirsher 	int num;
6859bcc9736cSJeff Kirsher 
6860bcc9736cSJeff Kirsher 	i = j = num = got_num = 0;
68616a3c910cSJoe Perches 	while (j < ETH_ALEN) {
6862bcc9736cSJeff Kirsher 		if (macaddr[i]) {
6863bcc9736cSJeff Kirsher 			int digit;
6864bcc9736cSJeff Kirsher 
6865bcc9736cSJeff Kirsher 			got_num = 1;
6866bcc9736cSJeff Kirsher 			digit = hex_to_bin(macaddr[i]);
6867bcc9736cSJeff Kirsher 			if (digit >= 0)
6868bcc9736cSJeff Kirsher 				num = num * 16 + digit;
6869bcc9736cSJeff Kirsher 			else if (':' == macaddr[i])
6870bcc9736cSJeff Kirsher 				got_num = 2;
6871bcc9736cSJeff Kirsher 			else
6872bcc9736cSJeff Kirsher 				break;
6873bcc9736cSJeff Kirsher 		} else if (got_num)
6874bcc9736cSJeff Kirsher 			got_num = 2;
6875bcc9736cSJeff Kirsher 		else
6876bcc9736cSJeff Kirsher 			break;
6877bcc9736cSJeff Kirsher 		if (2 == got_num) {
6878bcc9736cSJeff Kirsher 			if (MAIN_PORT == port) {
6879bcc9736cSJeff Kirsher 				hw_priv->hw.override_addr[j++] = (u8) num;
6880bcc9736cSJeff Kirsher 				hw_priv->hw.override_addr[5] +=
6881bcc9736cSJeff Kirsher 					hw_priv->hw.id;
6882bcc9736cSJeff Kirsher 			} else {
6883bcc9736cSJeff Kirsher 				hw_priv->hw.ksz_switch->other_addr[j++] =
6884bcc9736cSJeff Kirsher 					(u8) num;
6885bcc9736cSJeff Kirsher 				hw_priv->hw.ksz_switch->other_addr[5] +=
6886bcc9736cSJeff Kirsher 					hw_priv->hw.id;
6887bcc9736cSJeff Kirsher 			}
6888bcc9736cSJeff Kirsher 			num = got_num = 0;
6889bcc9736cSJeff Kirsher 		}
6890bcc9736cSJeff Kirsher 		i++;
6891bcc9736cSJeff Kirsher 	}
68926a3c910cSJoe Perches 	if (ETH_ALEN == j) {
6893bcc9736cSJeff Kirsher 		if (MAIN_PORT == port)
6894bcc9736cSJeff Kirsher 			hw_priv->hw.mac_override = 1;
6895bcc9736cSJeff Kirsher 	}
6896bcc9736cSJeff Kirsher }
6897bcc9736cSJeff Kirsher 
6898bcc9736cSJeff Kirsher #define KS884X_DMA_MASK			(~0x0UL)
6899bcc9736cSJeff Kirsher 
6900bcc9736cSJeff Kirsher static void read_other_addr(struct ksz_hw *hw)
6901bcc9736cSJeff Kirsher {
6902bcc9736cSJeff Kirsher 	int i;
6903bcc9736cSJeff Kirsher 	u16 data[3];
6904bcc9736cSJeff Kirsher 	struct ksz_switch *sw = hw->ksz_switch;
6905bcc9736cSJeff Kirsher 
6906bcc9736cSJeff Kirsher 	for (i = 0; i < 3; i++)
6907bcc9736cSJeff Kirsher 		data[i] = eeprom_read(hw, i + EEPROM_DATA_OTHER_MAC_ADDR);
6908bcc9736cSJeff Kirsher 	if ((data[0] || data[1] || data[2]) && data[0] != 0xffff) {
6909bcc9736cSJeff Kirsher 		sw->other_addr[5] = (u8) data[0];
6910bcc9736cSJeff Kirsher 		sw->other_addr[4] = (u8)(data[0] >> 8);
6911bcc9736cSJeff Kirsher 		sw->other_addr[3] = (u8) data[1];
6912bcc9736cSJeff Kirsher 		sw->other_addr[2] = (u8)(data[1] >> 8);
6913bcc9736cSJeff Kirsher 		sw->other_addr[1] = (u8) data[2];
6914bcc9736cSJeff Kirsher 		sw->other_addr[0] = (u8)(data[2] >> 8);
6915bcc9736cSJeff Kirsher 	}
6916bcc9736cSJeff Kirsher }
6917bcc9736cSJeff Kirsher 
6918bcc9736cSJeff Kirsher #ifndef PCI_VENDOR_ID_MICREL_KS
6919bcc9736cSJeff Kirsher #define PCI_VENDOR_ID_MICREL_KS		0x16c6
6920bcc9736cSJeff Kirsher #endif
6921bcc9736cSJeff Kirsher 
6922bcc9736cSJeff Kirsher static int __devinit pcidev_init(struct pci_dev *pdev,
6923bcc9736cSJeff Kirsher 	const struct pci_device_id *id)
6924bcc9736cSJeff Kirsher {
6925bcc9736cSJeff Kirsher 	struct net_device *dev;
6926bcc9736cSJeff Kirsher 	struct dev_priv *priv;
6927bcc9736cSJeff Kirsher 	struct dev_info *hw_priv;
6928bcc9736cSJeff Kirsher 	struct ksz_hw *hw;
6929bcc9736cSJeff Kirsher 	struct platform_info *info;
6930bcc9736cSJeff Kirsher 	struct ksz_port *port;
6931bcc9736cSJeff Kirsher 	unsigned long reg_base;
6932bcc9736cSJeff Kirsher 	unsigned long reg_len;
6933bcc9736cSJeff Kirsher 	int cnt;
6934bcc9736cSJeff Kirsher 	int i;
6935bcc9736cSJeff Kirsher 	int mib_port_count;
6936bcc9736cSJeff Kirsher 	int pi;
6937bcc9736cSJeff Kirsher 	int port_count;
6938bcc9736cSJeff Kirsher 	int result;
6939bcc9736cSJeff Kirsher 	char banner[sizeof(version)];
6940bcc9736cSJeff Kirsher 	struct ksz_switch *sw = NULL;
6941bcc9736cSJeff Kirsher 
6942bcc9736cSJeff Kirsher 	result = pci_enable_device(pdev);
6943bcc9736cSJeff Kirsher 	if (result)
6944bcc9736cSJeff Kirsher 		return result;
6945bcc9736cSJeff Kirsher 
6946bcc9736cSJeff Kirsher 	result = -ENODEV;
6947bcc9736cSJeff Kirsher 
6948bcc9736cSJeff Kirsher 	if (pci_set_dma_mask(pdev, DMA_BIT_MASK(32)) ||
6949bcc9736cSJeff Kirsher 			pci_set_consistent_dma_mask(pdev, DMA_BIT_MASK(32)))
6950bcc9736cSJeff Kirsher 		return result;
6951bcc9736cSJeff Kirsher 
6952bcc9736cSJeff Kirsher 	reg_base = pci_resource_start(pdev, 0);
6953bcc9736cSJeff Kirsher 	reg_len = pci_resource_len(pdev, 0);
6954bcc9736cSJeff Kirsher 	if ((pci_resource_flags(pdev, 0) & IORESOURCE_IO) != 0)
6955bcc9736cSJeff Kirsher 		return result;
6956bcc9736cSJeff Kirsher 
6957bcc9736cSJeff Kirsher 	if (!request_mem_region(reg_base, reg_len, DRV_NAME))
6958bcc9736cSJeff Kirsher 		return result;
6959bcc9736cSJeff Kirsher 	pci_set_master(pdev);
6960bcc9736cSJeff Kirsher 
6961bcc9736cSJeff Kirsher 	result = -ENOMEM;
6962bcc9736cSJeff Kirsher 
6963bcc9736cSJeff Kirsher 	info = kzalloc(sizeof(struct platform_info), GFP_KERNEL);
6964bcc9736cSJeff Kirsher 	if (!info)
6965bcc9736cSJeff Kirsher 		goto pcidev_init_dev_err;
6966bcc9736cSJeff Kirsher 
6967bcc9736cSJeff Kirsher 	hw_priv = &info->dev_info;
6968bcc9736cSJeff Kirsher 	hw_priv->pdev = pdev;
6969bcc9736cSJeff Kirsher 
6970bcc9736cSJeff Kirsher 	hw = &hw_priv->hw;
6971bcc9736cSJeff Kirsher 
6972bcc9736cSJeff Kirsher 	hw->io = ioremap(reg_base, reg_len);
6973bcc9736cSJeff Kirsher 	if (!hw->io)
6974bcc9736cSJeff Kirsher 		goto pcidev_init_io_err;
6975bcc9736cSJeff Kirsher 
6976bcc9736cSJeff Kirsher 	cnt = hw_init(hw);
6977bcc9736cSJeff Kirsher 	if (!cnt) {
6978bcc9736cSJeff Kirsher 		if (msg_enable & NETIF_MSG_PROBE)
6979bcc9736cSJeff Kirsher 			pr_alert("chip not detected\n");
6980bcc9736cSJeff Kirsher 		result = -ENODEV;
6981bcc9736cSJeff Kirsher 		goto pcidev_init_alloc_err;
6982bcc9736cSJeff Kirsher 	}
6983bcc9736cSJeff Kirsher 
6984bcc9736cSJeff Kirsher 	snprintf(banner, sizeof(banner), "%s", version);
6985bcc9736cSJeff Kirsher 	banner[13] = cnt + '0';		/* Replace x in "Micrel KSZ884x" */
6986bcc9736cSJeff Kirsher 	dev_info(&hw_priv->pdev->dev, "%s\n", banner);
6987bcc9736cSJeff Kirsher 	dev_dbg(&hw_priv->pdev->dev, "Mem = %p; IRQ = %d\n", hw->io, pdev->irq);
6988bcc9736cSJeff Kirsher 
6989bcc9736cSJeff Kirsher 	/* Assume device is KSZ8841. */
6990bcc9736cSJeff Kirsher 	hw->dev_count = 1;
6991bcc9736cSJeff Kirsher 	port_count = 1;
6992bcc9736cSJeff Kirsher 	mib_port_count = 1;
6993bcc9736cSJeff Kirsher 	hw->addr_list_size = 0;
6994bcc9736cSJeff Kirsher 	hw->mib_cnt = PORT_COUNTER_NUM;
6995bcc9736cSJeff Kirsher 	hw->mib_port_cnt = 1;
6996bcc9736cSJeff Kirsher 
6997bcc9736cSJeff Kirsher 	/* KSZ8842 has a switch with multiple ports. */
6998bcc9736cSJeff Kirsher 	if (2 == cnt) {
6999bcc9736cSJeff Kirsher 		if (fast_aging)
7000bcc9736cSJeff Kirsher 			hw->overrides |= FAST_AGING;
7001bcc9736cSJeff Kirsher 
7002bcc9736cSJeff Kirsher 		hw->mib_cnt = TOTAL_PORT_COUNTER_NUM;
7003bcc9736cSJeff Kirsher 
7004bcc9736cSJeff Kirsher 		/* Multiple network device interfaces are required. */
7005bcc9736cSJeff Kirsher 		if (multi_dev) {
7006bcc9736cSJeff Kirsher 			hw->dev_count = SWITCH_PORT_NUM;
7007bcc9736cSJeff Kirsher 			hw->addr_list_size = SWITCH_PORT_NUM - 1;
7008bcc9736cSJeff Kirsher 		}
7009bcc9736cSJeff Kirsher 
7010bcc9736cSJeff Kirsher 		/* Single network device has multiple ports. */
7011bcc9736cSJeff Kirsher 		if (1 == hw->dev_count) {
7012bcc9736cSJeff Kirsher 			port_count = SWITCH_PORT_NUM;
7013bcc9736cSJeff Kirsher 			mib_port_count = SWITCH_PORT_NUM;
7014bcc9736cSJeff Kirsher 		}
7015bcc9736cSJeff Kirsher 		hw->mib_port_cnt = TOTAL_PORT_NUM;
7016bcc9736cSJeff Kirsher 		hw->ksz_switch = kzalloc(sizeof(struct ksz_switch), GFP_KERNEL);
7017bcc9736cSJeff Kirsher 		if (!hw->ksz_switch)
7018bcc9736cSJeff Kirsher 			goto pcidev_init_alloc_err;
7019bcc9736cSJeff Kirsher 
7020bcc9736cSJeff Kirsher 		sw = hw->ksz_switch;
7021bcc9736cSJeff Kirsher 	}
7022bcc9736cSJeff Kirsher 	for (i = 0; i < hw->mib_port_cnt; i++)
7023bcc9736cSJeff Kirsher 		hw->port_mib[i].mib_start = 0;
7024bcc9736cSJeff Kirsher 
7025bcc9736cSJeff Kirsher 	hw->parent = hw_priv;
7026bcc9736cSJeff Kirsher 
7027bcc9736cSJeff Kirsher 	/* Default MTU is 1500. */
7028bcc9736cSJeff Kirsher 	hw_priv->mtu = (REGULAR_RX_BUF_SIZE + 3) & ~3;
7029bcc9736cSJeff Kirsher 
7030bcc9736cSJeff Kirsher 	if (ksz_alloc_mem(hw_priv))
7031bcc9736cSJeff Kirsher 		goto pcidev_init_mem_err;
7032bcc9736cSJeff Kirsher 
7033bcc9736cSJeff Kirsher 	hw_priv->hw.id = net_device_present;
7034bcc9736cSJeff Kirsher 
7035bcc9736cSJeff Kirsher 	spin_lock_init(&hw_priv->hwlock);
7036bcc9736cSJeff Kirsher 	mutex_init(&hw_priv->lock);
7037bcc9736cSJeff Kirsher 
7038bcc9736cSJeff Kirsher 	/* tasklet is enabled. */
7039bcc9736cSJeff Kirsher 	tasklet_init(&hw_priv->rx_tasklet, rx_proc_task,
7040bcc9736cSJeff Kirsher 		(unsigned long) hw_priv);
7041bcc9736cSJeff Kirsher 	tasklet_init(&hw_priv->tx_tasklet, tx_proc_task,
7042bcc9736cSJeff Kirsher 		(unsigned long) hw_priv);
7043bcc9736cSJeff Kirsher 
7044bcc9736cSJeff Kirsher 	/* tasklet_enable will decrement the atomic counter. */
7045bcc9736cSJeff Kirsher 	tasklet_disable(&hw_priv->rx_tasklet);
7046bcc9736cSJeff Kirsher 	tasklet_disable(&hw_priv->tx_tasklet);
7047bcc9736cSJeff Kirsher 
7048bcc9736cSJeff Kirsher 	for (i = 0; i < TOTAL_PORT_NUM; i++)
7049bcc9736cSJeff Kirsher 		init_waitqueue_head(&hw_priv->counter[i].counter);
7050bcc9736cSJeff Kirsher 
7051bcc9736cSJeff Kirsher 	if (macaddr[0] != ':')
7052bcc9736cSJeff Kirsher 		get_mac_addr(hw_priv, macaddr, MAIN_PORT);
7053bcc9736cSJeff Kirsher 
7054bcc9736cSJeff Kirsher 	/* Read MAC address and initialize override address if not overrided. */
7055bcc9736cSJeff Kirsher 	hw_read_addr(hw);
7056bcc9736cSJeff Kirsher 
7057bcc9736cSJeff Kirsher 	/* Multiple device interfaces mode requires a second MAC address. */
7058bcc9736cSJeff Kirsher 	if (hw->dev_count > 1) {
70596a3c910cSJoe Perches 		memcpy(sw->other_addr, hw->override_addr, ETH_ALEN);
7060bcc9736cSJeff Kirsher 		read_other_addr(hw);
7061bcc9736cSJeff Kirsher 		if (mac1addr[0] != ':')
7062bcc9736cSJeff Kirsher 			get_mac_addr(hw_priv, mac1addr, OTHER_PORT);
7063bcc9736cSJeff Kirsher 	}
7064bcc9736cSJeff Kirsher 
7065bcc9736cSJeff Kirsher 	hw_setup(hw);
7066bcc9736cSJeff Kirsher 	if (hw->ksz_switch)
7067bcc9736cSJeff Kirsher 		sw_setup(hw);
7068bcc9736cSJeff Kirsher 	else {
7069bcc9736cSJeff Kirsher 		hw_priv->wol_support = WOL_SUPPORT;
7070bcc9736cSJeff Kirsher 		hw_priv->wol_enable = 0;
7071bcc9736cSJeff Kirsher 	}
7072bcc9736cSJeff Kirsher 
7073bcc9736cSJeff Kirsher 	INIT_WORK(&hw_priv->mib_read, mib_read_work);
7074bcc9736cSJeff Kirsher 
7075bcc9736cSJeff Kirsher 	/* 500 ms timeout */
7076bcc9736cSJeff Kirsher 	ksz_init_timer(&hw_priv->mib_timer_info, 500 * HZ / 1000,
7077bcc9736cSJeff Kirsher 		mib_monitor, hw_priv);
7078bcc9736cSJeff Kirsher 
7079bcc9736cSJeff Kirsher 	for (i = 0; i < hw->dev_count; i++) {
7080bcc9736cSJeff Kirsher 		dev = alloc_etherdev(sizeof(struct dev_priv));
7081bcc9736cSJeff Kirsher 		if (!dev)
7082bcc9736cSJeff Kirsher 			goto pcidev_init_reg_err;
7083bcc9736cSJeff Kirsher 		info->netdev[i] = dev;
7084bcc9736cSJeff Kirsher 
7085bcc9736cSJeff Kirsher 		priv = netdev_priv(dev);
7086bcc9736cSJeff Kirsher 		priv->adapter = hw_priv;
7087bcc9736cSJeff Kirsher 		priv->id = net_device_present++;
7088bcc9736cSJeff Kirsher 
7089bcc9736cSJeff Kirsher 		port = &priv->port;
7090bcc9736cSJeff Kirsher 		port->port_cnt = port_count;
7091bcc9736cSJeff Kirsher 		port->mib_port_cnt = mib_port_count;
7092bcc9736cSJeff Kirsher 		port->first_port = i;
7093bcc9736cSJeff Kirsher 		port->flow_ctrl = PHY_FLOW_CTRL;
7094bcc9736cSJeff Kirsher 
7095bcc9736cSJeff Kirsher 		port->hw = hw;
7096bcc9736cSJeff Kirsher 		port->linked = &hw->port_info[port->first_port];
7097bcc9736cSJeff Kirsher 
7098bcc9736cSJeff Kirsher 		for (cnt = 0, pi = i; cnt < port_count; cnt++, pi++) {
7099bcc9736cSJeff Kirsher 			hw->port_info[pi].port_id = pi;
7100bcc9736cSJeff Kirsher 			hw->port_info[pi].pdev = dev;
7101bcc9736cSJeff Kirsher 			hw->port_info[pi].state = media_disconnected;
7102bcc9736cSJeff Kirsher 		}
7103bcc9736cSJeff Kirsher 
7104bcc9736cSJeff Kirsher 		dev->mem_start = (unsigned long) hw->io;
7105bcc9736cSJeff Kirsher 		dev->mem_end = dev->mem_start + reg_len - 1;
7106bcc9736cSJeff Kirsher 		dev->irq = pdev->irq;
7107bcc9736cSJeff Kirsher 		if (MAIN_PORT == i)
7108bcc9736cSJeff Kirsher 			memcpy(dev->dev_addr, hw_priv->hw.override_addr,
71096a3c910cSJoe Perches 			       ETH_ALEN);
7110bcc9736cSJeff Kirsher 		else {
71116a3c910cSJoe Perches 			memcpy(dev->dev_addr, sw->other_addr, ETH_ALEN);
7112bcc9736cSJeff Kirsher 			if (!memcmp(sw->other_addr, hw->override_addr,
71136a3c910cSJoe Perches 				    ETH_ALEN))
7114bcc9736cSJeff Kirsher 				dev->dev_addr[5] += port->first_port;
7115bcc9736cSJeff Kirsher 		}
7116bcc9736cSJeff Kirsher 
7117bcc9736cSJeff Kirsher 		dev->netdev_ops = &netdev_ops;
7118bcc9736cSJeff Kirsher 		SET_ETHTOOL_OPS(dev, &netdev_ethtool_ops);
7119bcc9736cSJeff Kirsher 		if (register_netdev(dev))
7120bcc9736cSJeff Kirsher 			goto pcidev_init_reg_err;
7121bcc9736cSJeff Kirsher 		port_set_power_saving(port, true);
7122bcc9736cSJeff Kirsher 	}
7123bcc9736cSJeff Kirsher 
7124bcc9736cSJeff Kirsher 	pci_dev_get(hw_priv->pdev);
7125bcc9736cSJeff Kirsher 	pci_set_drvdata(pdev, info);
7126bcc9736cSJeff Kirsher 	return 0;
7127bcc9736cSJeff Kirsher 
7128bcc9736cSJeff Kirsher pcidev_init_reg_err:
7129bcc9736cSJeff Kirsher 	for (i = 0; i < hw->dev_count; i++) {
7130bcc9736cSJeff Kirsher 		if (info->netdev[i]) {
7131bcc9736cSJeff Kirsher 			netdev_free(info->netdev[i]);
7132bcc9736cSJeff Kirsher 			info->netdev[i] = NULL;
7133bcc9736cSJeff Kirsher 		}
7134bcc9736cSJeff Kirsher 	}
7135bcc9736cSJeff Kirsher 
7136bcc9736cSJeff Kirsher pcidev_init_mem_err:
7137bcc9736cSJeff Kirsher 	ksz_free_mem(hw_priv);
7138bcc9736cSJeff Kirsher 	kfree(hw->ksz_switch);
7139bcc9736cSJeff Kirsher 
7140bcc9736cSJeff Kirsher pcidev_init_alloc_err:
7141bcc9736cSJeff Kirsher 	iounmap(hw->io);
7142bcc9736cSJeff Kirsher 
7143bcc9736cSJeff Kirsher pcidev_init_io_err:
7144bcc9736cSJeff Kirsher 	kfree(info);
7145bcc9736cSJeff Kirsher 
7146bcc9736cSJeff Kirsher pcidev_init_dev_err:
7147bcc9736cSJeff Kirsher 	release_mem_region(reg_base, reg_len);
7148bcc9736cSJeff Kirsher 
7149bcc9736cSJeff Kirsher 	return result;
7150bcc9736cSJeff Kirsher }
7151bcc9736cSJeff Kirsher 
7152bcc9736cSJeff Kirsher static void pcidev_exit(struct pci_dev *pdev)
7153bcc9736cSJeff Kirsher {
7154bcc9736cSJeff Kirsher 	int i;
7155bcc9736cSJeff Kirsher 	struct platform_info *info = pci_get_drvdata(pdev);
7156bcc9736cSJeff Kirsher 	struct dev_info *hw_priv = &info->dev_info;
7157bcc9736cSJeff Kirsher 
7158bcc9736cSJeff Kirsher 	pci_set_drvdata(pdev, NULL);
7159bcc9736cSJeff Kirsher 
7160bcc9736cSJeff Kirsher 	release_mem_region(pci_resource_start(pdev, 0),
7161bcc9736cSJeff Kirsher 		pci_resource_len(pdev, 0));
7162bcc9736cSJeff Kirsher 	for (i = 0; i < hw_priv->hw.dev_count; i++) {
7163bcc9736cSJeff Kirsher 		if (info->netdev[i])
7164bcc9736cSJeff Kirsher 			netdev_free(info->netdev[i]);
7165bcc9736cSJeff Kirsher 	}
7166bcc9736cSJeff Kirsher 	if (hw_priv->hw.io)
7167bcc9736cSJeff Kirsher 		iounmap(hw_priv->hw.io);
7168bcc9736cSJeff Kirsher 	ksz_free_mem(hw_priv);
7169bcc9736cSJeff Kirsher 	kfree(hw_priv->hw.ksz_switch);
7170bcc9736cSJeff Kirsher 	pci_dev_put(hw_priv->pdev);
7171bcc9736cSJeff Kirsher 	kfree(info);
7172bcc9736cSJeff Kirsher }
7173bcc9736cSJeff Kirsher 
7174bcc9736cSJeff Kirsher #ifdef CONFIG_PM
7175bcc9736cSJeff Kirsher static int pcidev_resume(struct pci_dev *pdev)
7176bcc9736cSJeff Kirsher {
7177bcc9736cSJeff Kirsher 	int i;
7178bcc9736cSJeff Kirsher 	struct platform_info *info = pci_get_drvdata(pdev);
7179bcc9736cSJeff Kirsher 	struct dev_info *hw_priv = &info->dev_info;
7180bcc9736cSJeff Kirsher 	struct ksz_hw *hw = &hw_priv->hw;
7181bcc9736cSJeff Kirsher 
7182bcc9736cSJeff Kirsher 	pci_set_power_state(pdev, PCI_D0);
7183bcc9736cSJeff Kirsher 	pci_restore_state(pdev);
7184bcc9736cSJeff Kirsher 	pci_enable_wake(pdev, PCI_D0, 0);
7185bcc9736cSJeff Kirsher 
7186bcc9736cSJeff Kirsher 	if (hw_priv->wol_enable)
7187bcc9736cSJeff Kirsher 		hw_cfg_wol_pme(hw, 0);
7188bcc9736cSJeff Kirsher 	for (i = 0; i < hw->dev_count; i++) {
7189bcc9736cSJeff Kirsher 		if (info->netdev[i]) {
7190bcc9736cSJeff Kirsher 			struct net_device *dev = info->netdev[i];
7191bcc9736cSJeff Kirsher 
7192bcc9736cSJeff Kirsher 			if (netif_running(dev)) {
7193bcc9736cSJeff Kirsher 				netdev_open(dev);
7194bcc9736cSJeff Kirsher 				netif_device_attach(dev);
7195bcc9736cSJeff Kirsher 			}
7196bcc9736cSJeff Kirsher 		}
7197bcc9736cSJeff Kirsher 	}
7198bcc9736cSJeff Kirsher 	return 0;
7199bcc9736cSJeff Kirsher }
7200bcc9736cSJeff Kirsher 
7201bcc9736cSJeff Kirsher static int pcidev_suspend(struct pci_dev *pdev, pm_message_t state)
7202bcc9736cSJeff Kirsher {
7203bcc9736cSJeff Kirsher 	int i;
7204bcc9736cSJeff Kirsher 	struct platform_info *info = pci_get_drvdata(pdev);
7205bcc9736cSJeff Kirsher 	struct dev_info *hw_priv = &info->dev_info;
7206bcc9736cSJeff Kirsher 	struct ksz_hw *hw = &hw_priv->hw;
7207bcc9736cSJeff Kirsher 
7208bcc9736cSJeff Kirsher 	/* Need to find a way to retrieve the device IP address. */
7209bcc9736cSJeff Kirsher 	static const u8 net_addr[] = { 192, 168, 1, 1 };
7210bcc9736cSJeff Kirsher 
7211bcc9736cSJeff Kirsher 	for (i = 0; i < hw->dev_count; i++) {
7212bcc9736cSJeff Kirsher 		if (info->netdev[i]) {
7213bcc9736cSJeff Kirsher 			struct net_device *dev = info->netdev[i];
7214bcc9736cSJeff Kirsher 
7215bcc9736cSJeff Kirsher 			if (netif_running(dev)) {
7216bcc9736cSJeff Kirsher 				netif_device_detach(dev);
7217bcc9736cSJeff Kirsher 				netdev_close(dev);
7218bcc9736cSJeff Kirsher 			}
7219bcc9736cSJeff Kirsher 		}
7220bcc9736cSJeff Kirsher 	}
7221bcc9736cSJeff Kirsher 	if (hw_priv->wol_enable) {
7222bcc9736cSJeff Kirsher 		hw_enable_wol(hw, hw_priv->wol_enable, net_addr);
7223bcc9736cSJeff Kirsher 		hw_cfg_wol_pme(hw, 1);
7224bcc9736cSJeff Kirsher 	}
7225bcc9736cSJeff Kirsher 
7226bcc9736cSJeff Kirsher 	pci_save_state(pdev);
7227bcc9736cSJeff Kirsher 	pci_enable_wake(pdev, pci_choose_state(pdev, state), 1);
7228bcc9736cSJeff Kirsher 	pci_set_power_state(pdev, pci_choose_state(pdev, state));
7229bcc9736cSJeff Kirsher 	return 0;
7230bcc9736cSJeff Kirsher }
7231bcc9736cSJeff Kirsher #endif
7232bcc9736cSJeff Kirsher 
7233bcc9736cSJeff Kirsher static char pcidev_name[] = "ksz884xp";
7234bcc9736cSJeff Kirsher 
7235bcc9736cSJeff Kirsher static struct pci_device_id pcidev_table[] = {
7236bcc9736cSJeff Kirsher 	{ PCI_VENDOR_ID_MICREL_KS, 0x8841,
7237bcc9736cSJeff Kirsher 		PCI_ANY_ID, PCI_ANY_ID, 0, 0, 0 },
7238bcc9736cSJeff Kirsher 	{ PCI_VENDOR_ID_MICREL_KS, 0x8842,
7239bcc9736cSJeff Kirsher 		PCI_ANY_ID, PCI_ANY_ID, 0, 0, 0 },
7240bcc9736cSJeff Kirsher 	{ 0 }
7241bcc9736cSJeff Kirsher };
7242bcc9736cSJeff Kirsher 
7243bcc9736cSJeff Kirsher MODULE_DEVICE_TABLE(pci, pcidev_table);
7244bcc9736cSJeff Kirsher 
7245bcc9736cSJeff Kirsher static struct pci_driver pci_device_driver = {
7246bcc9736cSJeff Kirsher #ifdef CONFIG_PM
7247bcc9736cSJeff Kirsher 	.suspend	= pcidev_suspend,
7248bcc9736cSJeff Kirsher 	.resume		= pcidev_resume,
7249bcc9736cSJeff Kirsher #endif
7250bcc9736cSJeff Kirsher 	.name		= pcidev_name,
7251bcc9736cSJeff Kirsher 	.id_table	= pcidev_table,
7252bcc9736cSJeff Kirsher 	.probe		= pcidev_init,
7253bcc9736cSJeff Kirsher 	.remove		= pcidev_exit
7254bcc9736cSJeff Kirsher };
7255bcc9736cSJeff Kirsher 
7256bcc9736cSJeff Kirsher static int __init ksz884x_init_module(void)
7257bcc9736cSJeff Kirsher {
7258bcc9736cSJeff Kirsher 	return pci_register_driver(&pci_device_driver);
7259bcc9736cSJeff Kirsher }
7260bcc9736cSJeff Kirsher 
7261bcc9736cSJeff Kirsher static void __exit ksz884x_cleanup_module(void)
7262bcc9736cSJeff Kirsher {
7263bcc9736cSJeff Kirsher 	pci_unregister_driver(&pci_device_driver);
7264bcc9736cSJeff Kirsher }
7265bcc9736cSJeff Kirsher 
7266bcc9736cSJeff Kirsher module_init(ksz884x_init_module);
7267bcc9736cSJeff Kirsher module_exit(ksz884x_cleanup_module);
7268bcc9736cSJeff Kirsher 
7269bcc9736cSJeff Kirsher MODULE_DESCRIPTION("KSZ8841/2 PCI network driver");
7270bcc9736cSJeff Kirsher MODULE_AUTHOR("Tristram Ha <Tristram.Ha@micrel.com>");
7271bcc9736cSJeff Kirsher MODULE_LICENSE("GPL");
7272bcc9736cSJeff Kirsher 
7273bcc9736cSJeff Kirsher module_param_named(message, msg_enable, int, 0);
7274bcc9736cSJeff Kirsher MODULE_PARM_DESC(message, "Message verbosity level (0=none, 31=all)");
7275bcc9736cSJeff Kirsher 
7276bcc9736cSJeff Kirsher module_param(macaddr, charp, 0);
7277bcc9736cSJeff Kirsher module_param(mac1addr, charp, 0);
7278bcc9736cSJeff Kirsher module_param(fast_aging, int, 0);
7279bcc9736cSJeff Kirsher module_param(multi_dev, int, 0);
7280bcc9736cSJeff Kirsher module_param(stp, int, 0);
7281bcc9736cSJeff Kirsher MODULE_PARM_DESC(macaddr, "MAC address");
7282bcc9736cSJeff Kirsher MODULE_PARM_DESC(mac1addr, "Second MAC address");
7283bcc9736cSJeff Kirsher MODULE_PARM_DESC(fast_aging, "Fast aging");
7284bcc9736cSJeff Kirsher MODULE_PARM_DESC(multi_dev, "Multiple device interfaces");
7285bcc9736cSJeff Kirsher MODULE_PARM_DESC(stp, "STP support");
7286