mdio_10g.h 8.4 KB

123456789101112131415161718192021222324252627282930313233343536373839404142434445464748495051525354555657585960616263646566676869707172737475767778798081828384858687888990919293949596979899100101102103104105106107108109110111112113114115116117118119120121122123124125126127128129130131132133134135136137138139140141142143144145146147148149150151152153154155156157158159160161162163164165166167168169170171172173174175176177178179180181182183184185186187188189190191192193194195196197198199200201202203204205206207208209210211212213214215216217218219220221222223224225226227228229230231232233234235236237238239240241242243244245246247248249250251252253254255
  1. /****************************************************************************
  2. * Driver for Solarflare Solarstorm network controllers and boards
  3. * Copyright 2006-2008 Solarflare Communications Inc.
  4. *
  5. * This program is free software; you can redistribute it and/or modify it
  6. * under the terms of the GNU General Public License version 2 as published
  7. * by the Free Software Foundation, incorporated herein by reference.
  8. */
  9. #ifndef EFX_MDIO_10G_H
  10. #define EFX_MDIO_10G_H
  11. /*
  12. * Definitions needed for doing 10G MDIO as specified in clause 45
  13. * MDIO, which do not appear in Linux yet. Also some helper functions.
  14. */
  15. #include "efx.h"
  16. #include "boards.h"
  17. /* Numbering of the MDIO Manageable Devices (MMDs) */
  18. /* Physical Medium Attachment/ Physical Medium Dependent sublayer */
  19. #define MDIO_MMD_PMAPMD (1)
  20. /* WAN Interface Sublayer */
  21. #define MDIO_MMD_WIS (2)
  22. /* Physical Coding Sublayer */
  23. #define MDIO_MMD_PCS (3)
  24. /* PHY Extender Sublayer */
  25. #define MDIO_MMD_PHYXS (4)
  26. /* Extender Sublayer */
  27. #define MDIO_MMD_DTEXS (5)
  28. /* Transmission convergence */
  29. #define MDIO_MMD_TC (6)
  30. /* Auto negotiation */
  31. #define MDIO_MMD_AN (7)
  32. /* Generic register locations */
  33. #define MDIO_MMDREG_CTRL1 (0)
  34. #define MDIO_MMDREG_STAT1 (1)
  35. #define MDIO_MMDREG_IDHI (2)
  36. #define MDIO_MMDREG_IDLOW (3)
  37. #define MDIO_MMDREG_SPEED (4)
  38. #define MDIO_MMDREG_DEVS0 (5)
  39. #define MDIO_MMDREG_DEVS1 (6)
  40. #define MDIO_MMDREG_CTRL2 (7)
  41. #define MDIO_MMDREG_STAT2 (8)
  42. #define MDIO_MMDREG_TXDIS (9)
  43. /* Bits in MMDREG_CTRL1 */
  44. /* Reset */
  45. #define MDIO_MMDREG_CTRL1_RESET_LBN (15)
  46. #define MDIO_MMDREG_CTRL1_RESET_WIDTH (1)
  47. /* Loopback */
  48. /* Loopback bit for WIS, PCS, PHYSX and DTEXS */
  49. #define MDIO_MMDREG_CTRL1_LBACK_LBN (14)
  50. #define MDIO_MMDREG_CTRL1_LBACK_WIDTH (1)
  51. /* Bits in MMDREG_STAT1 */
  52. #define MDIO_MMDREG_STAT1_FAULT_LBN (7)
  53. #define MDIO_MMDREG_STAT1_FAULT_WIDTH (1)
  54. /* Link state */
  55. #define MDIO_MMDREG_STAT1_LINK_LBN (2)
  56. #define MDIO_MMDREG_STAT1_LINK_WIDTH (1)
  57. /* Low power ability */
  58. #define MDIO_MMDREG_STAT1_LPABLE_LBN (1)
  59. #define MDIO_MMDREG_STAT1_LPABLE_WIDTH (1)
  60. /* Bits in ID reg */
  61. #define MDIO_ID_REV(_id32) (_id32 & 0xf)
  62. #define MDIO_ID_MODEL(_id32) ((_id32 >> 4) & 0x3f)
  63. #define MDIO_ID_OUI(_id32) (_id32 >> 10)
  64. /* Bits in MMDREG_DEVS0. Someone thoughtfully layed things out
  65. * so the 'bit present' bit number of an MMD is the number of
  66. * that MMD */
  67. #define DEV_PRESENT_BIT(_b) (1 << _b)
  68. #define MDIO_MMDREG_DEVS0_PHYXS DEV_PRESENT_BIT(MDIO_MMD_PHYXS)
  69. #define MDIO_MMDREG_DEVS0_PCS DEV_PRESENT_BIT(MDIO_MMD_PCS)
  70. #define MDIO_MMDREG_DEVS0_PMAPMD DEV_PRESENT_BIT(MDIO_MMD_PMAPMD)
  71. /* Bits in MMDREG_STAT2 */
  72. #define MDIO_MMDREG_STAT2_PRESENT_VAL (2)
  73. #define MDIO_MMDREG_STAT2_PRESENT_LBN (14)
  74. #define MDIO_MMDREG_STAT2_PRESENT_WIDTH (2)
  75. /* Bits in MMDREG_TXDIS */
  76. #define MDIO_MMDREG_TXDIS_GLOBAL_LBN (0)
  77. #define MDIO_MMDREG_TXDIS_GLOBAL_WIDTH (1)
  78. /* MMD-specific bits, ordered by MMD, then register */
  79. #define MDIO_PMAPMD_CTRL1_LBACK_LBN (0)
  80. #define MDIO_PMAPMD_CTRL1_LBACK_WIDTH (1)
  81. /* PMA type (4 bits) */
  82. #define MDIO_PMAPMD_CTRL2_10G_CX4 (0x0)
  83. #define MDIO_PMAPMD_CTRL2_10G_EW (0x1)
  84. #define MDIO_PMAPMD_CTRL2_10G_LW (0x2)
  85. #define MDIO_PMAPMD_CTRL2_10G_SW (0x3)
  86. #define MDIO_PMAPMD_CTRL2_10G_LX4 (0x4)
  87. #define MDIO_PMAPMD_CTRL2_10G_ER (0x5)
  88. #define MDIO_PMAPMD_CTRL2_10G_LR (0x6)
  89. #define MDIO_PMAPMD_CTRL2_10G_SR (0x7)
  90. /* Reserved */
  91. #define MDIO_PMAPMD_CTRL2_10G_BT (0x9)
  92. /* Reserved */
  93. /* Reserved */
  94. #define MDIO_PMAPMD_CTRL2_1G_BT (0xc)
  95. /* Reserved */
  96. #define MDIO_PMAPMD_CTRL2_100_BT (0xe)
  97. #define MDIO_PMAPMD_CTRL2_10_BT (0xf)
  98. #define MDIO_PMAPMD_CTRL2_TYPE_MASK (0xf)
  99. /* PHY XGXS lane state */
  100. #define MDIO_PHYXS_LANE_STATE (0x18)
  101. #define MDIO_PHYXS_LANE_ALIGNED_LBN (12)
  102. /* AN registers */
  103. #define MDIO_AN_STATUS (1)
  104. #define MDIO_AN_STATUS_XNP_LBN (7)
  105. #define MDIO_AN_STATUS_PAGE_LBN (6)
  106. #define MDIO_AN_STATUS_AN_DONE_LBN (5)
  107. #define MDIO_AN_STATUS_LP_AN_CAP_LBN (0)
  108. #define MDIO_AN_10GBT_STATUS (33)
  109. #define MDIO_AN_10GBT_STATUS_MS_FLT_LBN (15) /* MASTER/SLAVE config fault */
  110. #define MDIO_AN_10GBT_STATUS_MS_LBN (14) /* MASTER/SLAVE config */
  111. #define MDIO_AN_10GBT_STATUS_LOC_OK_LBN (13) /* Local OK */
  112. #define MDIO_AN_10GBT_STATUS_REM_OK_LBN (12) /* Remote OK */
  113. #define MDIO_AN_10GBT_STATUS_LP_10G_LBN (11) /* Link partner is 10GBT capable */
  114. #define MDIO_AN_10GBT_STATUS_LP_LTA_LBN (10) /* LP loop timing ability */
  115. #define MDIO_AN_10GBT_STATUS_LP_TRR_LBN (9) /* LP Training Reset Request */
  116. /* Packing of the prt and dev arguments of clause 45 style MDIO into a
  117. * single int so they can be passed into the mdio_read/write functions
  118. * that currently exist. Note that as Falcon is the only current user,
  119. * the packed form is chosen to match what Falcon needs to write into
  120. * a register. This is checked at compile-time so do not change it. If
  121. * your target chip needs things layed out differently you will need
  122. * to unpack the arguments in your chip-specific mdio functions.
  123. */
  124. /* These are defined by the standard. */
  125. #define MDIO45_PRT_ID_WIDTH (5)
  126. #define MDIO45_DEV_ID_WIDTH (5)
  127. /* The prt ID is just packed in immediately to the left of the dev ID */
  128. #define MDIO45_PRT_DEV_WIDTH (MDIO45_PRT_ID_WIDTH + MDIO45_DEV_ID_WIDTH)
  129. #define MDIO45_PRT_ID_MASK ((1 << MDIO45_PRT_DEV_WIDTH) - 1)
  130. /* This is the prt + dev extended by 1 bit to hold the 'is clause 45' flag. */
  131. #define MDIO45_XPRT_ID_WIDTH (MDIO45_PRT_DEV_WIDTH + 1)
  132. #define MDIO45_XPRT_ID_MASK ((1 << MDIO45_XPRT_ID_WIDTH) - 1)
  133. #define MDIO45_XPRT_ID_IS10G (1 << (MDIO45_XPRT_ID_WIDTH - 1))
  134. #define MDIO45_PRT_ID_COMP_LBN MDIO45_DEV_ID_WIDTH
  135. #define MDIO45_PRT_ID_COMP_WIDTH MDIO45_PRT_ID_WIDTH
  136. #define MDIO45_DEV_ID_COMP_LBN 0
  137. #define MDIO45_DEV_ID_COMP_WIDTH MDIO45_DEV_ID_WIDTH
  138. /* Compose port and device into a phy_id */
  139. static inline int mdio_clause45_pack(u8 prt, u8 dev)
  140. {
  141. efx_dword_t phy_id;
  142. EFX_POPULATE_DWORD_2(phy_id, MDIO45_PRT_ID_COMP, prt,
  143. MDIO45_DEV_ID_COMP, dev);
  144. return MDIO45_XPRT_ID_IS10G | EFX_DWORD_VAL(phy_id);
  145. }
  146. static inline void mdio_clause45_unpack(u32 val, u8 *prt, u8 *dev)
  147. {
  148. efx_dword_t phy_id;
  149. EFX_POPULATE_DWORD_1(phy_id, EFX_DWORD_0, val);
  150. *prt = EFX_DWORD_FIELD(phy_id, MDIO45_PRT_ID_COMP);
  151. *dev = EFX_DWORD_FIELD(phy_id, MDIO45_DEV_ID_COMP);
  152. }
  153. static inline int mdio_clause45_read(struct efx_nic *efx,
  154. u8 prt, u8 dev, u16 addr)
  155. {
  156. return efx->mii.mdio_read(efx->net_dev,
  157. mdio_clause45_pack(prt, dev), addr);
  158. }
  159. static inline void mdio_clause45_write(struct efx_nic *efx,
  160. u8 prt, u8 dev, u16 addr, int value)
  161. {
  162. efx->mii.mdio_write(efx->net_dev,
  163. mdio_clause45_pack(prt, dev), addr, value);
  164. }
  165. static inline u32 mdio_clause45_read_id(struct efx_nic *efx, int mmd)
  166. {
  167. int phy_id = efx->mii.phy_id;
  168. u16 id_low = mdio_clause45_read(efx, phy_id, mmd, MDIO_MMDREG_IDLOW);
  169. u16 id_hi = mdio_clause45_read(efx, phy_id, mmd, MDIO_MMDREG_IDHI);
  170. return (id_hi << 16) | (id_low);
  171. }
  172. static inline bool mdio_clause45_phyxgxs_lane_sync(struct efx_nic *efx)
  173. {
  174. int i, lane_status;
  175. bool sync;
  176. for (i = 0; i < 2; ++i)
  177. lane_status = mdio_clause45_read(efx, efx->mii.phy_id,
  178. MDIO_MMD_PHYXS,
  179. MDIO_PHYXS_LANE_STATE);
  180. sync = !!(lane_status & (1 << MDIO_PHYXS_LANE_ALIGNED_LBN));
  181. if (!sync)
  182. EFX_LOG(efx, "XGXS lane status: %x\n", lane_status);
  183. return sync;
  184. }
  185. extern const char *mdio_clause45_mmd_name(int mmd);
  186. /*
  187. * Reset a specific MMD and wait for reset to clear.
  188. * Return number of spins left (>0) on success, -%ETIMEDOUT on failure.
  189. *
  190. * This function will sleep
  191. */
  192. extern int mdio_clause45_reset_mmd(struct efx_nic *efx, int mmd,
  193. int spins, int spintime);
  194. /* As mdio_clause45_check_mmd but for multiple MMDs */
  195. int mdio_clause45_check_mmds(struct efx_nic *efx,
  196. unsigned int mmd_mask, unsigned int fatal_mask);
  197. /* Check the link status of specified mmds in bit mask */
  198. extern bool mdio_clause45_links_ok(struct efx_nic *efx,
  199. unsigned int mmd_mask);
  200. /* Generic transmit disable support though PMAPMD */
  201. extern void mdio_clause45_transmit_disable(struct efx_nic *efx);
  202. /* Generic part of reconfigure: set/clear loopback bits */
  203. extern void mdio_clause45_phy_reconfigure(struct efx_nic *efx);
  204. /* Read (some of) the PHY settings over MDIO */
  205. extern void mdio_clause45_get_settings(struct efx_nic *efx,
  206. struct ethtool_cmd *ecmd);
  207. /* Set (some of) the PHY settings over MDIO */
  208. extern int mdio_clause45_set_settings(struct efx_nic *efx,
  209. struct ethtool_cmd *ecmd);
  210. /* Wait for specified MMDs to exit reset within a timeout */
  211. extern int mdio_clause45_wait_reset_mmds(struct efx_nic *efx,
  212. unsigned int mmd_mask);
  213. #endif /* EFX_MDIO_10G_H */