]>
Commit | Line | Data |
---|---|---|
42d1f039 | 1 | /* |
97d80fc3 | 2 | * Freescale Three Speed Ethernet Controller driver |
42d1f039 | 3 | * |
aec84bf6 | 4 | * Copyright 2004-2011, 2013 Freescale Semiconductor, Inc. |
42d1f039 | 5 | * (C) Copyright 2003, Motorola, Inc. |
42d1f039 WD |
6 | * author Andy Fleming |
7 | * | |
9872b736 | 8 | * SPDX-License-Identifier: GPL-2.0+ |
42d1f039 WD |
9 | */ |
10 | ||
11 | #include <config.h> | |
42d1f039 | 12 | #include <common.h> |
9a1d6af5 | 13 | #include <dm.h> |
42d1f039 WD |
14 | #include <malloc.h> |
15 | #include <net.h> | |
16 | #include <command.h> | |
dd3d1f56 | 17 | #include <tsec.h> |
063c1263 | 18 | #include <fsl_mdio.h> |
1221ce45 | 19 | #include <linux/errno.h> |
aada81de | 20 | #include <asm/processor.h> |
52d00a81 | 21 | #include <asm/io.h> |
42d1f039 | 22 | |
d87080b7 WD |
23 | DECLARE_GLOBAL_DATA_PTR; |
24 | ||
9a1d6af5 | 25 | #ifndef CONFIG_DM_ETH |
75b9d4ae AF |
26 | /* Default initializations for TSEC controllers. */ |
27 | ||
28 | static struct tsec_info_struct tsec_info[] = { | |
29 | #ifdef CONFIG_TSEC1 | |
30 | STD_TSEC_INFO(1), /* TSEC1 */ | |
31 | #endif | |
32 | #ifdef CONFIG_TSEC2 | |
33 | STD_TSEC_INFO(2), /* TSEC2 */ | |
34 | #endif | |
35 | #ifdef CONFIG_MPC85XX_FEC | |
36 | { | |
aec84bf6 | 37 | .regs = TSEC_GET_REGS(2, 0x2000), |
75b9d4ae AF |
38 | .devname = CONFIG_MPC85XX_FEC_NAME, |
39 | .phyaddr = FEC_PHY_ADDR, | |
063c1263 AF |
40 | .flags = FEC_FLAGS, |
41 | .mii_devname = DEFAULT_MII_NAME | |
75b9d4ae AF |
42 | }, /* FEC */ |
43 | #endif | |
44 | #ifdef CONFIG_TSEC3 | |
45 | STD_TSEC_INFO(3), /* TSEC3 */ | |
46 | #endif | |
47 | #ifdef CONFIG_TSEC4 | |
48 | STD_TSEC_INFO(4), /* TSEC4 */ | |
49 | #endif | |
50 | }; | |
9a1d6af5 | 51 | #endif /* CONFIG_DM_ETH */ |
75b9d4ae | 52 | |
2abe361c AF |
53 | #define TBIANA_SETTINGS ( \ |
54 | TBIANA_ASYMMETRIC_PAUSE \ | |
55 | | TBIANA_SYMMETRIC_PAUSE \ | |
56 | | TBIANA_FULL_DUPLEX \ | |
57 | ) | |
58 | ||
90b5bf21 FR |
59 | /* By default force the TBI PHY into 1000Mbps full duplex when in SGMII mode */ |
60 | #ifndef CONFIG_TSEC_TBICR_SETTINGS | |
72c96a68 | 61 | #define CONFIG_TSEC_TBICR_SETTINGS ( \ |
2abe361c | 62 | TBICR_PHY_RESET \ |
72c96a68 | 63 | | TBICR_ANEG_ENABLE \ |
2abe361c AF |
64 | | TBICR_FULL_DUPLEX \ |
65 | | TBICR_SPEED1_SET \ | |
66 | ) | |
90b5bf21 | 67 | #endif /* CONFIG_TSEC_TBICR_SETTINGS */ |
46e91674 | 68 | |
2abe361c AF |
69 | /* Configure the TBI for SGMII operation */ |
70 | static void tsec_configure_serdes(struct tsec_private *priv) | |
71 | { | |
9872b736 BM |
72 | /* |
73 | * Access TBI PHY registers at given TSEC register offset as opposed | |
74 | * to the register offset used for external PHY accesses | |
75 | */ | |
063c1263 | 76 | tsec_local_mdio_write(priv->phyregs_sgmii, in_be32(&priv->regs->tbipa), |
d38de338 | 77 | 0, TBI_ANA, TBIANA_SETTINGS); |
063c1263 | 78 | tsec_local_mdio_write(priv->phyregs_sgmii, in_be32(&priv->regs->tbipa), |
d38de338 | 79 | 0, TBI_TBICON, TBICON_CLK_SELECT); |
063c1263 | 80 | tsec_local_mdio_write(priv->phyregs_sgmii, in_be32(&priv->regs->tbipa), |
d38de338 | 81 | 0, TBI_CR, CONFIG_TSEC_TBICR_SETTINGS); |
97d80fc3 | 82 | } |
42d1f039 | 83 | |
90751910 MH |
84 | #ifdef CONFIG_MCAST_TFTP |
85 | ||
86 | /* CREDITS: linux gianfar driver, slightly adjusted... thanx. */ | |
87 | ||
88 | /* Set the appropriate hash bit for the given addr */ | |
89 | ||
9872b736 BM |
90 | /* |
91 | * The algorithm works like so: | |
90751910 MH |
92 | * 1) Take the Destination Address (ie the multicast address), and |
93 | * do a CRC on it (little endian), and reverse the bits of the | |
94 | * result. | |
95 | * 2) Use the 8 most significant bits as a hash into a 256-entry | |
96 | * table. The table is controlled through 8 32-bit registers: | |
876d4515 CM |
97 | * gaddr0-7. gaddr0's MSB is entry 0, and gaddr7's LSB is entry |
98 | * 255. This means that the 3 most significant bits in the | |
90751910 MH |
99 | * hash index which gaddr register to use, and the 5 other bits |
100 | * indicate which bit (assuming an IBM numbering scheme, which | |
876d4515 | 101 | * for PowerPC (tm) is usually the case) in the register holds |
9872b736 BM |
102 | * the entry. |
103 | */ | |
9a1d6af5 | 104 | #ifndef CONFIG_DM_ETH |
9872b736 | 105 | static int tsec_mcast_addr(struct eth_device *dev, const u8 *mcast_mac, u8 set) |
9a1d6af5 BM |
106 | #else |
107 | static int tsec_mcast_addr(struct udevice *dev, const u8 *mcast_mac, int set) | |
108 | #endif | |
90751910 | 109 | { |
b200204e | 110 | struct tsec_private *priv = (struct tsec_private *)dev->priv; |
876d4515 CM |
111 | struct tsec __iomem *regs = priv->regs; |
112 | u32 result, value; | |
113 | u8 whichbit, whichreg; | |
114 | ||
115 | result = ether_crc(MAC_ADDR_LEN, mcast_mac); | |
116 | whichbit = (result >> 24) & 0x1f; /* the 5 LSB = which bit to set */ | |
117 | whichreg = result >> 29; /* the 3 MSB = which reg to set it in */ | |
118 | ||
d38de338 | 119 | value = BIT(31 - whichbit); |
876d4515 CM |
120 | |
121 | if (set) | |
122 | setbits_be32(®s->hash.gaddr0 + whichreg, value); | |
123 | else | |
124 | clrbits_be32(®s->hash.gaddr0 + whichreg, value); | |
125 | ||
90751910 MH |
126 | return 0; |
127 | } | |
128 | #endif /* Multicast TFTP ? */ | |
129 | ||
9872b736 BM |
130 | /* |
131 | * Initialized required registers to appropriate values, zeroing | |
90751910 MH |
132 | * those we don't care about (unless zero is bad, in which case, |
133 | * choose a more appropriate value) | |
134 | */ | |
aec84bf6 | 135 | static void init_registers(struct tsec __iomem *regs) |
90751910 MH |
136 | { |
137 | /* Clear IEVENT */ | |
138 | out_be32(®s->ievent, IEVENT_INIT_CLEAR); | |
139 | ||
140 | out_be32(®s->imask, IMASK_INIT_CLEAR); | |
141 | ||
142 | out_be32(®s->hash.iaddr0, 0); | |
143 | out_be32(®s->hash.iaddr1, 0); | |
144 | out_be32(®s->hash.iaddr2, 0); | |
145 | out_be32(®s->hash.iaddr3, 0); | |
146 | out_be32(®s->hash.iaddr4, 0); | |
147 | out_be32(®s->hash.iaddr5, 0); | |
148 | out_be32(®s->hash.iaddr6, 0); | |
149 | out_be32(®s->hash.iaddr7, 0); | |
150 | ||
151 | out_be32(®s->hash.gaddr0, 0); | |
152 | out_be32(®s->hash.gaddr1, 0); | |
153 | out_be32(®s->hash.gaddr2, 0); | |
154 | out_be32(®s->hash.gaddr3, 0); | |
155 | out_be32(®s->hash.gaddr4, 0); | |
156 | out_be32(®s->hash.gaddr5, 0); | |
157 | out_be32(®s->hash.gaddr6, 0); | |
158 | out_be32(®s->hash.gaddr7, 0); | |
159 | ||
160 | out_be32(®s->rctrl, 0x00000000); | |
161 | ||
162 | /* Init RMON mib registers */ | |
82ef75ca | 163 | memset((void *)®s->rmon, 0, sizeof(regs->rmon)); |
90751910 MH |
164 | |
165 | out_be32(®s->rmon.cam1, 0xffffffff); | |
166 | out_be32(®s->rmon.cam2, 0xffffffff); | |
167 | ||
168 | out_be32(®s->mrblr, MRBLR_INIT_SETTINGS); | |
169 | ||
170 | out_be32(®s->minflr, MINFLR_INIT_SETTINGS); | |
171 | ||
172 | out_be32(®s->attr, ATTR_INIT_SETTINGS); | |
173 | out_be32(®s->attreli, ATTRELI_INIT_SETTINGS); | |
90751910 MH |
174 | } |
175 | ||
9872b736 BM |
176 | /* |
177 | * Configure maccfg2 based on negotiated speed and duplex | |
90751910 MH |
178 | * reported by PHY handling code |
179 | */ | |
063c1263 | 180 | static void adjust_link(struct tsec_private *priv, struct phy_device *phydev) |
90751910 | 181 | { |
aec84bf6 | 182 | struct tsec __iomem *regs = priv->regs; |
90751910 MH |
183 | u32 ecntrl, maccfg2; |
184 | ||
063c1263 AF |
185 | if (!phydev->link) { |
186 | printf("%s: No link.\n", phydev->dev->name); | |
90751910 MH |
187 | return; |
188 | } | |
189 | ||
190 | /* clear all bits relative with interface mode */ | |
191 | ecntrl = in_be32(®s->ecntrl); | |
192 | ecntrl &= ~ECNTRL_R100; | |
193 | ||
194 | maccfg2 = in_be32(®s->maccfg2); | |
195 | maccfg2 &= ~(MACCFG2_IF | MACCFG2_FULL_DUPLEX); | |
196 | ||
063c1263 | 197 | if (phydev->duplex) |
90751910 MH |
198 | maccfg2 |= MACCFG2_FULL_DUPLEX; |
199 | ||
063c1263 | 200 | switch (phydev->speed) { |
90751910 MH |
201 | case 1000: |
202 | maccfg2 |= MACCFG2_GMII; | |
203 | break; | |
204 | case 100: | |
205 | case 10: | |
206 | maccfg2 |= MACCFG2_MII; | |
207 | ||
9872b736 BM |
208 | /* |
209 | * Set R100 bit in all modes although | |
90751910 MH |
210 | * it is only used in RGMII mode |
211 | */ | |
063c1263 | 212 | if (phydev->speed == 100) |
90751910 MH |
213 | ecntrl |= ECNTRL_R100; |
214 | break; | |
215 | default: | |
063c1263 | 216 | printf("%s: Speed was bad\n", phydev->dev->name); |
90751910 MH |
217 | break; |
218 | } | |
219 | ||
220 | out_be32(®s->ecntrl, ecntrl); | |
221 | out_be32(®s->maccfg2, maccfg2); | |
3dd7f0f0 | 222 | |
063c1263 | 223 | printf("Speed: %d, %s duplex%s\n", phydev->speed, |
d38de338 MS |
224 | (phydev->duplex) ? "full" : "half", |
225 | (phydev->port == PORT_FIBRE) ? ", fiber mode" : ""); | |
90751910 | 226 | } |
9d46ea4a | 227 | |
8ba50176 BM |
228 | /* |
229 | * This returns the status bits of the device. The return value | |
230 | * is never checked, and this is what the 8260 driver did, so we | |
231 | * do the same. Presumably, this would be zero if there were no | |
232 | * errors | |
233 | */ | |
9a1d6af5 | 234 | #ifndef CONFIG_DM_ETH |
8ba50176 | 235 | static int tsec_send(struct eth_device *dev, void *packet, int length) |
9a1d6af5 BM |
236 | #else |
237 | static int tsec_send(struct udevice *dev, void *packet, int length) | |
238 | #endif | |
8ba50176 BM |
239 | { |
240 | struct tsec_private *priv = (struct tsec_private *)dev->priv; | |
241 | struct tsec __iomem *regs = priv->regs; | |
d38de338 | 242 | u16 status; |
8ba50176 BM |
243 | int result = 0; |
244 | int i; | |
245 | ||
246 | /* Find an empty buffer descriptor */ | |
247 | for (i = 0; | |
248 | in_be16(&priv->txbd[priv->tx_idx].status) & TXBD_READY; | |
249 | i++) { | |
250 | if (i >= TOUT_LOOP) { | |
251 | debug("%s: tsec: tx buffers full\n", dev->name); | |
252 | return result; | |
253 | } | |
254 | } | |
255 | ||
256 | out_be32(&priv->txbd[priv->tx_idx].bufptr, (u32)packet); | |
257 | out_be16(&priv->txbd[priv->tx_idx].length, length); | |
258 | status = in_be16(&priv->txbd[priv->tx_idx].status); | |
259 | out_be16(&priv->txbd[priv->tx_idx].status, status | | |
260 | (TXBD_READY | TXBD_LAST | TXBD_CRC | TXBD_INTERRUPT)); | |
261 | ||
262 | /* Tell the DMA to go */ | |
263 | out_be32(®s->tstat, TSTAT_CLEAR_THALT); | |
264 | ||
265 | /* Wait for buffer to be transmitted */ | |
266 | for (i = 0; | |
267 | in_be16(&priv->txbd[priv->tx_idx].status) & TXBD_READY; | |
268 | i++) { | |
269 | if (i >= TOUT_LOOP) { | |
270 | debug("%s: tsec: tx error\n", dev->name); | |
271 | return result; | |
272 | } | |
273 | } | |
274 | ||
275 | priv->tx_idx = (priv->tx_idx + 1) % TX_BUF_CNT; | |
276 | result = in_be16(&priv->txbd[priv->tx_idx].status) & TXBD_STATS; | |
277 | ||
278 | return result; | |
279 | } | |
280 | ||
9a1d6af5 | 281 | #ifndef CONFIG_DM_ETH |
8ba50176 BM |
282 | static int tsec_recv(struct eth_device *dev) |
283 | { | |
284 | struct tsec_private *priv = (struct tsec_private *)dev->priv; | |
285 | struct tsec __iomem *regs = priv->regs; | |
286 | ||
287 | while (!(in_be16(&priv->rxbd[priv->rx_idx].status) & RXBD_EMPTY)) { | |
288 | int length = in_be16(&priv->rxbd[priv->rx_idx].length); | |
d38de338 | 289 | u16 status = in_be16(&priv->rxbd[priv->rx_idx].status); |
8ba50176 BM |
290 | uchar *packet = net_rx_packets[priv->rx_idx]; |
291 | ||
292 | /* Send the packet up if there were no errors */ | |
293 | if (!(status & RXBD_STATS)) | |
294 | net_process_received_packet(packet, length - 4); | |
295 | else | |
296 | printf("Got error %x\n", (status & RXBD_STATS)); | |
297 | ||
298 | out_be16(&priv->rxbd[priv->rx_idx].length, 0); | |
299 | ||
300 | status = RXBD_EMPTY; | |
301 | /* Set the wrap bit if this is the last element in the list */ | |
302 | if ((priv->rx_idx + 1) == PKTBUFSRX) | |
303 | status |= RXBD_WRAP; | |
304 | out_be16(&priv->rxbd[priv->rx_idx].status, status); | |
305 | ||
306 | priv->rx_idx = (priv->rx_idx + 1) % PKTBUFSRX; | |
307 | } | |
308 | ||
309 | if (in_be32(®s->ievent) & IEVENT_BSY) { | |
310 | out_be32(®s->ievent, IEVENT_BSY); | |
311 | out_be32(®s->rstat, RSTAT_CLEAR_RHALT); | |
312 | } | |
313 | ||
314 | return -1; | |
315 | } | |
9a1d6af5 BM |
316 | #else |
317 | static int tsec_recv(struct udevice *dev, int flags, uchar **packetp) | |
318 | { | |
319 | struct tsec_private *priv = (struct tsec_private *)dev->priv; | |
320 | struct tsec __iomem *regs = priv->regs; | |
321 | int ret = -1; | |
322 | ||
323 | if (!(in_be16(&priv->rxbd[priv->rx_idx].status) & RXBD_EMPTY)) { | |
324 | int length = in_be16(&priv->rxbd[priv->rx_idx].length); | |
d38de338 MS |
325 | u16 status = in_be16(&priv->rxbd[priv->rx_idx].status); |
326 | u32 buf; | |
9a1d6af5 BM |
327 | |
328 | /* Send the packet up if there were no errors */ | |
329 | if (!(status & RXBD_STATS)) { | |
330 | buf = in_be32(&priv->rxbd[priv->rx_idx].bufptr); | |
331 | *packetp = (uchar *)buf; | |
332 | ret = length - 4; | |
333 | } else { | |
334 | printf("Got error %x\n", (status & RXBD_STATS)); | |
335 | } | |
336 | } | |
337 | ||
338 | if (in_be32(®s->ievent) & IEVENT_BSY) { | |
339 | out_be32(®s->ievent, IEVENT_BSY); | |
340 | out_be32(®s->rstat, RSTAT_CLEAR_RHALT); | |
341 | } | |
342 | ||
343 | return ret; | |
344 | } | |
345 | ||
346 | static int tsec_free_pkt(struct udevice *dev, uchar *packet, int length) | |
347 | { | |
348 | struct tsec_private *priv = (struct tsec_private *)dev->priv; | |
d38de338 | 349 | u16 status; |
9a1d6af5 BM |
350 | |
351 | out_be16(&priv->rxbd[priv->rx_idx].length, 0); | |
352 | ||
353 | status = RXBD_EMPTY; | |
354 | /* Set the wrap bit if this is the last element in the list */ | |
355 | if ((priv->rx_idx + 1) == PKTBUFSRX) | |
356 | status |= RXBD_WRAP; | |
357 | out_be16(&priv->rxbd[priv->rx_idx].status, status); | |
358 | ||
359 | priv->rx_idx = (priv->rx_idx + 1) % PKTBUFSRX; | |
360 | ||
361 | return 0; | |
362 | } | |
363 | #endif | |
8ba50176 BM |
364 | |
365 | /* Stop the interface */ | |
9a1d6af5 | 366 | #ifndef CONFIG_DM_ETH |
8ba50176 | 367 | static void tsec_halt(struct eth_device *dev) |
9a1d6af5 BM |
368 | #else |
369 | static void tsec_halt(struct udevice *dev) | |
370 | #endif | |
8ba50176 BM |
371 | { |
372 | struct tsec_private *priv = (struct tsec_private *)dev->priv; | |
373 | struct tsec __iomem *regs = priv->regs; | |
374 | ||
375 | clrbits_be32(®s->dmactrl, DMACTRL_GRS | DMACTRL_GTS); | |
376 | setbits_be32(®s->dmactrl, DMACTRL_GRS | DMACTRL_GTS); | |
377 | ||
378 | while ((in_be32(®s->ievent) & (IEVENT_GRSC | IEVENT_GTSC)) | |
379 | != (IEVENT_GRSC | IEVENT_GTSC)) | |
380 | ; | |
381 | ||
382 | clrbits_be32(®s->maccfg1, MACCFG1_TX_EN | MACCFG1_RX_EN); | |
383 | ||
384 | /* Shut down the PHY, as needed */ | |
385 | phy_shutdown(priv->phydev); | |
386 | } | |
387 | ||
aada81de | 388 | #ifdef CONFIG_SYS_FSL_ERRATUM_NMG_ETSEC129 |
389 | /* | |
390 | * When MACCFG1[Rx_EN] is enabled during system boot as part | |
391 | * of the eTSEC port initialization sequence, | |
392 | * the eTSEC Rx logic may not be properly initialized. | |
393 | */ | |
56a27a1e | 394 | void redundant_init(struct tsec_private *priv) |
aada81de | 395 | { |
aec84bf6 | 396 | struct tsec __iomem *regs = priv->regs; |
aada81de | 397 | uint t, count = 0; |
398 | int fail = 1; | |
399 | static const u8 pkt[] = { | |
400 | 0x00, 0x1e, 0x4f, 0x12, 0xcb, 0x2c, 0x00, 0x25, | |
401 | 0x64, 0xbb, 0xd1, 0xab, 0x08, 0x00, 0x45, 0x00, | |
402 | 0x00, 0x5c, 0xdd, 0x22, 0x00, 0x00, 0x80, 0x01, | |
403 | 0x1f, 0x71, 0x0a, 0xc1, 0x14, 0x22, 0x0a, 0xc1, | |
404 | 0x14, 0x6a, 0x08, 0x00, 0xef, 0x7e, 0x02, 0x00, | |
405 | 0x94, 0x05, 0x61, 0x62, 0x63, 0x64, 0x65, 0x66, | |
406 | 0x67, 0x68, 0x69, 0x6a, 0x6b, 0x6c, 0x6d, 0x6e, | |
407 | 0x6f, 0x70, 0x71, 0x72, 0x73, 0x74, 0x75, 0x76, | |
408 | 0x77, 0x61, 0x62, 0x63, 0x64, 0x65, 0x66, 0x67, | |
409 | 0x68, 0x69, 0x6a, 0x6b, 0x6c, 0x6d, 0x6e, 0x6f, | |
410 | 0x70, 0x71, 0x72, 0x73, 0x74, 0x75, 0x76, 0x77, | |
411 | 0x61, 0x62, 0x63, 0x64, 0x65, 0x66, 0x67, 0x68, | |
412 | 0x69, 0x6a, 0x6b, 0x6c, 0x6d, 0x6e, 0x6f, 0x70, | |
413 | 0x71, 0x72}; | |
414 | ||
415 | /* Enable promiscuous mode */ | |
416 | setbits_be32(®s->rctrl, 0x8); | |
417 | /* Enable loopback mode */ | |
418 | setbits_be32(®s->maccfg1, MACCFG1_LOOPBACK); | |
419 | /* Enable transmit and receive */ | |
420 | setbits_be32(®s->maccfg1, MACCFG1_RX_EN | MACCFG1_TX_EN); | |
421 | ||
422 | /* Tell the DMA it is clear to go */ | |
423 | setbits_be32(®s->dmactrl, DMACTRL_INIT_SETTINGS); | |
424 | out_be32(®s->tstat, TSTAT_CLEAR_THALT); | |
425 | out_be32(®s->rstat, RSTAT_CLEAR_RHALT); | |
426 | clrbits_be32(®s->dmactrl, DMACTRL_GRS | DMACTRL_GTS); | |
427 | ||
428 | do { | |
d38de338 MS |
429 | u16 status; |
430 | ||
56a27a1e | 431 | tsec_send(priv->dev, (void *)pkt, sizeof(pkt)); |
aada81de | 432 | |
433 | /* Wait for buffer to be received */ | |
e677da97 BM |
434 | for (t = 0; |
435 | in_be16(&priv->rxbd[priv->rx_idx].status) & RXBD_EMPTY; | |
362b123f | 436 | t++) { |
aada81de | 437 | if (t >= 10 * TOUT_LOOP) { |
56a27a1e | 438 | printf("%s: tsec: rx error\n", priv->dev->name); |
aada81de | 439 | break; |
440 | } | |
441 | } | |
442 | ||
362b123f | 443 | if (!memcmp(pkt, net_rx_packets[priv->rx_idx], sizeof(pkt))) |
aada81de | 444 | fail = 0; |
445 | ||
e677da97 | 446 | out_be16(&priv->rxbd[priv->rx_idx].length, 0); |
9c9141fd | 447 | status = RXBD_EMPTY; |
362b123f | 448 | if ((priv->rx_idx + 1) == PKTBUFSRX) |
9c9141fd | 449 | status |= RXBD_WRAP; |
e677da97 | 450 | out_be16(&priv->rxbd[priv->rx_idx].status, status); |
362b123f | 451 | priv->rx_idx = (priv->rx_idx + 1) % PKTBUFSRX; |
aada81de | 452 | |
453 | if (in_be32(®s->ievent) & IEVENT_BSY) { | |
454 | out_be32(®s->ievent, IEVENT_BSY); | |
455 | out_be32(®s->rstat, RSTAT_CLEAR_RHALT); | |
456 | } | |
457 | if (fail) { | |
458 | printf("loopback recv packet error!\n"); | |
459 | clrbits_be32(®s->maccfg1, MACCFG1_RX_EN); | |
460 | udelay(1000); | |
461 | setbits_be32(®s->maccfg1, MACCFG1_RX_EN); | |
462 | } | |
463 | } while ((count++ < 4) && (fail == 1)); | |
464 | ||
465 | if (fail) | |
466 | panic("eTSEC init fail!\n"); | |
467 | /* Disable promiscuous mode */ | |
468 | clrbits_be32(®s->rctrl, 0x8); | |
469 | /* Disable loopback mode */ | |
470 | clrbits_be32(®s->maccfg1, MACCFG1_LOOPBACK); | |
471 | } | |
472 | #endif | |
473 | ||
9872b736 BM |
474 | /* |
475 | * Set up the buffers and their descriptors, and bring up the | |
90751910 | 476 | * interface |
89875e96 | 477 | */ |
56a27a1e | 478 | static void startup_tsec(struct tsec_private *priv) |
be5048f1 | 479 | { |
aec84bf6 | 480 | struct tsec __iomem *regs = priv->regs; |
d38de338 | 481 | u16 status; |
9c9141fd | 482 | int i; |
be5048f1 | 483 | |
063c1263 | 484 | /* reset the indices to zero */ |
362b123f BM |
485 | priv->rx_idx = 0; |
486 | priv->tx_idx = 0; | |
aada81de | 487 | #ifdef CONFIG_SYS_FSL_ERRATUM_NMG_ETSEC129 |
488 | uint svr; | |
489 | #endif | |
063c1263 | 490 | |
90751910 | 491 | /* Point to the buffer descriptors */ |
e677da97 BM |
492 | out_be32(®s->tbase, (u32)&priv->txbd[0]); |
493 | out_be32(®s->rbase, (u32)&priv->rxbd[0]); | |
be5048f1 | 494 | |
90751910 MH |
495 | /* Initialize the Rx Buffer descriptors */ |
496 | for (i = 0; i < PKTBUFSRX; i++) { | |
e677da97 BM |
497 | out_be16(&priv->rxbd[i].status, RXBD_EMPTY); |
498 | out_be16(&priv->rxbd[i].length, 0); | |
499 | out_be32(&priv->rxbd[i].bufptr, (u32)net_rx_packets[i]); | |
90751910 | 500 | } |
e677da97 BM |
501 | status = in_be16(&priv->rxbd[PKTBUFSRX - 1].status); |
502 | out_be16(&priv->rxbd[PKTBUFSRX - 1].status, status | RXBD_WRAP); | |
be5048f1 | 503 | |
90751910 MH |
504 | /* Initialize the TX Buffer Descriptors */ |
505 | for (i = 0; i < TX_BUF_CNT; i++) { | |
e677da97 BM |
506 | out_be16(&priv->txbd[i].status, 0); |
507 | out_be16(&priv->txbd[i].length, 0); | |
508 | out_be32(&priv->txbd[i].bufptr, 0); | |
be5048f1 | 509 | } |
e677da97 BM |
510 | status = in_be16(&priv->txbd[TX_BUF_CNT - 1].status); |
511 | out_be16(&priv->txbd[TX_BUF_CNT - 1].status, status | TXBD_WRAP); | |
be5048f1 | 512 | |
aada81de | 513 | #ifdef CONFIG_SYS_FSL_ERRATUM_NMG_ETSEC129 |
514 | svr = get_svr(); | |
515 | if ((SVR_MAJ(svr) == 1) || IS_SVR_REV(svr, 2, 0)) | |
56a27a1e | 516 | redundant_init(priv); |
aada81de | 517 | #endif |
90751910 MH |
518 | /* Enable Transmit and Receive */ |
519 | setbits_be32(®s->maccfg1, MACCFG1_RX_EN | MACCFG1_TX_EN); | |
520 | ||
521 | /* Tell the DMA it is clear to go */ | |
522 | setbits_be32(®s->dmactrl, DMACTRL_INIT_SETTINGS); | |
523 | out_be32(®s->tstat, TSTAT_CLEAR_THALT); | |
524 | out_be32(®s->rstat, RSTAT_CLEAR_RHALT); | |
525 | clrbits_be32(®s->dmactrl, DMACTRL_GRS | DMACTRL_GTS); | |
be5048f1 WD |
526 | } |
527 | ||
9872b736 BM |
528 | /* |
529 | * Initializes data structures and registers for the controller, | |
530 | * and brings the interface up. Returns the link status, meaning | |
90751910 | 531 | * that it returns success if the link is up, failure otherwise. |
9872b736 | 532 | * This allows U-Boot to find the first active controller. |
89875e96 | 533 | */ |
9a1d6af5 | 534 | #ifndef CONFIG_DM_ETH |
d38de338 | 535 | static int tsec_init(struct eth_device *dev, bd_t *bd) |
9a1d6af5 BM |
536 | #else |
537 | static int tsec_init(struct udevice *dev) | |
538 | #endif | |
97d80fc3 | 539 | { |
90751910 | 540 | struct tsec_private *priv = (struct tsec_private *)dev->priv; |
9a1d6af5 BM |
541 | #ifdef CONFIG_DM_ETH |
542 | struct eth_pdata *pdata = dev_get_platdata(dev); | |
543 | #endif | |
aec84bf6 | 544 | struct tsec __iomem *regs = priv->regs; |
b1690bc3 | 545 | u32 tempval; |
11af8d65 | 546 | int ret; |
97d80fc3 | 547 | |
90751910 MH |
548 | /* Make sure the controller is stopped */ |
549 | tsec_halt(dev); | |
97d80fc3 | 550 | |
90751910 MH |
551 | /* Init MACCFG2. Defaults to GMII */ |
552 | out_be32(®s->maccfg2, MACCFG2_INIT_SETTINGS); | |
97d80fc3 | 553 | |
90751910 MH |
554 | /* Init ECNTRL */ |
555 | out_be32(®s->ecntrl, ECNTRL_INIT_SETTINGS); | |
97d80fc3 | 556 | |
9872b736 BM |
557 | /* |
558 | * Copy the station address into the address registers. | |
b1690bc3 CM |
559 | * For a station address of 0x12345678ABCD in transmission |
560 | * order (BE), MACnADDR1 is set to 0xCDAB7856 and | |
561 | * MACnADDR2 is set to 0x34120000. | |
562 | */ | |
9a1d6af5 | 563 | #ifndef CONFIG_DM_ETH |
b1690bc3 CM |
564 | tempval = (dev->enetaddr[5] << 24) | (dev->enetaddr[4] << 16) | |
565 | (dev->enetaddr[3] << 8) | dev->enetaddr[2]; | |
9a1d6af5 BM |
566 | #else |
567 | tempval = (pdata->enetaddr[5] << 24) | (pdata->enetaddr[4] << 16) | | |
568 | (pdata->enetaddr[3] << 8) | pdata->enetaddr[2]; | |
569 | #endif | |
97d80fc3 | 570 | |
90751910 | 571 | out_be32(®s->macstnaddr1, tempval); |
97d80fc3 | 572 | |
9a1d6af5 | 573 | #ifndef CONFIG_DM_ETH |
b1690bc3 | 574 | tempval = (dev->enetaddr[1] << 24) | (dev->enetaddr[0] << 16); |
9a1d6af5 BM |
575 | #else |
576 | tempval = (pdata->enetaddr[1] << 24) | (pdata->enetaddr[0] << 16); | |
577 | #endif | |
97d80fc3 | 578 | |
90751910 | 579 | out_be32(®s->macstnaddr2, tempval); |
97d80fc3 | 580 | |
90751910 MH |
581 | /* Clear out (for the most part) the other registers */ |
582 | init_registers(regs); | |
583 | ||
584 | /* Ready the device for tx/rx */ | |
56a27a1e | 585 | startup_tsec(priv); |
90751910 | 586 | |
063c1263 | 587 | /* Start up the PHY */ |
11af8d65 TT |
588 | ret = phy_startup(priv->phydev); |
589 | if (ret) { | |
590 | printf("Could not initialize PHY %s\n", | |
591 | priv->phydev->dev->name); | |
592 | return ret; | |
593 | } | |
063c1263 AF |
594 | |
595 | adjust_link(priv, priv->phydev); | |
596 | ||
90751910 | 597 | /* If there's no link, fail */ |
063c1263 AF |
598 | return priv->phydev->link ? 0 : -1; |
599 | } | |
600 | ||
601 | static phy_interface_t tsec_get_interface(struct tsec_private *priv) | |
602 | { | |
aec84bf6 | 603 | struct tsec __iomem *regs = priv->regs; |
063c1263 AF |
604 | u32 ecntrl; |
605 | ||
606 | ecntrl = in_be32(®s->ecntrl); | |
607 | ||
608 | if (ecntrl & ECNTRL_SGMII_MODE) | |
609 | return PHY_INTERFACE_MODE_SGMII; | |
610 | ||
611 | if (ecntrl & ECNTRL_TBI_MODE) { | |
612 | if (ecntrl & ECNTRL_REDUCED_MODE) | |
613 | return PHY_INTERFACE_MODE_RTBI; | |
614 | else | |
615 | return PHY_INTERFACE_MODE_TBI; | |
616 | } | |
617 | ||
618 | if (ecntrl & ECNTRL_REDUCED_MODE) { | |
d38de338 MS |
619 | phy_interface_t interface; |
620 | ||
063c1263 AF |
621 | if (ecntrl & ECNTRL_REDUCED_MII_MODE) |
622 | return PHY_INTERFACE_MODE_RMII; | |
d38de338 MS |
623 | |
624 | interface = priv->interface; | |
625 | ||
626 | /* | |
627 | * This isn't autodetected, so it must | |
628 | * be set by the platform code. | |
629 | */ | |
630 | if (interface == PHY_INTERFACE_MODE_RGMII_ID || | |
631 | interface == PHY_INTERFACE_MODE_RGMII_TXID || | |
632 | interface == PHY_INTERFACE_MODE_RGMII_RXID) | |
633 | return interface; | |
634 | ||
635 | return PHY_INTERFACE_MODE_RGMII; | |
063c1263 AF |
636 | } |
637 | ||
638 | if (priv->flags & TSEC_GIGABIT) | |
639 | return PHY_INTERFACE_MODE_GMII; | |
640 | ||
641 | return PHY_INTERFACE_MODE_MII; | |
90751910 MH |
642 | } |
643 | ||
9872b736 BM |
644 | /* |
645 | * Discover which PHY is attached to the device, and configure it | |
90751910 MH |
646 | * properly. If the PHY is not recognized, then return 0 |
647 | * (failure). Otherwise, return 1 | |
7abf0c58 | 648 | */ |
56a27a1e | 649 | static int init_phy(struct tsec_private *priv) |
7abf0c58 | 650 | { |
063c1263 | 651 | struct phy_device *phydev; |
aec84bf6 | 652 | struct tsec __iomem *regs = priv->regs; |
063c1263 AF |
653 | u32 supported = (SUPPORTED_10baseT_Half | |
654 | SUPPORTED_10baseT_Full | | |
655 | SUPPORTED_100baseT_Half | | |
656 | SUPPORTED_100baseT_Full); | |
657 | ||
658 | if (priv->flags & TSEC_GIGABIT) | |
659 | supported |= SUPPORTED_1000baseT_Full; | |
97d80fc3 | 660 | |
90751910 | 661 | /* Assign a Physical address to the TBI */ |
a1c76c15 | 662 | out_be32(®s->tbipa, priv->tbiaddr); |
90751910 | 663 | |
063c1263 | 664 | priv->interface = tsec_get_interface(priv); |
90751910 | 665 | |
063c1263 AF |
666 | if (priv->interface == PHY_INTERFACE_MODE_SGMII) |
667 | tsec_configure_serdes(priv); | |
90751910 | 668 | |
56a27a1e BM |
669 | phydev = phy_connect(priv->bus, priv->phyaddr, priv->dev, |
670 | priv->interface); | |
7f233c05 CM |
671 | if (!phydev) |
672 | return 0; | |
7abf0c58 | 673 | |
063c1263 AF |
674 | phydev->supported &= supported; |
675 | phydev->advertising = phydev->supported; | |
7abf0c58 | 676 | |
063c1263 | 677 | priv->phydev = phydev; |
90751910 | 678 | |
063c1263 | 679 | phy_config(phydev); |
90751910 MH |
680 | |
681 | return 1; | |
7abf0c58 WD |
682 | } |
683 | ||
9a1d6af5 | 684 | #ifndef CONFIG_DM_ETH |
9872b736 BM |
685 | /* |
686 | * Initialize device structure. Returns success if PHY | |
90751910 | 687 | * initialization succeeded (i.e. if it recognizes the PHY) |
7abf0c58 | 688 | */ |
90751910 | 689 | static int tsec_initialize(bd_t *bis, struct tsec_info_struct *tsec_info) |
7abf0c58 | 690 | { |
90751910 MH |
691 | struct eth_device *dev; |
692 | int i; | |
693 | struct tsec_private *priv; | |
97d80fc3 | 694 | |
d38de338 | 695 | dev = (struct eth_device *)malloc(sizeof(*dev)); |
7abf0c58 | 696 | |
d38de338 | 697 | if (!dev) |
90751910 | 698 | return 0; |
7abf0c58 | 699 | |
d38de338 | 700 | memset(dev, 0, sizeof(*dev)); |
97d80fc3 | 701 | |
90751910 MH |
702 | priv = (struct tsec_private *)malloc(sizeof(*priv)); |
703 | ||
d38de338 | 704 | if (!priv) |
90751910 MH |
705 | return 0; |
706 | ||
90751910 | 707 | priv->regs = tsec_info->regs; |
90751910 MH |
708 | priv->phyregs_sgmii = tsec_info->miiregs_sgmii; |
709 | ||
710 | priv->phyaddr = tsec_info->phyaddr; | |
a1c76c15 | 711 | priv->tbiaddr = CONFIG_SYS_TBIPA_VALUE; |
90751910 | 712 | priv->flags = tsec_info->flags; |
97d80fc3 | 713 | |
192bc694 | 714 | strcpy(dev->name, tsec_info->devname); |
063c1263 AF |
715 | priv->interface = tsec_info->interface; |
716 | priv->bus = miiphy_get_dev_by_name(tsec_info->mii_devname); | |
56a27a1e | 717 | priv->dev = dev; |
90751910 MH |
718 | dev->iobase = 0; |
719 | dev->priv = priv; | |
720 | dev->init = tsec_init; | |
721 | dev->halt = tsec_halt; | |
722 | dev->send = tsec_send; | |
723 | dev->recv = tsec_recv; | |
53a5c424 | 724 | #ifdef CONFIG_MCAST_TFTP |
90751910 MH |
725 | dev->mcast = tsec_mcast_addr; |
726 | #endif | |
53a5c424 | 727 | |
9872b736 | 728 | /* Tell U-Boot to get the addr from the env */ |
90751910 MH |
729 | for (i = 0; i < 6; i++) |
730 | dev->enetaddr[i] = 0; | |
53a5c424 | 731 | |
90751910 | 732 | eth_register(dev); |
53a5c424 | 733 | |
90751910 MH |
734 | /* Reset the MAC */ |
735 | setbits_be32(&priv->regs->maccfg1, MACCFG1_SOFT_RESET); | |
736 | udelay(2); /* Soft Reset must be asserted for 3 TX clocks */ | |
737 | clrbits_be32(&priv->regs->maccfg1, MACCFG1_SOFT_RESET); | |
53a5c424 | 738 | |
90751910 | 739 | /* Try to initialize PHY here, and return */ |
56a27a1e | 740 | return init_phy(priv); |
90751910 | 741 | } |
53a5c424 | 742 | |
90751910 MH |
743 | /* |
744 | * Initialize all the TSEC devices | |
745 | * | |
746 | * Returns the number of TSEC devices that were initialized | |
747 | */ | |
748 | int tsec_eth_init(bd_t *bis, struct tsec_info_struct *tsecs, int num) | |
749 | { | |
750 | int i; | |
d38de338 | 751 | int count = 0; |
90751910 MH |
752 | |
753 | for (i = 0; i < num; i++) { | |
d38de338 MS |
754 | int ret = tsec_initialize(bis, &tsecs[i]); |
755 | ||
90751910 MH |
756 | if (ret > 0) |
757 | count += ret; | |
53a5c424 | 758 | } |
90751910 MH |
759 | |
760 | return count; | |
53a5c424 | 761 | } |
90751910 MH |
762 | |
763 | int tsec_standard_init(bd_t *bis) | |
764 | { | |
063c1263 AF |
765 | struct fsl_pq_mdio_info info; |
766 | ||
aec84bf6 | 767 | info.regs = TSEC_GET_MDIO_REGS_BASE(1); |
063c1263 AF |
768 | info.name = DEFAULT_MII_NAME; |
769 | ||
770 | fsl_pq_mdio_init(bis, &info); | |
771 | ||
90751910 MH |
772 | return tsec_eth_init(bis, tsec_info, ARRAY_SIZE(tsec_info)); |
773 | } | |
9a1d6af5 BM |
774 | #else /* CONFIG_DM_ETH */ |
775 | int tsec_probe(struct udevice *dev) | |
776 | { | |
777 | struct tsec_private *priv = dev_get_priv(dev); | |
778 | struct eth_pdata *pdata = dev_get_platdata(dev); | |
779 | struct fsl_pq_mdio_info mdio_info; | |
780 | int offset = 0; | |
781 | int reg; | |
782 | const char *phy_mode; | |
783 | int ret; | |
784 | ||
a821c4af | 785 | pdata->iobase = (phys_addr_t)devfdt_get_addr(dev); |
9a1d6af5 BM |
786 | priv->regs = (struct tsec *)pdata->iobase; |
787 | ||
e160f7d4 | 788 | offset = fdtdec_lookup_phandle(gd->fdt_blob, dev_of_offset(dev), |
9a1d6af5 BM |
789 | "phy-handle"); |
790 | if (offset > 0) { | |
791 | reg = fdtdec_get_int(gd->fdt_blob, offset, "reg", 0); | |
792 | priv->phyaddr = reg; | |
793 | } else { | |
794 | debug("phy-handle does not exist under tsec %s\n", dev->name); | |
795 | return -ENOENT; | |
796 | } | |
797 | ||
798 | offset = fdt_parent_offset(gd->fdt_blob, offset); | |
799 | if (offset > 0) { | |
800 | reg = fdtdec_get_int(gd->fdt_blob, offset, "reg", 0); | |
801 | priv->phyregs_sgmii = (struct tsec_mii_mng *)(reg + 0x520); | |
802 | } else { | |
803 | debug("No parent node for PHY?\n"); | |
804 | return -ENOENT; | |
805 | } | |
806 | ||
e160f7d4 | 807 | offset = fdtdec_lookup_phandle(gd->fdt_blob, dev_of_offset(dev), |
a1c76c15 BM |
808 | "tbi-handle"); |
809 | if (offset > 0) { | |
810 | reg = fdtdec_get_int(gd->fdt_blob, offset, "reg", | |
811 | CONFIG_SYS_TBIPA_VALUE); | |
812 | priv->tbiaddr = reg; | |
813 | } else { | |
814 | priv->tbiaddr = CONFIG_SYS_TBIPA_VALUE; | |
815 | } | |
816 | ||
e160f7d4 | 817 | phy_mode = fdt_getprop(gd->fdt_blob, dev_of_offset(dev), |
9a1d6af5 BM |
818 | "phy-connection-type", NULL); |
819 | if (phy_mode) | |
820 | pdata->phy_interface = phy_get_interface_by_name(phy_mode); | |
821 | if (pdata->phy_interface == -1) { | |
822 | debug("Invalid PHY interface '%s'\n", phy_mode); | |
823 | return -EINVAL; | |
824 | } | |
825 | priv->interface = pdata->phy_interface; | |
826 | ||
827 | /* Initialize flags */ | |
828 | priv->flags = TSEC_GIGABIT; | |
829 | if (priv->interface == PHY_INTERFACE_MODE_SGMII) | |
830 | priv->flags |= TSEC_SGMII; | |
831 | ||
832 | mdio_info.regs = priv->phyregs_sgmii; | |
833 | mdio_info.name = (char *)dev->name; | |
834 | ret = fsl_pq_mdio_init(NULL, &mdio_info); | |
835 | if (ret) | |
836 | return ret; | |
837 | ||
838 | /* Reset the MAC */ | |
839 | setbits_be32(&priv->regs->maccfg1, MACCFG1_SOFT_RESET); | |
840 | udelay(2); /* Soft Reset must be asserted for 3 TX clocks */ | |
841 | clrbits_be32(&priv->regs->maccfg1, MACCFG1_SOFT_RESET); | |
842 | ||
843 | priv->dev = dev; | |
844 | priv->bus = miiphy_get_dev_by_name(dev->name); | |
845 | ||
846 | /* Try to initialize PHY here, and return */ | |
847 | return !init_phy(priv); | |
848 | } | |
849 | ||
850 | int tsec_remove(struct udevice *dev) | |
851 | { | |
852 | struct tsec_private *priv = dev->priv; | |
853 | ||
854 | free(priv->phydev); | |
855 | mdio_unregister(priv->bus); | |
856 | mdio_free(priv->bus); | |
857 | ||
858 | return 0; | |
859 | } | |
860 | ||
861 | static const struct eth_ops tsec_ops = { | |
862 | .start = tsec_init, | |
863 | .send = tsec_send, | |
864 | .recv = tsec_recv, | |
865 | .free_pkt = tsec_free_pkt, | |
866 | .stop = tsec_halt, | |
867 | #ifdef CONFIG_MCAST_TFTP | |
868 | .mcast = tsec_mcast_addr, | |
869 | #endif | |
870 | }; | |
871 | ||
872 | static const struct udevice_id tsec_ids[] = { | |
873 | { .compatible = "fsl,tsec" }, | |
874 | { } | |
875 | }; | |
876 | ||
877 | U_BOOT_DRIVER(eth_tsec) = { | |
878 | .name = "tsec", | |
879 | .id = UCLASS_ETH, | |
880 | .of_match = tsec_ids, | |
881 | .probe = tsec_probe, | |
882 | .remove = tsec_remove, | |
883 | .ops = &tsec_ops, | |
884 | .priv_auto_alloc_size = sizeof(struct tsec_private), | |
885 | .platdata_auto_alloc_size = sizeof(struct eth_pdata), | |
886 | .flags = DM_FLAG_ALLOC_PRIV_DMA, | |
887 | }; | |
888 | #endif /* CONFIG_DM_ETH */ |