mirror of
https://github.com/AsahiLinux/u-boot
synced 2025-01-25 19:35:17 +00:00
1e94b46f73
This old patch was marked as deferred. Bring it back to life, to continue towards the removal of common.h Move this out of the common header and include it only where needed. Signed-off-by: Simon Glass <sjg@chromium.org>
1009 lines
24 KiB
C
1009 lines
24 KiB
C
// SPDX-License-Identifier: GPL-2.0+
|
|
/**
|
|
* sni_ave.c - Socionext UniPhier AVE ethernet driver
|
|
* Copyright 2016-2018 Socionext inc.
|
|
*/
|
|
|
|
#include <clk.h>
|
|
#include <cpu_func.h>
|
|
#include <dm.h>
|
|
#include <fdt_support.h>
|
|
#include <log.h>
|
|
#include <malloc.h>
|
|
#include <miiphy.h>
|
|
#include <net.h>
|
|
#include <regmap.h>
|
|
#include <reset.h>
|
|
#include <syscon.h>
|
|
#include <asm/cache.h>
|
|
#include <asm/global_data.h>
|
|
#include <dm/device_compat.h>
|
|
#include <linux/bitops.h>
|
|
#include <linux/delay.h>
|
|
#include <linux/err.h>
|
|
#include <linux/io.h>
|
|
#include <linux/iopoll.h>
|
|
#include <linux/printk.h>
|
|
|
|
#define AVE_GRST_DELAY_MSEC 40
|
|
#define AVE_MIN_XMITSIZE 60
|
|
#define AVE_SEND_TIMEOUT_COUNT 1000
|
|
#define AVE_MDIO_TIMEOUT_USEC 10000
|
|
#define AVE_HALT_TIMEOUT_USEC 10000
|
|
|
|
/* General Register Group */
|
|
#define AVE_IDR 0x000 /* ID */
|
|
#define AVE_VR 0x004 /* Version */
|
|
#define AVE_GRR 0x008 /* Global Reset */
|
|
#define AVE_CFGR 0x00c /* Configuration */
|
|
|
|
/* Interrupt Register Group */
|
|
#define AVE_GIMR 0x100 /* Global Interrupt Mask */
|
|
#define AVE_GISR 0x104 /* Global Interrupt Status */
|
|
|
|
/* MAC Register Group */
|
|
#define AVE_TXCR 0x200 /* TX Setup */
|
|
#define AVE_RXCR 0x204 /* RX Setup */
|
|
#define AVE_RXMAC1R 0x208 /* MAC address (lower) */
|
|
#define AVE_RXMAC2R 0x20c /* MAC address (upper) */
|
|
#define AVE_MDIOCTR 0x214 /* MDIO Control */
|
|
#define AVE_MDIOAR 0x218 /* MDIO Address */
|
|
#define AVE_MDIOWDR 0x21c /* MDIO Data */
|
|
#define AVE_MDIOSR 0x220 /* MDIO Status */
|
|
#define AVE_MDIORDR 0x224 /* MDIO Rd Data */
|
|
|
|
/* Descriptor Control Register Group */
|
|
#define AVE_DESCC 0x300 /* Descriptor Control */
|
|
#define AVE_TXDC 0x304 /* TX Descriptor Configuration */
|
|
#define AVE_RXDC 0x308 /* RX Descriptor Ring0 Configuration */
|
|
#define AVE_IIRQC 0x34c /* Interval IRQ Control */
|
|
|
|
/* 64bit descriptor memory */
|
|
#define AVE_DESC_SIZE_64 12 /* Descriptor Size */
|
|
#define AVE_TXDM_64 0x1000 /* Tx Descriptor Memory */
|
|
#define AVE_RXDM_64 0x1c00 /* Rx Descriptor Memory */
|
|
|
|
/* 32bit descriptor memory */
|
|
#define AVE_DESC_SIZE_32 8 /* Descriptor Size */
|
|
#define AVE_TXDM_32 0x1000 /* Tx Descriptor Memory */
|
|
#define AVE_RXDM_32 0x1800 /* Rx Descriptor Memory */
|
|
|
|
/* RMII Bridge Register Group */
|
|
#define AVE_RSTCTRL 0x8028 /* Reset control */
|
|
#define AVE_RSTCTRL_RMIIRST BIT(16)
|
|
#define AVE_LINKSEL 0x8034 /* Link speed setting */
|
|
#define AVE_LINKSEL_100M BIT(0)
|
|
|
|
/* AVE_GRR */
|
|
#define AVE_GRR_PHYRST BIT(4) /* Reset external PHY */
|
|
#define AVE_GRR_GRST BIT(0) /* Reset all MAC */
|
|
|
|
/* AVE_CFGR */
|
|
#define AVE_CFGR_MII BIT(27) /* Func mode (1:MII/RMII, 0:RGMII) */
|
|
|
|
/* AVE_GISR (common with GIMR) */
|
|
#define AVE_GIMR_CLR 0
|
|
#define AVE_GISR_CLR GENMASK(31, 0)
|
|
|
|
/* AVE_TXCR */
|
|
#define AVE_TXCR_FLOCTR BIT(18) /* Flow control */
|
|
#define AVE_TXCR_TXSPD_1G BIT(17)
|
|
#define AVE_TXCR_TXSPD_100 BIT(16)
|
|
|
|
/* AVE_RXCR */
|
|
#define AVE_RXCR_RXEN BIT(30) /* Rx enable */
|
|
#define AVE_RXCR_FDUPEN BIT(22) /* Interface mode */
|
|
#define AVE_RXCR_FLOCTR BIT(21) /* Flow control */
|
|
|
|
/* AVE_MDIOCTR */
|
|
#define AVE_MDIOCTR_RREQ BIT(3) /* Read request */
|
|
#define AVE_MDIOCTR_WREQ BIT(2) /* Write request */
|
|
|
|
/* AVE_MDIOSR */
|
|
#define AVE_MDIOSR_STS BIT(0) /* access status */
|
|
|
|
/* AVE_DESCC */
|
|
#define AVE_DESCC_RXDSTPSTS BIT(20)
|
|
#define AVE_DESCC_RD0 BIT(8) /* Enable Rx descriptor Ring0 */
|
|
#define AVE_DESCC_RXDSTP BIT(4) /* Pause Rx descriptor */
|
|
#define AVE_DESCC_TD BIT(0) /* Enable Tx descriptor */
|
|
|
|
/* AVE_TXDC/RXDC */
|
|
#define AVE_DESC_SIZE(priv, num) \
|
|
((num) * ((priv)->data->is_desc_64bit ? AVE_DESC_SIZE_64 : \
|
|
AVE_DESC_SIZE_32))
|
|
|
|
/* Command status for descriptor */
|
|
#define AVE_STS_OWN BIT(31) /* Descriptor ownership */
|
|
#define AVE_STS_OK BIT(27) /* Normal transmit */
|
|
#define AVE_STS_1ST BIT(26) /* Head of buffer chain */
|
|
#define AVE_STS_LAST BIT(25) /* Tail of buffer chain */
|
|
#define AVE_STS_PKTLEN_TX_MASK GENMASK(15, 0)
|
|
#define AVE_STS_PKTLEN_RX_MASK GENMASK(10, 0)
|
|
|
|
#define AVE_DESC_OFS_CMDSTS 0
|
|
#define AVE_DESC_OFS_ADDRL 4
|
|
#define AVE_DESC_OFS_ADDRU 8
|
|
|
|
/* Parameter for ethernet frame */
|
|
#define AVE_RXCR_MTU 1518
|
|
|
|
/* SG */
|
|
#define SG_ETPINMODE 0x540
|
|
#define SG_ETPINMODE_EXTPHY BIT(1) /* for LD11 */
|
|
#define SG_ETPINMODE_RMII(ins) BIT(ins)
|
|
|
|
#define AVE_MAX_CLKS 4
|
|
#define AVE_MAX_RSTS 2
|
|
|
|
enum desc_id {
|
|
AVE_DESCID_TX,
|
|
AVE_DESCID_RX,
|
|
};
|
|
|
|
struct ave_private {
|
|
phys_addr_t iobase;
|
|
unsigned int nclks;
|
|
struct clk clk[AVE_MAX_CLKS];
|
|
unsigned int nrsts;
|
|
struct reset_ctl rst[AVE_MAX_RSTS];
|
|
struct regmap *regmap;
|
|
unsigned int regmap_arg;
|
|
|
|
struct mii_dev *bus;
|
|
struct phy_device *phydev;
|
|
int phy_mode;
|
|
int max_speed;
|
|
|
|
int rx_pos;
|
|
int rx_siz;
|
|
int rx_off;
|
|
int tx_num;
|
|
|
|
u8 tx_adj_packetbuf[PKTSIZE_ALIGN + PKTALIGN];
|
|
void *tx_adj_buf;
|
|
|
|
const struct ave_soc_data *data;
|
|
};
|
|
|
|
struct ave_soc_data {
|
|
bool is_desc_64bit;
|
|
const char *clock_names[AVE_MAX_CLKS];
|
|
const char *reset_names[AVE_MAX_RSTS];
|
|
int (*get_pinmode)(struct ave_private *priv);
|
|
};
|
|
|
|
static u32 ave_desc_read(struct ave_private *priv, enum desc_id id, int entry,
|
|
int offset)
|
|
{
|
|
int desc_size;
|
|
u32 addr;
|
|
|
|
if (priv->data->is_desc_64bit) {
|
|
desc_size = AVE_DESC_SIZE_64;
|
|
addr = (id == AVE_DESCID_TX) ? AVE_TXDM_64 : AVE_RXDM_64;
|
|
} else {
|
|
desc_size = AVE_DESC_SIZE_32;
|
|
addr = (id == AVE_DESCID_TX) ? AVE_TXDM_32 : AVE_RXDM_32;
|
|
}
|
|
|
|
addr += entry * desc_size + offset;
|
|
|
|
return readl(priv->iobase + addr);
|
|
}
|
|
|
|
static u32 ave_desc_read_cmdsts(struct ave_private *priv, enum desc_id id,
|
|
int entry)
|
|
{
|
|
return ave_desc_read(priv, id, entry, AVE_DESC_OFS_CMDSTS);
|
|
}
|
|
|
|
static void ave_desc_write(struct ave_private *priv, enum desc_id id,
|
|
int entry, int offset, u32 val)
|
|
{
|
|
int desc_size;
|
|
u32 addr;
|
|
|
|
if (priv->data->is_desc_64bit) {
|
|
desc_size = AVE_DESC_SIZE_64;
|
|
addr = (id == AVE_DESCID_TX) ? AVE_TXDM_64 : AVE_RXDM_64;
|
|
} else {
|
|
desc_size = AVE_DESC_SIZE_32;
|
|
addr = (id == AVE_DESCID_TX) ? AVE_TXDM_32 : AVE_RXDM_32;
|
|
}
|
|
|
|
addr += entry * desc_size + offset;
|
|
writel(val, priv->iobase + addr);
|
|
}
|
|
|
|
static void ave_desc_write_cmdsts(struct ave_private *priv, enum desc_id id,
|
|
int entry, u32 val)
|
|
{
|
|
ave_desc_write(priv, id, entry, AVE_DESC_OFS_CMDSTS, val);
|
|
}
|
|
|
|
static void ave_desc_write_addr(struct ave_private *priv, enum desc_id id,
|
|
int entry, uintptr_t paddr)
|
|
{
|
|
ave_desc_write(priv, id, entry,
|
|
AVE_DESC_OFS_ADDRL, lower_32_bits(paddr));
|
|
if (priv->data->is_desc_64bit)
|
|
ave_desc_write(priv, id, entry,
|
|
AVE_DESC_OFS_ADDRU, upper_32_bits(paddr));
|
|
}
|
|
|
|
static void ave_cache_invalidate(uintptr_t vaddr, int len)
|
|
{
|
|
invalidate_dcache_range(rounddown(vaddr, ARCH_DMA_MINALIGN),
|
|
roundup(vaddr + len, ARCH_DMA_MINALIGN));
|
|
}
|
|
|
|
static void ave_cache_flush(uintptr_t vaddr, int len)
|
|
{
|
|
flush_dcache_range(rounddown(vaddr, ARCH_DMA_MINALIGN),
|
|
roundup(vaddr + len, ARCH_DMA_MINALIGN));
|
|
}
|
|
|
|
static int ave_mdiobus_read(struct mii_dev *bus,
|
|
int phyid, int devad, int regnum)
|
|
{
|
|
struct ave_private *priv = bus->priv;
|
|
u32 mdioctl, mdiosr;
|
|
int ret;
|
|
|
|
/* write address */
|
|
writel((phyid << 8) | regnum, priv->iobase + AVE_MDIOAR);
|
|
|
|
/* read request */
|
|
mdioctl = readl(priv->iobase + AVE_MDIOCTR);
|
|
writel(mdioctl | AVE_MDIOCTR_RREQ, priv->iobase + AVE_MDIOCTR);
|
|
|
|
ret = readl_poll_timeout(priv->iobase + AVE_MDIOSR, mdiosr,
|
|
!(mdiosr & AVE_MDIOSR_STS),
|
|
AVE_MDIO_TIMEOUT_USEC);
|
|
if (ret) {
|
|
pr_err("%s: failed to read from mdio (phy:%d reg:%x)\n",
|
|
priv->phydev->dev->name, phyid, regnum);
|
|
return ret;
|
|
}
|
|
|
|
return readl(priv->iobase + AVE_MDIORDR) & GENMASK(15, 0);
|
|
}
|
|
|
|
static int ave_mdiobus_write(struct mii_dev *bus,
|
|
int phyid, int devad, int regnum, u16 val)
|
|
{
|
|
struct ave_private *priv = bus->priv;
|
|
u32 mdioctl, mdiosr;
|
|
int ret;
|
|
|
|
/* write address */
|
|
writel((phyid << 8) | regnum, priv->iobase + AVE_MDIOAR);
|
|
|
|
/* write data */
|
|
writel(val, priv->iobase + AVE_MDIOWDR);
|
|
|
|
/* write request */
|
|
mdioctl = readl(priv->iobase + AVE_MDIOCTR);
|
|
writel((mdioctl | AVE_MDIOCTR_WREQ) & ~AVE_MDIOCTR_RREQ,
|
|
priv->iobase + AVE_MDIOCTR);
|
|
|
|
ret = readl_poll_timeout(priv->iobase + AVE_MDIOSR, mdiosr,
|
|
!(mdiosr & AVE_MDIOSR_STS),
|
|
AVE_MDIO_TIMEOUT_USEC);
|
|
if (ret)
|
|
pr_err("%s: failed to write to mdio (phy:%d reg:%x)\n",
|
|
priv->phydev->dev->name, phyid, regnum);
|
|
|
|
return ret;
|
|
}
|
|
|
|
static int ave_adjust_link(struct ave_private *priv)
|
|
{
|
|
struct phy_device *phydev = priv->phydev;
|
|
struct eth_pdata *pdata = dev_get_plat(phydev->dev);
|
|
u32 val, txcr, rxcr, rxcr_org;
|
|
u16 rmt_adv = 0, lcl_adv = 0;
|
|
u8 cap;
|
|
|
|
/* set RGMII speed */
|
|
val = readl(priv->iobase + AVE_TXCR);
|
|
val &= ~(AVE_TXCR_TXSPD_100 | AVE_TXCR_TXSPD_1G);
|
|
|
|
if (phy_interface_is_rgmii(phydev) && phydev->speed == SPEED_1000)
|
|
val |= AVE_TXCR_TXSPD_1G;
|
|
else if (phydev->speed == SPEED_100)
|
|
val |= AVE_TXCR_TXSPD_100;
|
|
|
|
writel(val, priv->iobase + AVE_TXCR);
|
|
|
|
/* set RMII speed (100M/10M only) */
|
|
if (!phy_interface_is_rgmii(phydev)) {
|
|
val = readl(priv->iobase + AVE_LINKSEL);
|
|
if (phydev->speed == SPEED_10)
|
|
val &= ~AVE_LINKSEL_100M;
|
|
else
|
|
val |= AVE_LINKSEL_100M;
|
|
writel(val, priv->iobase + AVE_LINKSEL);
|
|
}
|
|
|
|
/* check current RXCR/TXCR */
|
|
rxcr = readl(priv->iobase + AVE_RXCR);
|
|
txcr = readl(priv->iobase + AVE_TXCR);
|
|
rxcr_org = rxcr;
|
|
|
|
if (phydev->duplex) {
|
|
rxcr |= AVE_RXCR_FDUPEN;
|
|
|
|
if (phydev->pause)
|
|
rmt_adv |= LPA_PAUSE_CAP;
|
|
if (phydev->asym_pause)
|
|
rmt_adv |= LPA_PAUSE_ASYM;
|
|
if (phydev->advertising & ADVERTISED_Pause)
|
|
lcl_adv |= ADVERTISE_PAUSE_CAP;
|
|
if (phydev->advertising & ADVERTISED_Asym_Pause)
|
|
lcl_adv |= ADVERTISE_PAUSE_ASYM;
|
|
|
|
cap = mii_resolve_flowctrl_fdx(lcl_adv, rmt_adv);
|
|
if (cap & FLOW_CTRL_TX)
|
|
txcr |= AVE_TXCR_FLOCTR;
|
|
else
|
|
txcr &= ~AVE_TXCR_FLOCTR;
|
|
if (cap & FLOW_CTRL_RX)
|
|
rxcr |= AVE_RXCR_FLOCTR;
|
|
else
|
|
rxcr &= ~AVE_RXCR_FLOCTR;
|
|
} else {
|
|
rxcr &= ~AVE_RXCR_FDUPEN;
|
|
rxcr &= ~AVE_RXCR_FLOCTR;
|
|
txcr &= ~AVE_TXCR_FLOCTR;
|
|
}
|
|
|
|
if (rxcr_org != rxcr) {
|
|
/* disable Rx mac */
|
|
writel(rxcr & ~AVE_RXCR_RXEN, priv->iobase + AVE_RXCR);
|
|
/* change and enable TX/Rx mac */
|
|
writel(txcr, priv->iobase + AVE_TXCR);
|
|
writel(rxcr, priv->iobase + AVE_RXCR);
|
|
}
|
|
|
|
pr_notice("%s: phy:%s speed:%d mac:%pM\n",
|
|
phydev->dev->name, phydev->drv->name, phydev->speed,
|
|
pdata->enetaddr);
|
|
|
|
return phydev->link;
|
|
}
|
|
|
|
static int ave_mdiobus_init(struct ave_private *priv, const char *name)
|
|
{
|
|
struct mii_dev *bus = mdio_alloc();
|
|
|
|
if (!bus)
|
|
return -ENOMEM;
|
|
|
|
bus->read = ave_mdiobus_read;
|
|
bus->write = ave_mdiobus_write;
|
|
snprintf(bus->name, sizeof(bus->name), "%s", name);
|
|
bus->priv = priv;
|
|
|
|
return mdio_register(bus);
|
|
}
|
|
|
|
static int ave_phy_init(struct ave_private *priv, void *dev)
|
|
{
|
|
struct phy_device *phydev;
|
|
int ret;
|
|
|
|
phydev = phy_connect(priv->bus, -1, dev, priv->phy_mode);
|
|
if (!phydev)
|
|
return -ENODEV;
|
|
|
|
phydev->supported &= PHY_GBIT_FEATURES;
|
|
if (priv->max_speed) {
|
|
ret = phy_set_supported(phydev, priv->max_speed);
|
|
if (ret)
|
|
return ret;
|
|
}
|
|
phydev->advertising = phydev->supported;
|
|
|
|
priv->phydev = phydev;
|
|
phy_config(phydev);
|
|
|
|
return 0;
|
|
}
|
|
|
|
static void ave_stop(struct udevice *dev)
|
|
{
|
|
struct ave_private *priv = dev_get_priv(dev);
|
|
u32 val;
|
|
int ret;
|
|
|
|
val = readl(priv->iobase + AVE_GRR);
|
|
if (val)
|
|
return;
|
|
|
|
val = readl(priv->iobase + AVE_RXCR);
|
|
val &= ~AVE_RXCR_RXEN;
|
|
writel(val, priv->iobase + AVE_RXCR);
|
|
|
|
writel(0, priv->iobase + AVE_DESCC);
|
|
ret = readl_poll_timeout(priv->iobase + AVE_DESCC, val, !val,
|
|
AVE_HALT_TIMEOUT_USEC);
|
|
if (ret)
|
|
pr_warn("%s: halt timeout\n", priv->phydev->dev->name);
|
|
|
|
writel(AVE_GRR_GRST, priv->iobase + AVE_GRR);
|
|
|
|
phy_shutdown(priv->phydev);
|
|
}
|
|
|
|
static void ave_reset(struct ave_private *priv)
|
|
{
|
|
u32 val;
|
|
|
|
/* reset RMII register */
|
|
val = readl(priv->iobase + AVE_RSTCTRL);
|
|
val &= ~AVE_RSTCTRL_RMIIRST;
|
|
writel(val, priv->iobase + AVE_RSTCTRL);
|
|
|
|
/* assert reset */
|
|
writel(AVE_GRR_GRST | AVE_GRR_PHYRST, priv->iobase + AVE_GRR);
|
|
mdelay(AVE_GRST_DELAY_MSEC);
|
|
|
|
/* 1st, negate PHY reset only */
|
|
writel(AVE_GRR_GRST, priv->iobase + AVE_GRR);
|
|
mdelay(AVE_GRST_DELAY_MSEC);
|
|
|
|
/* negate reset */
|
|
writel(0, priv->iobase + AVE_GRR);
|
|
mdelay(AVE_GRST_DELAY_MSEC);
|
|
|
|
/* negate RMII register */
|
|
val = readl(priv->iobase + AVE_RSTCTRL);
|
|
val |= AVE_RSTCTRL_RMIIRST;
|
|
writel(val, priv->iobase + AVE_RSTCTRL);
|
|
}
|
|
|
|
static int ave_start(struct udevice *dev)
|
|
{
|
|
struct ave_private *priv = dev_get_priv(dev);
|
|
uintptr_t paddr;
|
|
u32 val;
|
|
int i;
|
|
|
|
ave_reset(priv);
|
|
|
|
priv->rx_pos = 0;
|
|
priv->rx_off = 2; /* RX data has 2byte offsets */
|
|
priv->tx_num = 0;
|
|
priv->tx_adj_buf =
|
|
(void *)roundup((uintptr_t)&priv->tx_adj_packetbuf[0],
|
|
PKTALIGN);
|
|
priv->rx_siz = (PKTSIZE_ALIGN - priv->rx_off);
|
|
|
|
val = 0;
|
|
if (priv->phy_mode != PHY_INTERFACE_MODE_RGMII &&
|
|
priv->phy_mode != PHY_INTERFACE_MODE_RGMII_ID &&
|
|
priv->phy_mode != PHY_INTERFACE_MODE_RGMII_RXID &&
|
|
priv->phy_mode != PHY_INTERFACE_MODE_RGMII_TXID)
|
|
val |= AVE_CFGR_MII;
|
|
writel(val, priv->iobase + AVE_CFGR);
|
|
|
|
/* use one descriptor for Tx */
|
|
writel(AVE_DESC_SIZE(priv, 1) << 16, priv->iobase + AVE_TXDC);
|
|
ave_desc_write_cmdsts(priv, AVE_DESCID_TX, 0, 0);
|
|
ave_desc_write_addr(priv, AVE_DESCID_TX, 0, 0);
|
|
|
|
/* use PKTBUFSRX descriptors for Rx */
|
|
writel(AVE_DESC_SIZE(priv, PKTBUFSRX) << 16, priv->iobase + AVE_RXDC);
|
|
for (i = 0; i < PKTBUFSRX; i++) {
|
|
paddr = (uintptr_t)net_rx_packets[i];
|
|
ave_cache_flush(paddr, priv->rx_siz + priv->rx_off);
|
|
ave_desc_write_addr(priv, AVE_DESCID_RX, i, paddr);
|
|
ave_desc_write_cmdsts(priv, AVE_DESCID_RX, i, priv->rx_siz);
|
|
}
|
|
|
|
writel(AVE_GISR_CLR, priv->iobase + AVE_GISR);
|
|
writel(AVE_GIMR_CLR, priv->iobase + AVE_GIMR);
|
|
|
|
writel(AVE_RXCR_RXEN | AVE_RXCR_FDUPEN | AVE_RXCR_FLOCTR | AVE_RXCR_MTU,
|
|
priv->iobase + AVE_RXCR);
|
|
writel(AVE_DESCC_RD0 | AVE_DESCC_TD, priv->iobase + AVE_DESCC);
|
|
|
|
phy_startup(priv->phydev);
|
|
ave_adjust_link(priv);
|
|
|
|
return 0;
|
|
}
|
|
|
|
static int ave_write_hwaddr(struct udevice *dev)
|
|
{
|
|
struct ave_private *priv = dev_get_priv(dev);
|
|
struct eth_pdata *pdata = dev_get_plat(dev);
|
|
u8 *mac = pdata->enetaddr;
|
|
|
|
writel(mac[0] | mac[1] << 8 | mac[2] << 16 | mac[3] << 24,
|
|
priv->iobase + AVE_RXMAC1R);
|
|
writel(mac[4] | mac[5] << 8, priv->iobase + AVE_RXMAC2R);
|
|
|
|
return 0;
|
|
}
|
|
|
|
static int ave_send(struct udevice *dev, void *packet, int length)
|
|
{
|
|
struct ave_private *priv = dev_get_priv(dev);
|
|
u32 val;
|
|
void *ptr = packet;
|
|
int count;
|
|
|
|
/* adjust alignment for descriptor */
|
|
if ((uintptr_t)ptr & 0x3) {
|
|
memcpy(priv->tx_adj_buf, (const void *)ptr, length);
|
|
ptr = priv->tx_adj_buf;
|
|
}
|
|
|
|
/* padding for minimum length */
|
|
if (length < AVE_MIN_XMITSIZE) {
|
|
memset(ptr + length, 0, AVE_MIN_XMITSIZE - length);
|
|
length = AVE_MIN_XMITSIZE;
|
|
}
|
|
|
|
/* check ownership and wait for previous xmit done */
|
|
count = AVE_SEND_TIMEOUT_COUNT;
|
|
do {
|
|
val = ave_desc_read_cmdsts(priv, AVE_DESCID_TX, 0);
|
|
} while ((val & AVE_STS_OWN) && --count);
|
|
if (!count)
|
|
return -ETIMEDOUT;
|
|
|
|
ave_cache_flush((uintptr_t)ptr, length);
|
|
ave_desc_write_addr(priv, AVE_DESCID_TX, 0, (uintptr_t)ptr);
|
|
|
|
val = AVE_STS_OWN | AVE_STS_1ST | AVE_STS_LAST |
|
|
(length & AVE_STS_PKTLEN_TX_MASK);
|
|
ave_desc_write_cmdsts(priv, AVE_DESCID_TX, 0, val);
|
|
priv->tx_num++;
|
|
|
|
count = AVE_SEND_TIMEOUT_COUNT;
|
|
do {
|
|
val = ave_desc_read_cmdsts(priv, AVE_DESCID_TX, 0);
|
|
} while ((val & AVE_STS_OWN) && --count);
|
|
if (!count)
|
|
return -ETIMEDOUT;
|
|
|
|
if (!(val & AVE_STS_OK))
|
|
pr_warn("%s: bad send packet status:%08x\n",
|
|
priv->phydev->dev->name, le32_to_cpu(val));
|
|
|
|
return 0;
|
|
}
|
|
|
|
static int ave_recv(struct udevice *dev, int flags, uchar **packetp)
|
|
{
|
|
struct ave_private *priv = dev_get_priv(dev);
|
|
uchar *ptr;
|
|
int length = 0;
|
|
u32 cmdsts;
|
|
|
|
while (1) {
|
|
cmdsts = ave_desc_read_cmdsts(priv, AVE_DESCID_RX,
|
|
priv->rx_pos);
|
|
if (!(cmdsts & AVE_STS_OWN))
|
|
/* hardware ownership, no received packets */
|
|
return -EAGAIN;
|
|
|
|
ptr = net_rx_packets[priv->rx_pos] + priv->rx_off;
|
|
if (cmdsts & AVE_STS_OK)
|
|
break;
|
|
|
|
pr_warn("%s: bad packet[%d] status:%08x ptr:%p\n",
|
|
priv->phydev->dev->name, priv->rx_pos,
|
|
le32_to_cpu(cmdsts), ptr);
|
|
}
|
|
|
|
length = cmdsts & AVE_STS_PKTLEN_RX_MASK;
|
|
|
|
/* invalidate after DMA is done */
|
|
ave_cache_invalidate((uintptr_t)ptr, length);
|
|
*packetp = ptr;
|
|
|
|
return length;
|
|
}
|
|
|
|
static int ave_free_packet(struct udevice *dev, uchar *packet, int length)
|
|
{
|
|
struct ave_private *priv = dev_get_priv(dev);
|
|
|
|
ave_cache_flush((uintptr_t)net_rx_packets[priv->rx_pos],
|
|
priv->rx_siz + priv->rx_off);
|
|
|
|
ave_desc_write_cmdsts(priv, AVE_DESCID_RX,
|
|
priv->rx_pos, priv->rx_siz);
|
|
|
|
if (++priv->rx_pos >= PKTBUFSRX)
|
|
priv->rx_pos = 0;
|
|
|
|
return 0;
|
|
}
|
|
|
|
static int ave_pro4_get_pinmode(struct ave_private *priv)
|
|
{
|
|
u32 reg, mask, val = 0;
|
|
|
|
if (priv->regmap_arg > 0)
|
|
return -EINVAL;
|
|
|
|
mask = SG_ETPINMODE_RMII(0);
|
|
|
|
switch (priv->phy_mode) {
|
|
case PHY_INTERFACE_MODE_RMII:
|
|
val = SG_ETPINMODE_RMII(0);
|
|
break;
|
|
case PHY_INTERFACE_MODE_MII:
|
|
case PHY_INTERFACE_MODE_RGMII:
|
|
case PHY_INTERFACE_MODE_RGMII_ID:
|
|
case PHY_INTERFACE_MODE_RGMII_RXID:
|
|
case PHY_INTERFACE_MODE_RGMII_TXID:
|
|
break;
|
|
default:
|
|
return -EINVAL;
|
|
}
|
|
|
|
regmap_read(priv->regmap, SG_ETPINMODE, ®);
|
|
reg &= ~mask;
|
|
reg |= val;
|
|
regmap_write(priv->regmap, SG_ETPINMODE, reg);
|
|
|
|
return 0;
|
|
}
|
|
|
|
static int ave_ld11_get_pinmode(struct ave_private *priv)
|
|
{
|
|
u32 reg, mask, val = 0;
|
|
|
|
if (priv->regmap_arg > 0)
|
|
return -EINVAL;
|
|
|
|
mask = SG_ETPINMODE_EXTPHY | SG_ETPINMODE_RMII(0);
|
|
|
|
switch (priv->phy_mode) {
|
|
case PHY_INTERFACE_MODE_INTERNAL:
|
|
break;
|
|
case PHY_INTERFACE_MODE_RMII:
|
|
val = SG_ETPINMODE_EXTPHY | SG_ETPINMODE_RMII(0);
|
|
break;
|
|
default:
|
|
return -EINVAL;
|
|
}
|
|
|
|
regmap_read(priv->regmap, SG_ETPINMODE, ®);
|
|
reg &= ~mask;
|
|
reg |= val;
|
|
regmap_write(priv->regmap, SG_ETPINMODE, reg);
|
|
|
|
return 0;
|
|
}
|
|
|
|
static int ave_ld20_get_pinmode(struct ave_private *priv)
|
|
{
|
|
u32 reg, mask, val = 0;
|
|
|
|
if (priv->regmap_arg > 0)
|
|
return -EINVAL;
|
|
|
|
mask = SG_ETPINMODE_RMII(0);
|
|
|
|
switch (priv->phy_mode) {
|
|
case PHY_INTERFACE_MODE_RMII:
|
|
val = SG_ETPINMODE_RMII(0);
|
|
break;
|
|
case PHY_INTERFACE_MODE_RGMII:
|
|
case PHY_INTERFACE_MODE_RGMII_ID:
|
|
case PHY_INTERFACE_MODE_RGMII_RXID:
|
|
case PHY_INTERFACE_MODE_RGMII_TXID:
|
|
break;
|
|
default:
|
|
return -EINVAL;
|
|
}
|
|
|
|
regmap_read(priv->regmap, SG_ETPINMODE, ®);
|
|
reg &= ~mask;
|
|
reg |= val;
|
|
regmap_write(priv->regmap, SG_ETPINMODE, reg);
|
|
|
|
return 0;
|
|
}
|
|
|
|
static int ave_pxs3_get_pinmode(struct ave_private *priv)
|
|
{
|
|
u32 reg, mask, val = 0;
|
|
|
|
if (priv->regmap_arg > 1)
|
|
return -EINVAL;
|
|
|
|
mask = SG_ETPINMODE_RMII(priv->regmap_arg);
|
|
|
|
switch (priv->phy_mode) {
|
|
case PHY_INTERFACE_MODE_RMII:
|
|
val = SG_ETPINMODE_RMII(priv->regmap_arg);
|
|
break;
|
|
case PHY_INTERFACE_MODE_RGMII:
|
|
case PHY_INTERFACE_MODE_RGMII_ID:
|
|
case PHY_INTERFACE_MODE_RGMII_RXID:
|
|
case PHY_INTERFACE_MODE_RGMII_TXID:
|
|
break;
|
|
default:
|
|
return -EINVAL;
|
|
}
|
|
|
|
regmap_read(priv->regmap, SG_ETPINMODE, ®);
|
|
reg &= ~mask;
|
|
reg |= val;
|
|
regmap_write(priv->regmap, SG_ETPINMODE, reg);
|
|
|
|
return 0;
|
|
}
|
|
|
|
static int ave_of_to_plat(struct udevice *dev)
|
|
{
|
|
struct eth_pdata *pdata = dev_get_plat(dev);
|
|
struct ave_private *priv = dev_get_priv(dev);
|
|
struct ofnode_phandle_args args;
|
|
const u32 *valp;
|
|
int ret, nc, nr;
|
|
const char *name;
|
|
|
|
priv->data = (const struct ave_soc_data *)dev_get_driver_data(dev);
|
|
if (!priv->data)
|
|
return -EINVAL;
|
|
|
|
pdata->iobase = dev_read_addr(dev);
|
|
|
|
pdata->phy_interface = dev_read_phy_mode(dev);
|
|
if (pdata->phy_interface == PHY_INTERFACE_MODE_NA)
|
|
return -EINVAL;
|
|
|
|
pdata->max_speed = 0;
|
|
valp = fdt_getprop(gd->fdt_blob, dev_of_offset(dev), "max-speed",
|
|
NULL);
|
|
if (valp)
|
|
pdata->max_speed = fdt32_to_cpu(*valp);
|
|
|
|
for (nc = 0; nc < AVE_MAX_CLKS; nc++) {
|
|
name = priv->data->clock_names[nc];
|
|
if (!name)
|
|
break;
|
|
ret = clk_get_by_name(dev, name, &priv->clk[nc]);
|
|
if (ret) {
|
|
dev_err(dev, "Failed to get clocks property: %d\n",
|
|
ret);
|
|
goto out_clk_free;
|
|
}
|
|
priv->nclks++;
|
|
}
|
|
|
|
for (nr = 0; nr < AVE_MAX_RSTS; nr++) {
|
|
name = priv->data->reset_names[nr];
|
|
if (!name)
|
|
break;
|
|
ret = reset_get_by_name(dev, name, &priv->rst[nr]);
|
|
if (ret) {
|
|
dev_err(dev, "Failed to get resets property: %d\n",
|
|
ret);
|
|
goto out_reset_free;
|
|
}
|
|
priv->nrsts++;
|
|
}
|
|
|
|
ret = dev_read_phandle_with_args(dev, "socionext,syscon-phy-mode",
|
|
NULL, 1, 0, &args);
|
|
if (ret) {
|
|
dev_err(dev, "Failed to get syscon-phy-mode property: %d\n",
|
|
ret);
|
|
goto out_reset_free;
|
|
}
|
|
|
|
priv->regmap = syscon_node_to_regmap(args.node);
|
|
if (IS_ERR(priv->regmap)) {
|
|
ret = PTR_ERR(priv->regmap);
|
|
dev_err(dev, "can't get syscon: %d\n", ret);
|
|
goto out_reset_free;
|
|
}
|
|
|
|
if (args.args_count != 1) {
|
|
ret = -EINVAL;
|
|
dev_err(dev, "Invalid argument of syscon-phy-mode\n");
|
|
goto out_reset_free;
|
|
}
|
|
|
|
priv->regmap_arg = args.args[0];
|
|
|
|
return 0;
|
|
|
|
out_reset_free:
|
|
while (--nr >= 0)
|
|
reset_free(&priv->rst[nr]);
|
|
out_clk_free:
|
|
while (--nc >= 0)
|
|
clk_free(&priv->clk[nc]);
|
|
|
|
return ret;
|
|
}
|
|
|
|
static int ave_probe(struct udevice *dev)
|
|
{
|
|
struct eth_pdata *pdata = dev_get_plat(dev);
|
|
struct ave_private *priv = dev_get_priv(dev);
|
|
int ret, nc, nr;
|
|
|
|
priv->data = (const struct ave_soc_data *)dev_get_driver_data(dev);
|
|
if (!priv->data)
|
|
return -EINVAL;
|
|
|
|
priv->iobase = pdata->iobase;
|
|
priv->phy_mode = pdata->phy_interface;
|
|
priv->max_speed = pdata->max_speed;
|
|
|
|
ret = priv->data->get_pinmode(priv);
|
|
if (ret) {
|
|
dev_err(dev, "Invalid phy-mode\n");
|
|
return -EINVAL;
|
|
}
|
|
|
|
for (nc = 0; nc < priv->nclks; nc++) {
|
|
ret = clk_enable(&priv->clk[nc]);
|
|
if (ret) {
|
|
dev_err(dev, "Failed to enable clk: %d\n", ret);
|
|
goto out_clk_release;
|
|
}
|
|
}
|
|
|
|
for (nr = 0; nr < priv->nrsts; nr++) {
|
|
ret = reset_deassert(&priv->rst[nr]);
|
|
if (ret) {
|
|
dev_err(dev, "Failed to deassert reset: %d\n", ret);
|
|
goto out_reset_release;
|
|
}
|
|
}
|
|
|
|
ave_reset(priv);
|
|
|
|
ret = ave_mdiobus_init(priv, dev->name);
|
|
if (ret) {
|
|
dev_err(dev, "Failed to initialize mdiobus: %d\n", ret);
|
|
goto out_reset_release;
|
|
}
|
|
|
|
priv->bus = miiphy_get_dev_by_name(dev->name);
|
|
|
|
ret = ave_phy_init(priv, dev);
|
|
if (ret) {
|
|
dev_err(dev, "Failed to initialize phy: %d\n", ret);
|
|
goto out_mdiobus_release;
|
|
}
|
|
|
|
return 0;
|
|
|
|
out_mdiobus_release:
|
|
mdio_unregister(priv->bus);
|
|
mdio_free(priv->bus);
|
|
out_reset_release:
|
|
reset_release_all(priv->rst, nr);
|
|
out_clk_release:
|
|
clk_release_all(priv->clk, nc);
|
|
|
|
return ret;
|
|
}
|
|
|
|
static int ave_remove(struct udevice *dev)
|
|
{
|
|
struct ave_private *priv = dev_get_priv(dev);
|
|
|
|
free(priv->phydev);
|
|
mdio_unregister(priv->bus);
|
|
mdio_free(priv->bus);
|
|
reset_release_all(priv->rst, priv->nrsts);
|
|
clk_release_all(priv->clk, priv->nclks);
|
|
|
|
return 0;
|
|
}
|
|
|
|
static const struct eth_ops ave_ops = {
|
|
.start = ave_start,
|
|
.stop = ave_stop,
|
|
.send = ave_send,
|
|
.recv = ave_recv,
|
|
.free_pkt = ave_free_packet,
|
|
.write_hwaddr = ave_write_hwaddr,
|
|
};
|
|
|
|
static const struct ave_soc_data ave_pro4_data = {
|
|
.is_desc_64bit = false,
|
|
.clock_names = {
|
|
"gio", "ether", "ether-gb", "ether-phy",
|
|
},
|
|
.reset_names = {
|
|
"gio", "ether",
|
|
},
|
|
.get_pinmode = ave_pro4_get_pinmode,
|
|
};
|
|
|
|
static const struct ave_soc_data ave_pxs2_data = {
|
|
.is_desc_64bit = false,
|
|
.clock_names = {
|
|
"ether",
|
|
},
|
|
.reset_names = {
|
|
"ether",
|
|
},
|
|
.get_pinmode = ave_pro4_get_pinmode,
|
|
};
|
|
|
|
static const struct ave_soc_data ave_ld11_data = {
|
|
.is_desc_64bit = false,
|
|
.clock_names = {
|
|
"ether",
|
|
},
|
|
.reset_names = {
|
|
"ether",
|
|
},
|
|
.get_pinmode = ave_ld11_get_pinmode,
|
|
};
|
|
|
|
static const struct ave_soc_data ave_ld20_data = {
|
|
.is_desc_64bit = true,
|
|
.clock_names = {
|
|
"ether",
|
|
},
|
|
.reset_names = {
|
|
"ether",
|
|
},
|
|
.get_pinmode = ave_ld20_get_pinmode,
|
|
};
|
|
|
|
static const struct ave_soc_data ave_pxs3_data = {
|
|
.is_desc_64bit = false,
|
|
.clock_names = {
|
|
"ether",
|
|
},
|
|
.reset_names = {
|
|
"ether",
|
|
},
|
|
.get_pinmode = ave_pxs3_get_pinmode,
|
|
};
|
|
|
|
static const struct udevice_id ave_ids[] = {
|
|
{
|
|
.compatible = "socionext,uniphier-pro4-ave4",
|
|
.data = (ulong)&ave_pro4_data,
|
|
},
|
|
{
|
|
.compatible = "socionext,uniphier-pxs2-ave4",
|
|
.data = (ulong)&ave_pxs2_data,
|
|
},
|
|
{
|
|
.compatible = "socionext,uniphier-ld11-ave4",
|
|
.data = (ulong)&ave_ld11_data,
|
|
},
|
|
{
|
|
.compatible = "socionext,uniphier-ld20-ave4",
|
|
.data = (ulong)&ave_ld20_data,
|
|
},
|
|
{
|
|
.compatible = "socionext,uniphier-pxs3-ave4",
|
|
.data = (ulong)&ave_pxs3_data,
|
|
},
|
|
{ /* Sentinel */ }
|
|
};
|
|
|
|
U_BOOT_DRIVER(ave) = {
|
|
.name = "ave",
|
|
.id = UCLASS_ETH,
|
|
.of_match = ave_ids,
|
|
.probe = ave_probe,
|
|
.remove = ave_remove,
|
|
.of_to_plat = ave_of_to_plat,
|
|
.ops = &ave_ops,
|
|
.priv_auto = sizeof(struct ave_private),
|
|
.plat_auto = sizeof(struct eth_pdata),
|
|
};
|