703-09-v5.17-net-phylink-add-generic-validate-implementation.patch 12 KB

123456789101112131415161718192021222324252627282930313233343536373839404142434445464748495051525354555657585960616263646566676869707172737475767778798081828384858687888990919293949596979899100101102103104105106107108109110111112113114115116117118119120121122123124125126127128129130131132133134135136137138139140141142143144145146147148149150151152153154155156157158159160161162163164165166167168169170171172173174175176177178179180181182183184185186187188189190191192193194195196197198199200201202203204205206207208209210211212213214215216217218219220221222223224225226227228229230231232233234235236237238239240241242243244245246247248249250251252253254255256257258259260261262263264265266267268269270271272273274275276277278279280281282283284285286287288289290291292293294295296297298299300301302303304305306307308309310311312313314315316317318319320321322323324325326327328329330331332333334335336337338339340341
  1. From 34ae2c09d46a2d0abd907e139b466f798e4095a8 Mon Sep 17 00:00:00 2001
  2. From: "Russell King (Oracle)" <[email protected]>
  3. Date: Mon, 15 Nov 2021 10:00:27 +0000
  4. Subject: [PATCH] net: phylink: add generic validate implementation
  5. Add a generic validate() implementation using the supported_interfaces
  6. and a bitmask of MAC pause/speed/duplex capabilities. This allows us
  7. to entirely eliminate many driver private validate() implementations.
  8. We expose the underlying phylink_get_linkmodes() function so that
  9. drivers which have special needs can still benefit from conversion.
  10. Signed-off-by: Russell King (Oracle) <[email protected]>
  11. Signed-off-by: David S. Miller <[email protected]>
  12. ---
  13. drivers/net/phy/phylink.c | 252 ++++++++++++++++++++++++++++++++++++++
  14. include/linux/phylink.h | 31 +++++
  15. 2 files changed, 283 insertions(+)
  16. --- a/drivers/net/phy/phylink.c
  17. +++ b/drivers/net/phy/phylink.c
  18. @@ -172,6 +172,258 @@ static int phylink_validate_mac_and_pcs(
  19. return phylink_is_empty_linkmode(supported) ? -EINVAL : 0;
  20. }
  21. +static void phylink_caps_to_linkmodes(unsigned long *linkmodes,
  22. + unsigned long caps)
  23. +{
  24. + if (caps & MAC_SYM_PAUSE)
  25. + __set_bit(ETHTOOL_LINK_MODE_Pause_BIT, linkmodes);
  26. +
  27. + if (caps & MAC_ASYM_PAUSE)
  28. + __set_bit(ETHTOOL_LINK_MODE_Asym_Pause_BIT, linkmodes);
  29. +
  30. + if (caps & MAC_10HD)
  31. + __set_bit(ETHTOOL_LINK_MODE_10baseT_Half_BIT, linkmodes);
  32. +
  33. + if (caps & MAC_10FD)
  34. + __set_bit(ETHTOOL_LINK_MODE_10baseT_Full_BIT, linkmodes);
  35. +
  36. + if (caps & MAC_100HD) {
  37. + __set_bit(ETHTOOL_LINK_MODE_100baseT_Half_BIT, linkmodes);
  38. + __set_bit(ETHTOOL_LINK_MODE_100baseFX_Half_BIT, linkmodes);
  39. + }
  40. +
  41. + if (caps & MAC_100FD) {
  42. + __set_bit(ETHTOOL_LINK_MODE_100baseT_Full_BIT, linkmodes);
  43. + __set_bit(ETHTOOL_LINK_MODE_100baseT1_Full_BIT, linkmodes);
  44. + __set_bit(ETHTOOL_LINK_MODE_100baseFX_Full_BIT, linkmodes);
  45. + }
  46. +
  47. + if (caps & MAC_1000HD)
  48. + __set_bit(ETHTOOL_LINK_MODE_1000baseT_Half_BIT, linkmodes);
  49. +
  50. + if (caps & MAC_1000FD) {
  51. + __set_bit(ETHTOOL_LINK_MODE_1000baseT_Full_BIT, linkmodes);
  52. + __set_bit(ETHTOOL_LINK_MODE_1000baseX_Full_BIT, linkmodes);
  53. + __set_bit(ETHTOOL_LINK_MODE_1000baseT1_Full_BIT, linkmodes);
  54. + }
  55. +
  56. + if (caps & MAC_2500FD) {
  57. + __set_bit(ETHTOOL_LINK_MODE_2500baseT_Full_BIT, linkmodes);
  58. + __set_bit(ETHTOOL_LINK_MODE_2500baseX_Full_BIT, linkmodes);
  59. + }
  60. +
  61. + if (caps & MAC_5000FD)
  62. + __set_bit(ETHTOOL_LINK_MODE_5000baseT_Full_BIT, linkmodes);
  63. +
  64. + if (caps & MAC_10000FD) {
  65. + __set_bit(ETHTOOL_LINK_MODE_10000baseT_Full_BIT, linkmodes);
  66. + __set_bit(ETHTOOL_LINK_MODE_10000baseKX4_Full_BIT, linkmodes);
  67. + __set_bit(ETHTOOL_LINK_MODE_10000baseKR_Full_BIT, linkmodes);
  68. + __set_bit(ETHTOOL_LINK_MODE_10000baseR_FEC_BIT, linkmodes);
  69. + __set_bit(ETHTOOL_LINK_MODE_10000baseCR_Full_BIT, linkmodes);
  70. + __set_bit(ETHTOOL_LINK_MODE_10000baseSR_Full_BIT, linkmodes);
  71. + __set_bit(ETHTOOL_LINK_MODE_10000baseLR_Full_BIT, linkmodes);
  72. + __set_bit(ETHTOOL_LINK_MODE_10000baseLRM_Full_BIT, linkmodes);
  73. + __set_bit(ETHTOOL_LINK_MODE_10000baseER_Full_BIT, linkmodes);
  74. + }
  75. +
  76. + if (caps & MAC_25000FD) {
  77. + __set_bit(ETHTOOL_LINK_MODE_25000baseCR_Full_BIT, linkmodes);
  78. + __set_bit(ETHTOOL_LINK_MODE_25000baseKR_Full_BIT, linkmodes);
  79. + __set_bit(ETHTOOL_LINK_MODE_25000baseSR_Full_BIT, linkmodes);
  80. + }
  81. +
  82. + if (caps & MAC_40000FD) {
  83. + __set_bit(ETHTOOL_LINK_MODE_40000baseKR4_Full_BIT, linkmodes);
  84. + __set_bit(ETHTOOL_LINK_MODE_40000baseCR4_Full_BIT, linkmodes);
  85. + __set_bit(ETHTOOL_LINK_MODE_40000baseSR4_Full_BIT, linkmodes);
  86. + __set_bit(ETHTOOL_LINK_MODE_40000baseLR4_Full_BIT, linkmodes);
  87. + }
  88. +
  89. + if (caps & MAC_50000FD) {
  90. + __set_bit(ETHTOOL_LINK_MODE_50000baseCR2_Full_BIT, linkmodes);
  91. + __set_bit(ETHTOOL_LINK_MODE_50000baseKR2_Full_BIT, linkmodes);
  92. + __set_bit(ETHTOOL_LINK_MODE_50000baseSR2_Full_BIT, linkmodes);
  93. + __set_bit(ETHTOOL_LINK_MODE_50000baseKR_Full_BIT, linkmodes);
  94. + __set_bit(ETHTOOL_LINK_MODE_50000baseSR_Full_BIT, linkmodes);
  95. + __set_bit(ETHTOOL_LINK_MODE_50000baseCR_Full_BIT, linkmodes);
  96. + __set_bit(ETHTOOL_LINK_MODE_50000baseLR_ER_FR_Full_BIT,
  97. + linkmodes);
  98. + __set_bit(ETHTOOL_LINK_MODE_50000baseDR_Full_BIT, linkmodes);
  99. + }
  100. +
  101. + if (caps & MAC_56000FD) {
  102. + __set_bit(ETHTOOL_LINK_MODE_56000baseKR4_Full_BIT, linkmodes);
  103. + __set_bit(ETHTOOL_LINK_MODE_56000baseCR4_Full_BIT, linkmodes);
  104. + __set_bit(ETHTOOL_LINK_MODE_56000baseSR4_Full_BIT, linkmodes);
  105. + __set_bit(ETHTOOL_LINK_MODE_56000baseLR4_Full_BIT, linkmodes);
  106. + }
  107. +
  108. + if (caps & MAC_100000FD) {
  109. + __set_bit(ETHTOOL_LINK_MODE_100000baseKR4_Full_BIT, linkmodes);
  110. + __set_bit(ETHTOOL_LINK_MODE_100000baseSR4_Full_BIT, linkmodes);
  111. + __set_bit(ETHTOOL_LINK_MODE_100000baseCR4_Full_BIT, linkmodes);
  112. + __set_bit(ETHTOOL_LINK_MODE_100000baseLR4_ER4_Full_BIT,
  113. + linkmodes);
  114. + __set_bit(ETHTOOL_LINK_MODE_100000baseKR2_Full_BIT, linkmodes);
  115. + __set_bit(ETHTOOL_LINK_MODE_100000baseSR2_Full_BIT, linkmodes);
  116. + __set_bit(ETHTOOL_LINK_MODE_100000baseCR2_Full_BIT, linkmodes);
  117. + __set_bit(ETHTOOL_LINK_MODE_100000baseLR2_ER2_FR2_Full_BIT,
  118. + linkmodes);
  119. + __set_bit(ETHTOOL_LINK_MODE_100000baseDR2_Full_BIT, linkmodes);
  120. + __set_bit(ETHTOOL_LINK_MODE_100000baseKR_Full_BIT, linkmodes);
  121. + __set_bit(ETHTOOL_LINK_MODE_100000baseSR_Full_BIT, linkmodes);
  122. + __set_bit(ETHTOOL_LINK_MODE_100000baseLR_ER_FR_Full_BIT,
  123. + linkmodes);
  124. + __set_bit(ETHTOOL_LINK_MODE_100000baseCR_Full_BIT, linkmodes);
  125. + __set_bit(ETHTOOL_LINK_MODE_100000baseDR_Full_BIT, linkmodes);
  126. + }
  127. +
  128. + if (caps & MAC_200000FD) {
  129. + __set_bit(ETHTOOL_LINK_MODE_200000baseKR4_Full_BIT, linkmodes);
  130. + __set_bit(ETHTOOL_LINK_MODE_200000baseSR4_Full_BIT, linkmodes);
  131. + __set_bit(ETHTOOL_LINK_MODE_200000baseLR4_ER4_FR4_Full_BIT,
  132. + linkmodes);
  133. + __set_bit(ETHTOOL_LINK_MODE_200000baseDR4_Full_BIT, linkmodes);
  134. + __set_bit(ETHTOOL_LINK_MODE_200000baseCR4_Full_BIT, linkmodes);
  135. + __set_bit(ETHTOOL_LINK_MODE_200000baseKR2_Full_BIT, linkmodes);
  136. + __set_bit(ETHTOOL_LINK_MODE_200000baseSR2_Full_BIT, linkmodes);
  137. + __set_bit(ETHTOOL_LINK_MODE_200000baseLR2_ER2_FR2_Full_BIT,
  138. + linkmodes);
  139. + __set_bit(ETHTOOL_LINK_MODE_200000baseDR2_Full_BIT, linkmodes);
  140. + __set_bit(ETHTOOL_LINK_MODE_200000baseCR2_Full_BIT, linkmodes);
  141. + }
  142. +
  143. + if (caps & MAC_400000FD) {
  144. + __set_bit(ETHTOOL_LINK_MODE_400000baseKR8_Full_BIT, linkmodes);
  145. + __set_bit(ETHTOOL_LINK_MODE_400000baseSR8_Full_BIT, linkmodes);
  146. + __set_bit(ETHTOOL_LINK_MODE_400000baseLR8_ER8_FR8_Full_BIT,
  147. + linkmodes);
  148. + __set_bit(ETHTOOL_LINK_MODE_400000baseDR8_Full_BIT, linkmodes);
  149. + __set_bit(ETHTOOL_LINK_MODE_400000baseCR8_Full_BIT, linkmodes);
  150. + __set_bit(ETHTOOL_LINK_MODE_400000baseKR4_Full_BIT, linkmodes);
  151. + __set_bit(ETHTOOL_LINK_MODE_400000baseSR4_Full_BIT, linkmodes);
  152. + __set_bit(ETHTOOL_LINK_MODE_400000baseLR4_ER4_FR4_Full_BIT,
  153. + linkmodes);
  154. + __set_bit(ETHTOOL_LINK_MODE_400000baseDR4_Full_BIT, linkmodes);
  155. + __set_bit(ETHTOOL_LINK_MODE_400000baseCR4_Full_BIT, linkmodes);
  156. + }
  157. +}
  158. +
  159. +/**
  160. + * phylink_get_linkmodes() - get acceptable link modes
  161. + * @linkmodes: ethtool linkmode mask (must be already initialised)
  162. + * @interface: phy interface mode defined by &typedef phy_interface_t
  163. + * @mac_capabilities: bitmask of MAC capabilities
  164. + *
  165. + * Set all possible pause, speed and duplex linkmodes in @linkmodes that
  166. + * are supported by the @interface mode and @mac_capabilities. @linkmodes
  167. + * must have been initialised previously.
  168. + */
  169. +void phylink_get_linkmodes(unsigned long *linkmodes, phy_interface_t interface,
  170. + unsigned long mac_capabilities)
  171. +{
  172. + unsigned long caps = MAC_SYM_PAUSE | MAC_ASYM_PAUSE;
  173. +
  174. + switch (interface) {
  175. + case PHY_INTERFACE_MODE_USXGMII:
  176. + caps |= MAC_10000FD | MAC_5000FD | MAC_2500FD;
  177. + fallthrough;
  178. +
  179. + case PHY_INTERFACE_MODE_RGMII_TXID:
  180. + case PHY_INTERFACE_MODE_RGMII_RXID:
  181. + case PHY_INTERFACE_MODE_RGMII_ID:
  182. + case PHY_INTERFACE_MODE_RGMII:
  183. + case PHY_INTERFACE_MODE_QSGMII:
  184. + case PHY_INTERFACE_MODE_SGMII:
  185. + case PHY_INTERFACE_MODE_GMII:
  186. + caps |= MAC_1000HD | MAC_1000FD;
  187. + fallthrough;
  188. +
  189. + case PHY_INTERFACE_MODE_REVRMII:
  190. + case PHY_INTERFACE_MODE_RMII:
  191. + case PHY_INTERFACE_MODE_REVMII:
  192. + case PHY_INTERFACE_MODE_MII:
  193. + caps |= MAC_10HD | MAC_10FD;
  194. + fallthrough;
  195. +
  196. + case PHY_INTERFACE_MODE_100BASEX:
  197. + caps |= MAC_100HD | MAC_100FD;
  198. + break;
  199. +
  200. + case PHY_INTERFACE_MODE_TBI:
  201. + case PHY_INTERFACE_MODE_MOCA:
  202. + case PHY_INTERFACE_MODE_RTBI:
  203. + case PHY_INTERFACE_MODE_1000BASEX:
  204. + caps |= MAC_1000HD;
  205. + fallthrough;
  206. + case PHY_INTERFACE_MODE_TRGMII:
  207. + caps |= MAC_1000FD;
  208. + break;
  209. +
  210. + case PHY_INTERFACE_MODE_2500BASEX:
  211. + caps |= MAC_2500FD;
  212. + break;
  213. +
  214. + case PHY_INTERFACE_MODE_5GBASER:
  215. + caps |= MAC_5000FD;
  216. + break;
  217. +
  218. + case PHY_INTERFACE_MODE_XGMII:
  219. + case PHY_INTERFACE_MODE_RXAUI:
  220. + case PHY_INTERFACE_MODE_XAUI:
  221. + case PHY_INTERFACE_MODE_10GBASER:
  222. + case PHY_INTERFACE_MODE_10GKR:
  223. + caps |= MAC_10000FD;
  224. + break;
  225. +
  226. + case PHY_INTERFACE_MODE_25GBASER:
  227. + caps |= MAC_25000FD;
  228. + break;
  229. +
  230. + case PHY_INTERFACE_MODE_XLGMII:
  231. + caps |= MAC_40000FD;
  232. + break;
  233. +
  234. + case PHY_INTERFACE_MODE_INTERNAL:
  235. + caps |= ~0;
  236. + break;
  237. +
  238. + case PHY_INTERFACE_MODE_NA:
  239. + case PHY_INTERFACE_MODE_MAX:
  240. + case PHY_INTERFACE_MODE_SMII:
  241. + break;
  242. + }
  243. +
  244. + phylink_caps_to_linkmodes(linkmodes, caps & mac_capabilities);
  245. +}
  246. +EXPORT_SYMBOL_GPL(phylink_get_linkmodes);
  247. +
  248. +/**
  249. + * phylink_generic_validate() - generic validate() callback implementation
  250. + * @config: a pointer to a &struct phylink_config.
  251. + * @supported: ethtool bitmask for supported link modes.
  252. + * @state: a pointer to a &struct phylink_link_state.
  253. + *
  254. + * Generic implementation of the validate() callback that MAC drivers can
  255. + * use when they pass the range of supported interfaces and MAC capabilities.
  256. + * This makes use of phylink_get_linkmodes().
  257. + */
  258. +void phylink_generic_validate(struct phylink_config *config,
  259. + unsigned long *supported,
  260. + struct phylink_link_state *state)
  261. +{
  262. + __ETHTOOL_DECLARE_LINK_MODE_MASK(mask) = { 0, };
  263. +
  264. + phylink_set_port_modes(mask);
  265. + phylink_set(mask, Autoneg);
  266. + phylink_get_linkmodes(mask, state->interface, config->mac_capabilities);
  267. +
  268. + linkmode_and(supported, supported, mask);
  269. + linkmode_and(state->advertising, state->advertising, mask);
  270. +}
  271. +EXPORT_SYMBOL_GPL(phylink_generic_validate);
  272. +
  273. static int phylink_validate_any(struct phylink *pl, unsigned long *supported,
  274. struct phylink_link_state *state)
  275. {
  276. --- a/include/linux/phylink.h
  277. +++ b/include/linux/phylink.h
  278. @@ -20,6 +20,29 @@ enum {
  279. MLO_AN_PHY = 0, /* Conventional PHY */
  280. MLO_AN_FIXED, /* Fixed-link mode */
  281. MLO_AN_INBAND, /* In-band protocol */
  282. +
  283. + MAC_SYM_PAUSE = BIT(0),
  284. + MAC_ASYM_PAUSE = BIT(1),
  285. + MAC_10HD = BIT(2),
  286. + MAC_10FD = BIT(3),
  287. + MAC_10 = MAC_10HD | MAC_10FD,
  288. + MAC_100HD = BIT(4),
  289. + MAC_100FD = BIT(5),
  290. + MAC_100 = MAC_100HD | MAC_100FD,
  291. + MAC_1000HD = BIT(6),
  292. + MAC_1000FD = BIT(7),
  293. + MAC_1000 = MAC_1000HD | MAC_1000FD,
  294. + MAC_2500FD = BIT(8),
  295. + MAC_5000FD = BIT(9),
  296. + MAC_10000FD = BIT(10),
  297. + MAC_20000FD = BIT(11),
  298. + MAC_25000FD = BIT(12),
  299. + MAC_40000FD = BIT(13),
  300. + MAC_50000FD = BIT(14),
  301. + MAC_56000FD = BIT(15),
  302. + MAC_100000FD = BIT(16),
  303. + MAC_200000FD = BIT(17),
  304. + MAC_400000FD = BIT(18),
  305. };
  306. static inline bool phylink_autoneg_inband(unsigned int mode)
  307. @@ -70,6 +93,7 @@ enum phylink_op_type {
  308. * if MAC link is at %MLO_AN_FIXED mode.
  309. * @supported_interfaces: bitmap describing which PHY_INTERFACE_MODE_xxx
  310. * are supported by the MAC/PCS.
  311. + * @mac_capabilities: MAC pause/speed/duplex capabilities.
  312. */
  313. struct phylink_config {
  314. struct device *dev;
  315. @@ -81,6 +105,7 @@ struct phylink_config {
  316. void (*get_fixed_state)(struct phylink_config *config,
  317. struct phylink_link_state *state);
  318. DECLARE_PHY_INTERFACE_MASK(supported_interfaces);
  319. + unsigned long mac_capabilities;
  320. };
  321. /**
  322. @@ -462,6 +487,12 @@ void pcs_link_up(struct phylink_pcs *pcs
  323. phy_interface_t interface, int speed, int duplex);
  324. #endif
  325. +void phylink_get_linkmodes(unsigned long *linkmodes, phy_interface_t interface,
  326. + unsigned long mac_capabilities);
  327. +void phylink_generic_validate(struct phylink_config *config,
  328. + unsigned long *supported,
  329. + struct phylink_link_state *state);
  330. +
  331. struct phylink *phylink_create(struct phylink_config *, struct fwnode_handle *,
  332. phy_interface_t iface,
  333. const struct phylink_mac_ops *mac_ops);