blob: f701790194cb27eba95157e247c4f21205c1de32 [file] [log] [blame]
Radu Pirea (NXP OSS)f2d36cb2021-06-18 21:58:30 +03001// SPDX-License-Identifier: GPL-2.0+
2/*
3 * NXP C45 PHY driver
4 *
5 * Copyright 2021 NXP
6 * Author: Radu Pirea <radu-nicolae.pirea@oss.nxp.com>
7 */
8#include <common.h>
9#include <dm.h>
10#include <dm/devres.h>
11#include <linux/delay.h>
12#include <linux/math64.h>
13#include <linux/mdio.h>
14#include <phy.h>
15
16#define PHY_ID_TJA_1103 0x001BB010
17
18#define VEND1_DEVICE_CONTROL 0x0040
19#define DEVICE_CONTROL_RESET BIT(15)
20#define DEVICE_CONTROL_CONFIG_GLOBAL_EN BIT(14)
21#define DEVICE_CONTROL_CONFIG_ALL_EN BIT(13)
22
23#define VEND1_PORT_CONTROL 0x8040
24#define PORT_CONTROL_EN BIT(14)
25
26#define VEND1_PHY_CONTROL 0x8100
27#define PHY_CONFIG_EN BIT(14)
28#define PHY_START_OP BIT(0)
29
30#define VEND1_PHY_CONFIG 0x8108
31#define PHY_CONFIG_AUTO BIT(0)
32
33#define VEND1_PORT_INFRA_CONTROL 0xAC00
34#define PORT_INFRA_CONTROL_EN BIT(14)
35
36#define VEND1_RXID 0xAFCC
37#define VEND1_TXID 0xAFCD
38#define ID_ENABLE BIT(15)
39
40#define VEND1_ABILITIES 0xAFC4
41#define RGMII_ID_ABILITY BIT(15)
42#define RGMII_ABILITY BIT(14)
43#define RMII_ABILITY BIT(10)
44#define REVMII_ABILITY BIT(9)
45#define MII_ABILITY BIT(8)
46#define SGMII_ABILITY BIT(0)
47
48#define VEND1_MII_BASIC_CONFIG 0xAFC6
49#define MII_BASIC_CONFIG_REV BIT(8)
50#define MII_BASIC_CONFIG_SGMII 0x9
51#define MII_BASIC_CONFIG_RGMII 0x7
52#define MII_BASIC_CONFIG_RMII 0x5
53#define MII_BASIC_CONFIG_MII 0x4
54
55#define RGMII_PERIOD_PS 8000U
56#define PS_PER_DEGREE div_u64(RGMII_PERIOD_PS, 360)
57#define MIN_ID_PS 1644U
58#define MAX_ID_PS 2260U
59#define DEFAULT_ID_PS 2000U
60
61#define RESET_DELAY_MS 25
62#define CONF_EN_DELAY_US 450
63
64struct nxp_c45_phy {
65 u32 tx_delay;
66 u32 rx_delay;
67};
68
69static int nxp_c45_soft_reset(struct phy_device *phydev)
70{
71 int tries = 10, ret;
72
73 ret = phy_write_mmd(phydev, MDIO_MMD_VEND1, VEND1_DEVICE_CONTROL,
74 DEVICE_CONTROL_RESET);
75 if (ret)
76 return ret;
77
78 do {
79 ret = phy_read_mmd(phydev, MDIO_MMD_VEND1,
80 VEND1_DEVICE_CONTROL);
81 if (!(ret & DEVICE_CONTROL_RESET))
82 return 0;
83 mdelay(RESET_DELAY_MS);
84 } while (tries--);
85
86 return -EIO;
87}
88
89static int nxp_c45_start_op(struct phy_device *phydev)
90{
91 return phy_write_mmd(phydev, MDIO_MMD_VEND1, VEND1_PHY_CONTROL,
92 PHY_START_OP);
93}
94
95static int nxp_c45_config_enable(struct phy_device *phydev)
96{
97 phy_write_mmd(phydev, MDIO_MMD_VEND1, VEND1_DEVICE_CONTROL,
98 DEVICE_CONTROL_CONFIG_GLOBAL_EN |
99 DEVICE_CONTROL_CONFIG_ALL_EN);
100 udelay(CONF_EN_DELAY_US);
101
102 phy_write_mmd(phydev, MDIO_MMD_VEND1, VEND1_PORT_CONTROL,
103 PORT_CONTROL_EN);
104 phy_write_mmd(phydev, MDIO_MMD_VEND1, VEND1_PHY_CONTROL,
105 PHY_CONFIG_EN);
106 phy_write_mmd(phydev, MDIO_MMD_VEND1, VEND1_PORT_INFRA_CONTROL,
107 PORT_INFRA_CONTROL_EN);
108
109 return 0;
110}
111
112static u64 nxp_c45_get_phase_shift(u64 phase_offset_raw)
113{
114 /* The delay in degree phase is 73.8 + phase_offset_raw * 0.9.
115 * To avoid floating point operations we'll multiply by 10
116 * and get 1 decimal point precision.
117 */
118 phase_offset_raw *= 10;
119 phase_offset_raw -= 738;
120 return div_u64(phase_offset_raw, 9);
121}
122
123static void nxp_c45_disable_delays(struct phy_device *phydev)
124{
125 phy_write_mmd(phydev, MDIO_MMD_VEND1, VEND1_TXID, 0);
126 phy_write_mmd(phydev, MDIO_MMD_VEND1, VEND1_RXID, 0);
127}
128
129static int nxp_c45_check_delay(struct phy_device *phydev, u32 delay)
130{
131 if (delay < MIN_ID_PS) {
132 pr_err("%s: delay value smaller than %u\n",
133 phydev->drv->name, MIN_ID_PS);
134 return -EINVAL;
135 }
136
137 if (delay > MAX_ID_PS) {
138 pr_err("%s: delay value higher than %u\n",
139 phydev->drv->name, MAX_ID_PS);
140 return -EINVAL;
141 }
142
143 return 0;
144}
145
146static int nxp_c45_get_delays(struct phy_device *phydev)
147{
148 struct nxp_c45_phy *priv = phydev->priv;
149 int ret;
150
151 if (phydev->interface == PHY_INTERFACE_MODE_RGMII_ID ||
152 phydev->interface == PHY_INTERFACE_MODE_RGMII_TXID) {
153 ret = dev_read_u32(phydev->dev, "tx-internal-delay-ps",
154 &priv->tx_delay);
155 if (ret)
156 priv->tx_delay = DEFAULT_ID_PS;
157
158 ret = nxp_c45_check_delay(phydev, priv->tx_delay);
159 if (ret) {
160 pr_err("%s: tx-internal-delay-ps invalid value\n",
161 phydev->drv->name);
162 return ret;
163 }
164 }
165
166 if (phydev->interface == PHY_INTERFACE_MODE_RGMII_ID ||
167 phydev->interface == PHY_INTERFACE_MODE_RGMII_RXID) {
168 ret = dev_read_u32(phydev->dev, "rx-internal-delay-ps",
169 &priv->rx_delay);
170 if (ret)
171 priv->rx_delay = DEFAULT_ID_PS;
172
173 ret = nxp_c45_check_delay(phydev, priv->rx_delay);
174 if (ret) {
175 pr_err("%s: rx-internal-delay-ps invalid value\n",
176 phydev->drv->name);
177 return ret;
178 }
179 }
180
181 return 0;
182}
183
184static void nxp_c45_set_delays(struct phy_device *phydev)
185{
186 struct nxp_c45_phy *priv = phydev->priv;
187 u64 tx_delay = priv->tx_delay;
188 u64 rx_delay = priv->rx_delay;
189 u64 degree;
190
191 if (phydev->interface == PHY_INTERFACE_MODE_RGMII_ID ||
192 phydev->interface == PHY_INTERFACE_MODE_RGMII_TXID) {
193 degree = div_u64(tx_delay, PS_PER_DEGREE);
194 phy_write_mmd(phydev, MDIO_MMD_VEND1, VEND1_TXID,
195 ID_ENABLE | nxp_c45_get_phase_shift(degree));
196 } else {
197 phy_write_mmd(phydev, MDIO_MMD_VEND1, VEND1_TXID, 0);
198 }
199
200 if (phydev->interface == PHY_INTERFACE_MODE_RGMII_ID ||
201 phydev->interface == PHY_INTERFACE_MODE_RGMII_RXID) {
202 degree = div_u64(rx_delay, PS_PER_DEGREE);
203 phy_write_mmd(phydev, MDIO_MMD_VEND1, VEND1_RXID,
204 ID_ENABLE | nxp_c45_get_phase_shift(degree));
205 } else {
206 phy_write_mmd(phydev, MDIO_MMD_VEND1, VEND1_RXID, 0);
207 }
208}
209
210static int nxp_c45_set_phy_mode(struct phy_device *phydev)
211{
212 int ret;
213
214 ret = phy_read_mmd(phydev, MDIO_MMD_VEND1, VEND1_ABILITIES);
215 pr_debug("%s: Clause 45 managed PHY abilities 0x%x\n",
216 phydev->drv->name, ret);
217
218 switch (phydev->interface) {
219 case PHY_INTERFACE_MODE_RGMII:
220 if (!(ret & RGMII_ABILITY)) {
221 pr_err("%s: rgmii mode not supported\n",
222 phydev->drv->name);
223 return -EINVAL;
224 }
225 phy_write_mmd(phydev, MDIO_MMD_VEND1, VEND1_MII_BASIC_CONFIG,
226 MII_BASIC_CONFIG_RGMII);
227 nxp_c45_disable_delays(phydev);
228 break;
229 case PHY_INTERFACE_MODE_RGMII_ID:
230 case PHY_INTERFACE_MODE_RGMII_TXID:
231 case PHY_INTERFACE_MODE_RGMII_RXID:
232 if (!(ret & RGMII_ID_ABILITY)) {
233 pr_err("%s: rgmii-id, rgmii-txid, rgmii-rxid modes are not supported\n",
234 phydev->drv->name);
235 return -EINVAL;
236 }
237 phy_write_mmd(phydev, MDIO_MMD_VEND1, VEND1_MII_BASIC_CONFIG,
238 MII_BASIC_CONFIG_RGMII);
239 ret = nxp_c45_get_delays(phydev);
240 if (ret)
241 return ret;
242
243 nxp_c45_set_delays(phydev);
244 break;
245 case PHY_INTERFACE_MODE_MII:
246 if (!(ret & MII_ABILITY)) {
247 pr_err("%s: mii mode not supported\n",
248 phydev->drv->name);
249 return -EINVAL;
250 }
251 phy_write_mmd(phydev, MDIO_MMD_VEND1, VEND1_MII_BASIC_CONFIG,
252 MII_BASIC_CONFIG_MII);
253 break;
254 case PHY_INTERFACE_MODE_RMII:
255 if (!(ret & RMII_ABILITY)) {
256 pr_err("%s: rmii mode not supported\n",
257 phydev->drv->name);
258 return -EINVAL;
259 }
260 phy_write_mmd(phydev, MDIO_MMD_VEND1, VEND1_MII_BASIC_CONFIG,
261 MII_BASIC_CONFIG_RMII);
262 break;
263 case PHY_INTERFACE_MODE_SGMII:
264 if (!(ret & SGMII_ABILITY)) {
265 pr_err("%s: sgmii mode not supported\n",
266 phydev->drv->name);
267 return -EINVAL;
268 }
269 phy_write_mmd(phydev, MDIO_MMD_VEND1, VEND1_MII_BASIC_CONFIG,
270 MII_BASIC_CONFIG_SGMII);
271 break;
272 case PHY_INTERFACE_MODE_INTERNAL:
273 break;
274 default:
275 return -EINVAL;
276 }
277
278 return 0;
279}
280
281static int nxp_c45_config(struct phy_device *phydev)
282{
283 int ret;
284
285 ret = nxp_c45_soft_reset(phydev);
286 if (ret)
287 return ret;
288
289 ret = nxp_c45_config_enable(phydev);
290 if (ret) {
291 pr_err("%s: Failed to enable config\n", phydev->drv->name);
292 return ret;
293 }
294
295 phy_write_mmd(phydev, MDIO_MMD_VEND1, VEND1_PHY_CONFIG,
296 PHY_CONFIG_AUTO);
297
298 ret = nxp_c45_set_phy_mode(phydev);
299 if (ret) {
300 pr_err("%s: Failed to set phy mode\n", phydev->drv->name);
301 return ret;
302 }
303
304 phydev->autoneg = AUTONEG_DISABLE;
305
306 return nxp_c45_start_op(phydev);
307}
308
309static int nxp_c45_startup(struct phy_device *phydev)
310{
311 u32 reg;
312
313 reg = phy_read_mmd(phydev, MDIO_MMD_PMAPMD, MDIO_STAT1);
314 phydev->link = !!(reg & MDIO_STAT1_LSTATUS);
315 phydev->speed = SPEED_100;
316 phydev->duplex = DUPLEX_FULL;
317 return 0;
318}
319
320static int nxp_c45_probe(struct phy_device *phydev)
321{
322 struct nxp_c45_phy *priv;
323
324 priv = devm_kzalloc(phydev->priv, sizeof(*priv), GFP_KERNEL);
325 if (!priv)
326 return -ENOMEM;
327
328 phydev->priv = priv;
329
330 return 0;
331}
332
Marek Vasut49eacfd2023-03-19 18:02:59 +0100333U_BOOT_PHY_DRIVER(nxp_c45_tja11xx) = {
Radu Pirea (NXP OSS)f2d36cb2021-06-18 21:58:30 +0300334 .name = "NXP C45 TJA1103",
335 .uid = PHY_ID_TJA_1103,
336 .mask = 0xfffff0,
337 .features = PHY_100BT1_FEATURES,
338 .probe = &nxp_c45_probe,
339 .config = &nxp_c45_config,
340 .startup = &nxp_c45_startup,
341 .shutdown = &genphy_shutdown,
342};