dwmac-ingenic.c 11 KB

123456789101112131415161718192021222324252627282930313233343536373839404142434445464748495051525354555657585960616263646566676869707172737475767778798081828384858687888990919293949596979899100101102103104105106107108109110111112113114115116117118119120121122123124125126127128129130131132133134135136137138139140141142143144145146147148149150151152153154155156157158159160161162163164165166167168169170171172173174175176177178179180181182183184185186187188189190191192193194195196197198199200201202203204205206207208209210211212213214215216217218219220221222223224225226227228229230231232233234235236237238239240241242243244245246247248249250251252253254255256257258259260261262263264265266267268269270271272273274275276277278279280281282283284285286287288289290291292293294295296297298299300301302303304305306307308309310311312313314315316317318319320321322323324325326327328329330331332333334335336337338339340341342343344345346347348349350351352353354355356357358359360361362363364365366367368369370371372373374375376377378379380381382383384385386387388389390391392393394395396397398399400
  1. // SPDX-License-Identifier: GPL-2.0
  2. /*
  3. * dwmac-ingenic.c - Ingenic SoCs DWMAC specific glue layer
  4. *
  5. * Copyright (c) 2021 周琰杰 (Zhou Yanjie) <[email protected]>
  6. */
  7. #include <linux/bitfield.h>
  8. #include <linux/clk.h>
  9. #include <linux/kernel.h>
  10. #include <linux/mfd/syscon.h>
  11. #include <linux/module.h>
  12. #include <linux/of.h>
  13. #include <linux/of_device.h>
  14. #include <linux/of_net.h>
  15. #include <linux/phy.h>
  16. #include <linux/platform_device.h>
  17. #include <linux/regmap.h>
  18. #include <linux/slab.h>
  19. #include <linux/stmmac.h>
  20. #include "stmmac_platform.h"
  21. #define MACPHYC_TXCLK_SEL_MASK GENMASK(31, 31)
  22. #define MACPHYC_TXCLK_SEL_OUTPUT 0x1
  23. #define MACPHYC_TXCLK_SEL_INPUT 0x0
  24. #define MACPHYC_MODE_SEL_MASK GENMASK(31, 31)
  25. #define MACPHYC_MODE_SEL_RMII 0x0
  26. #define MACPHYC_TX_SEL_MASK GENMASK(19, 19)
  27. #define MACPHYC_TX_SEL_ORIGIN 0x0
  28. #define MACPHYC_TX_SEL_DELAY 0x1
  29. #define MACPHYC_TX_DELAY_MASK GENMASK(18, 12)
  30. #define MACPHYC_RX_SEL_MASK GENMASK(11, 11)
  31. #define MACPHYC_RX_SEL_ORIGIN 0x0
  32. #define MACPHYC_RX_SEL_DELAY 0x1
  33. #define MACPHYC_RX_DELAY_MASK GENMASK(10, 4)
  34. #define MACPHYC_SOFT_RST_MASK GENMASK(3, 3)
  35. #define MACPHYC_PHY_INFT_MASK GENMASK(2, 0)
  36. #define MACPHYC_PHY_INFT_RMII 0x4
  37. #define MACPHYC_PHY_INFT_RGMII 0x1
  38. #define MACPHYC_PHY_INFT_GMII 0x0
  39. #define MACPHYC_PHY_INFT_MII 0x0
  40. #define MACPHYC_TX_DELAY_PS_MAX 2496
  41. #define MACPHYC_TX_DELAY_PS_MIN 20
  42. #define MACPHYC_RX_DELAY_PS_MAX 2496
  43. #define MACPHYC_RX_DELAY_PS_MIN 20
  44. enum ingenic_mac_version {
  45. ID_JZ4775,
  46. ID_X1000,
  47. ID_X1600,
  48. ID_X1830,
  49. ID_X2000,
  50. };
  51. struct ingenic_mac {
  52. const struct ingenic_soc_info *soc_info;
  53. struct device *dev;
  54. struct regmap *regmap;
  55. int rx_delay;
  56. int tx_delay;
  57. };
  58. struct ingenic_soc_info {
  59. enum ingenic_mac_version version;
  60. u32 mask;
  61. int (*set_mode)(struct plat_stmmacenet_data *plat_dat);
  62. };
  63. static int ingenic_mac_init(struct plat_stmmacenet_data *plat_dat)
  64. {
  65. struct ingenic_mac *mac = plat_dat->bsp_priv;
  66. int ret;
  67. if (mac->soc_info->set_mode) {
  68. ret = mac->soc_info->set_mode(plat_dat);
  69. if (ret)
  70. return ret;
  71. }
  72. return 0;
  73. }
  74. static int jz4775_mac_set_mode(struct plat_stmmacenet_data *plat_dat)
  75. {
  76. struct ingenic_mac *mac = plat_dat->bsp_priv;
  77. unsigned int val;
  78. switch (plat_dat->interface) {
  79. case PHY_INTERFACE_MODE_MII:
  80. val = FIELD_PREP(MACPHYC_TXCLK_SEL_MASK, MACPHYC_TXCLK_SEL_INPUT) |
  81. FIELD_PREP(MACPHYC_PHY_INFT_MASK, MACPHYC_PHY_INFT_MII);
  82. dev_dbg(mac->dev, "MAC PHY Control Register: PHY_INTERFACE_MODE_MII\n");
  83. break;
  84. case PHY_INTERFACE_MODE_GMII:
  85. val = FIELD_PREP(MACPHYC_TXCLK_SEL_MASK, MACPHYC_TXCLK_SEL_INPUT) |
  86. FIELD_PREP(MACPHYC_PHY_INFT_MASK, MACPHYC_PHY_INFT_GMII);
  87. dev_dbg(mac->dev, "MAC PHY Control Register: PHY_INTERFACE_MODE_GMII\n");
  88. break;
  89. case PHY_INTERFACE_MODE_RMII:
  90. val = FIELD_PREP(MACPHYC_TXCLK_SEL_MASK, MACPHYC_TXCLK_SEL_INPUT) |
  91. FIELD_PREP(MACPHYC_PHY_INFT_MASK, MACPHYC_PHY_INFT_RMII);
  92. dev_dbg(mac->dev, "MAC PHY Control Register: PHY_INTERFACE_MODE_RMII\n");
  93. break;
  94. case PHY_INTERFACE_MODE_RGMII:
  95. case PHY_INTERFACE_MODE_RGMII_ID:
  96. case PHY_INTERFACE_MODE_RGMII_TXID:
  97. case PHY_INTERFACE_MODE_RGMII_RXID:
  98. val = FIELD_PREP(MACPHYC_TXCLK_SEL_MASK, MACPHYC_TXCLK_SEL_INPUT) |
  99. FIELD_PREP(MACPHYC_PHY_INFT_MASK, MACPHYC_PHY_INFT_RGMII);
  100. dev_dbg(mac->dev, "MAC PHY Control Register: PHY_INTERFACE_MODE_RGMII\n");
  101. break;
  102. default:
  103. dev_err(mac->dev, "Unsupported interface %d", plat_dat->interface);
  104. return -EINVAL;
  105. }
  106. /* Update MAC PHY control register */
  107. return regmap_update_bits(mac->regmap, 0, mac->soc_info->mask, val);
  108. }
  109. static int x1000_mac_set_mode(struct plat_stmmacenet_data *plat_dat)
  110. {
  111. struct ingenic_mac *mac = plat_dat->bsp_priv;
  112. switch (plat_dat->interface) {
  113. case PHY_INTERFACE_MODE_RMII:
  114. dev_dbg(mac->dev, "MAC PHY Control Register: PHY_INTERFACE_MODE_RMII\n");
  115. break;
  116. default:
  117. dev_err(mac->dev, "Unsupported interface %d", plat_dat->interface);
  118. return -EINVAL;
  119. }
  120. /* Update MAC PHY control register */
  121. return regmap_update_bits(mac->regmap, 0, mac->soc_info->mask, 0);
  122. }
  123. static int x1600_mac_set_mode(struct plat_stmmacenet_data *plat_dat)
  124. {
  125. struct ingenic_mac *mac = plat_dat->bsp_priv;
  126. unsigned int val;
  127. switch (plat_dat->interface) {
  128. case PHY_INTERFACE_MODE_RMII:
  129. val = FIELD_PREP(MACPHYC_PHY_INFT_MASK, MACPHYC_PHY_INFT_RMII);
  130. dev_dbg(mac->dev, "MAC PHY Control Register: PHY_INTERFACE_MODE_RMII\n");
  131. break;
  132. default:
  133. dev_err(mac->dev, "Unsupported interface %d", plat_dat->interface);
  134. return -EINVAL;
  135. }
  136. /* Update MAC PHY control register */
  137. return regmap_update_bits(mac->regmap, 0, mac->soc_info->mask, val);
  138. }
  139. static int x1830_mac_set_mode(struct plat_stmmacenet_data *plat_dat)
  140. {
  141. struct ingenic_mac *mac = plat_dat->bsp_priv;
  142. unsigned int val;
  143. switch (plat_dat->interface) {
  144. case PHY_INTERFACE_MODE_RMII:
  145. val = FIELD_PREP(MACPHYC_MODE_SEL_MASK, MACPHYC_MODE_SEL_RMII) |
  146. FIELD_PREP(MACPHYC_PHY_INFT_MASK, MACPHYC_PHY_INFT_RMII);
  147. dev_dbg(mac->dev, "MAC PHY Control Register: PHY_INTERFACE_MODE_RMII\n");
  148. break;
  149. default:
  150. dev_err(mac->dev, "Unsupported interface %d", plat_dat->interface);
  151. return -EINVAL;
  152. }
  153. /* Update MAC PHY control register */
  154. return regmap_update_bits(mac->regmap, 0, mac->soc_info->mask, val);
  155. }
  156. static int x2000_mac_set_mode(struct plat_stmmacenet_data *plat_dat)
  157. {
  158. struct ingenic_mac *mac = plat_dat->bsp_priv;
  159. unsigned int val;
  160. switch (plat_dat->interface) {
  161. case PHY_INTERFACE_MODE_RMII:
  162. val = FIELD_PREP(MACPHYC_TX_SEL_MASK, MACPHYC_TX_SEL_ORIGIN) |
  163. FIELD_PREP(MACPHYC_RX_SEL_MASK, MACPHYC_RX_SEL_ORIGIN) |
  164. FIELD_PREP(MACPHYC_PHY_INFT_MASK, MACPHYC_PHY_INFT_RMII);
  165. dev_dbg(mac->dev, "MAC PHY Control Register: PHY_INTERFACE_MODE_RMII\n");
  166. break;
  167. case PHY_INTERFACE_MODE_RGMII:
  168. case PHY_INTERFACE_MODE_RGMII_ID:
  169. case PHY_INTERFACE_MODE_RGMII_TXID:
  170. case PHY_INTERFACE_MODE_RGMII_RXID:
  171. val = FIELD_PREP(MACPHYC_PHY_INFT_MASK, MACPHYC_PHY_INFT_RGMII);
  172. if (mac->tx_delay == 0)
  173. val |= FIELD_PREP(MACPHYC_TX_SEL_MASK, MACPHYC_TX_SEL_ORIGIN);
  174. else
  175. val |= FIELD_PREP(MACPHYC_TX_SEL_MASK, MACPHYC_TX_SEL_DELAY) |
  176. FIELD_PREP(MACPHYC_TX_DELAY_MASK, (mac->tx_delay + 9750) / 19500 - 1);
  177. if (mac->rx_delay == 0)
  178. val |= FIELD_PREP(MACPHYC_RX_SEL_MASK, MACPHYC_RX_SEL_ORIGIN);
  179. else
  180. val |= FIELD_PREP(MACPHYC_RX_SEL_MASK, MACPHYC_RX_SEL_DELAY) |
  181. FIELD_PREP(MACPHYC_RX_DELAY_MASK, (mac->rx_delay + 9750) / 19500 - 1);
  182. dev_dbg(mac->dev, "MAC PHY Control Register: PHY_INTERFACE_MODE_RGMII\n");
  183. break;
  184. default:
  185. dev_err(mac->dev, "Unsupported interface %d", plat_dat->interface);
  186. return -EINVAL;
  187. }
  188. /* Update MAC PHY control register */
  189. return regmap_update_bits(mac->regmap, 0, mac->soc_info->mask, val);
  190. }
  191. static int ingenic_mac_probe(struct platform_device *pdev)
  192. {
  193. struct plat_stmmacenet_data *plat_dat;
  194. struct stmmac_resources stmmac_res;
  195. struct ingenic_mac *mac;
  196. const struct ingenic_soc_info *data;
  197. u32 tx_delay_ps, rx_delay_ps;
  198. int ret;
  199. ret = stmmac_get_platform_resources(pdev, &stmmac_res);
  200. if (ret)
  201. return ret;
  202. plat_dat = stmmac_probe_config_dt(pdev, stmmac_res.mac);
  203. if (IS_ERR(plat_dat))
  204. return PTR_ERR(plat_dat);
  205. mac = devm_kzalloc(&pdev->dev, sizeof(*mac), GFP_KERNEL);
  206. if (!mac) {
  207. ret = -ENOMEM;
  208. goto err_remove_config_dt;
  209. }
  210. data = of_device_get_match_data(&pdev->dev);
  211. if (!data) {
  212. dev_err(&pdev->dev, "No of match data provided\n");
  213. ret = -EINVAL;
  214. goto err_remove_config_dt;
  215. }
  216. /* Get MAC PHY control register */
  217. mac->regmap = syscon_regmap_lookup_by_phandle(pdev->dev.of_node, "mode-reg");
  218. if (IS_ERR(mac->regmap)) {
  219. dev_err(&pdev->dev, "%s: Failed to get syscon regmap\n", __func__);
  220. ret = PTR_ERR(mac->regmap);
  221. goto err_remove_config_dt;
  222. }
  223. if (!of_property_read_u32(pdev->dev.of_node, "tx-clk-delay-ps", &tx_delay_ps)) {
  224. if (tx_delay_ps >= MACPHYC_TX_DELAY_PS_MIN &&
  225. tx_delay_ps <= MACPHYC_TX_DELAY_PS_MAX) {
  226. mac->tx_delay = tx_delay_ps * 1000;
  227. } else {
  228. dev_err(&pdev->dev, "Invalid TX clock delay: %dps\n", tx_delay_ps);
  229. ret = -EINVAL;
  230. goto err_remove_config_dt;
  231. }
  232. }
  233. if (!of_property_read_u32(pdev->dev.of_node, "rx-clk-delay-ps", &rx_delay_ps)) {
  234. if (rx_delay_ps >= MACPHYC_RX_DELAY_PS_MIN &&
  235. rx_delay_ps <= MACPHYC_RX_DELAY_PS_MAX) {
  236. mac->rx_delay = rx_delay_ps * 1000;
  237. } else {
  238. dev_err(&pdev->dev, "Invalid RX clock delay: %dps\n", rx_delay_ps);
  239. ret = -EINVAL;
  240. goto err_remove_config_dt;
  241. }
  242. }
  243. mac->soc_info = data;
  244. mac->dev = &pdev->dev;
  245. plat_dat->bsp_priv = mac;
  246. ret = ingenic_mac_init(plat_dat);
  247. if (ret)
  248. goto err_remove_config_dt;
  249. ret = stmmac_dvr_probe(&pdev->dev, plat_dat, &stmmac_res);
  250. if (ret)
  251. goto err_remove_config_dt;
  252. return 0;
  253. err_remove_config_dt:
  254. stmmac_remove_config_dt(pdev, plat_dat);
  255. return ret;
  256. }
  257. #ifdef CONFIG_PM_SLEEP
  258. static int ingenic_mac_suspend(struct device *dev)
  259. {
  260. int ret;
  261. ret = stmmac_suspend(dev);
  262. return ret;
  263. }
  264. static int ingenic_mac_resume(struct device *dev)
  265. {
  266. struct net_device *ndev = dev_get_drvdata(dev);
  267. struct stmmac_priv *priv = netdev_priv(ndev);
  268. int ret;
  269. ret = ingenic_mac_init(priv->plat);
  270. if (ret)
  271. return ret;
  272. ret = stmmac_resume(dev);
  273. return ret;
  274. }
  275. #endif /* CONFIG_PM_SLEEP */
  276. static SIMPLE_DEV_PM_OPS(ingenic_mac_pm_ops, ingenic_mac_suspend, ingenic_mac_resume);
  277. static struct ingenic_soc_info jz4775_soc_info = {
  278. .version = ID_JZ4775,
  279. .mask = MACPHYC_TXCLK_SEL_MASK | MACPHYC_SOFT_RST_MASK | MACPHYC_PHY_INFT_MASK,
  280. .set_mode = jz4775_mac_set_mode,
  281. };
  282. static struct ingenic_soc_info x1000_soc_info = {
  283. .version = ID_X1000,
  284. .mask = MACPHYC_SOFT_RST_MASK,
  285. .set_mode = x1000_mac_set_mode,
  286. };
  287. static struct ingenic_soc_info x1600_soc_info = {
  288. .version = ID_X1600,
  289. .mask = MACPHYC_SOFT_RST_MASK | MACPHYC_PHY_INFT_MASK,
  290. .set_mode = x1600_mac_set_mode,
  291. };
  292. static struct ingenic_soc_info x1830_soc_info = {
  293. .version = ID_X1830,
  294. .mask = MACPHYC_MODE_SEL_MASK | MACPHYC_SOFT_RST_MASK | MACPHYC_PHY_INFT_MASK,
  295. .set_mode = x1830_mac_set_mode,
  296. };
  297. static struct ingenic_soc_info x2000_soc_info = {
  298. .version = ID_X2000,
  299. .mask = MACPHYC_TX_SEL_MASK | MACPHYC_TX_DELAY_MASK | MACPHYC_RX_SEL_MASK |
  300. MACPHYC_RX_DELAY_MASK | MACPHYC_SOFT_RST_MASK | MACPHYC_PHY_INFT_MASK,
  301. .set_mode = x2000_mac_set_mode,
  302. };
  303. static const struct of_device_id ingenic_mac_of_matches[] = {
  304. { .compatible = "ingenic,jz4775-mac", .data = &jz4775_soc_info },
  305. { .compatible = "ingenic,x1000-mac", .data = &x1000_soc_info },
  306. { .compatible = "ingenic,x1600-mac", .data = &x1600_soc_info },
  307. { .compatible = "ingenic,x1830-mac", .data = &x1830_soc_info },
  308. { .compatible = "ingenic,x2000-mac", .data = &x2000_soc_info },
  309. { }
  310. };
  311. MODULE_DEVICE_TABLE(of, ingenic_mac_of_matches);
  312. static struct platform_driver ingenic_mac_driver = {
  313. .probe = ingenic_mac_probe,
  314. .remove = stmmac_pltfr_remove,
  315. .driver = {
  316. .name = "ingenic-mac",
  317. .pm = pm_ptr(&ingenic_mac_pm_ops),
  318. .of_match_table = ingenic_mac_of_matches,
  319. },
  320. };
  321. module_platform_driver(ingenic_mac_driver);
  322. MODULE_AUTHOR("周琰杰 (Zhou Yanjie) <[email protected]>");
  323. MODULE_DESCRIPTION("Ingenic SoCs DWMAC specific glue layer");
  324. MODULE_LICENSE("GPL v2");