xref: /OK3568_Linux_fs/u-boot/board/gdsys/common/phy.c (revision 4882a59341e53eb6f0b4789bf948001014eff981)
1*4882a593Smuzhiyun /*
2*4882a593Smuzhiyun  * (C) Copyright 2014
3*4882a593Smuzhiyun  * Dirk Eibach,  Guntermann & Drunck GmbH, eibach@gdsys.de
4*4882a593Smuzhiyun  *
5*4882a593Smuzhiyun  * SPDX-License-Identifier:	GPL-2.0+
6*4882a593Smuzhiyun  */
7*4882a593Smuzhiyun 
8*4882a593Smuzhiyun #include <common.h>
9*4882a593Smuzhiyun 
10*4882a593Smuzhiyun #include <miiphy.h>
11*4882a593Smuzhiyun 
12*4882a593Smuzhiyun enum {
13*4882a593Smuzhiyun 	MIICMD_SET,
14*4882a593Smuzhiyun 	MIICMD_MODIFY,
15*4882a593Smuzhiyun 	MIICMD_VERIFY_VALUE,
16*4882a593Smuzhiyun 	MIICMD_WAIT_FOR_VALUE,
17*4882a593Smuzhiyun };
18*4882a593Smuzhiyun 
19*4882a593Smuzhiyun struct mii_setupcmd {
20*4882a593Smuzhiyun 	u8 token;
21*4882a593Smuzhiyun 	u8 reg;
22*4882a593Smuzhiyun 	u16 data;
23*4882a593Smuzhiyun 	u16 mask;
24*4882a593Smuzhiyun 	u32 timeout;
25*4882a593Smuzhiyun };
26*4882a593Smuzhiyun 
27*4882a593Smuzhiyun /*
28*4882a593Smuzhiyun  * verify we are talking to a 88e1518
29*4882a593Smuzhiyun  */
30*4882a593Smuzhiyun struct mii_setupcmd verify_88e1518[] = {
31*4882a593Smuzhiyun 	{ MIICMD_SET, 22, 0x0000 },
32*4882a593Smuzhiyun 	{ MIICMD_VERIFY_VALUE, 2, 0x0141, 0xffff },
33*4882a593Smuzhiyun 	{ MIICMD_VERIFY_VALUE, 3, 0x0dd0, 0xfff0 },
34*4882a593Smuzhiyun };
35*4882a593Smuzhiyun 
36*4882a593Smuzhiyun /*
37*4882a593Smuzhiyun  * workaround for erratum mentioned in 88E1518 release notes
38*4882a593Smuzhiyun  */
39*4882a593Smuzhiyun struct mii_setupcmd fixup_88e1518[] = {
40*4882a593Smuzhiyun 	{ MIICMD_SET, 22, 0x00ff },
41*4882a593Smuzhiyun 	{ MIICMD_SET, 17, 0x214b },
42*4882a593Smuzhiyun 	{ MIICMD_SET, 16, 0x2144 },
43*4882a593Smuzhiyun 	{ MIICMD_SET, 17, 0x0c28 },
44*4882a593Smuzhiyun 	{ MIICMD_SET, 16, 0x2146 },
45*4882a593Smuzhiyun 	{ MIICMD_SET, 17, 0xb233 },
46*4882a593Smuzhiyun 	{ MIICMD_SET, 16, 0x214d },
47*4882a593Smuzhiyun 	{ MIICMD_SET, 17, 0xcc0c },
48*4882a593Smuzhiyun 	{ MIICMD_SET, 16, 0x2159 },
49*4882a593Smuzhiyun 	{ MIICMD_SET, 22, 0x00fb },
50*4882a593Smuzhiyun 	{ MIICMD_SET,  7, 0xc00d },
51*4882a593Smuzhiyun 	{ MIICMD_SET, 22, 0x0000 },
52*4882a593Smuzhiyun };
53*4882a593Smuzhiyun 
54*4882a593Smuzhiyun /*
55*4882a593Smuzhiyun  * default initialization:
56*4882a593Smuzhiyun  * - set RGMII receive timing to "receive clock transition when data stable"
57*4882a593Smuzhiyun  * - set RGMII transmit timing to "transmit clock internally delayed"
58*4882a593Smuzhiyun  * - set RGMII output impedance target to 78,8 Ohm
59*4882a593Smuzhiyun  * - run output impedance calibration
60*4882a593Smuzhiyun  * - set autonegotiation advertise to 1000FD only
61*4882a593Smuzhiyun  */
62*4882a593Smuzhiyun struct mii_setupcmd default_88e1518[] = {
63*4882a593Smuzhiyun 	{ MIICMD_SET, 22, 0x0002 },
64*4882a593Smuzhiyun 	{ MIICMD_MODIFY, 21, 0x0030, 0x0030 },
65*4882a593Smuzhiyun 	{ MIICMD_MODIFY, 25, 0x0000, 0x0003 },
66*4882a593Smuzhiyun 	{ MIICMD_MODIFY, 24, 0x8000, 0x8000 },
67*4882a593Smuzhiyun 	{ MIICMD_WAIT_FOR_VALUE, 24, 0x4000, 0x4000, 2000 },
68*4882a593Smuzhiyun 	{ MIICMD_SET, 22, 0x0000 },
69*4882a593Smuzhiyun 	{ MIICMD_MODIFY, 4, 0x0000, 0x01e0 },
70*4882a593Smuzhiyun 	{ MIICMD_MODIFY, 9, 0x0200, 0x0300 },
71*4882a593Smuzhiyun };
72*4882a593Smuzhiyun 
73*4882a593Smuzhiyun /*
74*4882a593Smuzhiyun  * turn off CLK125 for PHY daughterboard
75*4882a593Smuzhiyun  */
76*4882a593Smuzhiyun struct mii_setupcmd ch1fix_88e1518[] = {
77*4882a593Smuzhiyun 	{ MIICMD_SET, 22, 0x0002 },
78*4882a593Smuzhiyun 	{ MIICMD_MODIFY, 16, 0x0006, 0x0006 },
79*4882a593Smuzhiyun 	{ MIICMD_SET, 22, 0x0000 },
80*4882a593Smuzhiyun };
81*4882a593Smuzhiyun 
82*4882a593Smuzhiyun /*
83*4882a593Smuzhiyun  * perform copper software reset
84*4882a593Smuzhiyun  */
85*4882a593Smuzhiyun struct mii_setupcmd swreset_88e1518[] = {
86*4882a593Smuzhiyun 	{ MIICMD_SET, 22, 0x0000 },
87*4882a593Smuzhiyun 	{ MIICMD_MODIFY, 0, 0x8000, 0x8000 },
88*4882a593Smuzhiyun 	{ MIICMD_WAIT_FOR_VALUE, 0, 0x0000, 0x8000, 2000 },
89*4882a593Smuzhiyun };
90*4882a593Smuzhiyun 
91*4882a593Smuzhiyun /*
92*4882a593Smuzhiyun  * special one for 88E1514:
93*4882a593Smuzhiyun  * Force SGMII to Copper mode
94*4882a593Smuzhiyun  */
95*4882a593Smuzhiyun struct mii_setupcmd mii_to_copper_88e1514[] = {
96*4882a593Smuzhiyun 	{ MIICMD_SET, 22, 0x0012 },
97*4882a593Smuzhiyun 	{ MIICMD_MODIFY, 20, 0x0001, 0x0007 },
98*4882a593Smuzhiyun 	{ MIICMD_MODIFY, 20, 0x8000, 0x8000 },
99*4882a593Smuzhiyun 	{ MIICMD_SET, 22, 0x0000 },
100*4882a593Smuzhiyun };
101*4882a593Smuzhiyun 
102*4882a593Smuzhiyun /*
103*4882a593Smuzhiyun  * turn off SGMII auto-negotiation
104*4882a593Smuzhiyun  */
105*4882a593Smuzhiyun struct mii_setupcmd sgmii_autoneg_off_88e1518[] = {
106*4882a593Smuzhiyun 	{ MIICMD_SET, 22, 0x0001 },
107*4882a593Smuzhiyun 	{ MIICMD_MODIFY, 0, 0x0000, 0x1000 },
108*4882a593Smuzhiyun 	{ MIICMD_MODIFY, 0, 0x8000, 0x8000 },
109*4882a593Smuzhiyun 	{ MIICMD_SET, 22, 0x0000 },
110*4882a593Smuzhiyun };
111*4882a593Smuzhiyun 
112*4882a593Smuzhiyun /*
113*4882a593Smuzhiyun  * invert LED2 polarity
114*4882a593Smuzhiyun  */
115*4882a593Smuzhiyun struct mii_setupcmd invert_led2_88e1514[] = {
116*4882a593Smuzhiyun 	{ MIICMD_SET, 22, 0x0003 },
117*4882a593Smuzhiyun 	{ MIICMD_MODIFY, 17, 0x0030, 0x0010 },
118*4882a593Smuzhiyun 	{ MIICMD_SET, 22, 0x0000 },
119*4882a593Smuzhiyun };
120*4882a593Smuzhiyun 
process_setupcmd(const char * bus,unsigned char addr,struct mii_setupcmd * setupcmd)121*4882a593Smuzhiyun static int process_setupcmd(const char *bus, unsigned char addr,
122*4882a593Smuzhiyun 			    struct mii_setupcmd *setupcmd)
123*4882a593Smuzhiyun {
124*4882a593Smuzhiyun 	int res;
125*4882a593Smuzhiyun 	u8 reg = setupcmd->reg;
126*4882a593Smuzhiyun 	u16 data = setupcmd->data;
127*4882a593Smuzhiyun 	u16 mask = setupcmd->mask;
128*4882a593Smuzhiyun 	u32 timeout = setupcmd->timeout;
129*4882a593Smuzhiyun 	u16 orig_data;
130*4882a593Smuzhiyun 	unsigned long start;
131*4882a593Smuzhiyun 
132*4882a593Smuzhiyun 	debug("mii %s:%u reg %2u ", bus, addr, reg);
133*4882a593Smuzhiyun 
134*4882a593Smuzhiyun 	switch (setupcmd->token) {
135*4882a593Smuzhiyun 	case MIICMD_MODIFY:
136*4882a593Smuzhiyun 		res = miiphy_read(bus, addr, reg, &orig_data);
137*4882a593Smuzhiyun 		if (res)
138*4882a593Smuzhiyun 			break;
139*4882a593Smuzhiyun 		debug("is %04x. (value %04x mask %04x) ", orig_data, data,
140*4882a593Smuzhiyun 		      mask);
141*4882a593Smuzhiyun 		data = (orig_data & ~mask) | (data & mask);
142*4882a593Smuzhiyun 		/* fallthrough */
143*4882a593Smuzhiyun 	case MIICMD_SET:
144*4882a593Smuzhiyun 		debug("=> %04x\n", data);
145*4882a593Smuzhiyun 		res = miiphy_write(bus, addr, reg, data);
146*4882a593Smuzhiyun 		break;
147*4882a593Smuzhiyun 	case MIICMD_VERIFY_VALUE:
148*4882a593Smuzhiyun 		res = miiphy_read(bus, addr, reg, &orig_data);
149*4882a593Smuzhiyun 		if (res)
150*4882a593Smuzhiyun 			break;
151*4882a593Smuzhiyun 		if ((orig_data & mask) != (data & mask))
152*4882a593Smuzhiyun 			res = -1;
153*4882a593Smuzhiyun 		debug("(value %04x mask %04x) == %04x? %s\n", data, mask,
154*4882a593Smuzhiyun 		      orig_data, res ? "FAIL" : "PASS");
155*4882a593Smuzhiyun 		break;
156*4882a593Smuzhiyun 	case MIICMD_WAIT_FOR_VALUE:
157*4882a593Smuzhiyun 		res = -1;
158*4882a593Smuzhiyun 		start = get_timer(0);
159*4882a593Smuzhiyun 		while ((res != 0) && (get_timer(start) < timeout)) {
160*4882a593Smuzhiyun 			res = miiphy_read(bus, addr, reg, &orig_data);
161*4882a593Smuzhiyun 			if (res)
162*4882a593Smuzhiyun 				continue;
163*4882a593Smuzhiyun 			if ((orig_data & mask) != (data & mask))
164*4882a593Smuzhiyun 				res = -1;
165*4882a593Smuzhiyun 		}
166*4882a593Smuzhiyun 		debug("(value %04x mask %04x) == %04x? %s after %lu ms\n", data,
167*4882a593Smuzhiyun 		      mask, orig_data, res ? "FAIL" : "PASS",
168*4882a593Smuzhiyun 		      get_timer(start));
169*4882a593Smuzhiyun 		break;
170*4882a593Smuzhiyun 	default:
171*4882a593Smuzhiyun 		res = -1;
172*4882a593Smuzhiyun 		break;
173*4882a593Smuzhiyun 	}
174*4882a593Smuzhiyun 
175*4882a593Smuzhiyun 	return res;
176*4882a593Smuzhiyun }
177*4882a593Smuzhiyun 
process_setup(const char * bus,unsigned char addr,struct mii_setupcmd * setupcmd,unsigned int count)178*4882a593Smuzhiyun static int process_setup(const char *bus, unsigned char addr,
179*4882a593Smuzhiyun 			    struct mii_setupcmd *setupcmd, unsigned int count)
180*4882a593Smuzhiyun {
181*4882a593Smuzhiyun 	int res = 0;
182*4882a593Smuzhiyun 	unsigned int k;
183*4882a593Smuzhiyun 
184*4882a593Smuzhiyun 	for (k = 0; k < count; ++k) {
185*4882a593Smuzhiyun 		res = process_setupcmd(bus, addr, &setupcmd[k]);
186*4882a593Smuzhiyun 		if (res) {
187*4882a593Smuzhiyun 			printf("mii cmd %u on bus %s addr %u failed, aborting setup\n",
188*4882a593Smuzhiyun 			       setupcmd[k].token, bus, addr);
189*4882a593Smuzhiyun 			break;
190*4882a593Smuzhiyun 		}
191*4882a593Smuzhiyun 	}
192*4882a593Smuzhiyun 
193*4882a593Smuzhiyun 	return res;
194*4882a593Smuzhiyun }
195*4882a593Smuzhiyun 
setup_88e1518(const char * bus,unsigned char addr)196*4882a593Smuzhiyun int setup_88e1518(const char *bus, unsigned char addr)
197*4882a593Smuzhiyun {
198*4882a593Smuzhiyun 	int res;
199*4882a593Smuzhiyun 
200*4882a593Smuzhiyun 	res = process_setup(bus, addr,
201*4882a593Smuzhiyun 			    verify_88e1518, ARRAY_SIZE(verify_88e1518));
202*4882a593Smuzhiyun 	if (res)
203*4882a593Smuzhiyun 		return res;
204*4882a593Smuzhiyun 
205*4882a593Smuzhiyun 	res = process_setup(bus, addr,
206*4882a593Smuzhiyun 			    fixup_88e1518, ARRAY_SIZE(fixup_88e1518));
207*4882a593Smuzhiyun 	if (res)
208*4882a593Smuzhiyun 		return res;
209*4882a593Smuzhiyun 
210*4882a593Smuzhiyun 	res = process_setup(bus, addr,
211*4882a593Smuzhiyun 			    default_88e1518, ARRAY_SIZE(default_88e1518));
212*4882a593Smuzhiyun 	if (res)
213*4882a593Smuzhiyun 		return res;
214*4882a593Smuzhiyun 
215*4882a593Smuzhiyun 	if (addr) {
216*4882a593Smuzhiyun 		res = process_setup(bus, addr,
217*4882a593Smuzhiyun 				    ch1fix_88e1518, ARRAY_SIZE(ch1fix_88e1518));
218*4882a593Smuzhiyun 		if (res)
219*4882a593Smuzhiyun 			return res;
220*4882a593Smuzhiyun 	}
221*4882a593Smuzhiyun 
222*4882a593Smuzhiyun 	res = process_setup(bus, addr,
223*4882a593Smuzhiyun 			    swreset_88e1518, ARRAY_SIZE(swreset_88e1518));
224*4882a593Smuzhiyun 	if (res)
225*4882a593Smuzhiyun 		return res;
226*4882a593Smuzhiyun 
227*4882a593Smuzhiyun 	return 0;
228*4882a593Smuzhiyun }
229*4882a593Smuzhiyun 
setup_88e1514(const char * bus,unsigned char addr)230*4882a593Smuzhiyun int setup_88e1514(const char *bus, unsigned char addr)
231*4882a593Smuzhiyun {
232*4882a593Smuzhiyun 	int res;
233*4882a593Smuzhiyun 
234*4882a593Smuzhiyun 	res = process_setup(bus, addr,
235*4882a593Smuzhiyun 			    verify_88e1518, ARRAY_SIZE(verify_88e1518));
236*4882a593Smuzhiyun 	if (res)
237*4882a593Smuzhiyun 		return res;
238*4882a593Smuzhiyun 
239*4882a593Smuzhiyun 	res = process_setup(bus, addr,
240*4882a593Smuzhiyun 			    fixup_88e1518, ARRAY_SIZE(fixup_88e1518));
241*4882a593Smuzhiyun 	if (res)
242*4882a593Smuzhiyun 		return res;
243*4882a593Smuzhiyun 
244*4882a593Smuzhiyun 	res = process_setup(bus, addr,
245*4882a593Smuzhiyun 			    mii_to_copper_88e1514,
246*4882a593Smuzhiyun 			    ARRAY_SIZE(mii_to_copper_88e1514));
247*4882a593Smuzhiyun 	if (res)
248*4882a593Smuzhiyun 		return res;
249*4882a593Smuzhiyun 
250*4882a593Smuzhiyun 	res = process_setup(bus, addr,
251*4882a593Smuzhiyun 			    sgmii_autoneg_off_88e1518,
252*4882a593Smuzhiyun 			    ARRAY_SIZE(sgmii_autoneg_off_88e1518));
253*4882a593Smuzhiyun 	if (res)
254*4882a593Smuzhiyun 		return res;
255*4882a593Smuzhiyun 
256*4882a593Smuzhiyun 	res = process_setup(bus, addr,
257*4882a593Smuzhiyun 			    invert_led2_88e1514,
258*4882a593Smuzhiyun 			    ARRAY_SIZE(invert_led2_88e1514));
259*4882a593Smuzhiyun 	if (res)
260*4882a593Smuzhiyun 		return res;
261*4882a593Smuzhiyun 
262*4882a593Smuzhiyun 	res = process_setup(bus, addr,
263*4882a593Smuzhiyun 			    default_88e1518, ARRAY_SIZE(default_88e1518));
264*4882a593Smuzhiyun 	if (res)
265*4882a593Smuzhiyun 		return res;
266*4882a593Smuzhiyun 
267*4882a593Smuzhiyun 	if (addr) {
268*4882a593Smuzhiyun 		res = process_setup(bus, addr,
269*4882a593Smuzhiyun 				    ch1fix_88e1518, ARRAY_SIZE(ch1fix_88e1518));
270*4882a593Smuzhiyun 		if (res)
271*4882a593Smuzhiyun 			return res;
272*4882a593Smuzhiyun 	}
273*4882a593Smuzhiyun 
274*4882a593Smuzhiyun 	res = process_setup(bus, addr,
275*4882a593Smuzhiyun 			    swreset_88e1518, ARRAY_SIZE(swreset_88e1518));
276*4882a593Smuzhiyun 	if (res)
277*4882a593Smuzhiyun 		return res;
278*4882a593Smuzhiyun 
279*4882a593Smuzhiyun 	return 0;
280*4882a593Smuzhiyun }
281