mdio-mscc-miim.c 9.3 KB

123456789101112131415161718192021222324252627282930313233343536373839404142434445464748495051525354555657585960616263646566676869707172737475767778798081828384858687888990919293949596979899100101102103104105106107108109110111112113114115116117118119120121122123124125126127128129130131132133134135136137138139140141142143144145146147148149150151152153154155156157158159160161162163164165166167168169170171172173174175176177178179180181182183184185186187188189190191192193194195196197198199200201202203204205206207208209210211212213214215216217218219220221222223224225226227228229230231232233234235236237238239240241242243244245246247248249250251252253254255256257258259260261262263264265266267268269270271272273274275276277278279280281282283284285286287288289290291292293294295296297298299300301302303304305306307308309310311312313314315316317318319320321322323324325326327328329330331332333334335336337338339340341342343344345346347348349350351352353354355356357358359360361362363364365366367368369370371372373374375376377378379380381382383384385386387388389390391
  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. #include <linux/bitops.h>
  9. #include <linux/clk.h>
  10. #include <linux/io.h>
  11. #include <linux/iopoll.h>
  12. #include <linux/kernel.h>
  13. #include <linux/mdio/mdio-mscc-miim.h>
  14. #include <linux/mfd/ocelot.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. #include <linux/reset.h>
  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. #define MSCC_PHY_REG_PHY_CFG 0x0
  37. #define PHY_CFG_PHY_ENA (BIT(0) | BIT(1) | BIT(2) | BIT(3))
  38. #define PHY_CFG_PHY_COMMON_RESET BIT(4)
  39. #define PHY_CFG_PHY_RESET (BIT(5) | BIT(6) | BIT(7) | BIT(8))
  40. #define MSCC_PHY_REG_PHY_STATUS 0x4
  41. #define LAN966X_CUPHY_COMMON_CFG 0x0
  42. #define CUPHY_COMMON_CFG_RESET_N BIT(0)
  43. struct mscc_miim_info {
  44. unsigned int phy_reset_offset;
  45. unsigned int phy_reset_bits;
  46. };
  47. struct mscc_miim_dev {
  48. struct regmap *regs;
  49. int mii_status_offset;
  50. bool ignore_read_errors;
  51. struct regmap *phy_regs;
  52. const struct mscc_miim_info *info;
  53. struct clk *clk;
  54. u32 bus_freq;
  55. };
  56. /* When high resolution timers aren't built-in: we can't use usleep_range() as
  57. * we would sleep way too long. Use udelay() instead.
  58. */
  59. #define mscc_readx_poll_timeout(op, addr, val, cond, delay_us, timeout_us)\
  60. ({ \
  61. if (!IS_ENABLED(CONFIG_HIGH_RES_TIMERS)) \
  62. readx_poll_timeout_atomic(op, addr, val, cond, delay_us, \
  63. timeout_us); \
  64. readx_poll_timeout(op, addr, val, cond, delay_us, timeout_us); \
  65. })
  66. static int mscc_miim_status(struct mii_bus *bus)
  67. {
  68. struct mscc_miim_dev *miim = bus->priv;
  69. int val, ret;
  70. ret = regmap_read(miim->regs,
  71. MSCC_MIIM_REG_STATUS + miim->mii_status_offset, &val);
  72. if (ret < 0) {
  73. WARN_ONCE(1, "mscc miim status read error %d\n", ret);
  74. return ret;
  75. }
  76. return val;
  77. }
  78. static int mscc_miim_wait_ready(struct mii_bus *bus)
  79. {
  80. u32 val;
  81. return mscc_readx_poll_timeout(mscc_miim_status, bus, val,
  82. !(val & MSCC_MIIM_STATUS_STAT_BUSY), 50,
  83. 10000);
  84. }
  85. static int mscc_miim_wait_pending(struct mii_bus *bus)
  86. {
  87. u32 val;
  88. return mscc_readx_poll_timeout(mscc_miim_status, bus, val,
  89. !(val & MSCC_MIIM_STATUS_STAT_PENDING),
  90. 50, 10000);
  91. }
  92. static int mscc_miim_read(struct mii_bus *bus, int mii_id, int regnum)
  93. {
  94. struct mscc_miim_dev *miim = bus->priv;
  95. u32 val;
  96. int ret;
  97. ret = mscc_miim_wait_pending(bus);
  98. if (ret)
  99. goto out;
  100. ret = regmap_write(miim->regs,
  101. MSCC_MIIM_REG_CMD + miim->mii_status_offset,
  102. MSCC_MIIM_CMD_VLD |
  103. (mii_id << MSCC_MIIM_CMD_PHYAD_SHIFT) |
  104. (regnum << MSCC_MIIM_CMD_REGAD_SHIFT) |
  105. MSCC_MIIM_CMD_OPR_READ);
  106. if (ret < 0) {
  107. WARN_ONCE(1, "mscc miim write cmd reg error %d\n", ret);
  108. goto out;
  109. }
  110. ret = mscc_miim_wait_ready(bus);
  111. if (ret)
  112. goto out;
  113. ret = regmap_read(miim->regs,
  114. MSCC_MIIM_REG_DATA + miim->mii_status_offset, &val);
  115. if (ret < 0) {
  116. WARN_ONCE(1, "mscc miim read data reg error %d\n", ret);
  117. goto out;
  118. }
  119. if (!miim->ignore_read_errors && !!(val & MSCC_MIIM_DATA_ERROR)) {
  120. ret = -EIO;
  121. goto out;
  122. }
  123. ret = val & 0xFFFF;
  124. out:
  125. return ret;
  126. }
  127. static int mscc_miim_write(struct mii_bus *bus, int mii_id,
  128. int regnum, u16 value)
  129. {
  130. struct mscc_miim_dev *miim = bus->priv;
  131. int ret;
  132. ret = mscc_miim_wait_pending(bus);
  133. if (ret < 0)
  134. goto out;
  135. ret = regmap_write(miim->regs,
  136. MSCC_MIIM_REG_CMD + miim->mii_status_offset,
  137. MSCC_MIIM_CMD_VLD |
  138. (mii_id << MSCC_MIIM_CMD_PHYAD_SHIFT) |
  139. (regnum << MSCC_MIIM_CMD_REGAD_SHIFT) |
  140. (value << MSCC_MIIM_CMD_WRDATA_SHIFT) |
  141. MSCC_MIIM_CMD_OPR_WRITE);
  142. if (ret < 0)
  143. WARN_ONCE(1, "mscc miim write error %d\n", ret);
  144. out:
  145. return ret;
  146. }
  147. static int mscc_miim_reset(struct mii_bus *bus)
  148. {
  149. struct mscc_miim_dev *miim = bus->priv;
  150. unsigned int offset, bits;
  151. int ret;
  152. if (!miim->phy_regs)
  153. return 0;
  154. offset = miim->info->phy_reset_offset;
  155. bits = miim->info->phy_reset_bits;
  156. ret = regmap_update_bits(miim->phy_regs, offset, bits, 0);
  157. if (ret < 0) {
  158. WARN_ONCE(1, "mscc reset set error %d\n", ret);
  159. return ret;
  160. }
  161. ret = regmap_update_bits(miim->phy_regs, offset, bits, bits);
  162. if (ret < 0) {
  163. WARN_ONCE(1, "mscc reset clear error %d\n", ret);
  164. return ret;
  165. }
  166. mdelay(500);
  167. return 0;
  168. }
  169. static const struct regmap_config mscc_miim_regmap_config = {
  170. .reg_bits = 32,
  171. .val_bits = 32,
  172. .reg_stride = 4,
  173. };
  174. static const struct regmap_config mscc_miim_phy_regmap_config = {
  175. .reg_bits = 32,
  176. .val_bits = 32,
  177. .reg_stride = 4,
  178. .name = "phy",
  179. };
  180. int mscc_miim_setup(struct device *dev, struct mii_bus **pbus, const char *name,
  181. struct regmap *mii_regmap, int status_offset,
  182. bool ignore_read_errors)
  183. {
  184. struct mscc_miim_dev *miim;
  185. struct mii_bus *bus;
  186. bus = devm_mdiobus_alloc_size(dev, sizeof(*miim));
  187. if (!bus)
  188. return -ENOMEM;
  189. bus->name = name;
  190. bus->read = mscc_miim_read;
  191. bus->write = mscc_miim_write;
  192. bus->reset = mscc_miim_reset;
  193. snprintf(bus->id, MII_BUS_ID_SIZE, "%s-mii", dev_name(dev));
  194. bus->parent = dev;
  195. miim = bus->priv;
  196. *pbus = bus;
  197. miim->regs = mii_regmap;
  198. miim->mii_status_offset = status_offset;
  199. miim->ignore_read_errors = ignore_read_errors;
  200. *pbus = bus;
  201. return 0;
  202. }
  203. EXPORT_SYMBOL(mscc_miim_setup);
  204. static int mscc_miim_clk_set(struct mii_bus *bus)
  205. {
  206. struct mscc_miim_dev *miim = bus->priv;
  207. unsigned long rate;
  208. u32 div;
  209. /* Keep the current settings */
  210. if (!miim->bus_freq)
  211. return 0;
  212. rate = clk_get_rate(miim->clk);
  213. div = DIV_ROUND_UP(rate, 2 * miim->bus_freq) - 1;
  214. if (div == 0 || div & ~MSCC_MIIM_CFG_PRESCALE_MASK) {
  215. dev_err(&bus->dev, "Incorrect MDIO clock frequency\n");
  216. return -EINVAL;
  217. }
  218. return regmap_update_bits(miim->regs, MSCC_MIIM_REG_CFG,
  219. MSCC_MIIM_CFG_PRESCALE_MASK, div);
  220. }
  221. static int mscc_miim_probe(struct platform_device *pdev)
  222. {
  223. struct device_node *np = pdev->dev.of_node;
  224. struct regmap *mii_regmap, *phy_regmap;
  225. struct device *dev = &pdev->dev;
  226. struct reset_control *reset;
  227. struct mscc_miim_dev *miim;
  228. struct mii_bus *bus;
  229. int ret;
  230. reset = devm_reset_control_get_optional_shared(dev, "switch");
  231. if (IS_ERR(reset))
  232. return dev_err_probe(dev, PTR_ERR(reset), "Failed to get reset\n");
  233. reset_control_reset(reset);
  234. mii_regmap = ocelot_regmap_from_resource(pdev, 0,
  235. &mscc_miim_regmap_config);
  236. if (IS_ERR(mii_regmap))
  237. return dev_err_probe(dev, PTR_ERR(mii_regmap),
  238. "Unable to create MIIM regmap\n");
  239. /* This resource is optional */
  240. phy_regmap = ocelot_regmap_from_resource_optional(pdev, 1,
  241. &mscc_miim_phy_regmap_config);
  242. if (IS_ERR(phy_regmap))
  243. return dev_err_probe(dev, PTR_ERR(phy_regmap),
  244. "Unable to create phy register regmap\n");
  245. ret = mscc_miim_setup(dev, &bus, "mscc_miim", mii_regmap, 0, false);
  246. if (ret < 0) {
  247. dev_err(dev, "Unable to setup the MDIO bus\n");
  248. return ret;
  249. }
  250. miim = bus->priv;
  251. miim->phy_regs = phy_regmap;
  252. miim->info = device_get_match_data(dev);
  253. if (!miim->info)
  254. return -EINVAL;
  255. miim->clk = devm_clk_get_optional(dev, NULL);
  256. if (IS_ERR(miim->clk))
  257. return PTR_ERR(miim->clk);
  258. of_property_read_u32(np, "clock-frequency", &miim->bus_freq);
  259. if (miim->bus_freq && !miim->clk) {
  260. dev_err(dev, "cannot use clock-frequency without a clock\n");
  261. return -EINVAL;
  262. }
  263. ret = clk_prepare_enable(miim->clk);
  264. if (ret)
  265. return ret;
  266. ret = mscc_miim_clk_set(bus);
  267. if (ret)
  268. goto out_disable_clk;
  269. ret = of_mdiobus_register(bus, np);
  270. if (ret < 0) {
  271. dev_err(dev, "Cannot register MDIO bus (%d)\n", ret);
  272. goto out_disable_clk;
  273. }
  274. platform_set_drvdata(pdev, bus);
  275. return 0;
  276. out_disable_clk:
  277. clk_disable_unprepare(miim->clk);
  278. return ret;
  279. }
  280. static void mscc_miim_remove(struct platform_device *pdev)
  281. {
  282. struct mii_bus *bus = platform_get_drvdata(pdev);
  283. struct mscc_miim_dev *miim = bus->priv;
  284. clk_disable_unprepare(miim->clk);
  285. mdiobus_unregister(bus);
  286. }
  287. static const struct mscc_miim_info mscc_ocelot_miim_info = {
  288. .phy_reset_offset = MSCC_PHY_REG_PHY_CFG,
  289. .phy_reset_bits = PHY_CFG_PHY_ENA | PHY_CFG_PHY_COMMON_RESET |
  290. PHY_CFG_PHY_RESET,
  291. };
  292. static const struct mscc_miim_info microchip_lan966x_miim_info = {
  293. .phy_reset_offset = LAN966X_CUPHY_COMMON_CFG,
  294. .phy_reset_bits = CUPHY_COMMON_CFG_RESET_N,
  295. };
  296. static const struct of_device_id mscc_miim_match[] = {
  297. {
  298. .compatible = "mscc,ocelot-miim",
  299. .data = &mscc_ocelot_miim_info
  300. }, {
  301. .compatible = "microchip,lan966x-miim",
  302. .data = &microchip_lan966x_miim_info
  303. },
  304. { }
  305. };
  306. MODULE_DEVICE_TABLE(of, mscc_miim_match);
  307. static struct platform_driver mscc_miim_driver = {
  308. .probe = mscc_miim_probe,
  309. .remove = mscc_miim_remove,
  310. .driver = {
  311. .name = "mscc-miim",
  312. .of_match_table = mscc_miim_match,
  313. },
  314. };
  315. module_platform_driver(mscc_miim_driver);
  316. MODULE_DESCRIPTION("Microsemi MIIM driver");
  317. MODULE_AUTHOR("Alexandre Belloni <alexandre.belloni@bootlin.com>");
  318. MODULE_LICENSE("Dual MIT/GPL");