vitesse.c 12 KB

123456789101112131415161718192021222324252627282930313233343536373839404142434445464748495051525354555657585960616263646566676869707172737475767778798081828384858687888990919293949596979899100101102103104105106107108109110111112113114115116117118119120121122123124125126127128129130131132133134135136137138139140141142143144145146147148149150151152153154155156157158159160161162163164165166167168169170171172173174175176177178179180181182183184185186187188189190191192193194195196197198199200201202203204205206207208209210211212213214215216217218219220221222223224225226227228229230231232233234235236237238239240241242243244245246247248249250251252253254255256257258259260261262263264265266267268269270271272273274275276277278279280281282283284285286287288289290291292293294295296297298299300301302303304305306307308309310311312313314315316317318319320321322323324325326327328329330331332333334335336337338339340341342343344345346347348349350351352353354355356357358359360361362363364365366367368369370371372373374375376377378379380381382383384385386387388389390391392393394395396397398399400401402403404405406407408409410411412413414415416417418419420421422423424425426427428429430431432433434435436437438439440441442443
  1. /*
  2. * Vitesse PHY drivers
  3. *
  4. * Copyright 2010-2014 Freescale Semiconductor, Inc.
  5. * Original Author: Andy Fleming
  6. * Add vsc8662 phy support - Priyanka Jain
  7. * SPDX-License-Identifier: GPL-2.0+
  8. */
  9. #include <miiphy.h>
  10. /* Cicada Auxiliary Control/Status Register */
  11. #define MIIM_CIS82xx_AUX_CONSTAT 0x1c
  12. #define MIIM_CIS82xx_AUXCONSTAT_INIT 0x0004
  13. #define MIIM_CIS82xx_AUXCONSTAT_DUPLEX 0x0020
  14. #define MIIM_CIS82xx_AUXCONSTAT_SPEED 0x0018
  15. #define MIIM_CIS82xx_AUXCONSTAT_GBIT 0x0010
  16. #define MIIM_CIS82xx_AUXCONSTAT_100 0x0008
  17. /* Cicada Extended Control Register 1 */
  18. #define MIIM_CIS82xx_EXT_CON1 0x17
  19. #define MIIM_CIS8201_EXTCON1_INIT 0x0000
  20. /* Cicada 8204 Extended PHY Control Register 1 */
  21. #define MIIM_CIS8204_EPHY_CON 0x17
  22. #define MIIM_CIS8204_EPHYCON_INIT 0x0006
  23. #define MIIM_CIS8204_EPHYCON_RGMII 0x1100
  24. /* Cicada 8204 Serial LED Control Register */
  25. #define MIIM_CIS8204_SLED_CON 0x1b
  26. #define MIIM_CIS8204_SLEDCON_INIT 0x1115
  27. /* Vitesse VSC8601 Extended PHY Control Register 1 */
  28. #define MII_VSC8601_EPHY_CTL 0x17
  29. #define MII_VSC8601_EPHY_CTL_RGMII_SKEW (1 << 8)
  30. #define PHY_EXT_PAGE_ACCESS 0x1f
  31. #define PHY_EXT_PAGE_ACCESS_GENERAL 0x10
  32. #define PHY_EXT_PAGE_ACCESS_EXTENDED3 0x3
  33. /* Vitesse VSC8574 control register */
  34. #define MIIM_VSC8574_MAC_SERDES_CON 0x10
  35. #define MIIM_VSC8574_MAC_SERDES_ANEG 0x80
  36. #define MIIM_VSC8574_GENERAL18 0x12
  37. #define MIIM_VSC8574_GENERAL19 0x13
  38. /* Vitesse VSC8574 gerenal purpose register 18 */
  39. #define MIIM_VSC8574_18G_SGMII 0x80f0
  40. #define MIIM_VSC8574_18G_QSGMII 0x80e0
  41. #define MIIM_VSC8574_18G_CMDSTAT 0x8000
  42. /* Vitesse VSC8514 control register */
  43. #define MIIM_VSC8514_MAC_SERDES_CON 0x10
  44. #define MIIM_VSC8514_GENERAL18 0x12
  45. #define MIIM_VSC8514_GENERAL19 0x13
  46. #define MIIM_VSC8514_GENERAL23 0x17
  47. /* Vitesse VSC8514 gerenal purpose register 18 */
  48. #define MIIM_VSC8514_18G_QSGMII 0x80e0
  49. #define MIIM_VSC8514_18G_CMDSTAT 0x8000
  50. /* Vitesse VSC8664 Control/Status Register */
  51. #define MIIM_VSC8664_SERDES_AND_SIGDET 0x13
  52. #define MIIM_VSC8664_ADDITIONAL_DEV 0x16
  53. #define MIIM_VSC8664_EPHY_CON 0x17
  54. #define MIIM_VSC8664_LED_CON 0x1E
  55. #define PHY_EXT_PAGE_ACCESS_EXTENDED 0x0001
  56. /* CIS8201 */
  57. static int vitesse_config(struct phy_device *phydev)
  58. {
  59. /* Override PHY config settings */
  60. phy_write(phydev, MDIO_DEVAD_NONE, MIIM_CIS82xx_AUX_CONSTAT,
  61. MIIM_CIS82xx_AUXCONSTAT_INIT);
  62. /* Set up the interface mode */
  63. phy_write(phydev, MDIO_DEVAD_NONE, MIIM_CIS82xx_EXT_CON1,
  64. MIIM_CIS8201_EXTCON1_INIT);
  65. genphy_config_aneg(phydev);
  66. return 0;
  67. }
  68. static int vitesse_parse_status(struct phy_device *phydev)
  69. {
  70. int speed;
  71. int mii_reg;
  72. mii_reg = phy_read(phydev, MDIO_DEVAD_NONE, MIIM_CIS82xx_AUX_CONSTAT);
  73. if (mii_reg & MIIM_CIS82xx_AUXCONSTAT_DUPLEX)
  74. phydev->duplex = DUPLEX_FULL;
  75. else
  76. phydev->duplex = DUPLEX_HALF;
  77. speed = mii_reg & MIIM_CIS82xx_AUXCONSTAT_SPEED;
  78. switch (speed) {
  79. case MIIM_CIS82xx_AUXCONSTAT_GBIT:
  80. phydev->speed = SPEED_1000;
  81. break;
  82. case MIIM_CIS82xx_AUXCONSTAT_100:
  83. phydev->speed = SPEED_100;
  84. break;
  85. default:
  86. phydev->speed = SPEED_10;
  87. break;
  88. }
  89. return 0;
  90. }
  91. static int vitesse_startup(struct phy_device *phydev)
  92. {
  93. int ret;
  94. ret = genphy_update_link(phydev);
  95. if (ret)
  96. return ret;
  97. return vitesse_parse_status(phydev);
  98. }
  99. static int cis8204_config(struct phy_device *phydev)
  100. {
  101. /* Override PHY config settings */
  102. phy_write(phydev, MDIO_DEVAD_NONE, MIIM_CIS82xx_AUX_CONSTAT,
  103. MIIM_CIS82xx_AUXCONSTAT_INIT);
  104. genphy_config_aneg(phydev);
  105. if (phy_interface_is_rgmii(phydev))
  106. phy_write(phydev, MDIO_DEVAD_NONE, MIIM_CIS8204_EPHY_CON,
  107. MIIM_CIS8204_EPHYCON_INIT |
  108. MIIM_CIS8204_EPHYCON_RGMII);
  109. else
  110. phy_write(phydev, MDIO_DEVAD_NONE, MIIM_CIS8204_EPHY_CON,
  111. MIIM_CIS8204_EPHYCON_INIT);
  112. return 0;
  113. }
  114. /* Vitesse VSC8601 */
  115. /* This adds a skew for both TX and RX clocks, so the skew should only be
  116. * applied to "rgmii-id" interfaces. It may not work as expected
  117. * on "rgmii-txid", "rgmii-rxid" or "rgmii" interfaces. */
  118. static int vsc8601_add_skew(struct phy_device *phydev)
  119. {
  120. int ret;
  121. ret = phy_read(phydev, MDIO_DEVAD_NONE, MII_VSC8601_EPHY_CTL);
  122. if (ret < 0)
  123. return ret;
  124. ret |= MII_VSC8601_EPHY_CTL_RGMII_SKEW;
  125. return phy_write(phydev, MDIO_DEVAD_NONE, MII_VSC8601_EPHY_CTL, ret);
  126. }
  127. static int vsc8601_config(struct phy_device *phydev)
  128. {
  129. int ret = 0;
  130. if (phydev->interface == PHY_INTERFACE_MODE_RGMII_ID)
  131. ret = vsc8601_add_skew(phydev);
  132. if (ret < 0)
  133. return ret;
  134. return genphy_config_aneg(phydev);
  135. }
  136. static int vsc8574_config(struct phy_device *phydev)
  137. {
  138. u32 val;
  139. /* configure register 19G for MAC */
  140. phy_write(phydev, MDIO_DEVAD_NONE, PHY_EXT_PAGE_ACCESS,
  141. PHY_EXT_PAGE_ACCESS_GENERAL);
  142. val = phy_read(phydev, MDIO_DEVAD_NONE, MIIM_VSC8574_GENERAL19);
  143. if (phydev->interface == PHY_INTERFACE_MODE_QSGMII) {
  144. /* set bit 15:14 to '01' for QSGMII mode */
  145. val = (val & 0x3fff) | (1 << 14);
  146. phy_write(phydev, MDIO_DEVAD_NONE,
  147. MIIM_VSC8574_GENERAL19, val);
  148. /* Enable 4 ports MAC QSGMII */
  149. phy_write(phydev, MDIO_DEVAD_NONE, MIIM_VSC8574_GENERAL18,
  150. MIIM_VSC8574_18G_QSGMII);
  151. } else {
  152. /* set bit 15:14 to '00' for SGMII mode */
  153. val = val & 0x3fff;
  154. phy_write(phydev, MDIO_DEVAD_NONE, MIIM_VSC8574_GENERAL19, val);
  155. /* Enable 4 ports MAC SGMII */
  156. phy_write(phydev, MDIO_DEVAD_NONE, MIIM_VSC8574_GENERAL18,
  157. MIIM_VSC8574_18G_SGMII);
  158. }
  159. val = phy_read(phydev, MDIO_DEVAD_NONE, MIIM_VSC8574_GENERAL18);
  160. /* When bit 15 is cleared the command has completed */
  161. while (val & MIIM_VSC8574_18G_CMDSTAT)
  162. val = phy_read(phydev, MDIO_DEVAD_NONE, MIIM_VSC8574_GENERAL18);
  163. /* Enable Serdes Auto-negotiation */
  164. phy_write(phydev, MDIO_DEVAD_NONE, PHY_EXT_PAGE_ACCESS,
  165. PHY_EXT_PAGE_ACCESS_EXTENDED3);
  166. val = phy_read(phydev, MDIO_DEVAD_NONE, MIIM_VSC8574_MAC_SERDES_CON);
  167. val = val | MIIM_VSC8574_MAC_SERDES_ANEG;
  168. phy_write(phydev, MDIO_DEVAD_NONE, MIIM_VSC8574_MAC_SERDES_CON, val);
  169. phy_write(phydev, MDIO_DEVAD_NONE, PHY_EXT_PAGE_ACCESS, 0);
  170. genphy_config_aneg(phydev);
  171. return 0;
  172. }
  173. static int vsc8514_config(struct phy_device *phydev)
  174. {
  175. u32 val;
  176. int timeout = 1000000;
  177. /* configure register to access 19G */
  178. phy_write(phydev, MDIO_DEVAD_NONE, PHY_EXT_PAGE_ACCESS,
  179. PHY_EXT_PAGE_ACCESS_GENERAL);
  180. val = phy_read(phydev, MDIO_DEVAD_NONE, MIIM_VSC8514_GENERAL19);
  181. if (phydev->interface == PHY_INTERFACE_MODE_QSGMII) {
  182. /* set bit 15:14 to '01' for QSGMII mode */
  183. val = (val & 0x3fff) | (1 << 14);
  184. phy_write(phydev, MDIO_DEVAD_NONE,
  185. MIIM_VSC8514_GENERAL19, val);
  186. /* Enable 4 ports MAC QSGMII */
  187. phy_write(phydev, MDIO_DEVAD_NONE, MIIM_VSC8514_GENERAL18,
  188. MIIM_VSC8514_18G_QSGMII);
  189. } else {
  190. /*TODO Add SGMII functionality once spec sheet
  191. * for VSC8514 defines complete functionality
  192. */
  193. }
  194. val = phy_read(phydev, MDIO_DEVAD_NONE, MIIM_VSC8514_GENERAL18);
  195. /* When bit 15 is cleared the command has completed */
  196. while ((val & MIIM_VSC8514_18G_CMDSTAT) && timeout--)
  197. val = phy_read(phydev, MDIO_DEVAD_NONE, MIIM_VSC8514_GENERAL18);
  198. if (0 == timeout) {
  199. printf("PHY 8514 config failed\n");
  200. return -1;
  201. }
  202. phy_write(phydev, MDIO_DEVAD_NONE, PHY_EXT_PAGE_ACCESS, 0);
  203. /* configure register to access 23 */
  204. val = phy_read(phydev, MDIO_DEVAD_NONE, MIIM_VSC8514_GENERAL23);
  205. /* set bits 10:8 to '000' */
  206. val = (val & 0xf8ff);
  207. phy_write(phydev, MDIO_DEVAD_NONE, MIIM_VSC8514_GENERAL23, val);
  208. /* Enable Serdes Auto-negotiation */
  209. phy_write(phydev, MDIO_DEVAD_NONE, PHY_EXT_PAGE_ACCESS,
  210. PHY_EXT_PAGE_ACCESS_EXTENDED3);
  211. val = phy_read(phydev, MDIO_DEVAD_NONE, MIIM_VSC8514_MAC_SERDES_CON);
  212. val = val | MIIM_VSC8574_MAC_SERDES_ANEG;
  213. phy_write(phydev, MDIO_DEVAD_NONE, MIIM_VSC8514_MAC_SERDES_CON, val);
  214. phy_write(phydev, MDIO_DEVAD_NONE, PHY_EXT_PAGE_ACCESS, 0);
  215. genphy_config_aneg(phydev);
  216. return 0;
  217. }
  218. static int vsc8664_config(struct phy_device *phydev)
  219. {
  220. u32 val;
  221. /* Enable MAC interface auto-negotiation */
  222. phy_write(phydev, MDIO_DEVAD_NONE, PHY_EXT_PAGE_ACCESS, 0);
  223. val = phy_read(phydev, MDIO_DEVAD_NONE, MIIM_VSC8664_EPHY_CON);
  224. val |= (1 << 13);
  225. phy_write(phydev, MDIO_DEVAD_NONE, MIIM_VSC8664_EPHY_CON, val);
  226. phy_write(phydev, MDIO_DEVAD_NONE, PHY_EXT_PAGE_ACCESS,
  227. PHY_EXT_PAGE_ACCESS_EXTENDED);
  228. val = phy_read(phydev, MDIO_DEVAD_NONE, MIIM_VSC8664_SERDES_AND_SIGDET);
  229. val |= (1 << 11);
  230. phy_write(phydev, MDIO_DEVAD_NONE, MIIM_VSC8664_SERDES_AND_SIGDET, val);
  231. phy_write(phydev, MDIO_DEVAD_NONE, PHY_EXT_PAGE_ACCESS, 0);
  232. /* Enable LED blink */
  233. val = phy_read(phydev, MDIO_DEVAD_NONE, MIIM_VSC8664_LED_CON);
  234. val &= ~(1 << 2);
  235. phy_write(phydev, MDIO_DEVAD_NONE, MIIM_VSC8664_LED_CON, val);
  236. genphy_config_aneg(phydev);
  237. return 0;
  238. }
  239. static struct phy_driver VSC8211_driver = {
  240. .name = "Vitesse VSC8211",
  241. .uid = 0xfc4b0,
  242. .mask = 0xffff0,
  243. .features = PHY_GBIT_FEATURES,
  244. .config = &vitesse_config,
  245. .startup = &vitesse_startup,
  246. .shutdown = &genphy_shutdown,
  247. };
  248. static struct phy_driver VSC8221_driver = {
  249. .name = "Vitesse VSC8221",
  250. .uid = 0xfc550,
  251. .mask = 0xffff0,
  252. .features = PHY_GBIT_FEATURES,
  253. .config = &genphy_config_aneg,
  254. .startup = &vitesse_startup,
  255. .shutdown = &genphy_shutdown,
  256. };
  257. static struct phy_driver VSC8244_driver = {
  258. .name = "Vitesse VSC8244",
  259. .uid = 0xfc6c0,
  260. .mask = 0xffff0,
  261. .features = PHY_GBIT_FEATURES,
  262. .config = &genphy_config_aneg,
  263. .startup = &vitesse_startup,
  264. .shutdown = &genphy_shutdown,
  265. };
  266. static struct phy_driver VSC8234_driver = {
  267. .name = "Vitesse VSC8234",
  268. .uid = 0xfc620,
  269. .mask = 0xffff0,
  270. .features = PHY_GBIT_FEATURES,
  271. .config = &genphy_config_aneg,
  272. .startup = &vitesse_startup,
  273. .shutdown = &genphy_shutdown,
  274. };
  275. static struct phy_driver VSC8574_driver = {
  276. .name = "Vitesse VSC8574",
  277. .uid = 0x704a0,
  278. .mask = 0xffff0,
  279. .features = PHY_GBIT_FEATURES,
  280. .config = &vsc8574_config,
  281. .startup = &vitesse_startup,
  282. .shutdown = &genphy_shutdown,
  283. };
  284. static struct phy_driver VSC8514_driver = {
  285. .name = "Vitesse VSC8514",
  286. .uid = 0x70670,
  287. .mask = 0xffff0,
  288. .features = PHY_GBIT_FEATURES,
  289. .config = &vsc8514_config,
  290. .startup = &vitesse_startup,
  291. .shutdown = &genphy_shutdown,
  292. };
  293. static struct phy_driver VSC8584_driver = {
  294. .name = "Vitesse VSC8584",
  295. .uid = 0x707c0,
  296. .mask = 0xffff0,
  297. .features = PHY_GBIT_FEATURES,
  298. .config = &vsc8574_config,
  299. .startup = &vitesse_startup,
  300. .shutdown = &genphy_shutdown,
  301. };
  302. static struct phy_driver VSC8601_driver = {
  303. .name = "Vitesse VSC8601",
  304. .uid = 0x70420,
  305. .mask = 0xffff0,
  306. .features = PHY_GBIT_FEATURES,
  307. .config = &vsc8601_config,
  308. .startup = &vitesse_startup,
  309. .shutdown = &genphy_shutdown,
  310. };
  311. static struct phy_driver VSC8641_driver = {
  312. .name = "Vitesse VSC8641",
  313. .uid = 0x70430,
  314. .mask = 0xffff0,
  315. .features = PHY_GBIT_FEATURES,
  316. .config = &genphy_config_aneg,
  317. .startup = &vitesse_startup,
  318. .shutdown = &genphy_shutdown,
  319. };
  320. static struct phy_driver VSC8662_driver = {
  321. .name = "Vitesse VSC8662",
  322. .uid = 0x70660,
  323. .mask = 0xffff0,
  324. .features = PHY_GBIT_FEATURES,
  325. .config = &genphy_config_aneg,
  326. .startup = &vitesse_startup,
  327. .shutdown = &genphy_shutdown,
  328. };
  329. static struct phy_driver VSC8664_driver = {
  330. .name = "Vitesse VSC8664",
  331. .uid = 0x70660,
  332. .mask = 0xffff0,
  333. .features = PHY_GBIT_FEATURES,
  334. .config = &vsc8664_config,
  335. .startup = &vitesse_startup,
  336. .shutdown = &genphy_shutdown,
  337. };
  338. /* Vitesse bought Cicada, so we'll put these here */
  339. static struct phy_driver cis8201_driver = {
  340. .name = "CIS8201",
  341. .uid = 0xfc410,
  342. .mask = 0xffff0,
  343. .features = PHY_GBIT_FEATURES,
  344. .config = &vitesse_config,
  345. .startup = &vitesse_startup,
  346. .shutdown = &genphy_shutdown,
  347. };
  348. static struct phy_driver cis8204_driver = {
  349. .name = "Cicada Cis8204",
  350. .uid = 0xfc440,
  351. .mask = 0xffff0,
  352. .features = PHY_GBIT_FEATURES,
  353. .config = &cis8204_config,
  354. .startup = &vitesse_startup,
  355. .shutdown = &genphy_shutdown,
  356. };
  357. int phy_vitesse_init(void)
  358. {
  359. phy_register(&VSC8641_driver);
  360. phy_register(&VSC8601_driver);
  361. phy_register(&VSC8234_driver);
  362. phy_register(&VSC8244_driver);
  363. phy_register(&VSC8211_driver);
  364. phy_register(&VSC8221_driver);
  365. phy_register(&VSC8574_driver);
  366. phy_register(&VSC8584_driver);
  367. phy_register(&VSC8514_driver);
  368. phy_register(&VSC8662_driver);
  369. phy_register(&VSC8664_driver);
  370. phy_register(&cis8201_driver);
  371. phy_register(&cis8204_driver);
  372. return 0;
  373. }