1*83d290c5STom Rini // SPDX-License-Identifier: GPL-2.0+
2db40c1aaSHannes Schmelzer /*
3db40c1aaSHannes Schmelzer * Fixed-Link phy
4db40c1aaSHannes Schmelzer *
5db40c1aaSHannes Schmelzer * Copyright 2017 Bernecker & Rainer Industrieelektronik GmbH
6db40c1aaSHannes Schmelzer */
7db40c1aaSHannes Schmelzer
8db40c1aaSHannes Schmelzer #include <config.h>
9db40c1aaSHannes Schmelzer #include <common.h>
10db40c1aaSHannes Schmelzer #include <phy.h>
11db40c1aaSHannes Schmelzer #include <dm.h>
12db40c1aaSHannes Schmelzer #include <fdt_support.h>
13db40c1aaSHannes Schmelzer
14db40c1aaSHannes Schmelzer DECLARE_GLOBAL_DATA_PTR;
15db40c1aaSHannes Schmelzer
fixedphy_probe(struct phy_device * phydev)16db40c1aaSHannes Schmelzer int fixedphy_probe(struct phy_device *phydev)
17db40c1aaSHannes Schmelzer {
18db40c1aaSHannes Schmelzer struct fixed_link *priv;
19db40c1aaSHannes Schmelzer int ofnode = phydev->addr;
20db40c1aaSHannes Schmelzer u32 val;
21db40c1aaSHannes Schmelzer
22db40c1aaSHannes Schmelzer /* check for mandatory properties within fixed-link node */
23db40c1aaSHannes Schmelzer val = fdt_getprop_u32_default_node(gd->fdt_blob,
24db40c1aaSHannes Schmelzer ofnode, 0, "speed", 0);
25db40c1aaSHannes Schmelzer if (val != SPEED_10 && val != SPEED_100 && val != SPEED_1000) {
26db40c1aaSHannes Schmelzer printf("ERROR: no/invalid speed given in fixed-link node!");
27db40c1aaSHannes Schmelzer return -EINVAL;
28db40c1aaSHannes Schmelzer }
29db40c1aaSHannes Schmelzer
30db40c1aaSHannes Schmelzer priv = malloc(sizeof(*priv));
31db40c1aaSHannes Schmelzer if (!priv)
32db40c1aaSHannes Schmelzer return -ENOMEM;
33db40c1aaSHannes Schmelzer memset(priv, 0, sizeof(*priv));
34db40c1aaSHannes Schmelzer
35db40c1aaSHannes Schmelzer phydev->priv = priv;
36db40c1aaSHannes Schmelzer
37db40c1aaSHannes Schmelzer priv->link_speed = val;
38db40c1aaSHannes Schmelzer priv->duplex = fdtdec_get_bool(gd->fdt_blob, ofnode, "full-duplex");
39db40c1aaSHannes Schmelzer priv->pause = fdtdec_get_bool(gd->fdt_blob, ofnode, "pause");
40db40c1aaSHannes Schmelzer priv->asym_pause = fdtdec_get_bool(gd->fdt_blob, ofnode, "asym-pause");
41db40c1aaSHannes Schmelzer
42db40c1aaSHannes Schmelzer /* fixed-link phy must not be reset by core phy code */
43db40c1aaSHannes Schmelzer phydev->flags |= PHY_FLAG_BROKEN_RESET;
44db40c1aaSHannes Schmelzer
45db40c1aaSHannes Schmelzer return 0;
46db40c1aaSHannes Schmelzer }
47db40c1aaSHannes Schmelzer
fixedphy_startup(struct phy_device * phydev)48db40c1aaSHannes Schmelzer int fixedphy_startup(struct phy_device *phydev)
49db40c1aaSHannes Schmelzer {
50db40c1aaSHannes Schmelzer struct fixed_link *priv = phydev->priv;
51db40c1aaSHannes Schmelzer
52db40c1aaSHannes Schmelzer phydev->asym_pause = priv->asym_pause;
53db40c1aaSHannes Schmelzer phydev->pause = priv->pause;
54db40c1aaSHannes Schmelzer phydev->duplex = priv->duplex;
55db40c1aaSHannes Schmelzer phydev->speed = priv->link_speed;
56db40c1aaSHannes Schmelzer phydev->link = 1;
57db40c1aaSHannes Schmelzer
58db40c1aaSHannes Schmelzer return 0;
59db40c1aaSHannes Schmelzer }
60db40c1aaSHannes Schmelzer
fixedphy_shutdown(struct phy_device * phydev)61db40c1aaSHannes Schmelzer int fixedphy_shutdown(struct phy_device *phydev)
62db40c1aaSHannes Schmelzer {
63db40c1aaSHannes Schmelzer return 0;
64db40c1aaSHannes Schmelzer }
65db40c1aaSHannes Schmelzer
66db40c1aaSHannes Schmelzer static struct phy_driver fixedphy_driver = {
67db40c1aaSHannes Schmelzer .uid = PHY_FIXED_ID,
68db40c1aaSHannes Schmelzer .mask = 0xffffffff,
69db40c1aaSHannes Schmelzer .name = "Fixed PHY",
70db40c1aaSHannes Schmelzer .features = PHY_GBIT_FEATURES | SUPPORTED_MII,
71db40c1aaSHannes Schmelzer .probe = fixedphy_probe,
72db40c1aaSHannes Schmelzer .startup = fixedphy_startup,
73db40c1aaSHannes Schmelzer .shutdown = fixedphy_shutdown,
74db40c1aaSHannes Schmelzer };
75db40c1aaSHannes Schmelzer
phy_fixed_init(void)76db40c1aaSHannes Schmelzer int phy_fixed_init(void)
77db40c1aaSHannes Schmelzer {
78db40c1aaSHannes Schmelzer phy_register(&fixedphy_driver);
79db40c1aaSHannes Schmelzer return 0;
80db40c1aaSHannes Schmelzer }
81