blob: db573c08317bc020baa3d090455fb3e44833c4b2 [file] [log] [blame]
Álvaro Fernández Rojas96229722018-12-01 19:00:32 +01001// SPDX-License-Identifier: GPL-2.0+
2/*
3 * Copyright (C) 2018 Álvaro Fernández Rojas <noltari@gmail.com>
4 *
5 * Derived from linux/drivers/net/ethernet/broadcom/bcm63xx_enet.c:
6 * Copyright (C) 2008 Maxime Bizon <mbizon@freebox.fr>
7 */
8
9#include <common.h>
10#include <clk.h>
11#include <dm.h>
12#include <dma.h>
Simon Glassf7ae49f2020-05-10 11:40:05 -060013#include <log.h>
Simon Glass336d4612020-02-03 07:36:16 -070014#include <malloc.h>
Álvaro Fernández Rojas96229722018-12-01 19:00:32 +010015#include <miiphy.h>
16#include <net.h>
17#include <reset.h>
18#include <wait_bit.h>
19#include <asm/io.h>
Simon Glass336d4612020-02-03 07:36:16 -070020#include <dm/device_compat.h>
Álvaro Fernández Rojas96229722018-12-01 19:00:32 +010021
22#define ETH_PORT_STR "brcm,enetsw-port"
23
24#define ETH_RX_DESC PKTBUFSRX
25#define ETH_ZLEN 60
26#define ETH_TIMEOUT 100
27
28#define ETH_MAX_PORT 8
29#define ETH_RGMII_PORT0 4
30
31/* Port traffic control */
32#define ETH_PTCTRL_REG(x) (0x0 + (x))
33#define ETH_PTCTRL_RXDIS_SHIFT 0
34#define ETH_PTCTRL_RXDIS_MASK (1 << ETH_PTCTRL_RXDIS_SHIFT)
35#define ETH_PTCTRL_TXDIS_SHIFT 1
36#define ETH_PTCTRL_TXDIS_MASK (1 << ETH_PTCTRL_TXDIS_SHIFT)
37
38/* Switch mode register */
39#define ETH_SWMODE_REG 0xb
40#define ETH_SWMODE_FWD_EN_SHIFT 1
41#define ETH_SWMODE_FWD_EN_MASK (1 << ETH_SWMODE_FWD_EN_SHIFT)
42
43/* IMP override Register */
44#define ETH_IMPOV_REG 0xe
45#define ETH_IMPOV_LINKUP_SHIFT 0
46#define ETH_IMPOV_LINKUP_MASK (1 << ETH_IMPOV_LINKUP_SHIFT)
47#define ETH_IMPOV_FDX_SHIFT 1
48#define ETH_IMPOV_FDX_MASK (1 << ETH_IMPOV_FDX_SHIFT)
49#define ETH_IMPOV_100_SHIFT 2
50#define ETH_IMPOV_100_MASK (1 << ETH_IMPOV_100_SHIFT)
51#define ETH_IMPOV_1000_SHIFT 3
52#define ETH_IMPOV_1000_MASK (1 << ETH_IMPOV_1000_SHIFT)
53#define ETH_IMPOV_RXFLOW_SHIFT 4
54#define ETH_IMPOV_RXFLOW_MASK (1 << ETH_IMPOV_RXFLOW_SHIFT)
55#define ETH_IMPOV_TXFLOW_SHIFT 5
56#define ETH_IMPOV_TXFLOW_MASK (1 << ETH_IMPOV_TXFLOW_SHIFT)
57#define ETH_IMPOV_FORCE_SHIFT 7
58#define ETH_IMPOV_FORCE_MASK (1 << ETH_IMPOV_FORCE_SHIFT)
59
60/* Port override Register */
61#define ETH_PORTOV_REG(x) (0x58 + (x))
62#define ETH_PORTOV_LINKUP_SHIFT 0
63#define ETH_PORTOV_LINKUP_MASK (1 << ETH_PORTOV_LINKUP_SHIFT)
64#define ETH_PORTOV_FDX_SHIFT 1
65#define ETH_PORTOV_FDX_MASK (1 << ETH_PORTOV_FDX_SHIFT)
66#define ETH_PORTOV_100_SHIFT 2
67#define ETH_PORTOV_100_MASK (1 << ETH_PORTOV_100_SHIFT)
68#define ETH_PORTOV_1000_SHIFT 3
69#define ETH_PORTOV_1000_MASK (1 << ETH_PORTOV_1000_SHIFT)
70#define ETH_PORTOV_RXFLOW_SHIFT 4
71#define ETH_PORTOV_RXFLOW_MASK (1 << ETH_PORTOV_RXFLOW_SHIFT)
72#define ETH_PORTOV_TXFLOW_SHIFT 5
73#define ETH_PORTOV_TXFLOW_MASK (1 << ETH_PORTOV_TXFLOW_SHIFT)
74#define ETH_PORTOV_ENABLE_SHIFT 6
75#define ETH_PORTOV_ENABLE_MASK (1 << ETH_PORTOV_ENABLE_SHIFT)
76
77/* Port RGMII control register */
78#define ETH_RGMII_CTRL_REG(x) (0x60 + (x))
79#define ETH_RGMII_CTRL_GMII_CLK_EN (1 << 7)
80#define ETH_RGMII_CTRL_MII_OVERRIDE_EN (1 << 6)
81#define ETH_RGMII_CTRL_MII_MODE_MASK (3 << 4)
82#define ETH_RGMII_CTRL_RGMII_MODE (0 << 4)
83#define ETH_RGMII_CTRL_MII_MODE (1 << 4)
84#define ETH_RGMII_CTRL_RVMII_MODE (2 << 4)
85#define ETH_RGMII_CTRL_TIMING_SEL_EN (1 << 0)
86
87/* Port RGMII timing register */
88#define ENETSW_RGMII_TIMING_REG(x) (0x68 + (x))
89
90/* MDIO control register */
91#define MII_SC_REG 0xb0
92#define MII_SC_EXT_SHIFT 16
93#define MII_SC_EXT_MASK (1 << MII_SC_EXT_SHIFT)
94#define MII_SC_REG_SHIFT 20
95#define MII_SC_PHYID_SHIFT 25
96#define MII_SC_RD_SHIFT 30
97#define MII_SC_RD_MASK (1 << MII_SC_RD_SHIFT)
98#define MII_SC_WR_SHIFT 31
99#define MII_SC_WR_MASK (1 << MII_SC_WR_SHIFT)
100
101/* MDIO data register */
102#define MII_DAT_REG 0xb4
103
104/* Global Management Configuration Register */
105#define ETH_GMCR_REG 0x200
106#define ETH_GMCR_RST_MIB_SHIFT 0
107#define ETH_GMCR_RST_MIB_MASK (1 << ETH_GMCR_RST_MIB_SHIFT)
108
109/* Jumbo control register port mask register */
110#define ETH_JMBCTL_PORT_REG 0x4004
111
112/* Jumbo control mib good frame register */
113#define ETH_JMBCTL_MAXSIZE_REG 0x4008
114
115/* ETH port data */
116struct bcm_enetsw_port {
117 bool used;
118 const char *name;
119 /* Config */
120 bool bypass_link;
121 int force_speed;
122 bool force_duplex_full;
123 /* PHY */
124 int phy_id;
125};
126
127/* ETH data */
128struct bcm6368_eth_priv {
129 void __iomem *base;
130 /* DMA */
131 struct dma rx_dma;
132 struct dma tx_dma;
133 /* Ports */
134 uint8_t num_ports;
135 struct bcm_enetsw_port used_ports[ETH_MAX_PORT];
136 int sw_port_link[ETH_MAX_PORT];
137 bool rgmii_override;
138 bool rgmii_timing;
139 /* PHY */
140 int phy_id;
141};
142
143static inline bool bcm_enet_port_is_rgmii(int portid)
144{
145 return portid >= ETH_RGMII_PORT0;
146}
147
148static int bcm6368_mdio_read(struct bcm6368_eth_priv *priv, uint8_t ext,
149 int phy_id, int reg)
150{
151 uint32_t val;
152
153 writel_be(0, priv->base + MII_SC_REG);
154
155 val = MII_SC_RD_MASK |
156 (phy_id << MII_SC_PHYID_SHIFT) |
157 (reg << MII_SC_REG_SHIFT);
158
159 if (ext)
160 val |= MII_SC_EXT_MASK;
161
162 writel_be(val, priv->base + MII_SC_REG);
163 udelay(50);
164
165 return readw_be(priv->base + MII_DAT_REG);
166}
167
168static int bcm6368_mdio_write(struct bcm6368_eth_priv *priv, uint8_t ext,
169 int phy_id, int reg, u16 data)
170{
171 uint32_t val;
172
173 writel_be(0, priv->base + MII_SC_REG);
174
175 val = MII_SC_WR_MASK |
176 (phy_id << MII_SC_PHYID_SHIFT) |
177 (reg << MII_SC_REG_SHIFT);
178
179 if (ext)
180 val |= MII_SC_EXT_MASK;
181
182 val |= data;
183
184 writel_be(val, priv->base + MII_SC_REG);
185 udelay(50);
186
187 return 0;
188}
189
190static int bcm6368_eth_free_pkt(struct udevice *dev, uchar *packet, int len)
191{
192 struct bcm6368_eth_priv *priv = dev_get_priv(dev);
193
194 return dma_prepare_rcv_buf(&priv->rx_dma, packet, len);
195}
196
197static int bcm6368_eth_recv(struct udevice *dev, int flags, uchar **packetp)
198{
199 struct bcm6368_eth_priv *priv = dev_get_priv(dev);
200
201 return dma_receive(&priv->rx_dma, (void**)packetp, NULL);
202}
203
204static int bcm6368_eth_send(struct udevice *dev, void *packet, int length)
205{
206 struct bcm6368_eth_priv *priv = dev_get_priv(dev);
207
208 /* pad packets smaller than ETH_ZLEN */
209 if (length < ETH_ZLEN) {
210 memset(packet + length, 0, ETH_ZLEN - length);
211 length = ETH_ZLEN;
212 }
213
214 return dma_send(&priv->tx_dma, packet, length, NULL);
215}
216
217static int bcm6368_eth_adjust_link(struct udevice *dev)
218{
219 struct bcm6368_eth_priv *priv = dev_get_priv(dev);
220 unsigned int i;
221
222 for (i = 0; i < priv->num_ports; i++) {
223 struct bcm_enetsw_port *port;
224 int val, j, up, adv, lpa, speed, duplex, media;
225 int external_phy = bcm_enet_port_is_rgmii(i);
226 u8 override;
227
228 port = &priv->used_ports[i];
229 if (!port->used)
230 continue;
231
232 if (port->bypass_link)
233 continue;
234
235 /* dummy read to clear */
236 for (j = 0; j < 2; j++)
237 val = bcm6368_mdio_read(priv, external_phy,
238 port->phy_id, MII_BMSR);
239
240 if (val == 0xffff)
241 continue;
242
243 up = (val & BMSR_LSTATUS) ? 1 : 0;
244 if (!(up ^ priv->sw_port_link[i]))
245 continue;
246
247 priv->sw_port_link[i] = up;
248
249 /* link changed */
250 if (!up) {
251 dev_info(&priv->pdev->dev, "link DOWN on %s\n",
252 port->name);
253 writeb_be(ETH_PORTOV_ENABLE_MASK,
254 priv->base + ETH_PORTOV_REG(i));
255 writeb_be(ETH_PTCTRL_RXDIS_MASK |
256 ETH_PTCTRL_TXDIS_MASK,
257 priv->base + ETH_PTCTRL_REG(i));
258 continue;
259 }
260
261 adv = bcm6368_mdio_read(priv, external_phy,
262 port->phy_id, MII_ADVERTISE);
263
264 lpa = bcm6368_mdio_read(priv, external_phy, port->phy_id,
265 MII_LPA);
266
267 /* figure out media and duplex from advertise and LPA values */
268 media = mii_nway_result(lpa & adv);
269 duplex = (media & ADVERTISE_FULL) ? 1 : 0;
270
271 if (media & (ADVERTISE_100FULL | ADVERTISE_100HALF))
272 speed = 100;
273 else
274 speed = 10;
275
276 if (val & BMSR_ESTATEN) {
277 adv = bcm6368_mdio_read(priv, external_phy,
278 port->phy_id, MII_CTRL1000);
279
280 lpa = bcm6368_mdio_read(priv, external_phy,
281 port->phy_id, MII_STAT1000);
282
283 if ((adv & (ADVERTISE_1000FULL | ADVERTISE_1000HALF)) &&
284 (lpa & (LPA_1000FULL | LPA_1000HALF))) {
285 speed = 1000;
286 duplex = (lpa & LPA_1000FULL);
287 }
288 }
289
290 pr_alert("link UP on %s, %dMbps, %s-duplex\n",
291 port->name, speed, duplex ? "full" : "half");
292
293 override = ETH_PORTOV_ENABLE_MASK |
294 ETH_PORTOV_LINKUP_MASK;
295
296 if (speed == 1000)
297 override |= ETH_PORTOV_1000_MASK;
298 else if (speed == 100)
299 override |= ETH_PORTOV_100_MASK;
300 if (duplex)
301 override |= ETH_PORTOV_FDX_MASK;
302
303 writeb_be(override, priv->base + ETH_PORTOV_REG(i));
304 writeb_be(0, priv->base + ETH_PTCTRL_REG(i));
305 }
306
307 return 0;
308}
309
310static int bcm6368_eth_start(struct udevice *dev)
311{
312 struct bcm6368_eth_priv *priv = dev_get_priv(dev);
313 uint8_t i;
314
Álvaro Fernández Rojasa4ae4222019-01-01 19:44:46 +0100315 /* disable all ports */
316 for (i = 0; i < priv->num_ports; i++) {
317 setbits_8(priv->base + ETH_PORTOV_REG(i),
318 ETH_PORTOV_ENABLE_MASK);
319 setbits_8(priv->base + ETH_PTCTRL_REG(i),
320 ETH_PTCTRL_RXDIS_MASK | ETH_PTCTRL_TXDIS_MASK);
321 priv->sw_port_link[i] = 0;
322 }
323
324 /* enable external ports */
325 for (i = ETH_RGMII_PORT0; i < priv->num_ports; i++) {
326 u8 rgmii_ctrl = ETH_RGMII_CTRL_GMII_CLK_EN;
327
328 if (!priv->used_ports[i].used)
329 continue;
330
331 if (priv->rgmii_override)
332 rgmii_ctrl |= ETH_RGMII_CTRL_MII_OVERRIDE_EN;
333 if (priv->rgmii_timing)
334 rgmii_ctrl |= ETH_RGMII_CTRL_TIMING_SEL_EN;
335
336 setbits_8(priv->base + ETH_RGMII_CTRL_REG(i), rgmii_ctrl);
337 }
338
339 /* reset mib */
340 setbits_8(priv->base + ETH_GMCR_REG, ETH_GMCR_RST_MIB_MASK);
341 mdelay(1);
342 clrbits_8(priv->base + ETH_GMCR_REG, ETH_GMCR_RST_MIB_MASK);
343 mdelay(1);
344
345 /* force CPU port state */
346 setbits_8(priv->base + ETH_IMPOV_REG,
347 ETH_IMPOV_FORCE_MASK | ETH_IMPOV_LINKUP_MASK);
348
349 /* enable switch forward engine */
350 setbits_8(priv->base + ETH_SWMODE_REG, ETH_SWMODE_FWD_EN_MASK);
351
Álvaro Fernández Rojas96229722018-12-01 19:00:32 +0100352 /* prepare rx dma buffers */
353 for (i = 0; i < ETH_RX_DESC; i++) {
354 int ret = dma_prepare_rcv_buf(&priv->rx_dma, net_rx_packets[i],
355 PKTSIZE_ALIGN);
356 if (ret < 0)
357 break;
358 }
359
360 /* enable dma rx channel */
361 dma_enable(&priv->rx_dma);
362
363 /* enable dma tx channel */
364 dma_enable(&priv->tx_dma);
365
366 /* apply override config for bypass_link ports here. */
367 for (i = 0; i < priv->num_ports; i++) {
368 struct bcm_enetsw_port *port;
369 u8 override;
370
371 port = &priv->used_ports[i];
372 if (!port->used)
373 continue;
374
375 if (!port->bypass_link)
376 continue;
377
378 override = ETH_PORTOV_ENABLE_MASK |
379 ETH_PORTOV_LINKUP_MASK;
380
381 switch (port->force_speed) {
382 case 1000:
383 override |= ETH_PORTOV_1000_MASK;
384 break;
385 case 100:
386 override |= ETH_PORTOV_100_MASK;
387 break;
388 case 10:
389 break;
390 default:
391 pr_warn("%s: invalid forced speed on port %s\n",
392 __func__, port->name);
393 break;
394 }
395
396 if (port->force_duplex_full)
397 override |= ETH_PORTOV_FDX_MASK;
398
399 writeb_be(override, priv->base + ETH_PORTOV_REG(i));
400 writeb_be(0, priv->base + ETH_PTCTRL_REG(i));
401 }
402
403 bcm6368_eth_adjust_link(dev);
404
405 return 0;
406}
407
408static void bcm6368_eth_stop(struct udevice *dev)
409{
410 struct bcm6368_eth_priv *priv = dev_get_priv(dev);
Álvaro Fernández Rojasa4ae4222019-01-01 19:44:46 +0100411 uint8_t i;
412
413 /* disable all ports */
414 for (i = 0; i < priv->num_ports; i++) {
415 setbits_8(priv->base + ETH_PORTOV_REG(i),
416 ETH_PORTOV_ENABLE_MASK);
417 setbits_8(priv->base + ETH_PTCTRL_REG(i),
418 ETH_PTCTRL_RXDIS_MASK | ETH_PTCTRL_TXDIS_MASK);
419 }
420
421 /* disable external ports */
422 for (i = ETH_RGMII_PORT0; i < priv->num_ports; i++) {
423 if (!priv->used_ports[i].used)
424 continue;
425
426 clrbits_8(priv->base + ETH_RGMII_CTRL_REG(i),
427 ETH_RGMII_CTRL_GMII_CLK_EN);
428 }
429
430 /* disable CPU port */
431 clrbits_8(priv->base + ETH_IMPOV_REG,
432 ETH_IMPOV_FORCE_MASK | ETH_IMPOV_LINKUP_MASK);
433
434 /* disable switch forward engine */
435 clrbits_8(priv->base + ETH_SWMODE_REG, ETH_SWMODE_FWD_EN_MASK);
Álvaro Fernández Rojas96229722018-12-01 19:00:32 +0100436
437 /* disable dma rx channel */
438 dma_disable(&priv->rx_dma);
439
440 /* disable dma tx channel */
441 dma_disable(&priv->tx_dma);
442}
443
444static const struct eth_ops bcm6368_eth_ops = {
445 .free_pkt = bcm6368_eth_free_pkt,
446 .recv = bcm6368_eth_recv,
447 .send = bcm6368_eth_send,
448 .start = bcm6368_eth_start,
449 .stop = bcm6368_eth_stop,
450};
451
452static const struct udevice_id bcm6368_eth_ids[] = {
453 { .compatible = "brcm,bcm6368-enet", },
454 { /* sentinel */ }
455};
456
457static bool bcm6368_phy_is_external(struct bcm6368_eth_priv *priv, int phy_id)
458{
459 uint8_t i;
460
461 for (i = 0; i < priv->num_ports; ++i) {
462 if (!priv->used_ports[i].used)
463 continue;
464 if (priv->used_ports[i].phy_id == phy_id)
465 return bcm_enet_port_is_rgmii(i);
466 }
467
468 return true;
469}
470
471static int bcm6368_mii_mdio_read(struct mii_dev *bus, int addr, int devaddr,
472 int reg)
473{
474 struct bcm6368_eth_priv *priv = bus->priv;
475 bool ext = bcm6368_phy_is_external(priv, addr);
476
477 return bcm6368_mdio_read(priv, ext, addr, reg);
478}
479
480static int bcm6368_mii_mdio_write(struct mii_dev *bus, int addr, int devaddr,
481 int reg, u16 data)
482{
483 struct bcm6368_eth_priv *priv = bus->priv;
484 bool ext = bcm6368_phy_is_external(priv, addr);
485
486 return bcm6368_mdio_write(priv, ext, addr, reg, data);
487}
488
489static int bcm6368_mdio_init(const char *name, struct bcm6368_eth_priv *priv)
490{
491 struct mii_dev *bus;
492
493 bus = mdio_alloc();
494 if (!bus) {
495 pr_err("%s: failed to allocate MDIO bus\n", __func__);
496 return -ENOMEM;
497 }
498
499 bus->read = bcm6368_mii_mdio_read;
500 bus->write = bcm6368_mii_mdio_write;
501 bus->priv = priv;
502 snprintf(bus->name, sizeof(bus->name), "%s", name);
503
504 return mdio_register(bus);
505}
506
507static int bcm6368_eth_probe(struct udevice *dev)
508{
509 struct eth_pdata *pdata = dev_get_platdata(dev);
510 struct bcm6368_eth_priv *priv = dev_get_priv(dev);
511 int num_ports, ret, i;
Álvaro Fernández Rojas96229722018-12-01 19:00:32 +0100512 ofnode node;
513
514 /* get base address */
515 priv->base = dev_remap_addr(dev);
516 if (!priv->base)
517 return -EINVAL;
518 pdata->iobase = (phys_addr_t) priv->base;
519
520 /* get number of ports */
521 num_ports = dev_read_u32_default(dev, "brcm,num-ports", ETH_MAX_PORT);
522 if (!num_ports || num_ports > ETH_MAX_PORT)
523 return -EINVAL;
524
525 /* get dma channels */
526 ret = dma_get_by_name(dev, "tx", &priv->tx_dma);
527 if (ret)
528 return -EINVAL;
529
530 ret = dma_get_by_name(dev, "rx", &priv->rx_dma);
531 if (ret)
532 return -EINVAL;
533
534 /* try to enable clocks */
535 for (i = 0; ; i++) {
536 struct clk clk;
537 int ret;
538
539 ret = clk_get_by_index(dev, i, &clk);
540 if (ret < 0)
541 break;
542
543 ret = clk_enable(&clk);
544 if (ret < 0) {
545 pr_err("%s: error enabling clock %d\n", __func__, i);
546 return ret;
547 }
548
549 ret = clk_free(&clk);
550 if (ret < 0) {
551 pr_err("%s: error freeing clock %d\n", __func__, i);
552 return ret;
553 }
554 }
555
556 /* try to perform resets */
557 for (i = 0; ; i++) {
558 struct reset_ctl reset;
559 int ret;
560
561 ret = reset_get_by_index(dev, i, &reset);
562 if (ret < 0)
563 break;
564
565 ret = reset_deassert(&reset);
566 if (ret < 0) {
567 pr_err("%s: error deasserting reset %d\n", __func__, i);
568 return ret;
569 }
570
571 ret = reset_free(&reset);
572 if (ret < 0) {
573 pr_err("%s: error freeing reset %d\n", __func__, i);
574 return ret;
575 }
576 }
577
578 /* set priv data */
579 priv->num_ports = num_ports;
580 if (dev_read_bool(dev, "brcm,rgmii-override"))
581 priv->rgmii_override = true;
582 if (dev_read_bool(dev, "brcm,rgmii-timing"))
583 priv->rgmii_timing = true;
584
585 /* get ports */
586 dev_for_each_subnode(node, dev) {
587 const char *comp;
588 const char *label;
589 unsigned int p;
590 int phy_id;
591 int speed;
592
593 comp = ofnode_read_string(node, "compatible");
594 if (!comp || memcmp(comp, ETH_PORT_STR, sizeof(ETH_PORT_STR)))
595 continue;
596
597 p = ofnode_read_u32_default(node, "reg", ETH_MAX_PORT);
598 if (p >= num_ports)
599 return -EINVAL;
600
601 label = ofnode_read_string(node, "label");
602 if (!label) {
603 debug("%s: node %s has no label\n", __func__,
604 ofnode_get_name(node));
605 return -EINVAL;
606 }
607
608 phy_id = ofnode_read_u32_default(node, "brcm,phy-id", -1);
609
610 priv->used_ports[p].used = true;
611 priv->used_ports[p].name = label;
612 priv->used_ports[p].phy_id = phy_id;
613
614 if (ofnode_read_bool(node, "full-duplex"))
615 priv->used_ports[p].force_duplex_full = true;
616 if (ofnode_read_bool(node, "bypass-link"))
617 priv->used_ports[p].bypass_link = true;
618 speed = ofnode_read_u32_default(node, "speed", 0);
619 if (speed)
620 priv->used_ports[p].force_speed = speed;
621 }
622
623 /* init mii bus */
624 ret = bcm6368_mdio_init(dev->name, priv);
625 if (ret)
626 return ret;
627
Álvaro Fernández Rojas96229722018-12-01 19:00:32 +0100628 /* enable jumbo on all ports */
629 writel_be(0x1ff, priv->base + ETH_JMBCTL_PORT_REG);
630 writew_be(9728, priv->base + ETH_JMBCTL_MAXSIZE_REG);
631
632 return 0;
633}
634
635U_BOOT_DRIVER(bcm6368_eth) = {
636 .name = "bcm6368_eth",
637 .id = UCLASS_ETH,
638 .of_match = bcm6368_eth_ids,
639 .ops = &bcm6368_eth_ops,
640 .platdata_auto_alloc_size = sizeof(struct eth_pdata),
641 .priv_auto_alloc_size = sizeof(struct bcm6368_eth_priv),
642 .probe = bcm6368_eth_probe,
643};