xref: /linux/drivers/net/ethernet/ti/cpsw_sl.c (revision c39f2d9db0fd81ea20bb5cce9b3f082ca63753e2)
1*a71a18f2SGrygorii Strashko // SPDX-License-Identifier: GPL-2.0
2*a71a18f2SGrygorii Strashko /*
3*a71a18f2SGrygorii Strashko  * Texas Instruments Ethernet Switch media-access-controller (MAC) submodule/
4*a71a18f2SGrygorii Strashko  * Ethernet MAC Sliver (CPGMAC_SL)
5*a71a18f2SGrygorii Strashko  *
6*a71a18f2SGrygorii Strashko  * Copyright (C) 2019 Texas Instruments
7*a71a18f2SGrygorii Strashko  *
8*a71a18f2SGrygorii Strashko  */
9*a71a18f2SGrygorii Strashko 
10*a71a18f2SGrygorii Strashko #include <linux/delay.h>
11*a71a18f2SGrygorii Strashko #include <linux/io.h>
12*a71a18f2SGrygorii Strashko #include <linux/kernel.h>
13*a71a18f2SGrygorii Strashko 
14*a71a18f2SGrygorii Strashko #include "cpsw_sl.h"
15*a71a18f2SGrygorii Strashko 
16*a71a18f2SGrygorii Strashko #define CPSW_SL_REG_NOTUSED U16_MAX
17*a71a18f2SGrygorii Strashko 
18*a71a18f2SGrygorii Strashko static const u16 cpsw_sl_reg_map_cpsw[] = {
19*a71a18f2SGrygorii Strashko 	[CPSW_SL_IDVER] = 0x00,
20*a71a18f2SGrygorii Strashko 	[CPSW_SL_MACCONTROL] = 0x04,
21*a71a18f2SGrygorii Strashko 	[CPSW_SL_MACSTATUS] = 0x08,
22*a71a18f2SGrygorii Strashko 	[CPSW_SL_SOFT_RESET] = 0x0c,
23*a71a18f2SGrygorii Strashko 	[CPSW_SL_RX_MAXLEN] = 0x10,
24*a71a18f2SGrygorii Strashko 	[CPSW_SL_BOFFTEST] = 0x14,
25*a71a18f2SGrygorii Strashko 	[CPSW_SL_RX_PAUSE] = 0x18,
26*a71a18f2SGrygorii Strashko 	[CPSW_SL_TX_PAUSE] = 0x1c,
27*a71a18f2SGrygorii Strashko 	[CPSW_SL_EMCONTROL] = 0x20,
28*a71a18f2SGrygorii Strashko 	[CPSW_SL_RX_PRI_MAP] = 0x24,
29*a71a18f2SGrygorii Strashko 	[CPSW_SL_TX_GAP] = 0x28,
30*a71a18f2SGrygorii Strashko };
31*a71a18f2SGrygorii Strashko 
32*a71a18f2SGrygorii Strashko static const u16 cpsw_sl_reg_map_66ak2hk[] = {
33*a71a18f2SGrygorii Strashko 	[CPSW_SL_IDVER] = 0x00,
34*a71a18f2SGrygorii Strashko 	[CPSW_SL_MACCONTROL] = 0x04,
35*a71a18f2SGrygorii Strashko 	[CPSW_SL_MACSTATUS] = 0x08,
36*a71a18f2SGrygorii Strashko 	[CPSW_SL_SOFT_RESET] = 0x0c,
37*a71a18f2SGrygorii Strashko 	[CPSW_SL_RX_MAXLEN] = 0x10,
38*a71a18f2SGrygorii Strashko 	[CPSW_SL_BOFFTEST] = CPSW_SL_REG_NOTUSED,
39*a71a18f2SGrygorii Strashko 	[CPSW_SL_RX_PAUSE] = 0x18,
40*a71a18f2SGrygorii Strashko 	[CPSW_SL_TX_PAUSE] = 0x1c,
41*a71a18f2SGrygorii Strashko 	[CPSW_SL_EMCONTROL] = 0x20,
42*a71a18f2SGrygorii Strashko 	[CPSW_SL_RX_PRI_MAP] = 0x24,
43*a71a18f2SGrygorii Strashko 	[CPSW_SL_TX_GAP] = CPSW_SL_REG_NOTUSED,
44*a71a18f2SGrygorii Strashko };
45*a71a18f2SGrygorii Strashko 
46*a71a18f2SGrygorii Strashko static const u16 cpsw_sl_reg_map_66ak2x_xgbe[] = {
47*a71a18f2SGrygorii Strashko 	[CPSW_SL_IDVER] = 0x00,
48*a71a18f2SGrygorii Strashko 	[CPSW_SL_MACCONTROL] = 0x04,
49*a71a18f2SGrygorii Strashko 	[CPSW_SL_MACSTATUS] = 0x08,
50*a71a18f2SGrygorii Strashko 	[CPSW_SL_SOFT_RESET] = 0x0c,
51*a71a18f2SGrygorii Strashko 	[CPSW_SL_RX_MAXLEN] = 0x10,
52*a71a18f2SGrygorii Strashko 	[CPSW_SL_BOFFTEST] = CPSW_SL_REG_NOTUSED,
53*a71a18f2SGrygorii Strashko 	[CPSW_SL_RX_PAUSE] = 0x18,
54*a71a18f2SGrygorii Strashko 	[CPSW_SL_TX_PAUSE] = 0x1c,
55*a71a18f2SGrygorii Strashko 	[CPSW_SL_EMCONTROL] = 0x20,
56*a71a18f2SGrygorii Strashko 	[CPSW_SL_RX_PRI_MAP] = CPSW_SL_REG_NOTUSED,
57*a71a18f2SGrygorii Strashko 	[CPSW_SL_TX_GAP] = 0x28,
58*a71a18f2SGrygorii Strashko };
59*a71a18f2SGrygorii Strashko 
60*a71a18f2SGrygorii Strashko static const u16 cpsw_sl_reg_map_66ak2elg_am65[] = {
61*a71a18f2SGrygorii Strashko 	[CPSW_SL_IDVER] = CPSW_SL_REG_NOTUSED,
62*a71a18f2SGrygorii Strashko 	[CPSW_SL_MACCONTROL] = 0x00,
63*a71a18f2SGrygorii Strashko 	[CPSW_SL_MACSTATUS] = 0x04,
64*a71a18f2SGrygorii Strashko 	[CPSW_SL_SOFT_RESET] = 0x08,
65*a71a18f2SGrygorii Strashko 	[CPSW_SL_RX_MAXLEN] = CPSW_SL_REG_NOTUSED,
66*a71a18f2SGrygorii Strashko 	[CPSW_SL_BOFFTEST] = 0x0c,
67*a71a18f2SGrygorii Strashko 	[CPSW_SL_RX_PAUSE] = 0x10,
68*a71a18f2SGrygorii Strashko 	[CPSW_SL_TX_PAUSE] = 0x40,
69*a71a18f2SGrygorii Strashko 	[CPSW_SL_EMCONTROL] = 0x70,
70*a71a18f2SGrygorii Strashko 	[CPSW_SL_RX_PRI_MAP] = CPSW_SL_REG_NOTUSED,
71*a71a18f2SGrygorii Strashko 	[CPSW_SL_TX_GAP] = 0x74,
72*a71a18f2SGrygorii Strashko };
73*a71a18f2SGrygorii Strashko 
74*a71a18f2SGrygorii Strashko #define CPSW_SL_SOFT_RESET_BIT		BIT(0)
75*a71a18f2SGrygorii Strashko 
76*a71a18f2SGrygorii Strashko #define CPSW_SL_STATUS_PN_IDLE		BIT(31)
77*a71a18f2SGrygorii Strashko #define CPSW_SL_AM65_STATUS_PN_E_IDLE	BIT(30)
78*a71a18f2SGrygorii Strashko #define CPSW_SL_AM65_STATUS_PN_P_IDLE	BIT(29)
79*a71a18f2SGrygorii Strashko #define CPSW_SL_AM65_STATUS_PN_TX_IDLE	BIT(28)
80*a71a18f2SGrygorii Strashko 
81*a71a18f2SGrygorii Strashko #define CPSW_SL_STATUS_IDLE_MASK_BASE (CPSW_SL_STATUS_PN_IDLE)
82*a71a18f2SGrygorii Strashko 
83*a71a18f2SGrygorii Strashko #define CPSW_SL_STATUS_IDLE_MASK_K3 \
84*a71a18f2SGrygorii Strashko 	(CPSW_SL_STATUS_IDLE_MASK_BASE | CPSW_SL_AM65_STATUS_PN_E_IDLE | \
85*a71a18f2SGrygorii Strashko 	 CPSW_SL_AM65_STATUS_PN_P_IDLE | CPSW_SL_AM65_STATUS_PN_TX_IDLE)
86*a71a18f2SGrygorii Strashko 
87*a71a18f2SGrygorii Strashko #define CPSW_SL_CTL_FUNC_BASE \
88*a71a18f2SGrygorii Strashko 	(CPSW_SL_CTL_FULLDUPLEX |\
89*a71a18f2SGrygorii Strashko 	CPSW_SL_CTL_LOOPBACK |\
90*a71a18f2SGrygorii Strashko 	CPSW_SL_CTL_RX_FLOW_EN |\
91*a71a18f2SGrygorii Strashko 	CPSW_SL_CTL_TX_FLOW_EN |\
92*a71a18f2SGrygorii Strashko 	CPSW_SL_CTL_GMII_EN |\
93*a71a18f2SGrygorii Strashko 	CPSW_SL_CTL_TX_PACE |\
94*a71a18f2SGrygorii Strashko 	CPSW_SL_CTL_GIG |\
95*a71a18f2SGrygorii Strashko 	CPSW_SL_CTL_CMD_IDLE |\
96*a71a18f2SGrygorii Strashko 	CPSW_SL_CTL_IFCTL_A |\
97*a71a18f2SGrygorii Strashko 	CPSW_SL_CTL_IFCTL_B |\
98*a71a18f2SGrygorii Strashko 	CPSW_SL_CTL_GIG_FORCE |\
99*a71a18f2SGrygorii Strashko 	CPSW_SL_CTL_EXT_EN |\
100*a71a18f2SGrygorii Strashko 	CPSW_SL_CTL_RX_CEF_EN |\
101*a71a18f2SGrygorii Strashko 	CPSW_SL_CTL_RX_CSF_EN |\
102*a71a18f2SGrygorii Strashko 	CPSW_SL_CTL_RX_CMF_EN)
103*a71a18f2SGrygorii Strashko 
104*a71a18f2SGrygorii Strashko struct cpsw_sl {
105*a71a18f2SGrygorii Strashko 	struct device *dev;
106*a71a18f2SGrygorii Strashko 	void __iomem *sl_base;
107*a71a18f2SGrygorii Strashko 	const u16 *regs;
108*a71a18f2SGrygorii Strashko 	u32 control_features;
109*a71a18f2SGrygorii Strashko 	u32 idle_mask;
110*a71a18f2SGrygorii Strashko };
111*a71a18f2SGrygorii Strashko 
112*a71a18f2SGrygorii Strashko struct cpsw_sl_dev_id {
113*a71a18f2SGrygorii Strashko 	const char *device_id;
114*a71a18f2SGrygorii Strashko 	const u16 *regs;
115*a71a18f2SGrygorii Strashko 	const u32 control_features;
116*a71a18f2SGrygorii Strashko 	const u32 regs_offset;
117*a71a18f2SGrygorii Strashko 	const u32 idle_mask;
118*a71a18f2SGrygorii Strashko };
119*a71a18f2SGrygorii Strashko 
120*a71a18f2SGrygorii Strashko static const struct cpsw_sl_dev_id cpsw_sl_id_match[] = {
121*a71a18f2SGrygorii Strashko 	{
122*a71a18f2SGrygorii Strashko 		.device_id = "cpsw",
123*a71a18f2SGrygorii Strashko 		.regs = cpsw_sl_reg_map_cpsw,
124*a71a18f2SGrygorii Strashko 		.control_features = CPSW_SL_CTL_FUNC_BASE |
125*a71a18f2SGrygorii Strashko 				    CPSW_SL_CTL_MTEST |
126*a71a18f2SGrygorii Strashko 				    CPSW_SL_CTL_TX_SHORT_GAP_EN |
127*a71a18f2SGrygorii Strashko 				    CPSW_SL_CTL_TX_SG_LIM_EN,
128*a71a18f2SGrygorii Strashko 		.idle_mask = CPSW_SL_STATUS_IDLE_MASK_BASE,
129*a71a18f2SGrygorii Strashko 	},
130*a71a18f2SGrygorii Strashko 	{
131*a71a18f2SGrygorii Strashko 		.device_id = "66ak2hk",
132*a71a18f2SGrygorii Strashko 		.regs = cpsw_sl_reg_map_66ak2hk,
133*a71a18f2SGrygorii Strashko 		.control_features = CPSW_SL_CTL_FUNC_BASE |
134*a71a18f2SGrygorii Strashko 				    CPSW_SL_CTL_TX_SHORT_GAP_EN,
135*a71a18f2SGrygorii Strashko 		.idle_mask = CPSW_SL_STATUS_IDLE_MASK_BASE,
136*a71a18f2SGrygorii Strashko 	},
137*a71a18f2SGrygorii Strashko 	{
138*a71a18f2SGrygorii Strashko 		.device_id = "66ak2x_xgbe",
139*a71a18f2SGrygorii Strashko 		.regs = cpsw_sl_reg_map_66ak2x_xgbe,
140*a71a18f2SGrygorii Strashko 		.control_features = CPSW_SL_CTL_FUNC_BASE |
141*a71a18f2SGrygorii Strashko 				    CPSW_SL_CTL_XGIG |
142*a71a18f2SGrygorii Strashko 				    CPSW_SL_CTL_TX_SHORT_GAP_EN |
143*a71a18f2SGrygorii Strashko 				    CPSW_SL_CTL_CRC_TYPE |
144*a71a18f2SGrygorii Strashko 				    CPSW_SL_CTL_XGMII_EN,
145*a71a18f2SGrygorii Strashko 		.idle_mask = CPSW_SL_STATUS_IDLE_MASK_BASE,
146*a71a18f2SGrygorii Strashko 	},
147*a71a18f2SGrygorii Strashko 	{
148*a71a18f2SGrygorii Strashko 		.device_id = "66ak2el",
149*a71a18f2SGrygorii Strashko 		.regs = cpsw_sl_reg_map_66ak2elg_am65,
150*a71a18f2SGrygorii Strashko 		.regs_offset = 0x330,
151*a71a18f2SGrygorii Strashko 		.control_features = CPSW_SL_CTL_FUNC_BASE |
152*a71a18f2SGrygorii Strashko 				    CPSW_SL_CTL_MTEST |
153*a71a18f2SGrygorii Strashko 				    CPSW_SL_CTL_TX_SHORT_GAP_EN |
154*a71a18f2SGrygorii Strashko 				    CPSW_SL_CTL_CRC_TYPE |
155*a71a18f2SGrygorii Strashko 				    CPSW_SL_CTL_EXT_EN_RX_FLO |
156*a71a18f2SGrygorii Strashko 				    CPSW_SL_CTL_EXT_EN_TX_FLO |
157*a71a18f2SGrygorii Strashko 				    CPSW_SL_CTL_TX_SG_LIM_EN,
158*a71a18f2SGrygorii Strashko 		.idle_mask = CPSW_SL_STATUS_IDLE_MASK_BASE,
159*a71a18f2SGrygorii Strashko 	},
160*a71a18f2SGrygorii Strashko 	{
161*a71a18f2SGrygorii Strashko 		.device_id = "66ak2g",
162*a71a18f2SGrygorii Strashko 		.regs = cpsw_sl_reg_map_66ak2elg_am65,
163*a71a18f2SGrygorii Strashko 		.regs_offset = 0x330,
164*a71a18f2SGrygorii Strashko 		.control_features = CPSW_SL_CTL_FUNC_BASE |
165*a71a18f2SGrygorii Strashko 				    CPSW_SL_CTL_MTEST |
166*a71a18f2SGrygorii Strashko 				    CPSW_SL_CTL_CRC_TYPE |
167*a71a18f2SGrygorii Strashko 				    CPSW_SL_CTL_EXT_EN_RX_FLO |
168*a71a18f2SGrygorii Strashko 				    CPSW_SL_CTL_EXT_EN_TX_FLO,
169*a71a18f2SGrygorii Strashko 	},
170*a71a18f2SGrygorii Strashko 	{
171*a71a18f2SGrygorii Strashko 		.device_id = "am65",
172*a71a18f2SGrygorii Strashko 		.regs = cpsw_sl_reg_map_66ak2elg_am65,
173*a71a18f2SGrygorii Strashko 		.regs_offset = 0x330,
174*a71a18f2SGrygorii Strashko 		.control_features = CPSW_SL_CTL_FUNC_BASE |
175*a71a18f2SGrygorii Strashko 				    CPSW_SL_CTL_MTEST |
176*a71a18f2SGrygorii Strashko 				    CPSW_SL_CTL_XGIG |
177*a71a18f2SGrygorii Strashko 				    CPSW_SL_CTL_TX_SHORT_GAP_EN |
178*a71a18f2SGrygorii Strashko 				    CPSW_SL_CTL_CRC_TYPE |
179*a71a18f2SGrygorii Strashko 				    CPSW_SL_CTL_XGMII_EN |
180*a71a18f2SGrygorii Strashko 				    CPSW_SL_CTL_EXT_EN_RX_FLO |
181*a71a18f2SGrygorii Strashko 				    CPSW_SL_CTL_EXT_EN_TX_FLO |
182*a71a18f2SGrygorii Strashko 				    CPSW_SL_CTL_TX_SG_LIM_EN |
183*a71a18f2SGrygorii Strashko 				    CPSW_SL_CTL_EXT_EN_XGIG,
184*a71a18f2SGrygorii Strashko 		.idle_mask = CPSW_SL_STATUS_IDLE_MASK_K3,
185*a71a18f2SGrygorii Strashko 	},
186*a71a18f2SGrygorii Strashko 	{ },
187*a71a18f2SGrygorii Strashko };
188*a71a18f2SGrygorii Strashko 
cpsw_sl_reg_read(struct cpsw_sl * sl,enum cpsw_sl_regs reg)189*a71a18f2SGrygorii Strashko u32 cpsw_sl_reg_read(struct cpsw_sl *sl, enum cpsw_sl_regs reg)
190*a71a18f2SGrygorii Strashko {
191*a71a18f2SGrygorii Strashko 	int val;
192*a71a18f2SGrygorii Strashko 
193*a71a18f2SGrygorii Strashko 	if (sl->regs[reg] == CPSW_SL_REG_NOTUSED) {
194*a71a18f2SGrygorii Strashko 		dev_err(sl->dev, "cpsw_sl: not sup r reg: %04X\n",
195*a71a18f2SGrygorii Strashko 			sl->regs[reg]);
196*a71a18f2SGrygorii Strashko 		return 0;
197*a71a18f2SGrygorii Strashko 	}
198*a71a18f2SGrygorii Strashko 
199*a71a18f2SGrygorii Strashko 	val = readl(sl->sl_base + sl->regs[reg]);
200*a71a18f2SGrygorii Strashko 	dev_dbg(sl->dev, "cpsw_sl: reg: %04X r 0x%08X\n", sl->regs[reg], val);
201*a71a18f2SGrygorii Strashko 	return val;
202*a71a18f2SGrygorii Strashko }
203*a71a18f2SGrygorii Strashko 
cpsw_sl_reg_write(struct cpsw_sl * sl,enum cpsw_sl_regs reg,u32 val)204*a71a18f2SGrygorii Strashko void cpsw_sl_reg_write(struct cpsw_sl *sl, enum cpsw_sl_regs reg, u32 val)
205*a71a18f2SGrygorii Strashko {
206*a71a18f2SGrygorii Strashko 	if (sl->regs[reg] == CPSW_SL_REG_NOTUSED) {
207*a71a18f2SGrygorii Strashko 		dev_err(sl->dev, "cpsw_sl: not sup w reg: %04X\n",
208*a71a18f2SGrygorii Strashko 			sl->regs[reg]);
209*a71a18f2SGrygorii Strashko 		return;
210*a71a18f2SGrygorii Strashko 	}
211*a71a18f2SGrygorii Strashko 
212*a71a18f2SGrygorii Strashko 	dev_dbg(sl->dev, "cpsw_sl: reg: %04X w 0x%08X\n", sl->regs[reg], val);
213*a71a18f2SGrygorii Strashko 	writel(val, sl->sl_base + sl->regs[reg]);
214*a71a18f2SGrygorii Strashko }
215*a71a18f2SGrygorii Strashko 
cpsw_sl_match_id(const struct cpsw_sl_dev_id * id,const char * device_id)216*a71a18f2SGrygorii Strashko static const struct cpsw_sl_dev_id *cpsw_sl_match_id(
217*a71a18f2SGrygorii Strashko 		const struct cpsw_sl_dev_id *id,
218*a71a18f2SGrygorii Strashko 		const char *device_id)
219*a71a18f2SGrygorii Strashko {
220*a71a18f2SGrygorii Strashko 	if (!id || !device_id)
221*a71a18f2SGrygorii Strashko 		return NULL;
222*a71a18f2SGrygorii Strashko 
223*a71a18f2SGrygorii Strashko 	while (id->device_id) {
224*a71a18f2SGrygorii Strashko 		if (strcmp(device_id, id->device_id) == 0)
225*a71a18f2SGrygorii Strashko 			return id;
226*a71a18f2SGrygorii Strashko 		id++;
227*a71a18f2SGrygorii Strashko 	}
228*a71a18f2SGrygorii Strashko 	return NULL;
229*a71a18f2SGrygorii Strashko }
230*a71a18f2SGrygorii Strashko 
cpsw_sl_get(const char * device_id,struct device * dev,void __iomem * sl_base)231*a71a18f2SGrygorii Strashko struct cpsw_sl *cpsw_sl_get(const char *device_id, struct device *dev,
232*a71a18f2SGrygorii Strashko 			    void __iomem *sl_base)
233*a71a18f2SGrygorii Strashko {
234*a71a18f2SGrygorii Strashko 	const struct cpsw_sl_dev_id *sl_dev_id;
235*a71a18f2SGrygorii Strashko 	struct cpsw_sl *sl;
236*a71a18f2SGrygorii Strashko 
237*a71a18f2SGrygorii Strashko 	sl = devm_kzalloc(dev, sizeof(struct cpsw_sl), GFP_KERNEL);
238*a71a18f2SGrygorii Strashko 	if (!sl)
239*a71a18f2SGrygorii Strashko 		return ERR_PTR(-ENOMEM);
240*a71a18f2SGrygorii Strashko 	sl->dev = dev;
241*a71a18f2SGrygorii Strashko 	sl->sl_base = sl_base;
242*a71a18f2SGrygorii Strashko 
243*a71a18f2SGrygorii Strashko 	sl_dev_id = cpsw_sl_match_id(cpsw_sl_id_match, device_id);
244*a71a18f2SGrygorii Strashko 	if (!sl_dev_id) {
245*a71a18f2SGrygorii Strashko 		dev_err(sl->dev, "cpsw_sl: dev_id %s not found.\n", device_id);
246*a71a18f2SGrygorii Strashko 		return ERR_PTR(-EINVAL);
247*a71a18f2SGrygorii Strashko 	}
248*a71a18f2SGrygorii Strashko 	sl->regs = sl_dev_id->regs;
249*a71a18f2SGrygorii Strashko 	sl->control_features = sl_dev_id->control_features;
250*a71a18f2SGrygorii Strashko 	sl->idle_mask = sl_dev_id->idle_mask;
251*a71a18f2SGrygorii Strashko 	sl->sl_base += sl_dev_id->regs_offset;
252*a71a18f2SGrygorii Strashko 
253*a71a18f2SGrygorii Strashko 	return sl;
254*a71a18f2SGrygorii Strashko }
255*a71a18f2SGrygorii Strashko 
cpsw_sl_reset(struct cpsw_sl * sl,unsigned long tmo)256*a71a18f2SGrygorii Strashko void cpsw_sl_reset(struct cpsw_sl *sl, unsigned long tmo)
257*a71a18f2SGrygorii Strashko {
258*a71a18f2SGrygorii Strashko 	unsigned long timeout = jiffies + msecs_to_jiffies(tmo);
259*a71a18f2SGrygorii Strashko 
260*a71a18f2SGrygorii Strashko 	/* Set the soft reset bit */
261*a71a18f2SGrygorii Strashko 	cpsw_sl_reg_write(sl, CPSW_SL_SOFT_RESET, CPSW_SL_SOFT_RESET_BIT);
262*a71a18f2SGrygorii Strashko 
263*a71a18f2SGrygorii Strashko 	/* Wait for the bit to clear */
264*a71a18f2SGrygorii Strashko 	do {
265*a71a18f2SGrygorii Strashko 		usleep_range(100, 200);
266*a71a18f2SGrygorii Strashko 	} while ((cpsw_sl_reg_read(sl, CPSW_SL_SOFT_RESET) &
267*a71a18f2SGrygorii Strashko 		  CPSW_SL_SOFT_RESET_BIT) &&
268*a71a18f2SGrygorii Strashko 		  time_after(timeout, jiffies));
269*a71a18f2SGrygorii Strashko 
270*a71a18f2SGrygorii Strashko 	if (cpsw_sl_reg_read(sl, CPSW_SL_SOFT_RESET) & CPSW_SL_SOFT_RESET_BIT)
271*a71a18f2SGrygorii Strashko 		dev_err(sl->dev, "cpsw_sl failed to soft-reset.\n");
272*a71a18f2SGrygorii Strashko }
273*a71a18f2SGrygorii Strashko 
cpsw_sl_ctl_set(struct cpsw_sl * sl,u32 ctl_funcs)274*a71a18f2SGrygorii Strashko u32 cpsw_sl_ctl_set(struct cpsw_sl *sl, u32 ctl_funcs)
275*a71a18f2SGrygorii Strashko {
276*a71a18f2SGrygorii Strashko 	u32 val;
277*a71a18f2SGrygorii Strashko 
278*a71a18f2SGrygorii Strashko 	if (ctl_funcs & ~sl->control_features) {
279*a71a18f2SGrygorii Strashko 		dev_err(sl->dev, "cpsw_sl: unsupported func 0x%08X\n",
280*a71a18f2SGrygorii Strashko 			ctl_funcs & (~sl->control_features));
281*a71a18f2SGrygorii Strashko 		return -EINVAL;
282*a71a18f2SGrygorii Strashko 	}
283*a71a18f2SGrygorii Strashko 
284*a71a18f2SGrygorii Strashko 	val = cpsw_sl_reg_read(sl, CPSW_SL_MACCONTROL);
285*a71a18f2SGrygorii Strashko 	val |= ctl_funcs;
286*a71a18f2SGrygorii Strashko 	cpsw_sl_reg_write(sl, CPSW_SL_MACCONTROL, val);
287*a71a18f2SGrygorii Strashko 
288*a71a18f2SGrygorii Strashko 	return 0;
289*a71a18f2SGrygorii Strashko }
290*a71a18f2SGrygorii Strashko 
cpsw_sl_ctl_clr(struct cpsw_sl * sl,u32 ctl_funcs)291*a71a18f2SGrygorii Strashko u32 cpsw_sl_ctl_clr(struct cpsw_sl *sl, u32 ctl_funcs)
292*a71a18f2SGrygorii Strashko {
293*a71a18f2SGrygorii Strashko 	u32 val;
294*a71a18f2SGrygorii Strashko 
295*a71a18f2SGrygorii Strashko 	if (ctl_funcs & ~sl->control_features) {
296*a71a18f2SGrygorii Strashko 		dev_err(sl->dev, "cpsw_sl: unsupported func 0x%08X\n",
297*a71a18f2SGrygorii Strashko 			ctl_funcs & (~sl->control_features));
298*a71a18f2SGrygorii Strashko 		return -EINVAL;
299*a71a18f2SGrygorii Strashko 	}
300*a71a18f2SGrygorii Strashko 
301*a71a18f2SGrygorii Strashko 	val = cpsw_sl_reg_read(sl, CPSW_SL_MACCONTROL);
302*a71a18f2SGrygorii Strashko 	val &= ~ctl_funcs;
303*a71a18f2SGrygorii Strashko 	cpsw_sl_reg_write(sl, CPSW_SL_MACCONTROL, val);
304*a71a18f2SGrygorii Strashko 
305*a71a18f2SGrygorii Strashko 	return 0;
306*a71a18f2SGrygorii Strashko }
307*a71a18f2SGrygorii Strashko 
cpsw_sl_ctl_reset(struct cpsw_sl * sl)308*a71a18f2SGrygorii Strashko void cpsw_sl_ctl_reset(struct cpsw_sl *sl)
309*a71a18f2SGrygorii Strashko {
310*a71a18f2SGrygorii Strashko 	cpsw_sl_reg_write(sl, CPSW_SL_MACCONTROL, 0);
311*a71a18f2SGrygorii Strashko }
312*a71a18f2SGrygorii Strashko 
cpsw_sl_wait_for_idle(struct cpsw_sl * sl,unsigned long tmo)313*a71a18f2SGrygorii Strashko int cpsw_sl_wait_for_idle(struct cpsw_sl *sl, unsigned long tmo)
314*a71a18f2SGrygorii Strashko {
315*a71a18f2SGrygorii Strashko 	unsigned long timeout = jiffies + msecs_to_jiffies(tmo);
316*a71a18f2SGrygorii Strashko 
317*a71a18f2SGrygorii Strashko 	do {
318*a71a18f2SGrygorii Strashko 		usleep_range(100, 200);
319*a71a18f2SGrygorii Strashko 	} while (!(cpsw_sl_reg_read(sl, CPSW_SL_MACSTATUS) &
320*a71a18f2SGrygorii Strashko 		  sl->idle_mask) && time_after(timeout, jiffies));
321*a71a18f2SGrygorii Strashko 
322*a71a18f2SGrygorii Strashko 	if (!(cpsw_sl_reg_read(sl, CPSW_SL_MACSTATUS) & sl->idle_mask)) {
323*a71a18f2SGrygorii Strashko 		dev_err(sl->dev, "cpsw_sl failed to soft-reset.\n");
324*a71a18f2SGrygorii Strashko 		return -ETIMEDOUT;
325*a71a18f2SGrygorii Strashko 	}
326*a71a18f2SGrygorii Strashko 
327*a71a18f2SGrygorii Strashko 	return 0;
328*a71a18f2SGrygorii Strashko }
329