mtk-phy-lib.c 9.4 KB

123456789101112131415161718192021222324252627282930313233343536373839404142434445464748495051525354555657585960616263646566676869707172737475767778798081828384858687888990919293949596979899100101102103104105106107108109110111112113114115116117118119120121122123124125126127128129130131132133134135136137138139140141142143144145146147148149150151152153154155156157158159160161162163164165166167168169170171172173174175176177178179180181182183184185186187188189190191192193194195196197198199200201202203204205206207208209210211212213214215216217218219220221222223224225226227228229230231232233234235236237238239240241242243244245246247248249250251252253254255256257258259260261262263264265266267268269270271272273274275276277278279280281282283284285286287288289290291292293294295296297298299300301302303304305306307308309310311312313314315316317318319320321322323324325326327328329330331332333334335336337338339340341342343344345346347
  1. // SPDX-License-Identifier: GPL-2.0
  2. #include <linux/phy.h>
  3. #include <linux/module.h>
  4. #include <linux/netdevice.h>
  5. #include "mtk.h"
  6. /* Difference between functions with mtk_tr* and __mtk_tr* prefixes is
  7. * mtk_tr* functions: wrapped by page switching operations
  8. * __mtk_tr* functions: no page switching operations
  9. */
  10. static void __mtk_tr_access(struct phy_device *phydev, bool read, u8 ch_addr,
  11. u8 node_addr, u8 data_addr)
  12. {
  13. u16 tr_cmd = BIT(15); /* bit 14 & 0 are reserved */
  14. if (read)
  15. tr_cmd |= BIT(13);
  16. tr_cmd |= (((ch_addr & 0x3) << 11) |
  17. ((node_addr & 0xf) << 7) |
  18. ((data_addr & 0x3f) << 1));
  19. dev_dbg(&phydev->mdio.dev, "tr_cmd: 0x%x\n", tr_cmd);
  20. __phy_write(phydev, 0x10, tr_cmd);
  21. }
  22. static void __mtk_tr_read(struct phy_device *phydev, u8 ch_addr, u8 node_addr,
  23. u8 data_addr, u16 *tr_high, u16 *tr_low)
  24. {
  25. __mtk_tr_access(phydev, true, ch_addr, node_addr, data_addr);
  26. *tr_low = __phy_read(phydev, 0x11);
  27. *tr_high = __phy_read(phydev, 0x12);
  28. dev_dbg(&phydev->mdio.dev, "tr_high read: 0x%x, tr_low read: 0x%x\n",
  29. *tr_high, *tr_low);
  30. }
  31. static void __mtk_tr_write(struct phy_device *phydev, u8 ch_addr, u8 node_addr,
  32. u8 data_addr, u32 tr_data)
  33. {
  34. __phy_write(phydev, 0x11, tr_data & 0xffff);
  35. __phy_write(phydev, 0x12, tr_data >> 16);
  36. dev_dbg(&phydev->mdio.dev, "tr_high write: 0x%x, tr_low write: 0x%x\n",
  37. tr_data >> 16, tr_data & 0xffff);
  38. __mtk_tr_access(phydev, false, ch_addr, node_addr, data_addr);
  39. }
  40. void __mtk_tr_modify(struct phy_device *phydev, u8 ch_addr, u8 node_addr,
  41. u8 data_addr, u32 mask, u32 set)
  42. {
  43. u32 tr_data;
  44. u16 tr_high;
  45. u16 tr_low;
  46. __mtk_tr_read(phydev, ch_addr, node_addr, data_addr, &tr_high, &tr_low);
  47. tr_data = (tr_high << 16) | tr_low;
  48. tr_data = (tr_data & ~mask) | set;
  49. __mtk_tr_write(phydev, ch_addr, node_addr, data_addr, tr_data);
  50. }
  51. EXPORT_SYMBOL_GPL(__mtk_tr_modify);
  52. void mtk_tr_modify(struct phy_device *phydev, u8 ch_addr, u8 node_addr,
  53. u8 data_addr, u32 mask, u32 set)
  54. {
  55. phy_select_page(phydev, MTK_PHY_PAGE_EXTENDED_52B5);
  56. __mtk_tr_modify(phydev, ch_addr, node_addr, data_addr, mask, set);
  57. phy_restore_page(phydev, MTK_PHY_PAGE_STANDARD, 0);
  58. }
  59. EXPORT_SYMBOL_GPL(mtk_tr_modify);
  60. void __mtk_tr_set_bits(struct phy_device *phydev, u8 ch_addr, u8 node_addr,
  61. u8 data_addr, u32 set)
  62. {
  63. __mtk_tr_modify(phydev, ch_addr, node_addr, data_addr, 0, set);
  64. }
  65. EXPORT_SYMBOL_GPL(__mtk_tr_set_bits);
  66. void __mtk_tr_clr_bits(struct phy_device *phydev, u8 ch_addr, u8 node_addr,
  67. u8 data_addr, u32 clr)
  68. {
  69. __mtk_tr_modify(phydev, ch_addr, node_addr, data_addr, clr, 0);
  70. }
  71. EXPORT_SYMBOL_GPL(__mtk_tr_clr_bits);
  72. int mtk_phy_read_page(struct phy_device *phydev)
  73. {
  74. return __phy_read(phydev, MTK_EXT_PAGE_ACCESS);
  75. }
  76. EXPORT_SYMBOL_GPL(mtk_phy_read_page);
  77. int mtk_phy_write_page(struct phy_device *phydev, int page)
  78. {
  79. return __phy_write(phydev, MTK_EXT_PAGE_ACCESS, page);
  80. }
  81. EXPORT_SYMBOL_GPL(mtk_phy_write_page);
  82. int mtk_phy_led_hw_is_supported(struct phy_device *phydev, u8 index,
  83. unsigned long rules,
  84. unsigned long supported_triggers)
  85. {
  86. if (index > 1)
  87. return -EINVAL;
  88. /* All combinations of the supported triggers are allowed */
  89. if (rules & ~supported_triggers)
  90. return -EOPNOTSUPP;
  91. return 0;
  92. }
  93. EXPORT_SYMBOL_GPL(mtk_phy_led_hw_is_supported);
  94. int mtk_phy_led_hw_ctrl_get(struct phy_device *phydev, u8 index,
  95. unsigned long *rules, u16 on_set,
  96. u16 rx_blink_set, u16 tx_blink_set)
  97. {
  98. unsigned int bit_blink = MTK_PHY_LED_STATE_FORCE_BLINK +
  99. (index ? 16 : 0);
  100. unsigned int bit_netdev = MTK_PHY_LED_STATE_NETDEV + (index ? 16 : 0);
  101. unsigned int bit_on = MTK_PHY_LED_STATE_FORCE_ON + (index ? 16 : 0);
  102. struct mtk_socphy_priv *priv = phydev->priv;
  103. int on, blink;
  104. if (index > 1)
  105. return -EINVAL;
  106. on = phy_read_mmd(phydev, MDIO_MMD_VEND2,
  107. index ? MTK_PHY_LED1_ON_CTRL : MTK_PHY_LED0_ON_CTRL);
  108. if (on < 0)
  109. return -EIO;
  110. blink = phy_read_mmd(phydev, MDIO_MMD_VEND2,
  111. index ? MTK_PHY_LED1_BLINK_CTRL :
  112. MTK_PHY_LED0_BLINK_CTRL);
  113. if (blink < 0)
  114. return -EIO;
  115. if ((on & (on_set | MTK_PHY_LED_ON_FDX |
  116. MTK_PHY_LED_ON_HDX | MTK_PHY_LED_ON_LINKDOWN)) ||
  117. (blink & (rx_blink_set | tx_blink_set)))
  118. set_bit(bit_netdev, &priv->led_state);
  119. else
  120. clear_bit(bit_netdev, &priv->led_state);
  121. if (on & MTK_PHY_LED_ON_FORCE_ON)
  122. set_bit(bit_on, &priv->led_state);
  123. else
  124. clear_bit(bit_on, &priv->led_state);
  125. if (blink & MTK_PHY_LED_BLINK_FORCE_BLINK)
  126. set_bit(bit_blink, &priv->led_state);
  127. else
  128. clear_bit(bit_blink, &priv->led_state);
  129. if (!rules)
  130. return 0;
  131. if (on & on_set)
  132. *rules |= BIT(TRIGGER_NETDEV_LINK);
  133. if (on & MTK_PHY_LED_ON_LINK10)
  134. *rules |= BIT(TRIGGER_NETDEV_LINK_10);
  135. if (on & MTK_PHY_LED_ON_LINK100)
  136. *rules |= BIT(TRIGGER_NETDEV_LINK_100);
  137. if (on & MTK_PHY_LED_ON_LINK1000)
  138. *rules |= BIT(TRIGGER_NETDEV_LINK_1000);
  139. if (on & MTK_PHY_LED_ON_LINK2500)
  140. *rules |= BIT(TRIGGER_NETDEV_LINK_2500);
  141. if (on & MTK_PHY_LED_ON_FDX)
  142. *rules |= BIT(TRIGGER_NETDEV_FULL_DUPLEX);
  143. if (on & MTK_PHY_LED_ON_HDX)
  144. *rules |= BIT(TRIGGER_NETDEV_HALF_DUPLEX);
  145. if (blink & rx_blink_set)
  146. *rules |= BIT(TRIGGER_NETDEV_RX);
  147. if (blink & tx_blink_set)
  148. *rules |= BIT(TRIGGER_NETDEV_TX);
  149. return 0;
  150. }
  151. EXPORT_SYMBOL_GPL(mtk_phy_led_hw_ctrl_get);
  152. int mtk_phy_led_hw_ctrl_set(struct phy_device *phydev, u8 index,
  153. unsigned long rules, u16 on_set,
  154. u16 rx_blink_set, u16 tx_blink_set)
  155. {
  156. unsigned int bit_netdev = MTK_PHY_LED_STATE_NETDEV + (index ? 16 : 0);
  157. struct mtk_socphy_priv *priv = phydev->priv;
  158. u16 on = 0, blink = 0;
  159. int ret;
  160. if (index > 1)
  161. return -EINVAL;
  162. if (rules & BIT(TRIGGER_NETDEV_FULL_DUPLEX))
  163. on |= MTK_PHY_LED_ON_FDX;
  164. if (rules & BIT(TRIGGER_NETDEV_HALF_DUPLEX))
  165. on |= MTK_PHY_LED_ON_HDX;
  166. if (rules & (BIT(TRIGGER_NETDEV_LINK_10) | BIT(TRIGGER_NETDEV_LINK)))
  167. on |= MTK_PHY_LED_ON_LINK10;
  168. if (rules & (BIT(TRIGGER_NETDEV_LINK_100) | BIT(TRIGGER_NETDEV_LINK)))
  169. on |= MTK_PHY_LED_ON_LINK100;
  170. if (rules & (BIT(TRIGGER_NETDEV_LINK_1000) | BIT(TRIGGER_NETDEV_LINK)))
  171. on |= MTK_PHY_LED_ON_LINK1000;
  172. if (rules & (BIT(TRIGGER_NETDEV_LINK_2500) | BIT(TRIGGER_NETDEV_LINK)))
  173. on |= MTK_PHY_LED_ON_LINK2500;
  174. if (rules & BIT(TRIGGER_NETDEV_RX)) {
  175. if (on & on_set) {
  176. if (on & MTK_PHY_LED_ON_LINK10)
  177. blink |= MTK_PHY_LED_BLINK_10RX;
  178. if (on & MTK_PHY_LED_ON_LINK100)
  179. blink |= MTK_PHY_LED_BLINK_100RX;
  180. if (on & MTK_PHY_LED_ON_LINK1000)
  181. blink |= MTK_PHY_LED_BLINK_1000RX;
  182. if (on & MTK_PHY_LED_ON_LINK2500)
  183. blink |= MTK_PHY_LED_BLINK_2500RX;
  184. } else {
  185. blink |= rx_blink_set;
  186. }
  187. }
  188. if (rules & BIT(TRIGGER_NETDEV_TX)) {
  189. if (on & on_set) {
  190. if (on & MTK_PHY_LED_ON_LINK10)
  191. blink |= MTK_PHY_LED_BLINK_10TX;
  192. if (on & MTK_PHY_LED_ON_LINK100)
  193. blink |= MTK_PHY_LED_BLINK_100TX;
  194. if (on & MTK_PHY_LED_ON_LINK1000)
  195. blink |= MTK_PHY_LED_BLINK_1000TX;
  196. if (on & MTK_PHY_LED_ON_LINK2500)
  197. blink |= MTK_PHY_LED_BLINK_2500TX;
  198. } else {
  199. blink |= tx_blink_set;
  200. }
  201. }
  202. if (blink || on)
  203. set_bit(bit_netdev, &priv->led_state);
  204. else
  205. clear_bit(bit_netdev, &priv->led_state);
  206. ret = phy_modify_mmd(phydev, MDIO_MMD_VEND2, index ?
  207. MTK_PHY_LED1_ON_CTRL : MTK_PHY_LED0_ON_CTRL,
  208. MTK_PHY_LED_ON_FDX | MTK_PHY_LED_ON_HDX | on_set,
  209. on);
  210. if (ret)
  211. return ret;
  212. return phy_write_mmd(phydev, MDIO_MMD_VEND2, index ?
  213. MTK_PHY_LED1_BLINK_CTRL :
  214. MTK_PHY_LED0_BLINK_CTRL, blink);
  215. }
  216. EXPORT_SYMBOL_GPL(mtk_phy_led_hw_ctrl_set);
  217. int mtk_phy_led_num_dly_cfg(u8 index, unsigned long *delay_on,
  218. unsigned long *delay_off, bool *blinking)
  219. {
  220. if (index > 1)
  221. return -EINVAL;
  222. if (delay_on && delay_off && (*delay_on > 0) && (*delay_off > 0)) {
  223. *blinking = true;
  224. *delay_on = 50;
  225. *delay_off = 50;
  226. }
  227. return 0;
  228. }
  229. EXPORT_SYMBOL_GPL(mtk_phy_led_num_dly_cfg);
  230. int mtk_phy_hw_led_on_set(struct phy_device *phydev, u8 index,
  231. u16 led_on_mask, bool on)
  232. {
  233. unsigned int bit_on = MTK_PHY_LED_STATE_FORCE_ON + (index ? 16 : 0);
  234. struct mtk_socphy_priv *priv = phydev->priv;
  235. bool changed;
  236. if (on)
  237. changed = !test_and_set_bit(bit_on, &priv->led_state);
  238. else
  239. changed = !!test_and_clear_bit(bit_on, &priv->led_state);
  240. changed |= !!test_and_clear_bit(MTK_PHY_LED_STATE_NETDEV +
  241. (index ? 16 : 0), &priv->led_state);
  242. if (changed)
  243. return phy_modify_mmd(phydev, MDIO_MMD_VEND2, index ?
  244. MTK_PHY_LED1_ON_CTRL :
  245. MTK_PHY_LED0_ON_CTRL,
  246. led_on_mask,
  247. on ? MTK_PHY_LED_ON_FORCE_ON : 0);
  248. else
  249. return 0;
  250. }
  251. EXPORT_SYMBOL_GPL(mtk_phy_hw_led_on_set);
  252. int mtk_phy_hw_led_blink_set(struct phy_device *phydev, u8 index, bool blinking)
  253. {
  254. unsigned int bit_blink = MTK_PHY_LED_STATE_FORCE_BLINK +
  255. (index ? 16 : 0);
  256. struct mtk_socphy_priv *priv = phydev->priv;
  257. bool changed;
  258. if (blinking)
  259. changed = !test_and_set_bit(bit_blink, &priv->led_state);
  260. else
  261. changed = !!test_and_clear_bit(bit_blink, &priv->led_state);
  262. changed |= !!test_bit(MTK_PHY_LED_STATE_NETDEV +
  263. (index ? 16 : 0), &priv->led_state);
  264. if (changed)
  265. return phy_write_mmd(phydev, MDIO_MMD_VEND2, index ?
  266. MTK_PHY_LED1_BLINK_CTRL :
  267. MTK_PHY_LED0_BLINK_CTRL,
  268. blinking ?
  269. MTK_PHY_LED_BLINK_FORCE_BLINK : 0);
  270. else
  271. return 0;
  272. }
  273. EXPORT_SYMBOL_GPL(mtk_phy_hw_led_blink_set);
  274. void mtk_phy_leds_state_init(struct phy_device *phydev)
  275. {
  276. int i;
  277. for (i = 0; i < 2; ++i)
  278. phydev->drv->led_hw_control_get(phydev, i, NULL);
  279. }
  280. EXPORT_SYMBOL_GPL(mtk_phy_leds_state_init);
  281. MODULE_DESCRIPTION("MediaTek Ethernet PHY driver common");
  282. MODULE_AUTHOR("Sky Huang <SkyLake.Huang@mediatek.com>");
  283. MODULE_AUTHOR("Daniel Golle <daniel@makrotopia.org>");
  284. MODULE_LICENSE("GPL");