1 /*
2 * Copyright (c) 2023 PHOENIX CONTACT Electronics GmbH
3 *
4 * SPDX-License-Identifier: Apache-2.0
5 */
6
7 #include <zephyr/logging/log.h>
8 LOG_MODULE_REGISTER(mdio_adin2111, CONFIG_MDIO_LOG_LEVEL);
9
10 #define DT_DRV_COMPAT adi_adin2111_mdio
11
12 #include <stdint.h>
13 #include <errno.h>
14 #include <zephyr/device.h>
15 #include <zephyr/kernel.h>
16 #include <zephyr/drivers/mdio.h>
17 #include <zephyr/drivers/ethernet/eth_adin2111.h>
18
19 /* MDIO ready check retry delay */
20 #define ADIN2111_MDIO_READY_AWAIT_DELAY_POLL_US 5U
21 /* Number of retries for MDIO ready check */
22 #define ADIN2111_MDIO_READY_AWAIT_RETRY_COUNT 10U
23
24 /* MDIO Access Register 1 */
25 #define ADIN2111_MDIOACC0 0x20U
26 /* MDIO Access Register 2 */
27 #define ADIN2111_MDIOACC1 0x21U
28
29 /* MDIO MDIOACC Transaction Done */
30 #define ADIN211_MDIOACC_MDIO_TRDONE BIT(31)
31
32 struct mdio_adin2111_config {
33 const struct device *adin;
34 };
35
mdio_adin2111_wait_ready(const struct device * dev,uint16_t reg,uint32_t * out)36 static int mdio_adin2111_wait_ready(const struct device *dev, uint16_t reg,
37 uint32_t *out)
38 {
39 const struct mdio_adin2111_config *const cfg = dev->config;
40 uint32_t count;
41 int ret;
42
43 for (count = 0U; count < ADIN2111_MDIO_READY_AWAIT_RETRY_COUNT; ++count) {
44 ret = eth_adin2111_reg_read(cfg->adin, reg, out);
45 if (ret >= 0) {
46 if ((*out) & ADIN211_MDIOACC_MDIO_TRDONE) {
47 break;
48 }
49 ret = -ETIMEDOUT;
50 }
51 k_sleep(K_USEC(ADIN2111_MDIO_READY_AWAIT_DELAY_POLL_US));
52 }
53
54 return ret;
55 }
56
57
mdio_adin2111_read_c45(const struct device * dev,uint8_t prtad,uint8_t devad,uint16_t regad,uint16_t * data)58 static int mdio_adin2111_read_c45(const struct device *dev, uint8_t prtad,
59 uint8_t devad, uint16_t regad,
60 uint16_t *data)
61 {
62 const struct mdio_adin2111_config *const cfg = dev->config;
63 uint32_t rdy;
64 uint32_t cmd;
65 int ret;
66
67 /* address op */
68 cmd = (prtad & 0x1FU) << 21;
69 cmd |= (devad & 0x1FU) << 16;
70 cmd |= regad;
71
72 ret = eth_adin2111_reg_write(cfg->adin, ADIN2111_MDIOACC0, cmd);
73 if (ret < 0) {
74 return ret;
75 }
76
77 /* read op */
78 cmd = (cmd & ~UINT16_MAX) | (0x3U << 26);
79
80 ret = eth_adin2111_reg_write(cfg->adin, ADIN2111_MDIOACC1, cmd);
81 if (ret < 0) {
82 return ret;
83 }
84
85 ret = mdio_adin2111_wait_ready(dev, ADIN2111_MDIOACC1, &rdy);
86 if (ret < 0) {
87 return ret;
88 }
89
90 /* read out */
91 ret = eth_adin2111_reg_read(cfg->adin, ADIN2111_MDIOACC1, &cmd);
92
93 *data = cmd & UINT16_MAX;
94
95 return ret;
96 }
97
mdio_adin2111_write_c45(const struct device * dev,uint8_t prtad,uint8_t devad,uint16_t regad,uint16_t data)98 static int mdio_adin2111_write_c45(const struct device *dev, uint8_t prtad,
99 uint8_t devad, uint16_t regad,
100 uint16_t data)
101 {
102 const struct mdio_adin2111_config *const cfg = dev->config;
103
104 uint32_t rdy;
105 uint32_t cmd;
106 int ret;
107
108 /* address op */
109 cmd = (prtad & 0x1FU) << 21;
110 cmd |= (devad & 0x1FU) << 16;
111 cmd |= regad;
112
113 ret = eth_adin2111_reg_write(cfg->adin, ADIN2111_MDIOACC0, cmd);
114 if (ret < 0) {
115 return ret;
116 }
117
118 /* write op */
119 cmd |= BIT(26);
120 cmd = (cmd & ~UINT16_MAX) | data;
121
122 ret = eth_adin2111_reg_write(cfg->adin, ADIN2111_MDIOACC1, cmd);
123 if (ret < 0) {
124 return ret;
125 }
126
127 ret = mdio_adin2111_wait_ready(dev, ADIN2111_MDIOACC1, &rdy);
128
129 return ret;
130 }
131
mdio_adin2111_read(const struct device * dev,uint8_t prtad,uint8_t regad,uint16_t * data)132 static int mdio_adin2111_read(const struct device *dev, uint8_t prtad,
133 uint8_t regad, uint16_t *data)
134 {
135 const struct mdio_adin2111_config *const cfg = dev->config;
136 uint32_t read;
137 uint32_t cmd;
138 int ret;
139
140 cmd = BIT(28);
141 cmd |= 0x3U << 26;
142 cmd |= (prtad & 0x1FU) << 21;
143 cmd |= (regad & 0x1FU) << 16;
144
145 ret = eth_adin2111_reg_write(cfg->adin, ADIN2111_MDIOACC0, cmd);
146 if (ret >= 0) {
147 ret = mdio_adin2111_wait_ready(dev, ADIN2111_MDIOACC0, &read);
148 *data = read & UINT16_MAX;
149 }
150
151 return ret;
152 }
153
mdio_adin2111_write(const struct device * dev,uint8_t prtad,uint8_t regad,uint16_t data)154 static int mdio_adin2111_write(const struct device *dev, uint8_t prtad,
155 uint8_t regad, uint16_t data)
156 {
157 const struct mdio_adin2111_config *const cfg = dev->config;
158 uint32_t cmd;
159 uint32_t rdy;
160 int ret;
161
162 cmd = BIT(28);
163 cmd |= BIT(26);
164 cmd |= (prtad & 0x1FU) << 21;
165 cmd |= (regad & 0x1FU) << 16;
166 cmd |= data;
167
168 ret = eth_adin2111_reg_write(cfg->adin, ADIN2111_MDIOACC0, cmd);
169 if (ret >= 0) {
170 ret = mdio_adin2111_wait_ready(dev, ADIN2111_MDIOACC0, &rdy);
171 }
172
173 return ret;
174 }
175
mdio_adin2111_bus_enable(const struct device * dev)176 static void mdio_adin2111_bus_enable(const struct device *dev)
177 {
178 const struct mdio_adin2111_config *const cfg = dev->config;
179
180 eth_adin2111_lock(cfg->adin, K_FOREVER);
181 }
182
mdio_adin2111_bus_disable(const struct device * dev)183 static void mdio_adin2111_bus_disable(const struct device *dev)
184 {
185 const struct mdio_adin2111_config *const cfg = dev->config;
186
187 eth_adin2111_unlock(cfg->adin);
188 }
189
190 static DEVICE_API(mdio, mdio_adin2111_api) = {
191 .read = mdio_adin2111_read,
192 .write = mdio_adin2111_write,
193 .read_c45 = mdio_adin2111_read_c45,
194 .write_c45 = mdio_adin2111_write_c45,
195 .bus_enable = mdio_adin2111_bus_enable,
196 .bus_disable = mdio_adin2111_bus_disable
197 };
198
199 #define ADIN2111_MDIO_INIT(n) \
200 static const struct mdio_adin2111_config mdio_adin2111_config_##n = { \
201 .adin = DEVICE_DT_GET(DT_INST_BUS(n)), \
202 }; \
203 DEVICE_DT_INST_DEFINE(n, NULL, NULL, \
204 NULL, &mdio_adin2111_config_##n, \
205 POST_KERNEL, CONFIG_MDIO_INIT_PRIORITY, \
206 &mdio_adin2111_api);
207
208 DT_INST_FOREACH_STATUS_OKAY(ADIN2111_MDIO_INIT)
209