blob: 8c53059ae2ae7232b7971f5fe392a8df94d6b5d8 [file] [log] [blame]
Tom Rini10e47792018-05-06 17:58:06 -04001// SPDX-License-Identifier: GPL-2.0+
wdenk9c53f402003-10-15 23:53:47 +00002/*
wdenka445ddf2004-06-09 00:34:46 +00003 * Freescale Three Speed Ethernet Controller driver
wdenk9c53f402003-10-15 23:53:47 +00004 *
Claudiu Manoilcd0c4122013-09-30 12:44:42 +03005 * Copyright 2004-2011, 2013 Freescale Semiconductor, Inc.
wdenk9c53f402003-10-15 23:53:47 +00006 * (C) Copyright 2003, Motorola, Inc.
wdenk9c53f402003-10-15 23:53:47 +00007 * author Andy Fleming
wdenk9c53f402003-10-15 23:53:47 +00008 */
9
10#include <config.h>
wdenk9c53f402003-10-15 23:53:47 +000011#include <common.h>
Bin Meng1048f612016-01-11 22:41:24 -080012#include <dm.h>
wdenk9c53f402003-10-15 23:53:47 +000013#include <malloc.h>
14#include <net.h>
15#include <command.h>
Andy Flemingc067fc12008-08-31 16:33:25 -050016#include <tsec.h>
Andy Fleming422effd2011-04-08 02:10:54 -050017#include <fsl_mdio.h>
Simon Glass4dcacfc2020-05-10 11:40:13 -060018#include <linux/bitops.h>
Simon Glassdbd79542020-05-10 11:40:11 -060019#include <linux/delay.h>
Masahiro Yamada56a931c2016-09-21 11:28:55 +090020#include <linux/errno.h>
Hou Zhiqiangd35de972020-07-16 18:09:12 +080021#include <miiphy.h>
chenhui zhaoc8caa8a2011-10-03 08:38:50 -050022#include <asm/processor.h>
Alison Wang32cc5912014-09-05 13:52:38 +080023#include <asm/io.h>
wdenk9c53f402003-10-15 23:53:47 +000024
Bin Meng1048f612016-01-11 22:41:24 -080025#ifndef CONFIG_DM_ETH
Andy Flemingfecff2b2008-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 Manoilcd0c4122013-09-30 12:44:42 +030037 .regs = TSEC_GET_REGS(2, 0x2000),
Andy Flemingfecff2b2008-08-31 16:33:26 -050038 .devname = CONFIG_MPC85XX_FEC_NAME,
39 .phyaddr = FEC_PHY_ADDR,
Andy Fleming422effd2011-04-08 02:10:54 -050040 .flags = FEC_FLAGS,
41 .mii_devname = DEFAULT_MII_NAME
Andy Flemingfecff2b2008-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 Meng1048f612016-01-11 22:41:24 -080051#endif /* CONFIG_DM_ETH */
Andy Flemingfecff2b2008-08-31 16:33:26 -050052
Andy Flemingac65e072008-08-31 16:33:27 -050053#define TBIANA_SETTINGS ( \
54 TBIANA_ASYMMETRIC_PAUSE \
55 | TBIANA_SYMMETRIC_PAUSE \
56 | TBIANA_FULL_DUPLEX \
57 )
58
Felix Radensky27f98e02010-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 Galac1457f92010-12-01 22:55:54 -060061#define CONFIG_TSEC_TBICR_SETTINGS ( \
Andy Flemingac65e072008-08-31 16:33:27 -050062 TBICR_PHY_RESET \
Kumar Galac1457f92010-12-01 22:55:54 -060063 | TBICR_ANEG_ENABLE \
Andy Flemingac65e072008-08-31 16:33:27 -050064 | TBICR_FULL_DUPLEX \
65 | TBICR_SPEED1_SET \
66 )
Felix Radensky27f98e02010-06-28 01:57:39 +030067#endif /* CONFIG_TSEC_TBICR_SETTINGS */
Peter Tyser583c1f42009-11-03 17:52:07 -060068
Andy Flemingac65e072008-08-31 16:33:27 -050069/* Configure the TBI for SGMII operation */
70static void tsec_configure_serdes(struct tsec_private *priv)
71{
Bin Meng79cd33a2016-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 Fleming422effd2011-04-08 02:10:54 -050076 tsec_local_mdio_write(priv->phyregs_sgmii, in_be32(&priv->regs->tbipa),
Mario Sixc29fcc72018-01-15 11:08:21 +010077 0, TBI_ANA, TBIANA_SETTINGS);
Andy Fleming422effd2011-04-08 02:10:54 -050078 tsec_local_mdio_write(priv->phyregs_sgmii, in_be32(&priv->regs->tbipa),
Mario Sixc29fcc72018-01-15 11:08:21 +010079 0, TBI_TBICON, TBICON_CLK_SELECT);
Andy Fleming422effd2011-04-08 02:10:54 -050080 tsec_local_mdio_write(priv->phyregs_sgmii, in_be32(&priv->regs->tbipa),
Mario Sixc29fcc72018-01-15 11:08:21 +010081 0, TBI_CR, CONFIG_TSEC_TBICR_SETTINGS);
David Updegraff0451b012007-04-20 14:34:48 -050082}
83
Chris Packhambbe18572018-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
Mingkai Hue0653bf2011-01-27 12:52:46 +0800109/* CREDITS: linux gianfar driver, slightly adjusted... thanx. */
110
111/* Set the appropriate hash bit for the given addr */
112
Bin Meng79cd33a2016-01-11 22:41:18 -0800113/*
114 * The algorithm works like so:
Mingkai Hue0653bf2011-01-27 12:52:46 +0800115 * 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 Manoil461511b2013-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
Mingkai Hue0653bf2011-01-27 12:52:46 +0800122 * hash index which gaddr register to use, and the 5 other bits
123 * indicate which bit (assuming an IBM numbering scheme, which
Claudiu Manoil461511b2013-09-30 12:44:40 +0300124 * for PowerPC (tm) is usually the case) in the register holds
Bin Meng79cd33a2016-01-11 22:41:18 -0800125 * the entry.
126 */
Bin Meng1048f612016-01-11 22:41:24 -0800127#ifndef CONFIG_DM_ETH
Chris Packhama55ef7f2018-11-26 21:00:29 +1300128static int tsec_mcast_addr(struct eth_device *dev, const u8 *mcast_mac,
129 int join)
Bin Meng1048f612016-01-11 22:41:24 -0800130#else
Chris Packhama55ef7f2018-11-26 21:00:29 +1300131static int tsec_mcast_addr(struct udevice *dev, const u8 *mcast_mac, int join)
Bin Meng1048f612016-01-11 22:41:24 -0800132#endif
Mingkai Hue0653bf2011-01-27 12:52:46 +0800133{
Claudiu Manoil766c8942013-09-30 12:44:41 +0300134 struct tsec_private *priv = (struct tsec_private *)dev->priv;
Claudiu Manoil461511b2013-09-30 12:44:40 +0300135 struct tsec __iomem *regs = priv->regs;
136 u32 result, value;
137 u8 whichbit, whichreg;
Mingkai Hue0653bf2011-01-27 12:52:46 +0800138
Claudiu Manoil461511b2013-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 */
Mingkai Hue0653bf2011-01-27 12:52:46 +0800142
Mario Sixc29fcc72018-01-15 11:08:21 +0100143 value = BIT(31 - whichbit);
Claudiu Manoil461511b2013-09-30 12:44:40 +0300144
Chris Packhama55ef7f2018-11-26 21:00:29 +1300145 if (join)
Claudiu Manoil461511b2013-09-30 12:44:40 +0300146 setbits_be32(&regs->hash.gaddr0 + whichreg, value);
147 else
148 clrbits_be32(&regs->hash.gaddr0 + whichreg, value);
Mingkai Hue0653bf2011-01-27 12:52:46 +0800149
Mingkai Hue0653bf2011-01-27 12:52:46 +0800150 return 0;
151}
Mingkai Hue0653bf2011-01-27 12:52:46 +0800152
Bin Meng79cd33a2016-01-11 22:41:18 -0800153/*
154 * Initialized required registers to appropriate values, zeroing
Mingkai Hue0653bf2011-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 Manoilcd0c4122013-09-30 12:44:42 +0300158static void init_registers(struct tsec __iomem *regs)
Mingkai Hue0653bf2011-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 Manoila18ab902013-09-30 12:44:46 +0300186 memset((void *)&regs->rmon, 0, sizeof(regs->rmon));
Mingkai Hue0653bf2011-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 Hue0653bf2011-01-27 12:52:46 +0800197}
198
Bin Meng79cd33a2016-01-11 22:41:18 -0800199/*
200 * Configure maccfg2 based on negotiated speed and duplex
Mingkai Hue0653bf2011-01-27 12:52:46 +0800201 * reported by PHY handling code
202 */
Andy Fleming422effd2011-04-08 02:10:54 -0500203static void adjust_link(struct tsec_private *priv, struct phy_device *phydev)
Mingkai Hue0653bf2011-01-27 12:52:46 +0800204{
Claudiu Manoilcd0c4122013-09-30 12:44:42 +0300205 struct tsec __iomem *regs = priv->regs;
Mingkai Hue0653bf2011-01-27 12:52:46 +0800206 u32 ecntrl, maccfg2;
207
Andy Fleming422effd2011-04-08 02:10:54 -0500208 if (!phydev->link) {
209 printf("%s: No link.\n", phydev->dev->name);
Mingkai Hue0653bf2011-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 Fleming422effd2011-04-08 02:10:54 -0500220 if (phydev->duplex)
Mingkai Hue0653bf2011-01-27 12:52:46 +0800221 maccfg2 |= MACCFG2_FULL_DUPLEX;
222
Andy Fleming422effd2011-04-08 02:10:54 -0500223 switch (phydev->speed) {
Mingkai Hue0653bf2011-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 Meng79cd33a2016-01-11 22:41:18 -0800231 /*
232 * Set R100 bit in all modes although
Mingkai Hue0653bf2011-01-27 12:52:46 +0800233 * it is only used in RGMII mode
234 */
Andy Fleming422effd2011-04-08 02:10:54 -0500235 if (phydev->speed == 100)
Mingkai Hue0653bf2011-01-27 12:52:46 +0800236 ecntrl |= ECNTRL_R100;
237 break;
238 default:
Andy Fleming422effd2011-04-08 02:10:54 -0500239 printf("%s: Speed was bad\n", phydev->dev->name);
Mingkai Hue0653bf2011-01-27 12:52:46 +0800240 break;
241 }
242
243 out_be32(&regs->ecntrl, ecntrl);
244 out_be32(&regs->maccfg2, maccfg2);
wdenkf41ff3b2005-04-04 23:43:44 +0000245
Andy Fleming422effd2011-04-08 02:10:54 -0500246 printf("Speed: %d, %s duplex%s\n", phydev->speed,
Mario Sixc29fcc72018-01-15 11:08:21 +0100247 (phydev->duplex) ? "full" : "half",
248 (phydev->port == PORT_FIBRE) ? ", fiber mode" : "");
Mingkai Hue0653bf2011-01-27 12:52:46 +0800249}
wdenkbfad55d2005-03-14 23:56:42 +0000250
Bin Meng80b1a1c2016-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 Meng1048f612016-01-11 22:41:24 -0800257#ifndef CONFIG_DM_ETH
Bin Meng80b1a1c2016-01-11 22:41:21 -0800258static int tsec_send(struct eth_device *dev, void *packet, int length)
Bin Meng1048f612016-01-11 22:41:24 -0800259#else
260static int tsec_send(struct udevice *dev, void *packet, int length)
261#endif
Bin Meng80b1a1c2016-01-11 22:41:21 -0800262{
263 struct tsec_private *priv = (struct tsec_private *)dev->priv;
264 struct tsec __iomem *regs = priv->regs;
Bin Meng80b1a1c2016-01-11 22:41:21 -0800265 int result = 0;
Vladimir Olteana11c89d2019-07-19 00:29:55 +0300266 u16 status;
Bin Meng80b1a1c2016-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 Oltean8ec8eaa2019-07-19 00:29:56 +0300274 printf("%s: tsec: tx buffers full\n", dev->name);
Bin Meng80b1a1c2016-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 Oltean8ec8eaa2019-07-19 00:29:56 +0300293 printf("%s: tsec: tx error\n", dev->name);
Bin Meng80b1a1c2016-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 Meng1048f612016-01-11 22:41:24 -0800304#ifndef CONFIG_DM_ETH
Bin Meng80b1a1c2016-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 Sixc29fcc72018-01-15 11:08:21 +0100312 u16 status = in_be16(&priv->rxbd[priv->rx_idx].status);
Bin Meng80b1a1c2016-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 Meng1048f612016-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 Sixc29fcc72018-01-15 11:08:21 +0100348 u16 status = in_be16(&priv->rxbd[priv->rx_idx].status);
349 u32 buf;
Bin Meng1048f612016-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 Sixc29fcc72018-01-15 11:08:21 +0100372 u16 status;
Bin Meng1048f612016-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 Meng80b1a1c2016-01-11 22:41:21 -0800387
388/* Stop the interface */
Bin Meng1048f612016-01-11 22:41:24 -0800389#ifndef CONFIG_DM_ETH
Bin Meng80b1a1c2016-01-11 22:41:21 -0800390static void tsec_halt(struct eth_device *dev)
Bin Meng1048f612016-01-11 22:41:24 -0800391#else
392static void tsec_halt(struct udevice *dev)
393#endif
Bin Meng80b1a1c2016-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 zhaoc8caa8a2011-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 Menge86a6cd2016-01-11 22:41:22 -0800417void redundant_init(struct tsec_private *priv)
chenhui zhaoc8caa8a2011-10-03 08:38:50 -0500418{
Claudiu Manoilcd0c4122013-09-30 12:44:42 +0300419 struct tsec __iomem *regs = priv->regs;
chenhui zhaoc8caa8a2011-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 Sixc29fcc72018-01-15 11:08:21 +0100452 u16 status;
453
Bin Menge86a6cd2016-01-11 22:41:22 -0800454 tsec_send(priv->dev, (void *)pkt, sizeof(pkt));
chenhui zhaoc8caa8a2011-10-03 08:38:50 -0500455
456 /* Wait for buffer to be received */
Bin Meng1120c542016-01-11 22:41:20 -0800457 for (t = 0;
458 in_be16(&priv->rxbd[priv->rx_idx].status) & RXBD_EMPTY;
Bin Meng76f53992016-01-11 22:41:19 -0800459 t++) {
chenhui zhaoc8caa8a2011-10-03 08:38:50 -0500460 if (t >= 10 * TOUT_LOOP) {
Bin Menge86a6cd2016-01-11 22:41:22 -0800461 printf("%s: tsec: rx error\n", priv->dev->name);
chenhui zhaoc8caa8a2011-10-03 08:38:50 -0500462 break;
463 }
464 }
465
Bin Meng76f53992016-01-11 22:41:19 -0800466 if (!memcmp(pkt, net_rx_packets[priv->rx_idx], sizeof(pkt)))
chenhui zhaoc8caa8a2011-10-03 08:38:50 -0500467 fail = 0;
468
Bin Meng1120c542016-01-11 22:41:20 -0800469 out_be16(&priv->rxbd[priv->rx_idx].length, 0);
Claudiu Manoileec416b2013-10-04 19:13:53 +0300470 status = RXBD_EMPTY;
Bin Meng76f53992016-01-11 22:41:19 -0800471 if ((priv->rx_idx + 1) == PKTBUFSRX)
Claudiu Manoileec416b2013-10-04 19:13:53 +0300472 status |= RXBD_WRAP;
Bin Meng1120c542016-01-11 22:41:20 -0800473 out_be16(&priv->rxbd[priv->rx_idx].status, status);
Bin Meng76f53992016-01-11 22:41:19 -0800474 priv->rx_idx = (priv->rx_idx + 1) % PKTBUFSRX;
chenhui zhaoc8caa8a2011-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 Meng79cd33a2016-01-11 22:41:18 -0800497/*
498 * Set up the buffers and their descriptors, and bring up the
Mingkai Hue0653bf2011-01-27 12:52:46 +0800499 * interface
Jon Loeligerb7ced082006-10-10 17:03:43 -0500500 */
Bin Menge86a6cd2016-01-11 22:41:22 -0800501static void startup_tsec(struct tsec_private *priv)
Wolfgang Denkf0c4e462006-03-12 22:50:55 +0100502{
Claudiu Manoilcd0c4122013-09-30 12:44:42 +0300503 struct tsec __iomem *regs = priv->regs;
Mario Sixc29fcc72018-01-15 11:08:21 +0100504 u16 status;
Claudiu Manoileec416b2013-10-04 19:13:53 +0300505 int i;
Wolfgang Denkf0c4e462006-03-12 22:50:55 +0100506
Andy Fleming422effd2011-04-08 02:10:54 -0500507 /* reset the indices to zero */
Bin Meng76f53992016-01-11 22:41:19 -0800508 priv->rx_idx = 0;
509 priv->tx_idx = 0;
chenhui zhaoc8caa8a2011-10-03 08:38:50 -0500510#ifdef CONFIG_SYS_FSL_ERRATUM_NMG_ETSEC129
511 uint svr;
512#endif
Andy Fleming422effd2011-04-08 02:10:54 -0500513
Mingkai Hue0653bf2011-01-27 12:52:46 +0800514 /* Point to the buffer descriptors */
Bin Meng1120c542016-01-11 22:41:20 -0800515 out_be32(&regs->tbase, (u32)&priv->txbd[0]);
516 out_be32(&regs->rbase, (u32)&priv->rxbd[0]);
Wolfgang Denkf0c4e462006-03-12 22:50:55 +0100517
Mingkai Hue0653bf2011-01-27 12:52:46 +0800518 /* Initialize the Rx Buffer descriptors */
519 for (i = 0; i < PKTBUFSRX; i++) {
Bin Meng1120c542016-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 Hue0653bf2011-01-27 12:52:46 +0800523 }
Bin Meng1120c542016-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);
Wolfgang Denkf0c4e462006-03-12 22:50:55 +0100526
Mingkai Hue0653bf2011-01-27 12:52:46 +0800527 /* Initialize the TX Buffer Descriptors */
528 for (i = 0; i < TX_BUF_CNT; i++) {
Bin Meng1120c542016-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);
Wolfgang Denkf0c4e462006-03-12 22:50:55 +0100532 }
Bin Meng1120c542016-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);
Wolfgang Denkf0c4e462006-03-12 22:50:55 +0100535
chenhui zhaoc8caa8a2011-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 Menge86a6cd2016-01-11 22:41:22 -0800539 redundant_init(priv);
chenhui zhaoc8caa8a2011-10-03 08:38:50 -0500540#endif
Mingkai Hue0653bf2011-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);
Wolfgang Denkf0c4e462006-03-12 22:50:55 +0100549}
550
Bin Meng79cd33a2016-01-11 22:41:18 -0800551/*
Bin Meng79cd33a2016-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 Hue0653bf2011-01-27 12:52:46 +0800554 * that it returns success if the link is up, failure otherwise.
Bin Meng79cd33a2016-01-11 22:41:18 -0800555 * This allows U-Boot to find the first active controller.
Jon Loeligerb7ced082006-10-10 17:03:43 -0500556 */
Bin Meng1048f612016-01-11 22:41:24 -0800557#ifndef CONFIG_DM_ETH
Masahiro Yamadaf7ed78b2020-06-26 15:13:33 +0900558static int tsec_init(struct eth_device *dev, struct bd_info *bd)
Bin Meng1048f612016-01-11 22:41:24 -0800559#else
560static int tsec_init(struct udevice *dev)
561#endif
wdenka445ddf2004-06-09 00:34:46 +0000562{
Mingkai Hue0653bf2011-01-27 12:52:46 +0800563 struct tsec_private *priv = (struct tsec_private *)dev->priv;
Bin Meng1048f612016-01-11 22:41:24 -0800564#ifdef CONFIG_DM_ETH
565 struct eth_pdata *pdata = dev_get_platdata(dev);
Vladimir Oltean9953c792019-07-19 00:29:57 +0300566#else
567 struct eth_device *pdata = dev;
Bin Meng1048f612016-01-11 22:41:24 -0800568#endif
Claudiu Manoilcd0c4122013-09-30 12:44:42 +0300569 struct tsec __iomem *regs = priv->regs;
Claudiu Manoildcb38fe2013-09-30 12:44:47 +0300570 u32 tempval;
Timur Tabi42387462012-07-09 08:52:43 +0000571 int ret;
wdenka445ddf2004-06-09 00:34:46 +0000572
Mingkai Hue0653bf2011-01-27 12:52:46 +0800573 /* Make sure the controller is stopped */
574 tsec_halt(dev);
wdenka445ddf2004-06-09 00:34:46 +0000575
Mingkai Hue0653bf2011-01-27 12:52:46 +0800576 /* Init MACCFG2. Defaults to GMII */
577 out_be32(&regs->maccfg2, MACCFG2_INIT_SETTINGS);
wdenka445ddf2004-06-09 00:34:46 +0000578
Mingkai Hue0653bf2011-01-27 12:52:46 +0800579 /* Init ECNTRL */
580 out_be32(&regs->ecntrl, ECNTRL_INIT_SETTINGS);
wdenka445ddf2004-06-09 00:34:46 +0000581
Bin Meng79cd33a2016-01-11 22:41:18 -0800582 /*
583 * Copy the station address into the address registers.
Claudiu Manoildcb38fe2013-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 Meng1048f612016-01-11 22:41:24 -0800588 tempval = (pdata->enetaddr[5] << 24) | (pdata->enetaddr[4] << 16) |
589 (pdata->enetaddr[3] << 8) | pdata->enetaddr[2];
wdenka445ddf2004-06-09 00:34:46 +0000590
Mingkai Hue0653bf2011-01-27 12:52:46 +0800591 out_be32(&regs->macstnaddr1, tempval);
wdenka445ddf2004-06-09 00:34:46 +0000592
Bin Meng1048f612016-01-11 22:41:24 -0800593 tempval = (pdata->enetaddr[1] << 24) | (pdata->enetaddr[0] << 16);
wdenka445ddf2004-06-09 00:34:46 +0000594
Mingkai Hue0653bf2011-01-27 12:52:46 +0800595 out_be32(&regs->macstnaddr2, tempval);
wdenka445ddf2004-06-09 00:34:46 +0000596
Mingkai Hue0653bf2011-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 Menge86a6cd2016-01-11 22:41:22 -0800601 startup_tsec(priv);
Mingkai Hue0653bf2011-01-27 12:52:46 +0800602
Andy Fleming422effd2011-04-08 02:10:54 -0500603 /* Start up the PHY */
Timur Tabi42387462012-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 Fleming422effd2011-04-08 02:10:54 -0500610
611 adjust_link(priv, priv->phydev);
612
Mingkai Hue0653bf2011-01-27 12:52:46 +0800613 /* If there's no link, fail */
Andy Fleming422effd2011-04-08 02:10:54 -0500614 return priv->phydev->link ? 0 : -1;
615}
616
617static phy_interface_t tsec_get_interface(struct tsec_private *priv)
618{
Claudiu Manoilcd0c4122013-09-30 12:44:42 +0300619 struct tsec __iomem *regs = priv->regs;
Andy Fleming422effd2011-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 Sixc29fcc72018-01-15 11:08:21 +0100635 phy_interface_t interface;
636
Andy Fleming422effd2011-04-08 02:10:54 -0500637 if (ecntrl & ECNTRL_REDUCED_MII_MODE)
638 return PHY_INTERFACE_MODE_RMII;
Andy Fleming422effd2011-04-08 02:10:54 -0500639
Mario Sixc29fcc72018-01-15 11:08:21 +0100640 interface = priv->interface;
Andy Fleming422effd2011-04-08 02:10:54 -0500641
Mario Sixc29fcc72018-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 Fleming422effd2011-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;
Mingkai Hue0653bf2011-01-27 12:52:46 +0800658}
659
Bin Meng79cd33a2016-01-11 22:41:18 -0800660/*
661 * Discover which PHY is attached to the device, and configure it
Mingkai Hue0653bf2011-01-27 12:52:46 +0800662 * properly. If the PHY is not recognized, then return 0
663 * (failure). Otherwise, return 1
wdenk78924a72004-04-18 21:45:42 +0000664 */
Bin Menge86a6cd2016-01-11 22:41:22 -0800665static int init_phy(struct tsec_private *priv)
wdenk78924a72004-04-18 21:45:42 +0000666{
Andy Fleming422effd2011-04-08 02:10:54 -0500667 struct phy_device *phydev;
Claudiu Manoilcd0c4122013-09-30 12:44:42 +0300668 struct tsec __iomem *regs = priv->regs;
Andy Fleming422effd2011-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;
wdenk78924a72004-04-18 21:45:42 +0000676
Mingkai Hue0653bf2011-01-27 12:52:46 +0800677 /* Assign a Physical address to the TBI */
Bin Meng74314f12016-01-11 22:41:25 -0800678 out_be32(&regs->tbipa, priv->tbiaddr);
Mingkai Hue0653bf2011-01-27 12:52:46 +0800679
Andy Fleming422effd2011-04-08 02:10:54 -0500680 priv->interface = tsec_get_interface(priv);
Mingkai Hue0653bf2011-01-27 12:52:46 +0800681
Andy Fleming422effd2011-04-08 02:10:54 -0500682 if (priv->interface == PHY_INTERFACE_MODE_SGMII)
683 tsec_configure_serdes(priv);
Mingkai Hue0653bf2011-01-27 12:52:46 +0800684
Hou Zhiqiangd35de972020-07-16 18:09:12 +0800685#if defined(CONFIG_DM_ETH) && defined(CONFIG_DM_MDIO)
Hou Zhiqiang1addbbf2020-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 Zhiqiangd35de972020-07-16 18:09:12 +0800690#else
Bin Menge86a6cd2016-01-11 22:41:22 -0800691 phydev = phy_connect(priv->bus, priv->phyaddr, priv->dev,
692 priv->interface);
Hou Zhiqiangd35de972020-07-16 18:09:12 +0800693#endif
Claudiu Manoilfe56fec2013-12-10 15:21:04 +0200694 if (!phydev)
695 return 0;
Mingkai Hue0653bf2011-01-27 12:52:46 +0800696
Andy Fleming422effd2011-04-08 02:10:54 -0500697 phydev->supported &= supported;
698 phydev->advertising = phydev->supported;
wdenka445ddf2004-06-09 00:34:46 +0000699
Andy Fleming422effd2011-04-08 02:10:54 -0500700 priv->phydev = phydev;
wdenk78924a72004-04-18 21:45:42 +0000701
Andy Fleming422effd2011-04-08 02:10:54 -0500702 phy_config(phydev);
Mingkai Hue0653bf2011-01-27 12:52:46 +0800703
704 return 1;
wdenk78924a72004-04-18 21:45:42 +0000705}
706
Bin Meng1048f612016-01-11 22:41:24 -0800707#ifndef CONFIG_DM_ETH
Bin Meng79cd33a2016-01-11 22:41:18 -0800708/*
709 * Initialize device structure. Returns success if PHY
Mingkai Hue0653bf2011-01-27 12:52:46 +0800710 * initialization succeeded (i.e. if it recognizes the PHY)
wdenk78924a72004-04-18 21:45:42 +0000711 */
Masahiro Yamadaf7ed78b2020-06-26 15:13:33 +0900712static int tsec_initialize(struct bd_info *bis,
713 struct tsec_info_struct *tsec_info)
wdenk78924a72004-04-18 21:45:42 +0000714{
Vladimir Olteana11c89d2019-07-19 00:29:55 +0300715 struct tsec_private *priv;
Mingkai Hue0653bf2011-01-27 12:52:46 +0800716 struct eth_device *dev;
717 int i;
wdenka445ddf2004-06-09 00:34:46 +0000718
Mario Sixc29fcc72018-01-15 11:08:21 +0100719 dev = (struct eth_device *)malloc(sizeof(*dev));
wdenk78924a72004-04-18 21:45:42 +0000720
Mario Sixc29fcc72018-01-15 11:08:21 +0100721 if (!dev)
Mingkai Hue0653bf2011-01-27 12:52:46 +0800722 return 0;
wdenk78924a72004-04-18 21:45:42 +0000723
Mario Sixc29fcc72018-01-15 11:08:21 +0100724 memset(dev, 0, sizeof(*dev));
wdenka445ddf2004-06-09 00:34:46 +0000725
Mingkai Hue0653bf2011-01-27 12:52:46 +0800726 priv = (struct tsec_private *)malloc(sizeof(*priv));
727
Mario Six878b1d22018-01-15 11:08:22 +0100728 if (!priv) {
729 free(dev);
Mingkai Hue0653bf2011-01-27 12:52:46 +0800730 return 0;
Mario Six878b1d22018-01-15 11:08:22 +0100731 }
Mingkai Hue0653bf2011-01-27 12:52:46 +0800732
Mingkai Hue0653bf2011-01-27 12:52:46 +0800733 priv->regs = tsec_info->regs;
Mingkai Hue0653bf2011-01-27 12:52:46 +0800734 priv->phyregs_sgmii = tsec_info->miiregs_sgmii;
735
736 priv->phyaddr = tsec_info->phyaddr;
Bin Meng74314f12016-01-11 22:41:25 -0800737 priv->tbiaddr = CONFIG_SYS_TBIPA_VALUE;
Mingkai Hue0653bf2011-01-27 12:52:46 +0800738 priv->flags = tsec_info->flags;
wdenka445ddf2004-06-09 00:34:46 +0000739
Ben Whitten34fd6c92015-12-30 13:05:58 +0000740 strcpy(dev->name, tsec_info->devname);
Andy Fleming422effd2011-04-08 02:10:54 -0500741 priv->interface = tsec_info->interface;
742 priv->bus = miiphy_get_dev_by_name(tsec_info->mii_devname);
Bin Menge86a6cd2016-01-11 22:41:22 -0800743 priv->dev = dev;
Mingkai Hue0653bf2011-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 Hue0653bf2011-01-27 12:52:46 +0800750 dev->mcast = tsec_mcast_addr;
David Updegraff7280da72007-06-11 10:41:07 -0500751
Bin Meng79cd33a2016-01-11 22:41:18 -0800752 /* Tell U-Boot to get the addr from the env */
Mingkai Hue0653bf2011-01-27 12:52:46 +0800753 for (i = 0; i < 6; i++)
754 dev->enetaddr[i] = 0;
David Updegraff7280da72007-06-11 10:41:07 -0500755
Mingkai Hue0653bf2011-01-27 12:52:46 +0800756 eth_register(dev);
David Updegraff7280da72007-06-11 10:41:07 -0500757
Mingkai Hue0653bf2011-01-27 12:52:46 +0800758 /* 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);
David Updegraff7280da72007-06-11 10:41:07 -0500762
Mingkai Hue0653bf2011-01-27 12:52:46 +0800763 /* Try to initialize PHY here, and return */
Bin Menge86a6cd2016-01-11 22:41:22 -0800764 return init_phy(priv);
Mingkai Hue0653bf2011-01-27 12:52:46 +0800765}
David Updegraff7280da72007-06-11 10:41:07 -0500766
Mingkai Hue0653bf2011-01-27 12:52:46 +0800767/*
768 * Initialize all the TSEC devices
769 *
770 * Returns the number of TSEC devices that were initialized
771 */
Masahiro Yamadaf7ed78b2020-06-26 15:13:33 +0900772int tsec_eth_init(struct bd_info *bis, struct tsec_info_struct *tsecs,
773 int num)
Mingkai Hue0653bf2011-01-27 12:52:46 +0800774{
775 int i;
Mario Sixc29fcc72018-01-15 11:08:21 +0100776 int count = 0;
Mingkai Hue0653bf2011-01-27 12:52:46 +0800777
778 for (i = 0; i < num; i++) {
Mario Sixc29fcc72018-01-15 11:08:21 +0100779 int ret = tsec_initialize(bis, &tsecs[i]);
780
Mingkai Hue0653bf2011-01-27 12:52:46 +0800781 if (ret > 0)
782 count += ret;
David Updegraff7280da72007-06-11 10:41:07 -0500783 }
Mingkai Hue0653bf2011-01-27 12:52:46 +0800784
785 return count;
David Updegraff7280da72007-06-11 10:41:07 -0500786}
Mingkai Hue0653bf2011-01-27 12:52:46 +0800787
Masahiro Yamadaf7ed78b2020-06-26 15:13:33 +0900788int tsec_standard_init(struct bd_info *bis)
Mingkai Hue0653bf2011-01-27 12:52:46 +0800789{
Andy Fleming422effd2011-04-08 02:10:54 -0500790 struct fsl_pq_mdio_info info;
791
Claudiu Manoilcd0c4122013-09-30 12:44:42 +0300792 info.regs = TSEC_GET_MDIO_REGS_BASE(1);
Andy Fleming422effd2011-04-08 02:10:54 -0500793 info.name = DEFAULT_MII_NAME;
794
795 fsl_pq_mdio_init(bis, &info);
796
Mingkai Hue0653bf2011-01-27 12:52:46 +0800797 return tsec_eth_init(bis, tsec_info, ARRAY_SIZE(tsec_info));
798}
Bin Meng1048f612016-01-11 22:41:24 -0800799#else /* CONFIG_DM_ETH */
800int tsec_probe(struct udevice *dev)
801{
Bin Meng1048f612016-01-11 22:41:24 -0800802 struct eth_pdata *pdata = dev_get_platdata(dev);
Vladimir Olteana11c89d2019-07-19 00:29:55 +0300803 struct tsec_private *priv = dev_get_priv(dev);
Mario Six00ba0552018-01-15 11:08:23 +0100804 struct ofnode_phandle_args phandle_args;
Vladimir Olteand6392202019-07-19 00:29:53 +0300805 u32 tbiaddr = CONFIG_SYS_TBIPA_VALUE;
Hou Zhiqiang5966b6d2020-07-16 18:09:14 +0800806 struct tsec_data *data;
Bin Meng1048f612016-01-11 22:41:24 -0800807 const char *phy_mode;
Vladimir Oltean3095e342019-07-19 00:29:54 +0300808 fdt_addr_t reg;
Vladimir Olteana11c89d2019-07-19 00:29:55 +0300809 ofnode parent;
Bin Meng1048f612016-01-11 22:41:24 -0800810 int ret;
811
Hou Zhiqiang5966b6d2020-07-16 18:09:14 +0800812 data = (struct tsec_data *)dev_get_driver_data(dev);
813
Mario Six00ba0552018-01-15 11:08:23 +0100814 pdata->iobase = (phys_addr_t)dev_read_addr(dev);
Hou Zhiqiang7b69ce72020-05-03 22:48:42 +0800815 priv->regs = dev_remap_addr(dev);
Bin Meng1048f612016-01-11 22:41:24 -0800816
Vladimir Olteand6392202019-07-19 00:29:53 +0300817 ret = dev_read_phandle_with_args(dev, "tbi-handle", NULL, 0, 0,
818 &phandle_args);
Hou Zhiqiang53907d52020-05-03 22:48:43 +0800819 if (ret == 0) {
Vladimir Olteand6392202019-07-19 00:29:53 +0300820 ofnode_read_u32(phandle_args.node, "reg", &tbiaddr);
821
Hou Zhiqiang53907d52020-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 Zhiqiang5966b6d2020-07-16 18:09:14 +0800834 priv->phyregs_sgmii = map_physmem(reg + data->mdio_regs_off,
Hou Zhiqiang53907d52020-05-03 22:48:43 +0800835 0, MAP_NOCACHE);
836 }
837
Vladimir Olteand6392202019-07-19 00:29:53 +0300838 priv->tbiaddr = tbiaddr;
Bin Meng74314f12016-01-11 22:41:25 -0800839
Mario Six00ba0552018-01-15 11:08:23 +0100840 phy_mode = dev_read_prop(dev, "phy-connection-type", NULL);
Bin Meng1048f612016-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 Oltean8ec8eaa2019-07-19 00:29:56 +0300844 printf("Invalid PHY interface '%s'\n", phy_mode);
Bin Meng1048f612016-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 Meng1048f612016-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 Meng1048f612016-01-11 22:41:24 -0800883 .mcast = tsec_mcast_addr,
Bin Meng1048f612016-01-11 22:41:24 -0800884};
885
Hou Zhiqiang5966b6d2020-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 Meng1048f612016-01-11 22:41:24 -0800894static const struct udevice_id tsec_ids[] = {
Hou Zhiqiang5966b6d2020-07-16 18:09:14 +0800895 { .compatible = "fsl,etsec2", .data = (ulong)&etsec2_data },
896 { .compatible = "gianfar", .data = (ulong)&gianfar_data },
Bin Meng1048f612016-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,
Simon Glass8a2b47f2020-12-03 16:55:17 -0700907 .priv_auto = sizeof(struct tsec_private),
Simon Glass71fa5b42020-12-03 16:55:18 -0700908 .plat_auto = sizeof(struct eth_pdata),
Bin Meng1048f612016-01-11 22:41:24 -0800909 .flags = DM_FLAG_ALLOC_PRIV_DMA,
910};
911#endif /* CONFIG_DM_ETH */