xref: /OK3568_Linux_fs/u-boot/net/mdio-uclass.c (revision 4882a59341e53eb6f0b4789bf948001014eff981)
1 // SPDX-License-Identifier: GPL-2.0+
2 /*
3  * (C) Copyright 2019
4  * Alex Marginean, NXP
5  */
6 
7 #include <common.h>
8 #include <dm.h>
9 #include <miiphy.h>
10 #include <dm/device-internal.h>
11 #include <dm/uclass-internal.h>
12 
dm_mdio_probe_devices(void)13 void dm_mdio_probe_devices(void)
14 {
15 	struct udevice *it;
16 	struct uclass *uc;
17 
18 	uclass_get(UCLASS_MDIO, &uc);
19 	uclass_foreach_dev(it, uc) {
20 		device_probe(it);
21 	}
22 }
23 
dm_mdio_post_bind(struct udevice * dev)24 static int dm_mdio_post_bind(struct udevice *dev)
25 {
26 	/*
27 	 * MDIO command doesn't like spaces in names, don't allow them to keep
28 	 * it happy
29 	 */
30 	if (strchr(dev->name, ' ')) {
31 		debug("\nError: MDIO device name \"%s\" has a space!\n",
32 		      dev->name);
33 		return -EINVAL;
34 	}
35 
36 	return 0;
37 }
38 
39 /*
40  * Following read/write/reset functions are registered with legacy MII code.
41  * These are called for PHY operations by upper layers and we further call the
42  * DM MDIO driver functions.
43  */
mdio_read(struct mii_dev * mii_bus,int addr,int devad,int reg)44 static int mdio_read(struct mii_dev *mii_bus, int addr, int devad, int reg)
45 {
46 	struct udevice *dev = mii_bus->priv;
47 
48 	return mdio_get_ops(dev)->read(dev, addr, devad, reg);
49 }
50 
mdio_write(struct mii_dev * mii_bus,int addr,int devad,int reg,u16 val)51 static int mdio_write(struct mii_dev *mii_bus, int addr, int devad, int reg,
52 		      u16 val)
53 {
54 	struct udevice *dev = mii_bus->priv;
55 
56 	return mdio_get_ops(dev)->write(dev, addr, devad, reg, val);
57 }
58 
mdio_reset(struct mii_dev * mii_bus)59 static int mdio_reset(struct mii_dev *mii_bus)
60 {
61 	struct udevice *dev = mii_bus->priv;
62 
63 	if (mdio_get_ops(dev)->reset)
64 		return mdio_get_ops(dev)->reset(dev);
65 	else
66 		return 0;
67 }
68 
dm_mdio_post_probe(struct udevice * dev)69 static int dm_mdio_post_probe(struct udevice *dev)
70 {
71 	struct mdio_perdev_priv *pdata = dev_get_uclass_priv(dev);
72 
73 	pdata->mii_bus = mdio_alloc();
74 	pdata->mii_bus->read = mdio_read;
75 	pdata->mii_bus->write = mdio_write;
76 	pdata->mii_bus->reset = mdio_reset;
77 	pdata->mii_bus->priv = dev;
78 	strncpy(pdata->mii_bus->name, dev->name, MDIO_NAME_LEN);
79 
80 	return mdio_register(pdata->mii_bus);
81 }
82 
dm_mdio_pre_remove(struct udevice * dev)83 static int dm_mdio_pre_remove(struct udevice *dev)
84 {
85 	struct mdio_perdev_priv *pdata = dev_get_uclass_priv(dev);
86 	struct mdio_ops *ops = mdio_get_ops(dev);
87 
88 	if (ops->reset)
89 		ops->reset(dev);
90 	mdio_unregister(pdata->mii_bus);
91 	mdio_free(pdata->mii_bus);
92 
93 	return 0;
94 }
95 
dm_mdio_phy_connect(struct udevice * dev,int addr,struct udevice * ethdev,phy_interface_t interface)96 struct phy_device *dm_mdio_phy_connect(struct udevice *dev, int addr,
97 				       struct udevice *ethdev,
98 				       phy_interface_t interface)
99 {
100 	struct mdio_perdev_priv *pdata = dev_get_uclass_priv(dev);
101 
102 	if (device_probe(dev))
103 		return 0;
104 
105 	return phy_connect(pdata->mii_bus, addr, ethdev, interface);
106 }
107 
108 UCLASS_DRIVER(mdio) = {
109 	.id = UCLASS_MDIO,
110 	.name = "mdio",
111 	.post_bind  = dm_mdio_post_bind,
112 	.post_probe = dm_mdio_post_probe,
113 	.pre_remove = dm_mdio_pre_remove,
114 	.per_device_auto_alloc_size = sizeof(struct mdio_perdev_priv),
115 };
116