/* * Xilinx xps_ll_temac ethernet driver for u-boot * * supports SDMA or FIFO access and MDIO bus communication * * Copyright (C) 2011 - 2012 Stephan Linz <linz@li-pro.net> * Copyright (C) 2008 - 2011 Michal Simek <monstr@monstr.eu> * Copyright (C) 2008 - 2011 PetaLogix * * Based on Yoshio Kashiwagi kashiwagi@co-nss.co.jp driver * Copyright (C) 2008 Nissin Systems Co.,Ltd. * March 2008 created * * SPDX-License-Identifier: GPL-2.0+ * * [0]: http://www.xilinx.com/support/documentation * * [S]: [0]/ip_documentation/xps_ll_temac.pdf * [A]: [0]/application_notes/xapp1041.pdf */ #include <config.h> #include <common.h> #include <net.h> #include <netdev.h> #include <malloc.h> #include <asm/io.h> #include <miiphy.h> #include "xilinx_ll_temac.h" #include "xilinx_ll_temac_fifo.h" #include "xilinx_ll_temac_sdma.h" #include "xilinx_ll_temac_mdio.h" #if !defined(CONFIG_MII) # error "LL_TEMAC requires MII -- missing CONFIG_MII" #endif #if !defined(CONFIG_PHYLIB) # error "LL_TEMAC requires PHYLIB -- missing CONFIG_PHYLIB" #endif struct ll_temac_info { int flags; unsigned long base_addr; unsigned long ctrl_addr; char *devname; unsigned int phyaddr; char *mdio_busname; }; /* Ethernet interface ready status */ int ll_temac_check_status(struct temac_reg *regs, u32 mask) { unsigned timeout = 50; /* 1usec * 50 = 50usec */ /* * Quote from LL TEMAC documentation: The bits in the RDY * register are asserted when there is no access in progress. * When an access is in progress, a bit corresponding to the * type of access is automatically de-asserted. The bit is * automatically re-asserted when the access is complete. */ while (timeout && (!(in_be32(®s->rdy) & mask))) { timeout--; udelay(1); } if (!timeout) { printf("%s: Timeout on 0x%08x @%p\n", __func__, mask, ®s->rdy); return 1; } return 0; } /* * Indirect write to ll_temac. * * http://www.xilinx.com/support/documentation/ip_documentation/xps_ll_temac.pdf * page 23, second paragraph, The use of CTL0 register or CTL1 register */ int ll_temac_indirect_set(struct temac_reg *regs, u16 regn, u32 reg_data) { out_be32(®s->lsw, (reg_data & MLSW_MASK)); out_be32(®s->ctl, CTL_WEN | (regn & CTL_ADDR_MASK)); if (ll_temac_check_status(regs, RSE_CFG_WR)) return 0; return 1; } /* * Indirect read from ll_temac. * * http://www.xilinx.com/support/documentation/ip_documentation/xps_ll_temac.pdf * page 23, second paragraph, The use of CTL0 register or CTL1 register */ int ll_temac_indirect_get(struct temac_reg *regs, u16 regn, u32* reg_data) { out_be32(®s->ctl, (regn & CTL_ADDR_MASK)); if (ll_temac_check_status(regs, RSE_CFG_RR)) return 0; *reg_data = in_be32(®s->lsw) & MLSW_MASK; return 1; } /* setting sub-controller and ll_temac to proper setting */ static int ll_temac_setup_ctrl(struct eth_device *dev) { struct ll_temac *ll_temac = dev->priv; struct temac_reg *regs = (struct temac_reg *)dev->iobase; if (ll_temac->ctrlreset && ll_temac->ctrlreset(dev)) return 0; if (ll_temac->ctrlinit && ll_temac->ctrlinit(dev)) return 0; /* Promiscuous mode disable */ if (!ll_temac_indirect_set(regs, TEMAC_AFM, 0)) return 0; /* Enable Receiver - RX bit */ if (!ll_temac_indirect_set(regs, TEMAC_RCW1, RCW1_RX)) return 0; /* Enable Transmitter - TX bit */ if (!ll_temac_indirect_set(regs, TEMAC_TC, TC_TX)) return 0; return 1; } /* * Configure ll_temac based on negotiated speed and duplex * reported by PHY handling code */ static int ll_temac_adjust_link(struct eth_device *dev) { unsigned int speed, emmc_reg; struct temac_reg *regs = (struct temac_reg *)dev->iobase; struct ll_temac *ll_temac = dev->priv; struct phy_device *phydev = ll_temac->phydev; if (!phydev->link) { printf("%s: No link.\n", phydev->dev->name); return 0; } switch (phydev->speed) { case 1000: speed = EMMC_LSPD_1000; break; case 100: speed = EMMC_LSPD_100; break; case 10: speed = EMMC_LSPD_10; break; default: return 0; } if (!ll_temac_indirect_get(regs, TEMAC_EMMC, &emmc_reg)) return 0; emmc_reg &= ~EMMC_LSPD_MASK; emmc_reg |= speed; if (!ll_temac_indirect_set(regs, TEMAC_EMMC, emmc_reg)) return 0; printf("%s: PHY is %s with %dbase%s, %s%s\n", dev->name, phydev->drv->name, phydev->speed, (phydev->port == PORT_TP) ? "T" : "X", (phydev->duplex) ? "FDX" : "HDX", (phydev->port == PORT_OTHER) ? ", unkown mode" : ""); return 1; } /* setup mac addr */ static int ll_temac_setup_mac_addr(struct eth_device *dev) { struct temac_reg *regs = (struct temac_reg *)dev->iobase; u32 val; /* set up unicast MAC address filter */ val = ((dev->enetaddr[3] << 24) | (dev->enetaddr[2] << 16) | (dev->enetaddr[1] << 8) | (dev->enetaddr[0])); val &= UAW0_UADDR_MASK; if (!ll_temac_indirect_set(regs, TEMAC_UAW0, val)) return 1; val = ((dev->enetaddr[5] << 8) | dev->enetaddr[4]); val &= UAW1_UADDR_MASK; if (!ll_temac_indirect_set(regs, TEMAC_UAW1, val)) return 1; return 0; } /* halt device */ static void ll_temac_halt(struct eth_device *dev) { struct ll_temac *ll_temac = dev->priv; struct temac_reg *regs = (struct temac_reg *)dev->iobase; /* Disable Receiver */ ll_temac_indirect_set(regs, TEMAC_RCW0, 0); /* Disable Transmitter */ ll_temac_indirect_set(regs, TEMAC_TC, 0); if (ll_temac->ctrlhalt) ll_temac->ctrlhalt(dev); /* Shut down the PHY, as needed */ phy_shutdown(ll_temac->phydev); } static int ll_temac_init(struct eth_device *dev, bd_t *bis) { struct ll_temac *ll_temac = dev->priv; int ret; printf("%s: Xilinx XPS LocalLink Tri-Mode Ether MAC #%d at 0x%08X.\n", dev->name, dev->index, dev->iobase); if (!ll_temac_setup_ctrl(dev)) return -1; /* Start up the PHY */ ret = phy_startup(ll_temac->phydev); if (ret) { printf("%s: Could not initialize PHY %s\n", dev->name, ll_temac->phydev->dev->name); return ret; } if (!ll_temac_adjust_link(dev)) { ll_temac_halt(dev); return -1; } /* If there's no link, fail */ return ll_temac->phydev->link ? 0 : -1; } /* * Discover which PHY is attached to the device, and configure it * properly. If the PHY is not recognized, then return 0 * (failure). Otherwise, return 1 */ static int ll_temac_phy_init(struct eth_device *dev) { struct ll_temac *ll_temac = dev->priv; struct phy_device *phydev; unsigned int supported = PHY_GBIT_FEATURES; /* interface - look at driver/net/tsec.c */ phydev = phy_connect(ll_temac->bus, ll_temac->phyaddr, dev, PHY_INTERFACE_MODE_NONE); phydev->supported &= supported; phydev->advertising = phydev->supported; ll_temac->phydev = phydev; phy_config(phydev); return 1; } /* * Initialize a single ll_temac devices * * Returns the result of ll_temac phy interface that were initialized */ int xilinx_ll_temac_initialize(bd_t *bis, struct ll_temac_info *devinf) { struct eth_device *dev; struct ll_temac *ll_temac; dev = calloc(1, sizeof(*dev)); if (dev == NULL) return 0; ll_temac = calloc(1, sizeof(struct ll_temac)); if (ll_temac == NULL) { free(dev); return 0; } /* use given name or generate its own unique name */ if (devinf->devname) { strncpy(dev->name, devinf->devname, sizeof(dev->name)); } else { snprintf(dev->name, sizeof(dev->name), "lltemac.%lx", devinf->base_addr); devinf->devname = dev->name; } dev->iobase = devinf->base_addr; dev->priv = ll_temac; dev->init = ll_temac_init; dev->halt = ll_temac_halt; dev->write_hwaddr = ll_temac_setup_mac_addr; ll_temac->ctrladdr = devinf->ctrl_addr; if (devinf->flags & XILINX_LL_TEMAC_M_SDMA_PLB) { #if defined(CONFIG_XILINX_440) || defined(CONFIG_XILINX_405) if (devinf->flags & XILINX_LL_TEMAC_M_SDMA_DCR) { ll_temac_collect_xldcr_sdma_reg_addr(dev); ll_temac->in32 = ll_temac_xldcr_in32; ll_temac->out32 = ll_temac_xldcr_out32; } else #endif { ll_temac_collect_xlplb_sdma_reg_addr(dev); ll_temac->in32 = ll_temac_xlplb_in32; ll_temac->out32 = ll_temac_xlplb_out32; } ll_temac->ctrlinit = ll_temac_init_sdma; ll_temac->ctrlhalt = ll_temac_halt_sdma; ll_temac->ctrlreset = ll_temac_reset_sdma; dev->recv = ll_temac_recv_sdma; dev->send = ll_temac_send_sdma; } else { ll_temac->in32 = NULL; ll_temac->out32 = NULL; ll_temac->ctrlinit = NULL; ll_temac->ctrlhalt = NULL; ll_temac->ctrlreset = ll_temac_reset_fifo; dev->recv = ll_temac_recv_fifo; dev->send = ll_temac_send_fifo; } /* Link to specified MDIO bus */ strncpy(ll_temac->mdio_busname, devinf->mdio_busname, MDIO_NAME_LEN); ll_temac->bus = miiphy_get_dev_by_name(ll_temac->mdio_busname); /* Looking for a valid PHY address if it is not yet set */ if (devinf->phyaddr == -1) ll_temac->phyaddr = ll_temac_phy_addr(ll_temac->bus); else ll_temac->phyaddr = devinf->phyaddr; eth_register(dev); /* Try to initialize PHY here, and return */ return ll_temac_phy_init(dev); } /* * Initialize a single ll_temac device with its mdio bus behind ll_temac * * Returns 1 if the ll_temac device and the mdio bus were initialized * otherwise returns 0 */ int xilinx_ll_temac_eth_init(bd_t *bis, unsigned long base_addr, int flags, unsigned long ctrl_addr) { struct ll_temac_info devinf; struct ll_temac_mdio_info mdioinf; int ret; /* prepare the internal driver informations */ devinf.flags = flags; devinf.base_addr = base_addr; devinf.ctrl_addr = ctrl_addr; devinf.devname = NULL; devinf.phyaddr = -1; mdioinf.name = devinf.mdio_busname = NULL; mdioinf.regs = (struct temac_reg *)devinf.base_addr; ret = xilinx_ll_temac_mdio_initialize(bis, &mdioinf); if (ret >= 0) { /* * If there was no MDIO bus name then take over the * new automaticaly generated by the MDIO init code. */ if (mdioinf.name != devinf.mdio_busname) devinf.mdio_busname = mdioinf.name; ret = xilinx_ll_temac_initialize(bis, &devinf); if (ret > 0) return 1; } return 0; }