mirror of
https://xff.cz/git/u-boot/
synced 2025-10-26 16:13:55 +01:00
net: fec: add support for DM_MDIO
Add support for DM_MDIO by registering a UCLASS_MDIO driver and attempting to use it. This is necessary if wanting to use a DSA driver for example hanging off of the FEC MAC. Care is taken to fallback to non DM_MDIO mii bus as several boards define DM_MDIO without having the proper device-tree configuration necessary such as an mdio subnode, a phy-mode prop, and either a valid phy-handle prop or fixed-phy subnode which will cause dm_eth_phy_connect() to fail. Signed-off-by: Tim Harvey <tharvey@gateworks.com> Reviewed-by: Fabio Estevam <festevam@denx.de>
This commit is contained in:
@@ -30,6 +30,8 @@
|
|||||||
#include <asm/arch/imx-regs.h>
|
#include <asm/arch/imx-regs.h>
|
||||||
#include <asm/mach-imx/sys_proto.h>
|
#include <asm/mach-imx/sys_proto.h>
|
||||||
#include <asm-generic/gpio.h>
|
#include <asm-generic/gpio.h>
|
||||||
|
#include <dm/device_compat.h>
|
||||||
|
#include <dm/lists.h>
|
||||||
|
|
||||||
#include "fec_mxc.h"
|
#include "fec_mxc.h"
|
||||||
#include <eth_phy.h>
|
#include <eth_phy.h>
|
||||||
@@ -1019,6 +1021,81 @@ struct mii_dev *fec_get_miibus(ulong base_addr, int dev_id)
|
|||||||
return bus;
|
return bus;
|
||||||
}
|
}
|
||||||
|
|
||||||
|
#ifdef CONFIG_DM_MDIO
|
||||||
|
struct dm_fec_mdio_priv {
|
||||||
|
struct ethernet_regs *regs;
|
||||||
|
};
|
||||||
|
|
||||||
|
static int dm_fec_mdio_read(struct udevice *dev, int addr, int devad, int reg)
|
||||||
|
{
|
||||||
|
struct dm_fec_mdio_priv *priv = dev_get_priv(dev);
|
||||||
|
|
||||||
|
return fec_mdio_read(priv->regs, addr, reg);
|
||||||
|
}
|
||||||
|
|
||||||
|
static int dm_fec_mdio_write(struct udevice *dev, int addr, int devad, int reg, u16 data)
|
||||||
|
{
|
||||||
|
struct dm_fec_mdio_priv *priv = dev_get_priv(dev);
|
||||||
|
|
||||||
|
return fec_mdio_write(priv->regs, addr, reg, data);
|
||||||
|
}
|
||||||
|
|
||||||
|
static const struct mdio_ops dm_fec_mdio_ops = {
|
||||||
|
.read = dm_fec_mdio_read,
|
||||||
|
.write = dm_fec_mdio_write,
|
||||||
|
};
|
||||||
|
|
||||||
|
static int dm_fec_mdio_probe(struct udevice *dev)
|
||||||
|
{
|
||||||
|
struct dm_fec_mdio_priv *priv = dev_get_priv(dev);
|
||||||
|
|
||||||
|
priv->regs = (struct ethernet_regs *)ofnode_get_addr(dev_ofnode(dev->parent));
|
||||||
|
|
||||||
|
return 0;
|
||||||
|
}
|
||||||
|
|
||||||
|
U_BOOT_DRIVER(fec_mdio) = {
|
||||||
|
.name = "fec_mdio",
|
||||||
|
.id = UCLASS_MDIO,
|
||||||
|
.probe = dm_fec_mdio_probe,
|
||||||
|
.ops = &dm_fec_mdio_ops,
|
||||||
|
.priv_auto = sizeof(struct dm_fec_mdio_priv),
|
||||||
|
};
|
||||||
|
|
||||||
|
static int dm_fec_bind_mdio(struct udevice *dev)
|
||||||
|
{
|
||||||
|
struct udevice *mdiodev;
|
||||||
|
const char *name;
|
||||||
|
ofnode mdio;
|
||||||
|
int ret = -ENODEV;
|
||||||
|
|
||||||
|
/* for a UCLASS_MDIO driver we need to bind and probe manually
|
||||||
|
* for an internal MDIO bus that has no dt compatible of its own
|
||||||
|
*/
|
||||||
|
ofnode_for_each_subnode(mdio, dev_ofnode(dev)) {
|
||||||
|
name = ofnode_get_name(mdio);
|
||||||
|
|
||||||
|
if (strcmp(name, "mdio"))
|
||||||
|
continue;
|
||||||
|
|
||||||
|
ret = device_bind_driver_to_node(dev, "fec_mdio",
|
||||||
|
name, mdio, &mdiodev);
|
||||||
|
if (ret) {
|
||||||
|
printf("%s bind %s failed: %d\n", __func__, name, ret);
|
||||||
|
break;
|
||||||
|
}
|
||||||
|
|
||||||
|
/* need to probe it as there is no compatible to do so */
|
||||||
|
ret = uclass_get_device_by_ofnode(UCLASS_MDIO, mdio, &mdiodev);
|
||||||
|
if (!ret)
|
||||||
|
return 0;
|
||||||
|
printf("%s probe %s failed: %d\n", __func__, name, ret);
|
||||||
|
}
|
||||||
|
|
||||||
|
return ret;
|
||||||
|
}
|
||||||
|
#endif
|
||||||
|
|
||||||
static int fecmxc_read_rom_hwaddr(struct udevice *dev)
|
static int fecmxc_read_rom_hwaddr(struct udevice *dev)
|
||||||
{
|
{
|
||||||
struct fec_priv *priv = dev_get_priv(dev);
|
struct fec_priv *priv = dev_get_priv(dev);
|
||||||
@@ -1082,7 +1159,7 @@ static int device_get_phy_addr(struct fec_priv *priv, struct udevice *dev)
|
|||||||
|
|
||||||
static int fec_phy_init(struct fec_priv *priv, struct udevice *dev)
|
static int fec_phy_init(struct fec_priv *priv, struct udevice *dev)
|
||||||
{
|
{
|
||||||
struct phy_device *phydev;
|
struct phy_device *phydev = NULL;
|
||||||
int addr;
|
int addr;
|
||||||
|
|
||||||
addr = device_get_phy_addr(priv, dev);
|
addr = device_get_phy_addr(priv, dev);
|
||||||
@@ -1090,7 +1167,10 @@ static int fec_phy_init(struct fec_priv *priv, struct udevice *dev)
|
|||||||
addr = CFG_FEC_MXC_PHYADDR;
|
addr = CFG_FEC_MXC_PHYADDR;
|
||||||
#endif
|
#endif
|
||||||
|
|
||||||
phydev = phy_connect(priv->bus, addr, dev, priv->interface);
|
if (IS_ENABLED(CONFIG_DM_MDIO))
|
||||||
|
phydev = dm_eth_phy_connect(dev);
|
||||||
|
if (!phydev)
|
||||||
|
phydev = phy_connect(priv->bus, addr, dev, priv->interface);
|
||||||
if (!phydev)
|
if (!phydev)
|
||||||
return -ENODEV;
|
return -ENODEV;
|
||||||
|
|
||||||
@@ -1221,6 +1301,12 @@ static int fecmxc_probe(struct udevice *dev)
|
|||||||
|
|
||||||
priv->dev_id = dev_seq(dev);
|
priv->dev_id = dev_seq(dev);
|
||||||
|
|
||||||
|
#ifdef CONFIG_DM_MDIO
|
||||||
|
ret = dm_fec_bind_mdio(dev);
|
||||||
|
if (ret && ret != -ENODEV)
|
||||||
|
return ret;
|
||||||
|
#endif
|
||||||
|
|
||||||
#ifdef CONFIG_DM_ETH_PHY
|
#ifdef CONFIG_DM_ETH_PHY
|
||||||
bus = eth_phy_get_mdio_bus(dev);
|
bus = eth_phy_get_mdio_bus(dev);
|
||||||
#endif
|
#endif
|
||||||
|
|||||||
Reference in New Issue
Block a user