blob: c436b8317d995d2d55ec3e0f6d0f066e41120527 [file] [log] [blame]
Tom Rini83d290c2018-05-06 17:58:06 -04001// SPDX-License-Identifier: GPL-2.0+
wdenk42d1f032003-10-15 23:53:47 +00002/*
wdenk97d80fc2004-06-09 00:34:46 +00003 * Freescale Three Speed Ethernet Controller driver
wdenk42d1f032003-10-15 23:53:47 +00004 *
Claudiu Manoilaec84bf2013-09-30 12:44:42 +03005 * Copyright 2004-2011, 2013 Freescale Semiconductor, Inc.
wdenk42d1f032003-10-15 23:53:47 +00006 * (C) Copyright 2003, Motorola, Inc.
wdenk42d1f032003-10-15 23:53:47 +00007 * author Andy Fleming
wdenk42d1f032003-10-15 23:53:47 +00008 */
9
10#include <config.h>
wdenk42d1f032003-10-15 23:53:47 +000011#include <common.h>
Bin Meng9a1d6af2016-01-11 22:41:24 -080012#include <dm.h>
wdenk42d1f032003-10-15 23:53:47 +000013#include <malloc.h>
14#include <net.h>
15#include <command.h>
Andy Flemingdd3d1f52008-08-31 16:33:25 -050016#include <tsec.h>
Andy Fleming063c1262011-04-08 02:10:54 -050017#include <fsl_mdio.h>
Simon Glasscd93d622020-05-10 11:40:13 -060018#include <linux/bitops.h>
Simon Glassc05ed002020-05-10 11:40:11 -060019#include <linux/delay.h>
Masahiro Yamada1221ce42016-09-21 11:28:55 +090020#include <linux/errno.h>
Hou Zhiqiangb4eb9cf2020-07-16 18:09:12 +080021#include <miiphy.h>
chenhui zhaoaada81d2011-10-03 08:38:50 -050022#include <asm/processor.h>
Alison Wang52d00a82014-09-05 13:52:38 +080023#include <asm/io.h>
wdenk42d1f032003-10-15 23:53:47 +000024
Bin Meng9a1d6af2016-01-11 22:41:24 -080025#ifndef CONFIG_DM_ETH
Andy Fleming75b9d4a2008-08-31 16:33:26 -050026/* Default initializations for TSEC controllers. */
27
28static 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 {
Claudiu Manoilaec84bf2013-09-30 12:44:42 +030037 .regs = TSEC_GET_REGS(2, 0x2000),
Andy Fleming75b9d4a2008-08-31 16:33:26 -050038 .devname = CONFIG_MPC85XX_FEC_NAME,
39 .phyaddr = FEC_PHY_ADDR,
Andy Fleming063c1262011-04-08 02:10:54 -050040 .flags = FEC_FLAGS,
41 .mii_devname = DEFAULT_MII_NAME
Andy Fleming75b9d4a2008-08-31 16:33:26 -050042 }, /* 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};
Bin Meng9a1d6af2016-01-11 22:41:24 -080051#endif /* CONFIG_DM_ETH */
Andy Fleming75b9d4a2008-08-31 16:33:26 -050052
Andy Fleming2abe3612008-08-31 16:33:27 -050053#define TBIANA_SETTINGS ( \
54 TBIANA_ASYMMETRIC_PAUSE \
55 | TBIANA_SYMMETRIC_PAUSE \
56 | TBIANA_FULL_DUPLEX \
57 )
58
Felix Radensky90b5bf22010-06-28 01:57:39 +030059/* By default force the TBI PHY into 1000Mbps full duplex when in SGMII mode */
60#ifndef CONFIG_TSEC_TBICR_SETTINGS
Kumar Gala72c96a62010-12-01 22:55:54 -060061#define CONFIG_TSEC_TBICR_SETTINGS ( \
Andy Fleming2abe3612008-08-31 16:33:27 -050062 TBICR_PHY_RESET \
Kumar Gala72c96a62010-12-01 22:55:54 -060063 | TBICR_ANEG_ENABLE \
Andy Fleming2abe3612008-08-31 16:33:27 -050064 | TBICR_FULL_DUPLEX \
65 | TBICR_SPEED1_SET \
66 )
Felix Radensky90b5bf22010-06-28 01:57:39 +030067#endif /* CONFIG_TSEC_TBICR_SETTINGS */
Peter Tyser46e91672009-11-03 17:52:07 -060068
Andy Fleming2abe3612008-08-31 16:33:27 -050069/* Configure the TBI for SGMII operation */
70static void tsec_configure_serdes(struct tsec_private *priv)
71{
Bin Meng9872b732016-01-11 22:41:18 -080072 /*
73 * Access TBI PHY registers at given TSEC register offset as opposed
74 * to the register offset used for external PHY accesses
75 */
Andy Fleming063c1262011-04-08 02:10:54 -050076 tsec_local_mdio_write(priv->phyregs_sgmii, in_be32(&priv->regs->tbipa),
Mario Sixd38de332018-01-15 11:08:21 +010077 0, TBI_ANA, TBIANA_SETTINGS);
Andy Fleming063c1262011-04-08 02:10:54 -050078 tsec_local_mdio_write(priv->phyregs_sgmii, in_be32(&priv->regs->tbipa),
Mario Sixd38de332018-01-15 11:08:21 +010079 0, TBI_TBICON, TBICON_CLK_SELECT);
Andy Fleming063c1262011-04-08 02:10:54 -050080 tsec_local_mdio_write(priv->phyregs_sgmii, in_be32(&priv->regs->tbipa),
Mario Sixd38de332018-01-15 11:08:21 +010081 0, TBI_CR, CONFIG_TSEC_TBICR_SETTINGS);
Andy Fleming2abe3612008-08-31 16:33:27 -050082}
michael.firth@bt.com55fe7c52008-01-16 11:40:51 +000083
Chris Packham1a4af5c2018-11-26 21:00:28 +130084/* the 'way' for ethernet-CRC-32. Spliced in from Linux lib/crc32.c
85 * and this is the ethernet-crc method needed for TSEC -- and perhaps
86 * some other adapter -- hash tables
87 */
88#define CRCPOLY_LE 0xedb88320
89static u32 ether_crc(size_t len, unsigned char const *p)
90{
91 int i;
92 u32 crc;
93
94 crc = ~0;
95 while (len--) {
96 crc ^= *p++;
97 for (i = 0; i < 8; i++)
98 crc = (crc >> 1) ^ ((crc & 1) ? CRCPOLY_LE : 0);
99 }
100 /* an reverse the bits, cuz of way they arrive -- last-first */
101 crc = (crc >> 16) | (crc << 16);
102 crc = (crc >> 8 & 0x00ff00ff) | (crc << 8 & 0xff00ff00);
103 crc = (crc >> 4 & 0x0f0f0f0f) | (crc << 4 & 0xf0f0f0f0);
104 crc = (crc >> 2 & 0x33333333) | (crc << 2 & 0xcccccccc);
105 crc = (crc >> 1 & 0x55555555) | (crc << 1 & 0xaaaaaaaa);
106 return crc;
107}
108
David Updegraff53a5c422007-06-11 10:41:07 -0500109/* CREDITS: linux gianfar driver, slightly adjusted... thanx. */
110
111/* Set the appropriate hash bit for the given addr */
112
Bin Meng9872b732016-01-11 22:41:18 -0800113/*
114 * The algorithm works like so:
David Updegraff53a5c422007-06-11 10:41:07 -0500115 * 1) Take the Destination Address (ie the multicast address), and
116 * do a CRC on it (little endian), and reverse the bits of the
117 * result.
118 * 2) Use the 8 most significant bits as a hash into a 256-entry
119 * table. The table is controlled through 8 32-bit registers:
Claudiu Manoil876d4512013-09-30 12:44:40 +0300120 * gaddr0-7. gaddr0's MSB is entry 0, and gaddr7's LSB is entry
121 * 255. This means that the 3 most significant bits in the
David Updegraff53a5c422007-06-11 10:41:07 -0500122 * hash index which gaddr register to use, and the 5 other bits
123 * indicate which bit (assuming an IBM numbering scheme, which
Claudiu Manoil876d4512013-09-30 12:44:40 +0300124 * for PowerPC (tm) is usually the case) in the register holds
Bin Meng9872b732016-01-11 22:41:18 -0800125 * the entry.
126 */
Bin Meng9a1d6af2016-01-11 22:41:24 -0800127#ifndef CONFIG_DM_ETH
Chris Packham67bb9842018-11-26 21:00:29 +1300128static int tsec_mcast_addr(struct eth_device *dev, const u8 *mcast_mac,
129 int join)
Bin Meng9a1d6af2016-01-11 22:41:24 -0800130#else
Chris Packham67bb9842018-11-26 21:00:29 +1300131static int tsec_mcast_addr(struct udevice *dev, const u8 *mcast_mac, int join)
Bin Meng9a1d6af2016-01-11 22:41:24 -0800132#endif
David Updegraff53a5c422007-06-11 10:41:07 -0500133{
Claudiu Manoilb2002042013-09-30 12:44:41 +0300134 struct tsec_private *priv = (struct tsec_private *)dev->priv;
Claudiu Manoil876d4512013-09-30 12:44:40 +0300135 struct tsec __iomem *regs = priv->regs;
136 u32 result, value;
137 u8 whichbit, whichreg;
David Updegraff53a5c422007-06-11 10:41:07 -0500138
Claudiu Manoil876d4512013-09-30 12:44:40 +0300139 result = ether_crc(MAC_ADDR_LEN, mcast_mac);
140 whichbit = (result >> 24) & 0x1f; /* the 5 LSB = which bit to set */
141 whichreg = result >> 29; /* the 3 MSB = which reg to set it in */
David Updegraff53a5c422007-06-11 10:41:07 -0500142
Mario Sixd38de332018-01-15 11:08:21 +0100143 value = BIT(31 - whichbit);
David Updegraff53a5c422007-06-11 10:41:07 -0500144
Chris Packham67bb9842018-11-26 21:00:29 +1300145 if (join)
Claudiu Manoil876d4512013-09-30 12:44:40 +0300146 setbits_be32(&regs->hash.gaddr0 + whichreg, value);
147 else
148 clrbits_be32(&regs->hash.gaddr0 + whichreg, value);
149
David Updegraff53a5c422007-06-11 10:41:07 -0500150 return 0;
151}
Mingkai Hu90751912011-01-27 12:52:46 +0800152
Bin Meng9872b732016-01-11 22:41:18 -0800153/*
154 * Initialized required registers to appropriate values, zeroing
Mingkai Hu90751912011-01-27 12:52:46 +0800155 * those we don't care about (unless zero is bad, in which case,
156 * choose a more appropriate value)
157 */
Claudiu Manoilaec84bf2013-09-30 12:44:42 +0300158static void init_registers(struct tsec __iomem *regs)
Mingkai Hu90751912011-01-27 12:52:46 +0800159{
160 /* Clear IEVENT */
161 out_be32(&regs->ievent, IEVENT_INIT_CLEAR);
162
163 out_be32(&regs->imask, IMASK_INIT_CLEAR);
164
165 out_be32(&regs->hash.iaddr0, 0);
166 out_be32(&regs->hash.iaddr1, 0);
167 out_be32(&regs->hash.iaddr2, 0);
168 out_be32(&regs->hash.iaddr3, 0);
169 out_be32(&regs->hash.iaddr4, 0);
170 out_be32(&regs->hash.iaddr5, 0);
171 out_be32(&regs->hash.iaddr6, 0);
172 out_be32(&regs->hash.iaddr7, 0);
173
174 out_be32(&regs->hash.gaddr0, 0);
175 out_be32(&regs->hash.gaddr1, 0);
176 out_be32(&regs->hash.gaddr2, 0);
177 out_be32(&regs->hash.gaddr3, 0);
178 out_be32(&regs->hash.gaddr4, 0);
179 out_be32(&regs->hash.gaddr5, 0);
180 out_be32(&regs->hash.gaddr6, 0);
181 out_be32(&regs->hash.gaddr7, 0);
182
183 out_be32(&regs->rctrl, 0x00000000);
184
185 /* Init RMON mib registers */
Claudiu Manoil82ef75c2013-09-30 12:44:46 +0300186 memset((void *)&regs->rmon, 0, sizeof(regs->rmon));
Mingkai Hu90751912011-01-27 12:52:46 +0800187
188 out_be32(&regs->rmon.cam1, 0xffffffff);
189 out_be32(&regs->rmon.cam2, 0xffffffff);
190
191 out_be32(&regs->mrblr, MRBLR_INIT_SETTINGS);
192
193 out_be32(&regs->minflr, MINFLR_INIT_SETTINGS);
194
195 out_be32(&regs->attr, ATTR_INIT_SETTINGS);
196 out_be32(&regs->attreli, ATTRELI_INIT_SETTINGS);
Mingkai Hu90751912011-01-27 12:52:46 +0800197}
198
Bin Meng9872b732016-01-11 22:41:18 -0800199/*
200 * Configure maccfg2 based on negotiated speed and duplex
Mingkai Hu90751912011-01-27 12:52:46 +0800201 * reported by PHY handling code
202 */
Andy Fleming063c1262011-04-08 02:10:54 -0500203static void adjust_link(struct tsec_private *priv, struct phy_device *phydev)
Mingkai Hu90751912011-01-27 12:52:46 +0800204{
Claudiu Manoilaec84bf2013-09-30 12:44:42 +0300205 struct tsec __iomem *regs = priv->regs;
Mingkai Hu90751912011-01-27 12:52:46 +0800206 u32 ecntrl, maccfg2;
207
Andy Fleming063c1262011-04-08 02:10:54 -0500208 if (!phydev->link) {
209 printf("%s: No link.\n", phydev->dev->name);
Mingkai Hu90751912011-01-27 12:52:46 +0800210 return;
211 }
212
213 /* clear all bits relative with interface mode */
214 ecntrl = in_be32(&regs->ecntrl);
215 ecntrl &= ~ECNTRL_R100;
216
217 maccfg2 = in_be32(&regs->maccfg2);
218 maccfg2 &= ~(MACCFG2_IF | MACCFG2_FULL_DUPLEX);
219
Andy Fleming063c1262011-04-08 02:10:54 -0500220 if (phydev->duplex)
Mingkai Hu90751912011-01-27 12:52:46 +0800221 maccfg2 |= MACCFG2_FULL_DUPLEX;
222
Andy Fleming063c1262011-04-08 02:10:54 -0500223 switch (phydev->speed) {
Mingkai Hu90751912011-01-27 12:52:46 +0800224 case 1000:
225 maccfg2 |= MACCFG2_GMII;
226 break;
227 case 100:
228 case 10:
229 maccfg2 |= MACCFG2_MII;
230
Bin Meng9872b732016-01-11 22:41:18 -0800231 /*
232 * Set R100 bit in all modes although
Mingkai Hu90751912011-01-27 12:52:46 +0800233 * it is only used in RGMII mode
234 */
Andy Fleming063c1262011-04-08 02:10:54 -0500235 if (phydev->speed == 100)
Mingkai Hu90751912011-01-27 12:52:46 +0800236 ecntrl |= ECNTRL_R100;
237 break;
238 default:
Andy Fleming063c1262011-04-08 02:10:54 -0500239 printf("%s: Speed was bad\n", phydev->dev->name);
Mingkai Hu90751912011-01-27 12:52:46 +0800240 break;
241 }
242
243 out_be32(&regs->ecntrl, ecntrl);
244 out_be32(&regs->maccfg2, maccfg2);
245
Andy Fleming063c1262011-04-08 02:10:54 -0500246 printf("Speed: %d, %s duplex%s\n", phydev->speed,
Mario Sixd38de332018-01-15 11:08:21 +0100247 (phydev->duplex) ? "full" : "half",
248 (phydev->port == PORT_FIBRE) ? ", fiber mode" : "");
Mingkai Hu90751912011-01-27 12:52:46 +0800249}
250
Bin Meng8ba50172016-01-11 22:41:21 -0800251/*
252 * This returns the status bits of the device. The return value
253 * is never checked, and this is what the 8260 driver did, so we
254 * do the same. Presumably, this would be zero if there were no
255 * errors
256 */
Bin Meng9a1d6af2016-01-11 22:41:24 -0800257#ifndef CONFIG_DM_ETH
Bin Meng8ba50172016-01-11 22:41:21 -0800258static int tsec_send(struct eth_device *dev, void *packet, int length)
Bin Meng9a1d6af2016-01-11 22:41:24 -0800259#else
260static int tsec_send(struct udevice *dev, void *packet, int length)
261#endif
Bin Meng8ba50172016-01-11 22:41:21 -0800262{
263 struct tsec_private *priv = (struct tsec_private *)dev->priv;
264 struct tsec __iomem *regs = priv->regs;
Bin Meng8ba50172016-01-11 22:41:21 -0800265 int result = 0;
Vladimir Oltean07bd39f2019-07-19 00:29:55 +0300266 u16 status;
Bin Meng8ba50172016-01-11 22:41:21 -0800267 int i;
268
269 /* Find an empty buffer descriptor */
270 for (i = 0;
271 in_be16(&priv->txbd[priv->tx_idx].status) & TXBD_READY;
272 i++) {
273 if (i >= TOUT_LOOP) {
Vladimir Olteanb7be7762019-07-19 00:29:56 +0300274 printf("%s: tsec: tx buffers full\n", dev->name);
Bin Meng8ba50172016-01-11 22:41:21 -0800275 return result;
276 }
277 }
278
279 out_be32(&priv->txbd[priv->tx_idx].bufptr, (u32)packet);
280 out_be16(&priv->txbd[priv->tx_idx].length, length);
281 status = in_be16(&priv->txbd[priv->tx_idx].status);
282 out_be16(&priv->txbd[priv->tx_idx].status, status |
283 (TXBD_READY | TXBD_LAST | TXBD_CRC | TXBD_INTERRUPT));
284
285 /* Tell the DMA to go */
286 out_be32(&regs->tstat, TSTAT_CLEAR_THALT);
287
288 /* Wait for buffer to be transmitted */
289 for (i = 0;
290 in_be16(&priv->txbd[priv->tx_idx].status) & TXBD_READY;
291 i++) {
292 if (i >= TOUT_LOOP) {
Vladimir Olteanb7be7762019-07-19 00:29:56 +0300293 printf("%s: tsec: tx error\n", dev->name);
Bin Meng8ba50172016-01-11 22:41:21 -0800294 return result;
295 }
296 }
297
298 priv->tx_idx = (priv->tx_idx + 1) % TX_BUF_CNT;
299 result = in_be16(&priv->txbd[priv->tx_idx].status) & TXBD_STATS;
300
301 return result;
302}
303
Bin Meng9a1d6af2016-01-11 22:41:24 -0800304#ifndef CONFIG_DM_ETH
Bin Meng8ba50172016-01-11 22:41:21 -0800305static int tsec_recv(struct eth_device *dev)
306{
307 struct tsec_private *priv = (struct tsec_private *)dev->priv;
308 struct tsec __iomem *regs = priv->regs;
309
310 while (!(in_be16(&priv->rxbd[priv->rx_idx].status) & RXBD_EMPTY)) {
311 int length = in_be16(&priv->rxbd[priv->rx_idx].length);
Mario Sixd38de332018-01-15 11:08:21 +0100312 u16 status = in_be16(&priv->rxbd[priv->rx_idx].status);
Bin Meng8ba50172016-01-11 22:41:21 -0800313 uchar *packet = net_rx_packets[priv->rx_idx];
314
315 /* Send the packet up if there were no errors */
316 if (!(status & RXBD_STATS))
317 net_process_received_packet(packet, length - 4);
318 else
319 printf("Got error %x\n", (status & RXBD_STATS));
320
321 out_be16(&priv->rxbd[priv->rx_idx].length, 0);
322
323 status = RXBD_EMPTY;
324 /* Set the wrap bit if this is the last element in the list */
325 if ((priv->rx_idx + 1) == PKTBUFSRX)
326 status |= RXBD_WRAP;
327 out_be16(&priv->rxbd[priv->rx_idx].status, status);
328
329 priv->rx_idx = (priv->rx_idx + 1) % PKTBUFSRX;
330 }
331
332 if (in_be32(&regs->ievent) & IEVENT_BSY) {
333 out_be32(&regs->ievent, IEVENT_BSY);
334 out_be32(&regs->rstat, RSTAT_CLEAR_RHALT);
335 }
336
337 return -1;
338}
Bin Meng9a1d6af2016-01-11 22:41:24 -0800339#else
340static int tsec_recv(struct udevice *dev, int flags, uchar **packetp)
341{
342 struct tsec_private *priv = (struct tsec_private *)dev->priv;
343 struct tsec __iomem *regs = priv->regs;
344 int ret = -1;
345
346 if (!(in_be16(&priv->rxbd[priv->rx_idx].status) & RXBD_EMPTY)) {
347 int length = in_be16(&priv->rxbd[priv->rx_idx].length);
Mario Sixd38de332018-01-15 11:08:21 +0100348 u16 status = in_be16(&priv->rxbd[priv->rx_idx].status);
349 u32 buf;
Bin Meng9a1d6af2016-01-11 22:41:24 -0800350
351 /* Send the packet up if there were no errors */
352 if (!(status & RXBD_STATS)) {
353 buf = in_be32(&priv->rxbd[priv->rx_idx].bufptr);
354 *packetp = (uchar *)buf;
355 ret = length - 4;
356 } else {
357 printf("Got error %x\n", (status & RXBD_STATS));
358 }
359 }
360
361 if (in_be32(&regs->ievent) & IEVENT_BSY) {
362 out_be32(&regs->ievent, IEVENT_BSY);
363 out_be32(&regs->rstat, RSTAT_CLEAR_RHALT);
364 }
365
366 return ret;
367}
368
369static int tsec_free_pkt(struct udevice *dev, uchar *packet, int length)
370{
371 struct tsec_private *priv = (struct tsec_private *)dev->priv;
Mario Sixd38de332018-01-15 11:08:21 +0100372 u16 status;
Bin Meng9a1d6af2016-01-11 22:41:24 -0800373
374 out_be16(&priv->rxbd[priv->rx_idx].length, 0);
375
376 status = RXBD_EMPTY;
377 /* Set the wrap bit if this is the last element in the list */
378 if ((priv->rx_idx + 1) == PKTBUFSRX)
379 status |= RXBD_WRAP;
380 out_be16(&priv->rxbd[priv->rx_idx].status, status);
381
382 priv->rx_idx = (priv->rx_idx + 1) % PKTBUFSRX;
383
384 return 0;
385}
386#endif
Bin Meng8ba50172016-01-11 22:41:21 -0800387
388/* Stop the interface */
Bin Meng9a1d6af2016-01-11 22:41:24 -0800389#ifndef CONFIG_DM_ETH
Bin Meng8ba50172016-01-11 22:41:21 -0800390static void tsec_halt(struct eth_device *dev)
Bin Meng9a1d6af2016-01-11 22:41:24 -0800391#else
392static void tsec_halt(struct udevice *dev)
393#endif
Bin Meng8ba50172016-01-11 22:41:21 -0800394{
395 struct tsec_private *priv = (struct tsec_private *)dev->priv;
396 struct tsec __iomem *regs = priv->regs;
397
398 clrbits_be32(&regs->dmactrl, DMACTRL_GRS | DMACTRL_GTS);
399 setbits_be32(&regs->dmactrl, DMACTRL_GRS | DMACTRL_GTS);
400
401 while ((in_be32(&regs->ievent) & (IEVENT_GRSC | IEVENT_GTSC))
402 != (IEVENT_GRSC | IEVENT_GTSC))
403 ;
404
405 clrbits_be32(&regs->maccfg1, MACCFG1_TX_EN | MACCFG1_RX_EN);
406
407 /* Shut down the PHY, as needed */
408 phy_shutdown(priv->phydev);
409}
410
chenhui zhaoaada81d2011-10-03 08:38:50 -0500411#ifdef CONFIG_SYS_FSL_ERRATUM_NMG_ETSEC129
412/*
413 * When MACCFG1[Rx_EN] is enabled during system boot as part
414 * of the eTSEC port initialization sequence,
415 * the eTSEC Rx logic may not be properly initialized.
416 */
Bin Meng56a27a12016-01-11 22:41:22 -0800417void redundant_init(struct tsec_private *priv)
chenhui zhaoaada81d2011-10-03 08:38:50 -0500418{
Claudiu Manoilaec84bf2013-09-30 12:44:42 +0300419 struct tsec __iomem *regs = priv->regs;
chenhui zhaoaada81d2011-10-03 08:38:50 -0500420 uint t, count = 0;
421 int fail = 1;
422 static const u8 pkt[] = {
423 0x00, 0x1e, 0x4f, 0x12, 0xcb, 0x2c, 0x00, 0x25,
424 0x64, 0xbb, 0xd1, 0xab, 0x08, 0x00, 0x45, 0x00,
425 0x00, 0x5c, 0xdd, 0x22, 0x00, 0x00, 0x80, 0x01,
426 0x1f, 0x71, 0x0a, 0xc1, 0x14, 0x22, 0x0a, 0xc1,
427 0x14, 0x6a, 0x08, 0x00, 0xef, 0x7e, 0x02, 0x00,
428 0x94, 0x05, 0x61, 0x62, 0x63, 0x64, 0x65, 0x66,
429 0x67, 0x68, 0x69, 0x6a, 0x6b, 0x6c, 0x6d, 0x6e,
430 0x6f, 0x70, 0x71, 0x72, 0x73, 0x74, 0x75, 0x76,
431 0x77, 0x61, 0x62, 0x63, 0x64, 0x65, 0x66, 0x67,
432 0x68, 0x69, 0x6a, 0x6b, 0x6c, 0x6d, 0x6e, 0x6f,
433 0x70, 0x71, 0x72, 0x73, 0x74, 0x75, 0x76, 0x77,
434 0x61, 0x62, 0x63, 0x64, 0x65, 0x66, 0x67, 0x68,
435 0x69, 0x6a, 0x6b, 0x6c, 0x6d, 0x6e, 0x6f, 0x70,
436 0x71, 0x72};
437
438 /* Enable promiscuous mode */
439 setbits_be32(&regs->rctrl, 0x8);
440 /* Enable loopback mode */
441 setbits_be32(&regs->maccfg1, MACCFG1_LOOPBACK);
442 /* Enable transmit and receive */
443 setbits_be32(&regs->maccfg1, MACCFG1_RX_EN | MACCFG1_TX_EN);
444
445 /* Tell the DMA it is clear to go */
446 setbits_be32(&regs->dmactrl, DMACTRL_INIT_SETTINGS);
447 out_be32(&regs->tstat, TSTAT_CLEAR_THALT);
448 out_be32(&regs->rstat, RSTAT_CLEAR_RHALT);
449 clrbits_be32(&regs->dmactrl, DMACTRL_GRS | DMACTRL_GTS);
450
451 do {
Mario Sixd38de332018-01-15 11:08:21 +0100452 u16 status;
453
Bin Meng56a27a12016-01-11 22:41:22 -0800454 tsec_send(priv->dev, (void *)pkt, sizeof(pkt));
chenhui zhaoaada81d2011-10-03 08:38:50 -0500455
456 /* Wait for buffer to be received */
Bin Menge677da92016-01-11 22:41:20 -0800457 for (t = 0;
458 in_be16(&priv->rxbd[priv->rx_idx].status) & RXBD_EMPTY;
Bin Meng362b1232016-01-11 22:41:19 -0800459 t++) {
chenhui zhaoaada81d2011-10-03 08:38:50 -0500460 if (t >= 10 * TOUT_LOOP) {
Bin Meng56a27a12016-01-11 22:41:22 -0800461 printf("%s: tsec: rx error\n", priv->dev->name);
chenhui zhaoaada81d2011-10-03 08:38:50 -0500462 break;
463 }
464 }
465
Bin Meng362b1232016-01-11 22:41:19 -0800466 if (!memcmp(pkt, net_rx_packets[priv->rx_idx], sizeof(pkt)))
chenhui zhaoaada81d2011-10-03 08:38:50 -0500467 fail = 0;
468
Bin Menge677da92016-01-11 22:41:20 -0800469 out_be16(&priv->rxbd[priv->rx_idx].length, 0);
Claudiu Manoil9c9141f2013-10-04 19:13:53 +0300470 status = RXBD_EMPTY;
Bin Meng362b1232016-01-11 22:41:19 -0800471 if ((priv->rx_idx + 1) == PKTBUFSRX)
Claudiu Manoil9c9141f2013-10-04 19:13:53 +0300472 status |= RXBD_WRAP;
Bin Menge677da92016-01-11 22:41:20 -0800473 out_be16(&priv->rxbd[priv->rx_idx].status, status);
Bin Meng362b1232016-01-11 22:41:19 -0800474 priv->rx_idx = (priv->rx_idx + 1) % PKTBUFSRX;
chenhui zhaoaada81d2011-10-03 08:38:50 -0500475
476 if (in_be32(&regs->ievent) & IEVENT_BSY) {
477 out_be32(&regs->ievent, IEVENT_BSY);
478 out_be32(&regs->rstat, RSTAT_CLEAR_RHALT);
479 }
480 if (fail) {
481 printf("loopback recv packet error!\n");
482 clrbits_be32(&regs->maccfg1, MACCFG1_RX_EN);
483 udelay(1000);
484 setbits_be32(&regs->maccfg1, MACCFG1_RX_EN);
485 }
486 } while ((count++ < 4) && (fail == 1));
487
488 if (fail)
489 panic("eTSEC init fail!\n");
490 /* Disable promiscuous mode */
491 clrbits_be32(&regs->rctrl, 0x8);
492 /* Disable loopback mode */
493 clrbits_be32(&regs->maccfg1, MACCFG1_LOOPBACK);
494}
495#endif
496
Bin Meng9872b732016-01-11 22:41:18 -0800497/*
498 * Set up the buffers and their descriptors, and bring up the
Mingkai Hu90751912011-01-27 12:52:46 +0800499 * interface
500 */
Bin Meng56a27a12016-01-11 22:41:22 -0800501static void startup_tsec(struct tsec_private *priv)
Mingkai Hu90751912011-01-27 12:52:46 +0800502{
Claudiu Manoilaec84bf2013-09-30 12:44:42 +0300503 struct tsec __iomem *regs = priv->regs;
Mario Sixd38de332018-01-15 11:08:21 +0100504 u16 status;
Claudiu Manoil9c9141f2013-10-04 19:13:53 +0300505 int i;
Mingkai Hu90751912011-01-27 12:52:46 +0800506
Andy Fleming063c1262011-04-08 02:10:54 -0500507 /* reset the indices to zero */
Bin Meng362b1232016-01-11 22:41:19 -0800508 priv->rx_idx = 0;
509 priv->tx_idx = 0;
chenhui zhaoaada81d2011-10-03 08:38:50 -0500510#ifdef CONFIG_SYS_FSL_ERRATUM_NMG_ETSEC129
511 uint svr;
512#endif
Andy Fleming063c1262011-04-08 02:10:54 -0500513
Mingkai Hu90751912011-01-27 12:52:46 +0800514 /* Point to the buffer descriptors */
Bin Menge677da92016-01-11 22:41:20 -0800515 out_be32(&regs->tbase, (u32)&priv->txbd[0]);
516 out_be32(&regs->rbase, (u32)&priv->rxbd[0]);
Mingkai Hu90751912011-01-27 12:52:46 +0800517
518 /* Initialize the Rx Buffer descriptors */
519 for (i = 0; i < PKTBUFSRX; i++) {
Bin Menge677da92016-01-11 22:41:20 -0800520 out_be16(&priv->rxbd[i].status, RXBD_EMPTY);
521 out_be16(&priv->rxbd[i].length, 0);
522 out_be32(&priv->rxbd[i].bufptr, (u32)net_rx_packets[i]);
Mingkai Hu90751912011-01-27 12:52:46 +0800523 }
Bin Menge677da92016-01-11 22:41:20 -0800524 status = in_be16(&priv->rxbd[PKTBUFSRX - 1].status);
525 out_be16(&priv->rxbd[PKTBUFSRX - 1].status, status | RXBD_WRAP);
Mingkai Hu90751912011-01-27 12:52:46 +0800526
527 /* Initialize the TX Buffer Descriptors */
528 for (i = 0; i < TX_BUF_CNT; i++) {
Bin Menge677da92016-01-11 22:41:20 -0800529 out_be16(&priv->txbd[i].status, 0);
530 out_be16(&priv->txbd[i].length, 0);
531 out_be32(&priv->txbd[i].bufptr, 0);
Mingkai Hu90751912011-01-27 12:52:46 +0800532 }
Bin Menge677da92016-01-11 22:41:20 -0800533 status = in_be16(&priv->txbd[TX_BUF_CNT - 1].status);
534 out_be16(&priv->txbd[TX_BUF_CNT - 1].status, status | TXBD_WRAP);
Mingkai Hu90751912011-01-27 12:52:46 +0800535
chenhui zhaoaada81d2011-10-03 08:38:50 -0500536#ifdef CONFIG_SYS_FSL_ERRATUM_NMG_ETSEC129
537 svr = get_svr();
538 if ((SVR_MAJ(svr) == 1) || IS_SVR_REV(svr, 2, 0))
Bin Meng56a27a12016-01-11 22:41:22 -0800539 redundant_init(priv);
chenhui zhaoaada81d2011-10-03 08:38:50 -0500540#endif
Mingkai Hu90751912011-01-27 12:52:46 +0800541 /* Enable Transmit and Receive */
542 setbits_be32(&regs->maccfg1, MACCFG1_RX_EN | MACCFG1_TX_EN);
543
544 /* Tell the DMA it is clear to go */
545 setbits_be32(&regs->dmactrl, DMACTRL_INIT_SETTINGS);
546 out_be32(&regs->tstat, TSTAT_CLEAR_THALT);
547 out_be32(&regs->rstat, RSTAT_CLEAR_RHALT);
548 clrbits_be32(&regs->dmactrl, DMACTRL_GRS | DMACTRL_GTS);
549}
550
Bin Meng9872b732016-01-11 22:41:18 -0800551/*
Bin Meng9872b732016-01-11 22:41:18 -0800552 * Initializes data structures and registers for the controller,
553 * and brings the interface up. Returns the link status, meaning
Mingkai Hu90751912011-01-27 12:52:46 +0800554 * that it returns success if the link is up, failure otherwise.
Bin Meng9872b732016-01-11 22:41:18 -0800555 * This allows U-Boot to find the first active controller.
Mingkai Hu90751912011-01-27 12:52:46 +0800556 */
Bin Meng9a1d6af2016-01-11 22:41:24 -0800557#ifndef CONFIG_DM_ETH
Masahiro Yamadab75d8dc2020-06-26 15:13:33 +0900558static int tsec_init(struct eth_device *dev, struct bd_info *bd)
Bin Meng9a1d6af2016-01-11 22:41:24 -0800559#else
560static int tsec_init(struct udevice *dev)
561#endif
Mingkai Hu90751912011-01-27 12:52:46 +0800562{
Mingkai Hu90751912011-01-27 12:52:46 +0800563 struct tsec_private *priv = (struct tsec_private *)dev->priv;
Bin Meng9a1d6af2016-01-11 22:41:24 -0800564#ifdef CONFIG_DM_ETH
565 struct eth_pdata *pdata = dev_get_platdata(dev);
Vladimir Olteanf6297c02019-07-19 00:29:57 +0300566#else
567 struct eth_device *pdata = dev;
Bin Meng9a1d6af2016-01-11 22:41:24 -0800568#endif
Claudiu Manoilaec84bf2013-09-30 12:44:42 +0300569 struct tsec __iomem *regs = priv->regs;
Claudiu Manoilb1690bc2013-09-30 12:44:47 +0300570 u32 tempval;
Timur Tabi11af8d62012-07-09 08:52:43 +0000571 int ret;
Mingkai Hu90751912011-01-27 12:52:46 +0800572
573 /* Make sure the controller is stopped */
574 tsec_halt(dev);
575
576 /* Init MACCFG2. Defaults to GMII */
577 out_be32(&regs->maccfg2, MACCFG2_INIT_SETTINGS);
578
579 /* Init ECNTRL */
580 out_be32(&regs->ecntrl, ECNTRL_INIT_SETTINGS);
581
Bin Meng9872b732016-01-11 22:41:18 -0800582 /*
583 * Copy the station address into the address registers.
Claudiu Manoilb1690bc2013-09-30 12:44:47 +0300584 * For a station address of 0x12345678ABCD in transmission
585 * order (BE), MACnADDR1 is set to 0xCDAB7856 and
586 * MACnADDR2 is set to 0x34120000.
587 */
Bin Meng9a1d6af2016-01-11 22:41:24 -0800588 tempval = (pdata->enetaddr[5] << 24) | (pdata->enetaddr[4] << 16) |
589 (pdata->enetaddr[3] << 8) | pdata->enetaddr[2];
Mingkai Hu90751912011-01-27 12:52:46 +0800590
591 out_be32(&regs->macstnaddr1, tempval);
592
Bin Meng9a1d6af2016-01-11 22:41:24 -0800593 tempval = (pdata->enetaddr[1] << 24) | (pdata->enetaddr[0] << 16);
Mingkai Hu90751912011-01-27 12:52:46 +0800594
595 out_be32(&regs->macstnaddr2, tempval);
596
Mingkai Hu90751912011-01-27 12:52:46 +0800597 /* Clear out (for the most part) the other registers */
598 init_registers(regs);
599
600 /* Ready the device for tx/rx */
Bin Meng56a27a12016-01-11 22:41:22 -0800601 startup_tsec(priv);
Mingkai Hu90751912011-01-27 12:52:46 +0800602
Andy Fleming063c1262011-04-08 02:10:54 -0500603 /* Start up the PHY */
Timur Tabi11af8d62012-07-09 08:52:43 +0000604 ret = phy_startup(priv->phydev);
605 if (ret) {
606 printf("Could not initialize PHY %s\n",
607 priv->phydev->dev->name);
608 return ret;
609 }
Andy Fleming063c1262011-04-08 02:10:54 -0500610
611 adjust_link(priv, priv->phydev);
612
Mingkai Hu90751912011-01-27 12:52:46 +0800613 /* If there's no link, fail */
Andy Fleming063c1262011-04-08 02:10:54 -0500614 return priv->phydev->link ? 0 : -1;
Mingkai Hu90751912011-01-27 12:52:46 +0800615}
616
Andy Fleming063c1262011-04-08 02:10:54 -0500617static phy_interface_t tsec_get_interface(struct tsec_private *priv)
618{
Claudiu Manoilaec84bf2013-09-30 12:44:42 +0300619 struct tsec __iomem *regs = priv->regs;
Andy Fleming063c1262011-04-08 02:10:54 -0500620 u32 ecntrl;
621
622 ecntrl = in_be32(&regs->ecntrl);
623
624 if (ecntrl & ECNTRL_SGMII_MODE)
625 return PHY_INTERFACE_MODE_SGMII;
626
627 if (ecntrl & ECNTRL_TBI_MODE) {
628 if (ecntrl & ECNTRL_REDUCED_MODE)
629 return PHY_INTERFACE_MODE_RTBI;
630 else
631 return PHY_INTERFACE_MODE_TBI;
632 }
633
634 if (ecntrl & ECNTRL_REDUCED_MODE) {
Mario Sixd38de332018-01-15 11:08:21 +0100635 phy_interface_t interface;
636
Andy Fleming063c1262011-04-08 02:10:54 -0500637 if (ecntrl & ECNTRL_REDUCED_MII_MODE)
638 return PHY_INTERFACE_MODE_RMII;
Andy Fleming063c1262011-04-08 02:10:54 -0500639
Mario Sixd38de332018-01-15 11:08:21 +0100640 interface = priv->interface;
Andy Fleming063c1262011-04-08 02:10:54 -0500641
Mario Sixd38de332018-01-15 11:08:21 +0100642 /*
643 * This isn't autodetected, so it must
644 * be set by the platform code.
645 */
646 if (interface == PHY_INTERFACE_MODE_RGMII_ID ||
647 interface == PHY_INTERFACE_MODE_RGMII_TXID ||
648 interface == PHY_INTERFACE_MODE_RGMII_RXID)
649 return interface;
650
651 return PHY_INTERFACE_MODE_RGMII;
Andy Fleming063c1262011-04-08 02:10:54 -0500652 }
653
654 if (priv->flags & TSEC_GIGABIT)
655 return PHY_INTERFACE_MODE_GMII;
656
657 return PHY_INTERFACE_MODE_MII;
658}
659
Bin Meng9872b732016-01-11 22:41:18 -0800660/*
661 * Discover which PHY is attached to the device, and configure it
Mingkai Hu90751912011-01-27 12:52:46 +0800662 * properly. If the PHY is not recognized, then return 0
663 * (failure). Otherwise, return 1
664 */
Bin Meng56a27a12016-01-11 22:41:22 -0800665static int init_phy(struct tsec_private *priv)
Mingkai Hu90751912011-01-27 12:52:46 +0800666{
Andy Fleming063c1262011-04-08 02:10:54 -0500667 struct phy_device *phydev;
Claudiu Manoilaec84bf2013-09-30 12:44:42 +0300668 struct tsec __iomem *regs = priv->regs;
Andy Fleming063c1262011-04-08 02:10:54 -0500669 u32 supported = (SUPPORTED_10baseT_Half |
670 SUPPORTED_10baseT_Full |
671 SUPPORTED_100baseT_Half |
672 SUPPORTED_100baseT_Full);
673
674 if (priv->flags & TSEC_GIGABIT)
675 supported |= SUPPORTED_1000baseT_Full;
Mingkai Hu90751912011-01-27 12:52:46 +0800676
677 /* Assign a Physical address to the TBI */
Bin Menga1c76c12016-01-11 22:41:25 -0800678 out_be32(&regs->tbipa, priv->tbiaddr);
Mingkai Hu90751912011-01-27 12:52:46 +0800679
Andy Fleming063c1262011-04-08 02:10:54 -0500680 priv->interface = tsec_get_interface(priv);
Mingkai Hu90751912011-01-27 12:52:46 +0800681
Andy Fleming063c1262011-04-08 02:10:54 -0500682 if (priv->interface == PHY_INTERFACE_MODE_SGMII)
Mingkai Hu90751912011-01-27 12:52:46 +0800683 tsec_configure_serdes(priv);
684
Hou Zhiqiangb4eb9cf2020-07-16 18:09:12 +0800685#if defined(CONFIG_DM_ETH) && defined(CONFIG_DM_MDIO)
Hou Zhiqiang25a2e242020-07-16 18:09:13 +0800686 if (ofnode_valid(ofnode_find_subnode(priv->dev->node, "fixed-link")))
687 phydev = phy_connect(NULL, 0, priv->dev, priv->interface);
688 else
689 phydev = dm_eth_phy_connect(priv->dev);
Hou Zhiqiangb4eb9cf2020-07-16 18:09:12 +0800690#else
Bin Meng56a27a12016-01-11 22:41:22 -0800691 phydev = phy_connect(priv->bus, priv->phyaddr, priv->dev,
692 priv->interface);
Hou Zhiqiangb4eb9cf2020-07-16 18:09:12 +0800693#endif
Claudiu Manoil7f233c02013-12-10 15:21:04 +0200694 if (!phydev)
695 return 0;
Mingkai Hu90751912011-01-27 12:52:46 +0800696
Andy Fleming063c1262011-04-08 02:10:54 -0500697 phydev->supported &= supported;
698 phydev->advertising = phydev->supported;
699
700 priv->phydev = phydev;
701
702 phy_config(phydev);
Mingkai Hu90751912011-01-27 12:52:46 +0800703
704 return 1;
705}
706
Bin Meng9a1d6af2016-01-11 22:41:24 -0800707#ifndef CONFIG_DM_ETH
Bin Meng9872b732016-01-11 22:41:18 -0800708/*
709 * Initialize device structure. Returns success if PHY
Mingkai Hu90751912011-01-27 12:52:46 +0800710 * initialization succeeded (i.e. if it recognizes the PHY)
711 */
Masahiro Yamadab75d8dc2020-06-26 15:13:33 +0900712static int tsec_initialize(struct bd_info *bis,
713 struct tsec_info_struct *tsec_info)
Mingkai Hu90751912011-01-27 12:52:46 +0800714{
Vladimir Oltean07bd39f2019-07-19 00:29:55 +0300715 struct tsec_private *priv;
Mingkai Hu90751912011-01-27 12:52:46 +0800716 struct eth_device *dev;
717 int i;
Mingkai Hu90751912011-01-27 12:52:46 +0800718
Mario Sixd38de332018-01-15 11:08:21 +0100719 dev = (struct eth_device *)malloc(sizeof(*dev));
Mingkai Hu90751912011-01-27 12:52:46 +0800720
Mario Sixd38de332018-01-15 11:08:21 +0100721 if (!dev)
Mingkai Hu90751912011-01-27 12:52:46 +0800722 return 0;
723
Mario Sixd38de332018-01-15 11:08:21 +0100724 memset(dev, 0, sizeof(*dev));
Mingkai Hu90751912011-01-27 12:52:46 +0800725
726 priv = (struct tsec_private *)malloc(sizeof(*priv));
727
Mario Six5775f002018-01-15 11:08:22 +0100728 if (!priv) {
729 free(dev);
Mingkai Hu90751912011-01-27 12:52:46 +0800730 return 0;
Mario Six5775f002018-01-15 11:08:22 +0100731 }
Mingkai Hu90751912011-01-27 12:52:46 +0800732
Mingkai Hu90751912011-01-27 12:52:46 +0800733 priv->regs = tsec_info->regs;
Mingkai Hu90751912011-01-27 12:52:46 +0800734 priv->phyregs_sgmii = tsec_info->miiregs_sgmii;
735
736 priv->phyaddr = tsec_info->phyaddr;
Bin Menga1c76c12016-01-11 22:41:25 -0800737 priv->tbiaddr = CONFIG_SYS_TBIPA_VALUE;
Mingkai Hu90751912011-01-27 12:52:46 +0800738 priv->flags = tsec_info->flags;
739
Ben Whitten192bc692015-12-30 13:05:58 +0000740 strcpy(dev->name, tsec_info->devname);
Andy Fleming063c1262011-04-08 02:10:54 -0500741 priv->interface = tsec_info->interface;
742 priv->bus = miiphy_get_dev_by_name(tsec_info->mii_devname);
Bin Meng56a27a12016-01-11 22:41:22 -0800743 priv->dev = dev;
Mingkai Hu90751912011-01-27 12:52:46 +0800744 dev->iobase = 0;
745 dev->priv = priv;
746 dev->init = tsec_init;
747 dev->halt = tsec_halt;
748 dev->send = tsec_send;
749 dev->recv = tsec_recv;
Mingkai Hu90751912011-01-27 12:52:46 +0800750 dev->mcast = tsec_mcast_addr;
Mingkai Hu90751912011-01-27 12:52:46 +0800751
Bin Meng9872b732016-01-11 22:41:18 -0800752 /* Tell U-Boot to get the addr from the env */
Mingkai Hu90751912011-01-27 12:52:46 +0800753 for (i = 0; i < 6; i++)
754 dev->enetaddr[i] = 0;
755
756 eth_register(dev);
757
758 /* Reset the MAC */
759 setbits_be32(&priv->regs->maccfg1, MACCFG1_SOFT_RESET);
760 udelay(2); /* Soft Reset must be asserted for 3 TX clocks */
761 clrbits_be32(&priv->regs->maccfg1, MACCFG1_SOFT_RESET);
762
Mingkai Hu90751912011-01-27 12:52:46 +0800763 /* Try to initialize PHY here, and return */
Bin Meng56a27a12016-01-11 22:41:22 -0800764 return init_phy(priv);
Mingkai Hu90751912011-01-27 12:52:46 +0800765}
766
767/*
768 * Initialize all the TSEC devices
769 *
770 * Returns the number of TSEC devices that were initialized
771 */
Masahiro Yamadab75d8dc2020-06-26 15:13:33 +0900772int tsec_eth_init(struct bd_info *bis, struct tsec_info_struct *tsecs,
773 int num)
Mingkai Hu90751912011-01-27 12:52:46 +0800774{
775 int i;
Mario Sixd38de332018-01-15 11:08:21 +0100776 int count = 0;
Mingkai Hu90751912011-01-27 12:52:46 +0800777
778 for (i = 0; i < num; i++) {
Mario Sixd38de332018-01-15 11:08:21 +0100779 int ret = tsec_initialize(bis, &tsecs[i]);
780
Mingkai Hu90751912011-01-27 12:52:46 +0800781 if (ret > 0)
782 count += ret;
783 }
784
785 return count;
786}
787
Masahiro Yamadab75d8dc2020-06-26 15:13:33 +0900788int tsec_standard_init(struct bd_info *bis)
Mingkai Hu90751912011-01-27 12:52:46 +0800789{
Andy Fleming063c1262011-04-08 02:10:54 -0500790 struct fsl_pq_mdio_info info;
791
Claudiu Manoilaec84bf2013-09-30 12:44:42 +0300792 info.regs = TSEC_GET_MDIO_REGS_BASE(1);
Andy Fleming063c1262011-04-08 02:10:54 -0500793 info.name = DEFAULT_MII_NAME;
794
795 fsl_pq_mdio_init(bis, &info);
796
Mingkai Hu90751912011-01-27 12:52:46 +0800797 return tsec_eth_init(bis, tsec_info, ARRAY_SIZE(tsec_info));
798}
Bin Meng9a1d6af2016-01-11 22:41:24 -0800799#else /* CONFIG_DM_ETH */
800int tsec_probe(struct udevice *dev)
801{
Bin Meng9a1d6af2016-01-11 22:41:24 -0800802 struct eth_pdata *pdata = dev_get_platdata(dev);
Vladimir Oltean07bd39f2019-07-19 00:29:55 +0300803 struct tsec_private *priv = dev_get_priv(dev);
Mario Six1313aaf2018-01-15 11:08:23 +0100804 struct ofnode_phandle_args phandle_args;
Vladimir Oltean29db3102019-07-19 00:29:53 +0300805 u32 tbiaddr = CONFIG_SYS_TBIPA_VALUE;
Hou Zhiqiang7fb568d2020-07-16 18:09:14 +0800806 struct tsec_data *data;
Bin Meng9a1d6af2016-01-11 22:41:24 -0800807 const char *phy_mode;
Vladimir Olteanbca686a2019-07-19 00:29:54 +0300808 fdt_addr_t reg;
Vladimir Oltean07bd39f2019-07-19 00:29:55 +0300809 ofnode parent;
Bin Meng9a1d6af2016-01-11 22:41:24 -0800810 int ret;
811
Hou Zhiqiang7fb568d2020-07-16 18:09:14 +0800812 data = (struct tsec_data *)dev_get_driver_data(dev);
813
Mario Six1313aaf2018-01-15 11:08:23 +0100814 pdata->iobase = (phys_addr_t)dev_read_addr(dev);
Hou Zhiqiangbf67eb32020-05-03 22:48:42 +0800815 priv->regs = dev_remap_addr(dev);
Bin Meng9a1d6af2016-01-11 22:41:24 -0800816
Vladimir Oltean29db3102019-07-19 00:29:53 +0300817 ret = dev_read_phandle_with_args(dev, "tbi-handle", NULL, 0, 0,
818 &phandle_args);
Hou Zhiqianga0f47e02020-05-03 22:48:43 +0800819 if (ret == 0) {
Vladimir Oltean29db3102019-07-19 00:29:53 +0300820 ofnode_read_u32(phandle_args.node, "reg", &tbiaddr);
821
Hou Zhiqianga0f47e02020-05-03 22:48:43 +0800822 parent = ofnode_get_parent(phandle_args.node);
823 if (!ofnode_valid(parent)) {
824 printf("No parent node for TBI PHY?\n");
825 return -ENOENT;
826 }
827
828 reg = ofnode_get_addr_index(parent, 0);
829 if (reg == FDT_ADDR_T_NONE) {
830 printf("No 'reg' property of MII for TBI PHY\n");
831 return -ENOENT;
832 }
833
Hou Zhiqiang7fb568d2020-07-16 18:09:14 +0800834 priv->phyregs_sgmii = map_physmem(reg + data->mdio_regs_off,
Hou Zhiqianga0f47e02020-05-03 22:48:43 +0800835 0, MAP_NOCACHE);
836 }
837
Vladimir Oltean29db3102019-07-19 00:29:53 +0300838 priv->tbiaddr = tbiaddr;
Bin Menga1c76c12016-01-11 22:41:25 -0800839
Mario Six1313aaf2018-01-15 11:08:23 +0100840 phy_mode = dev_read_prop(dev, "phy-connection-type", NULL);
Bin Meng9a1d6af2016-01-11 22:41:24 -0800841 if (phy_mode)
842 pdata->phy_interface = phy_get_interface_by_name(phy_mode);
843 if (pdata->phy_interface == -1) {
Vladimir Olteanb7be7762019-07-19 00:29:56 +0300844 printf("Invalid PHY interface '%s'\n", phy_mode);
Bin Meng9a1d6af2016-01-11 22:41:24 -0800845 return -EINVAL;
846 }
847 priv->interface = pdata->phy_interface;
848
849 /* Initialize flags */
850 priv->flags = TSEC_GIGABIT;
851 if (priv->interface == PHY_INTERFACE_MODE_SGMII)
852 priv->flags |= TSEC_SGMII;
853
Bin Meng9a1d6af2016-01-11 22:41:24 -0800854 /* Reset the MAC */
855 setbits_be32(&priv->regs->maccfg1, MACCFG1_SOFT_RESET);
856 udelay(2); /* Soft Reset must be asserted for 3 TX clocks */
857 clrbits_be32(&priv->regs->maccfg1, MACCFG1_SOFT_RESET);
858
859 priv->dev = dev;
860 priv->bus = miiphy_get_dev_by_name(dev->name);
861
862 /* Try to initialize PHY here, and return */
863 return !init_phy(priv);
864}
865
866int tsec_remove(struct udevice *dev)
867{
868 struct tsec_private *priv = dev->priv;
869
870 free(priv->phydev);
871 mdio_unregister(priv->bus);
872 mdio_free(priv->bus);
873
874 return 0;
875}
876
877static const struct eth_ops tsec_ops = {
878 .start = tsec_init,
879 .send = tsec_send,
880 .recv = tsec_recv,
881 .free_pkt = tsec_free_pkt,
882 .stop = tsec_halt,
Bin Meng9a1d6af2016-01-11 22:41:24 -0800883 .mcast = tsec_mcast_addr,
Bin Meng9a1d6af2016-01-11 22:41:24 -0800884};
885
Hou Zhiqiang7fb568d2020-07-16 18:09:14 +0800886static struct tsec_data etsec2_data = {
887 .mdio_regs_off = TSEC_MDIO_REGS_OFFSET,
888};
889
890static struct tsec_data gianfar_data = {
891 .mdio_regs_off = 0x0,
892};
893
Bin Meng9a1d6af2016-01-11 22:41:24 -0800894static const struct udevice_id tsec_ids[] = {
Hou Zhiqiang7fb568d2020-07-16 18:09:14 +0800895 { .compatible = "fsl,etsec2", .data = (ulong)&etsec2_data },
896 { .compatible = "gianfar", .data = (ulong)&gianfar_data },
Bin Meng9a1d6af2016-01-11 22:41:24 -0800897 { }
898};
899
900U_BOOT_DRIVER(eth_tsec) = {
901 .name = "tsec",
902 .id = UCLASS_ETH,
903 .of_match = tsec_ids,
904 .probe = tsec_probe,
905 .remove = tsec_remove,
906 .ops = &tsec_ops,
907 .priv_auto_alloc_size = sizeof(struct tsec_private),
908 .platdata_auto_alloc_size = sizeof(struct eth_pdata),
909 .flags = DM_FLAG_ALLOC_PRIV_DMA,
910};
911#endif /* CONFIG_DM_ETH */