xref: /linux/drivers/net/mdio/mdio-mscc-miim.c (revision 24bce201d79807b668bf9d9e0aca801c5c0d5f78)
1 // SPDX-License-Identifier: (GPL-2.0 OR MIT)
2 /*
3  * Driver for the MDIO interface of Microsemi network switches.
4  *
5  * Author: Alexandre Belloni <alexandre.belloni@bootlin.com>
6  * Copyright (c) 2017 Microsemi Corporation
7  */
8 
9 #include <linux/bitops.h>
10 #include <linux/clk.h>
11 #include <linux/io.h>
12 #include <linux/iopoll.h>
13 #include <linux/kernel.h>
14 #include <linux/mdio/mdio-mscc-miim.h>
15 #include <linux/module.h>
16 #include <linux/of_mdio.h>
17 #include <linux/phy.h>
18 #include <linux/platform_device.h>
19 #include <linux/property.h>
20 #include <linux/regmap.h>
21 
22 #define MSCC_MIIM_REG_STATUS		0x0
23 #define		MSCC_MIIM_STATUS_STAT_PENDING	BIT(2)
24 #define		MSCC_MIIM_STATUS_STAT_BUSY	BIT(3)
25 #define MSCC_MIIM_REG_CMD		0x8
26 #define		MSCC_MIIM_CMD_OPR_WRITE		BIT(1)
27 #define		MSCC_MIIM_CMD_OPR_READ		BIT(2)
28 #define		MSCC_MIIM_CMD_WRDATA_SHIFT	4
29 #define		MSCC_MIIM_CMD_REGAD_SHIFT	20
30 #define		MSCC_MIIM_CMD_PHYAD_SHIFT	25
31 #define		MSCC_MIIM_CMD_VLD		BIT(31)
32 #define MSCC_MIIM_REG_DATA		0xC
33 #define		MSCC_MIIM_DATA_ERROR		(BIT(16) | BIT(17))
34 #define MSCC_MIIM_REG_CFG		0x10
35 #define		MSCC_MIIM_CFG_PRESCALE_MASK	GENMASK(7, 0)
36 
37 #define MSCC_PHY_REG_PHY_CFG	0x0
38 #define		PHY_CFG_PHY_ENA		(BIT(0) | BIT(1) | BIT(2) | BIT(3))
39 #define		PHY_CFG_PHY_COMMON_RESET BIT(4)
40 #define		PHY_CFG_PHY_RESET	(BIT(5) | BIT(6) | BIT(7) | BIT(8))
41 #define MSCC_PHY_REG_PHY_STATUS	0x4
42 
43 #define LAN966X_CUPHY_COMMON_CFG	0x0
44 #define		CUPHY_COMMON_CFG_RESET_N	BIT(0)
45 
46 struct mscc_miim_info {
47 	unsigned int phy_reset_offset;
48 	unsigned int phy_reset_bits;
49 };
50 
51 struct mscc_miim_dev {
52 	struct regmap *regs;
53 	int mii_status_offset;
54 	struct regmap *phy_regs;
55 	const struct mscc_miim_info *info;
56 	struct clk *clk;
57 	u32 bus_freq;
58 };
59 
60 /* When high resolution timers aren't built-in: we can't use usleep_range() as
61  * we would sleep way too long. Use udelay() instead.
62  */
63 #define mscc_readx_poll_timeout(op, addr, val, cond, delay_us, timeout_us)\
64 ({									  \
65 	if (!IS_ENABLED(CONFIG_HIGH_RES_TIMERS))			  \
66 		readx_poll_timeout_atomic(op, addr, val, cond, delay_us,  \
67 					  timeout_us);			  \
68 	readx_poll_timeout(op, addr, val, cond, delay_us, timeout_us);	  \
69 })
70 
71 static int mscc_miim_status(struct mii_bus *bus)
72 {
73 	struct mscc_miim_dev *miim = bus->priv;
74 	int val, ret;
75 
76 	ret = regmap_read(miim->regs,
77 			  MSCC_MIIM_REG_STATUS + miim->mii_status_offset, &val);
78 	if (ret < 0) {
79 		WARN_ONCE(1, "mscc miim status read error %d\n", ret);
80 		return ret;
81 	}
82 
83 	return val;
84 }
85 
86 static int mscc_miim_wait_ready(struct mii_bus *bus)
87 {
88 	u32 val;
89 
90 	return mscc_readx_poll_timeout(mscc_miim_status, bus, val,
91 				       !(val & MSCC_MIIM_STATUS_STAT_BUSY), 50,
92 				       10000);
93 }
94 
95 static int mscc_miim_wait_pending(struct mii_bus *bus)
96 {
97 	u32 val;
98 
99 	return mscc_readx_poll_timeout(mscc_miim_status, bus, val,
100 				       !(val & MSCC_MIIM_STATUS_STAT_PENDING),
101 				       50, 10000);
102 }
103 
104 static int mscc_miim_read(struct mii_bus *bus, int mii_id, int regnum)
105 {
106 	struct mscc_miim_dev *miim = bus->priv;
107 	u32 val;
108 	int ret;
109 
110 	if (regnum & MII_ADDR_C45)
111 		return -EOPNOTSUPP;
112 
113 	ret = mscc_miim_wait_pending(bus);
114 	if (ret)
115 		goto out;
116 
117 	ret = regmap_write(miim->regs,
118 			   MSCC_MIIM_REG_CMD + miim->mii_status_offset,
119 			   MSCC_MIIM_CMD_VLD |
120 			   (mii_id << MSCC_MIIM_CMD_PHYAD_SHIFT) |
121 			   (regnum << MSCC_MIIM_CMD_REGAD_SHIFT) |
122 			   MSCC_MIIM_CMD_OPR_READ);
123 
124 	if (ret < 0) {
125 		WARN_ONCE(1, "mscc miim write cmd reg error %d\n", ret);
126 		goto out;
127 	}
128 
129 	ret = mscc_miim_wait_ready(bus);
130 	if (ret)
131 		goto out;
132 
133 	ret = regmap_read(miim->regs,
134 			  MSCC_MIIM_REG_DATA + miim->mii_status_offset, &val);
135 	if (ret < 0) {
136 		WARN_ONCE(1, "mscc miim read data reg error %d\n", ret);
137 		goto out;
138 	}
139 
140 	if (val & MSCC_MIIM_DATA_ERROR) {
141 		ret = -EIO;
142 		goto out;
143 	}
144 
145 	ret = val & 0xFFFF;
146 out:
147 	return ret;
148 }
149 
150 static int mscc_miim_write(struct mii_bus *bus, int mii_id,
151 			   int regnum, u16 value)
152 {
153 	struct mscc_miim_dev *miim = bus->priv;
154 	int ret;
155 
156 	if (regnum & MII_ADDR_C45)
157 		return -EOPNOTSUPP;
158 
159 	ret = mscc_miim_wait_pending(bus);
160 	if (ret < 0)
161 		goto out;
162 
163 	ret = regmap_write(miim->regs,
164 			   MSCC_MIIM_REG_CMD + miim->mii_status_offset,
165 			   MSCC_MIIM_CMD_VLD |
166 			   (mii_id << MSCC_MIIM_CMD_PHYAD_SHIFT) |
167 			   (regnum << MSCC_MIIM_CMD_REGAD_SHIFT) |
168 			   (value << MSCC_MIIM_CMD_WRDATA_SHIFT) |
169 			   MSCC_MIIM_CMD_OPR_WRITE);
170 
171 	if (ret < 0)
172 		WARN_ONCE(1, "mscc miim write error %d\n", ret);
173 out:
174 	return ret;
175 }
176 
177 static int mscc_miim_reset(struct mii_bus *bus)
178 {
179 	struct mscc_miim_dev *miim = bus->priv;
180 	unsigned int offset, bits;
181 	int ret;
182 
183 	if (!miim->phy_regs)
184 		return 0;
185 
186 	offset = miim->info->phy_reset_offset;
187 	bits = miim->info->phy_reset_bits;
188 
189 	ret = regmap_update_bits(miim->phy_regs, offset, bits, 0);
190 	if (ret < 0) {
191 		WARN_ONCE(1, "mscc reset set error %d\n", ret);
192 		return ret;
193 	}
194 
195 	ret = regmap_update_bits(miim->phy_regs, offset, bits, bits);
196 	if (ret < 0) {
197 		WARN_ONCE(1, "mscc reset clear error %d\n", ret);
198 		return ret;
199 	}
200 
201 	mdelay(500);
202 
203 	return 0;
204 }
205 
206 static const struct regmap_config mscc_miim_regmap_config = {
207 	.reg_bits	= 32,
208 	.val_bits	= 32,
209 	.reg_stride	= 4,
210 };
211 
212 static const struct regmap_config mscc_miim_phy_regmap_config = {
213 	.reg_bits	= 32,
214 	.val_bits	= 32,
215 	.reg_stride	= 4,
216 	.name		= "phy",
217 };
218 
219 int mscc_miim_setup(struct device *dev, struct mii_bus **pbus, const char *name,
220 		    struct regmap *mii_regmap, int status_offset)
221 {
222 	struct mscc_miim_dev *miim;
223 	struct mii_bus *bus;
224 
225 	bus = devm_mdiobus_alloc_size(dev, sizeof(*miim));
226 	if (!bus)
227 		return -ENOMEM;
228 
229 	bus->name = name;
230 	bus->read = mscc_miim_read;
231 	bus->write = mscc_miim_write;
232 	bus->reset = mscc_miim_reset;
233 	snprintf(bus->id, MII_BUS_ID_SIZE, "%s-mii", dev_name(dev));
234 	bus->parent = dev;
235 
236 	miim = bus->priv;
237 
238 	*pbus = bus;
239 
240 	miim->regs = mii_regmap;
241 	miim->mii_status_offset = status_offset;
242 
243 	*pbus = bus;
244 
245 	return 0;
246 }
247 EXPORT_SYMBOL(mscc_miim_setup);
248 
249 static int mscc_miim_clk_set(struct mii_bus *bus)
250 {
251 	struct mscc_miim_dev *miim = bus->priv;
252 	unsigned long rate;
253 	u32 div;
254 
255 	/* Keep the current settings */
256 	if (!miim->bus_freq)
257 		return 0;
258 
259 	rate = clk_get_rate(miim->clk);
260 
261 	div = DIV_ROUND_UP(rate, 2 * miim->bus_freq) - 1;
262 	if (div == 0 || div & ~MSCC_MIIM_CFG_PRESCALE_MASK) {
263 		dev_err(&bus->dev, "Incorrect MDIO clock frequency\n");
264 		return -EINVAL;
265 	}
266 
267 	return regmap_update_bits(miim->regs, MSCC_MIIM_REG_CFG,
268 				  MSCC_MIIM_CFG_PRESCALE_MASK, div);
269 }
270 
271 static int mscc_miim_probe(struct platform_device *pdev)
272 {
273 	struct regmap *mii_regmap, *phy_regmap = NULL;
274 	struct device_node *np = pdev->dev.of_node;
275 	struct device *dev = &pdev->dev;
276 	void __iomem *regs, *phy_regs;
277 	struct mscc_miim_dev *miim;
278 	struct resource *res;
279 	struct mii_bus *bus;
280 	int ret;
281 
282 	regs = devm_platform_get_and_ioremap_resource(pdev, 0, NULL);
283 	if (IS_ERR(regs)) {
284 		dev_err(dev, "Unable to map MIIM registers\n");
285 		return PTR_ERR(regs);
286 	}
287 
288 	mii_regmap = devm_regmap_init_mmio(dev, regs, &mscc_miim_regmap_config);
289 
290 	if (IS_ERR(mii_regmap)) {
291 		dev_err(dev, "Unable to create MIIM regmap\n");
292 		return PTR_ERR(mii_regmap);
293 	}
294 
295 	/* This resource is optional */
296 	res = platform_get_resource(pdev, IORESOURCE_MEM, 1);
297 	if (res) {
298 		phy_regs = devm_ioremap_resource(dev, res);
299 		if (IS_ERR(phy_regs)) {
300 			dev_err(dev, "Unable to map internal phy registers\n");
301 			return PTR_ERR(phy_regs);
302 		}
303 
304 		phy_regmap = devm_regmap_init_mmio(dev, phy_regs,
305 						   &mscc_miim_phy_regmap_config);
306 		if (IS_ERR(phy_regmap)) {
307 			dev_err(dev, "Unable to create phy register regmap\n");
308 			return PTR_ERR(phy_regmap);
309 		}
310 	}
311 
312 	ret = mscc_miim_setup(dev, &bus, "mscc_miim", mii_regmap, 0);
313 	if (ret < 0) {
314 		dev_err(dev, "Unable to setup the MDIO bus\n");
315 		return ret;
316 	}
317 
318 	miim = bus->priv;
319 	miim->phy_regs = phy_regmap;
320 
321 	miim->info = device_get_match_data(dev);
322 	if (!miim->info)
323 		return -EINVAL;
324 
325 	miim->clk = devm_clk_get_optional(dev, NULL);
326 	if (IS_ERR(miim->clk))
327 		return PTR_ERR(miim->clk);
328 
329 	of_property_read_u32(np, "clock-frequency", &miim->bus_freq);
330 
331 	if (miim->bus_freq && !miim->clk) {
332 		dev_err(dev, "cannot use clock-frequency without a clock\n");
333 		return -EINVAL;
334 	}
335 
336 	ret = clk_prepare_enable(miim->clk);
337 	if (ret)
338 		return ret;
339 
340 	ret = mscc_miim_clk_set(bus);
341 	if (ret)
342 		goto out_disable_clk;
343 
344 	ret = of_mdiobus_register(bus, np);
345 	if (ret < 0) {
346 		dev_err(dev, "Cannot register MDIO bus (%d)\n", ret);
347 		goto out_disable_clk;
348 	}
349 
350 	platform_set_drvdata(pdev, bus);
351 
352 	return 0;
353 
354 out_disable_clk:
355 	clk_disable_unprepare(miim->clk);
356 	return ret;
357 }
358 
359 static int mscc_miim_remove(struct platform_device *pdev)
360 {
361 	struct mii_bus *bus = platform_get_drvdata(pdev);
362 	struct mscc_miim_dev *miim = bus->priv;
363 
364 	clk_disable_unprepare(miim->clk);
365 	mdiobus_unregister(bus);
366 
367 	return 0;
368 }
369 
370 static const struct mscc_miim_info mscc_ocelot_miim_info = {
371 	.phy_reset_offset = MSCC_PHY_REG_PHY_CFG,
372 	.phy_reset_bits = PHY_CFG_PHY_ENA | PHY_CFG_PHY_COMMON_RESET |
373 			  PHY_CFG_PHY_RESET,
374 };
375 
376 static const struct mscc_miim_info microchip_lan966x_miim_info = {
377 	.phy_reset_offset = LAN966X_CUPHY_COMMON_CFG,
378 	.phy_reset_bits = CUPHY_COMMON_CFG_RESET_N,
379 };
380 
381 static const struct of_device_id mscc_miim_match[] = {
382 	{
383 		.compatible = "mscc,ocelot-miim",
384 		.data = &mscc_ocelot_miim_info
385 	}, {
386 		.compatible = "microchip,lan966x-miim",
387 		.data = &microchip_lan966x_miim_info
388 	},
389 	{ }
390 };
391 MODULE_DEVICE_TABLE(of, mscc_miim_match);
392 
393 static struct platform_driver mscc_miim_driver = {
394 	.probe = mscc_miim_probe,
395 	.remove = mscc_miim_remove,
396 	.driver = {
397 		.name = "mscc-miim",
398 		.of_match_table = mscc_miim_match,
399 	},
400 };
401 
402 module_platform_driver(mscc_miim_driver);
403 
404 MODULE_DESCRIPTION("Microsemi MIIM driver");
405 MODULE_AUTHOR("Alexandre Belloni <alexandre.belloni@bootlin.com>");
406 MODULE_LICENSE("Dual MIT/GPL");
407