blob: 2f0823b83651a86ace38707695c45a0f9d4576c7 [file] [log] [blame]
Tom Rini10e47792018-05-06 17:58:06 -04001// SPDX-License-Identifier: GPL-2.0+
Hannes Schmelzerda494602017-03-23 15:11:43 +01002/*
3 * Fixed-Link phy
4 *
5 * Copyright 2017 Bernecker & Rainer Industrieelektronik GmbH
Hannes Schmelzerda494602017-03-23 15:11:43 +01006 */
7
8#include <config.h>
Tom Riniabb9a042024-05-18 20:20:43 -06009#include <common.h>
Simon Glass9bc15642020-02-03 07:36:16 -070010#include <malloc.h>
Hannes Schmelzerda494602017-03-23 15:11:43 +010011#include <phy.h>
12#include <dm.h>
13#include <fdt_support.h>
Simon Glass3ba929a2020-10-30 21:38:53 -060014#include <asm/global_data.h>
Hannes Schmelzerda494602017-03-23 15:11:43 +010015
16DECLARE_GLOBAL_DATA_PTR;
17
Bin Meng7d2129b2021-03-14 20:14:53 +080018static int fixedphy_probe(struct phy_device *phydev)
Hannes Schmelzerda494602017-03-23 15:11:43 +010019{
Vladimir Oltean6ca194a2021-03-14 20:14:48 +080020 /* fixed-link phy must not be reset by core phy code */
21 phydev->flags |= PHY_FLAG_BROKEN_RESET;
22
23 return 0;
24}
25
Bin Meng7d2129b2021-03-14 20:14:53 +080026static int fixedphy_config(struct phy_device *phydev)
Vladimir Oltean6ca194a2021-03-14 20:14:48 +080027{
28 ofnode node = phy_get_ofnode(phydev);
Hannes Schmelzerda494602017-03-23 15:11:43 +010029 struct fixed_link *priv;
Bin Meng13462672021-03-14 20:14:55 +080030 bool old_binding = false;
31 u32 old_val[5];
Hannes Schmelzerda494602017-03-23 15:11:43 +010032 u32 val;
33
Vladimir Oltean6ca194a2021-03-14 20:14:48 +080034 if (!ofnode_valid(node))
35 return -EINVAL;
36
Hannes Schmelzerda494602017-03-23 15:11:43 +010037 /* check for mandatory properties within fixed-link node */
Vladimir Oltean6ca194a2021-03-14 20:14:48 +080038 val = ofnode_read_u32_default(node, "speed", 0);
Bin Meng13462672021-03-14 20:14:55 +080039
40 if (!val) {
41 /* try old binding */
42 old_binding = true;
43 if (ofnode_read_u32_array(node, "fixed-link", old_val,
44 ARRAY_SIZE(old_val))) {
45 printf("ERROR: no/invalid <fixed-link> property!\n");
46 return -ENOENT;
47 }
48 val = old_val[2];
49 }
50
Vladimir Oltean312e15b2021-01-25 14:23:51 +020051 if (val != SPEED_10 && val != SPEED_100 && val != SPEED_1000 &&
52 val != SPEED_2500 && val != SPEED_10000) {
Bin Mengfbe5aea2021-03-14 20:14:54 +080053 printf("ERROR: no/invalid speed given in fixed-link node!\n");
Hannes Schmelzerda494602017-03-23 15:11:43 +010054 return -EINVAL;
55 }
56
57 priv = malloc(sizeof(*priv));
58 if (!priv)
59 return -ENOMEM;
60 memset(priv, 0, sizeof(*priv));
61
62 phydev->priv = priv;
Hannes Schmelzerda494602017-03-23 15:11:43 +010063
64 priv->link_speed = val;
Bin Meng13462672021-03-14 20:14:55 +080065 if (!old_binding) {
66 priv->duplex = ofnode_read_bool(node, "full-duplex");
67 priv->pause = ofnode_read_bool(node, "pause");
68 priv->asym_pause = ofnode_read_bool(node, "asym-pause");
69 } else {
70 priv->duplex = old_val[1];
71 priv->pause = old_val[3];
72 priv->asym_pause = old_val[4];
73 }
Hannes Schmelzerda494602017-03-23 15:11:43 +010074
75 return 0;
76}
77
Bin Meng7d2129b2021-03-14 20:14:53 +080078static int fixedphy_startup(struct phy_device *phydev)
Hannes Schmelzerda494602017-03-23 15:11:43 +010079{
80 struct fixed_link *priv = phydev->priv;
81
82 phydev->asym_pause = priv->asym_pause;
83 phydev->pause = priv->pause;
84 phydev->duplex = priv->duplex;
85 phydev->speed = priv->link_speed;
86 phydev->link = 1;
87
88 return 0;
89}
90
Bin Meng7d2129b2021-03-14 20:14:53 +080091static int fixedphy_shutdown(struct phy_device *phydev)
Hannes Schmelzerda494602017-03-23 15:11:43 +010092{
93 return 0;
94}
95
Marek Vasutfa976d92023-03-19 18:03:09 +010096U_BOOT_PHY_DRIVER(fixedphy) = {
Hannes Schmelzerda494602017-03-23 15:11:43 +010097 .uid = PHY_FIXED_ID,
98 .mask = 0xffffffff,
99 .name = "Fixed PHY",
100 .features = PHY_GBIT_FEATURES | SUPPORTED_MII,
101 .probe = fixedphy_probe,
Vladimir Oltean6ca194a2021-03-14 20:14:48 +0800102 .config = fixedphy_config,
Hannes Schmelzerda494602017-03-23 15:11:43 +0100103 .startup = fixedphy_startup,
104 .shutdown = fixedphy_shutdown,
105};