2018-11-27 06:19:11 +00:00
|
|
|
// SPDX-License-Identifier: GPL-2.0
|
|
|
|
/*
|
|
|
|
* Xilinx GMII2RGMII phy driver
|
|
|
|
*
|
|
|
|
* Copyright (C) 2018 Xilinx, Inc.
|
|
|
|
*/
|
|
|
|
|
2020-10-31 03:38:53 +00:00
|
|
|
#include <common.h>
|
2018-11-27 06:19:11 +00:00
|
|
|
#include <dm.h>
|
2020-05-10 17:40:05 +00:00
|
|
|
#include <log.h>
|
2018-11-27 06:19:11 +00:00
|
|
|
#include <phy.h>
|
2020-10-31 03:38:53 +00:00
|
|
|
#include <asm/global_data.h>
|
2018-11-27 06:19:11 +00:00
|
|
|
|
|
|
|
DECLARE_GLOBAL_DATA_PTR;
|
|
|
|
|
|
|
|
#define ZYNQ_GMII2RGMII_REG 0x10
|
|
|
|
#define ZYNQ_GMII2RGMII_SPEED_MASK (BMCR_SPEED1000 | BMCR_SPEED100)
|
|
|
|
|
|
|
|
static int xilinxgmiitorgmii_config(struct phy_device *phydev)
|
|
|
|
{
|
2021-03-14 12:14:50 +00:00
|
|
|
ofnode node = phy_get_ofnode(phydev);
|
|
|
|
struct phy_device *ext_phydev;
|
|
|
|
struct ofnode_phandle_args phandle;
|
|
|
|
int ext_phyaddr = -1;
|
|
|
|
int ret;
|
2018-11-27 06:19:11 +00:00
|
|
|
|
|
|
|
debug("%s\n", __func__);
|
2021-03-14 12:14:50 +00:00
|
|
|
|
2022-04-06 22:33:05 +00:00
|
|
|
if (phydev->interface != PHY_INTERFACE_MODE_GMII) {
|
|
|
|
printf("Incorrect interface type\n");
|
|
|
|
return -EINVAL;
|
|
|
|
}
|
|
|
|
|
2021-03-14 12:14:50 +00:00
|
|
|
if (!ofnode_valid(node))
|
|
|
|
return -EINVAL;
|
|
|
|
|
|
|
|
phydev->addr = ofnode_read_u32_default(node, "reg", -1);
|
|
|
|
ret = ofnode_parse_phandle_with_args(node, "phy-handle",
|
|
|
|
NULL, 0, 0, &phandle);
|
|
|
|
if (ret)
|
|
|
|
return ret;
|
|
|
|
|
|
|
|
ext_phyaddr = ofnode_read_u32_default(phandle.node, "reg", -1);
|
|
|
|
ext_phydev = phy_find_by_mask(phydev->bus,
|
2022-04-06 22:33:08 +00:00
|
|
|
1 << ext_phyaddr);
|
2021-03-14 12:14:50 +00:00
|
|
|
if (!ext_phydev) {
|
|
|
|
printf("%s, No external phy device found\n", __func__);
|
|
|
|
return -EINVAL;
|
|
|
|
}
|
|
|
|
|
2023-03-22 08:42:47 +00:00
|
|
|
ext_phydev->interface = ofnode_read_phy_mode(node);
|
|
|
|
if (ext_phydev->interface == PHY_INTERFACE_MODE_NA) {
|
|
|
|
ext_phydev->interface = PHY_INTERFACE_MODE_RGMII;
|
|
|
|
} else if (!phy_interface_is_rgmii(ext_phydev)) {
|
|
|
|
printf("Incorrect external interface type\n");
|
|
|
|
return -EINVAL;
|
|
|
|
}
|
|
|
|
|
2021-03-14 12:14:50 +00:00
|
|
|
ext_phydev->node = phandle.node;
|
|
|
|
phydev->priv = ext_phydev;
|
|
|
|
|
|
|
|
debug("%s, gmii2rgmmi:0x%x, extphy:0x%x\n", __func__, phydev->addr,
|
|
|
|
ext_phyaddr);
|
|
|
|
|
2018-11-27 06:19:11 +00:00
|
|
|
if (ext_phydev->drv->config)
|
|
|
|
ext_phydev->drv->config(ext_phydev);
|
|
|
|
|
|
|
|
return 0;
|
|
|
|
}
|
|
|
|
|
|
|
|
static int xilinxgmiitorgmii_extread(struct phy_device *phydev, int addr,
|
|
|
|
int devaddr, int regnum)
|
|
|
|
{
|
|
|
|
struct phy_device *ext_phydev = phydev->priv;
|
|
|
|
|
|
|
|
debug("%s\n", __func__);
|
|
|
|
if (ext_phydev->drv->readext)
|
|
|
|
ext_phydev->drv->readext(ext_phydev, addr, devaddr, regnum);
|
|
|
|
|
|
|
|
return 0;
|
|
|
|
}
|
|
|
|
|
|
|
|
static int xilinxgmiitorgmii_extwrite(struct phy_device *phydev, int addr,
|
|
|
|
int devaddr, int regnum, u16 val)
|
|
|
|
|
|
|
|
{
|
|
|
|
struct phy_device *ext_phydev = phydev->priv;
|
|
|
|
|
|
|
|
debug("%s\n", __func__);
|
|
|
|
if (ext_phydev->drv->writeext)
|
|
|
|
ext_phydev->drv->writeext(ext_phydev, addr, devaddr, regnum,
|
|
|
|
val);
|
|
|
|
|
|
|
|
return 0;
|
|
|
|
}
|
|
|
|
|
|
|
|
static int xilinxgmiitorgmii_startup(struct phy_device *phydev)
|
|
|
|
{
|
|
|
|
u16 val = 0;
|
|
|
|
struct phy_device *ext_phydev = phydev->priv;
|
|
|
|
|
|
|
|
debug("%s\n", __func__);
|
|
|
|
ext_phydev->dev = phydev->dev;
|
|
|
|
if (ext_phydev->drv->startup)
|
|
|
|
ext_phydev->drv->startup(ext_phydev);
|
|
|
|
|
|
|
|
val = phy_read(phydev, phydev->addr, ZYNQ_GMII2RGMII_REG);
|
|
|
|
val &= ~ZYNQ_GMII2RGMII_SPEED_MASK;
|
|
|
|
|
|
|
|
if (ext_phydev->speed == SPEED_1000)
|
|
|
|
val |= BMCR_SPEED1000;
|
|
|
|
else if (ext_phydev->speed == SPEED_100)
|
|
|
|
val |= BMCR_SPEED100;
|
|
|
|
|
|
|
|
phy_write(phydev, phydev->addr, ZYNQ_GMII2RGMII_REG, val |
|
|
|
|
BMCR_FULLDPLX);
|
|
|
|
|
|
|
|
phydev->duplex = ext_phydev->duplex;
|
|
|
|
phydev->speed = ext_phydev->speed;
|
|
|
|
phydev->link = ext_phydev->link;
|
|
|
|
|
|
|
|
return 0;
|
|
|
|
}
|
|
|
|
|
|
|
|
static int xilinxgmiitorgmii_probe(struct phy_device *phydev)
|
|
|
|
{
|
|
|
|
debug("%s\n", __func__);
|
|
|
|
|
|
|
|
phydev->flags |= PHY_FLAG_BROKEN_RESET;
|
|
|
|
|
|
|
|
return 0;
|
|
|
|
}
|
|
|
|
|
2023-03-19 17:03:11 +00:00
|
|
|
U_BOOT_PHY_DRIVER(gmii2rgmii) = {
|
2018-11-27 06:19:11 +00:00
|
|
|
.name = "XILINX GMII2RGMII",
|
|
|
|
.uid = PHY_GMII2RGMII_ID,
|
|
|
|
.mask = 0xffffffff,
|
|
|
|
.features = PHY_GBIT_FEATURES,
|
|
|
|
.probe = xilinxgmiitorgmii_probe,
|
|
|
|
.config = xilinxgmiitorgmii_config,
|
|
|
|
.startup = xilinxgmiitorgmii_startup,
|
|
|
|
.writeext = xilinxgmiitorgmii_extwrite,
|
|
|
|
.readext = xilinxgmiitorgmii_extread,
|
|
|
|
};
|