1*6285aa6aSThomas Monjalon /* SPDX-License-Identifier: BSD-3-Clause
2*6285aa6aSThomas Monjalon * Copyright (c) 2024 NVIDIA Corporation & Affiliates
3*6285aa6aSThomas Monjalon */
4*6285aa6aSThomas Monjalon
5*6285aa6aSThomas Monjalon #include <rte_bitops.h>
6*6285aa6aSThomas Monjalon
7*6285aa6aSThomas Monjalon #include "rte_ethdev.h"
8*6285aa6aSThomas Monjalon #include "ethdev_linux_ethtool.h"
9*6285aa6aSThomas Monjalon
10*6285aa6aSThomas Monjalon /* Link modes sorted with index as defined in ethtool.
11*6285aa6aSThomas Monjalon * Values are speed in Mbps with LSB indicating duplex.
12*6285aa6aSThomas Monjalon *
13*6285aa6aSThomas Monjalon * The ethtool bits definition should not change as it is a kernel API.
14*6285aa6aSThomas Monjalon * Using raw numbers directly avoids checking API availability
15*6285aa6aSThomas Monjalon * and allows to compile with new bits included even on an old kernel.
16*6285aa6aSThomas Monjalon *
17*6285aa6aSThomas Monjalon * The array below is built from bit definitions with this shell command:
18*6285aa6aSThomas Monjalon * sed -rn 's;.*(ETHTOOL_LINK_MODE_)([0-9]+)([0-9a-zA-Z_]*).*= *([0-9]*).*;'\
19*6285aa6aSThomas Monjalon * '[\4] = \2, /\* \1\2\3 *\/;p' /usr/include/linux/ethtool.h |
20*6285aa6aSThomas Monjalon * awk '/_Half_/{$3=$3+1","}1'
21*6285aa6aSThomas Monjalon */
22*6285aa6aSThomas Monjalon static const uint32_t link_modes[] = {
23*6285aa6aSThomas Monjalon [0] = 11, /* ETHTOOL_LINK_MODE_10baseT_Half_BIT */
24*6285aa6aSThomas Monjalon [1] = 10, /* ETHTOOL_LINK_MODE_10baseT_Full_BIT */
25*6285aa6aSThomas Monjalon [2] = 101, /* ETHTOOL_LINK_MODE_100baseT_Half_BIT */
26*6285aa6aSThomas Monjalon [3] = 100, /* ETHTOOL_LINK_MODE_100baseT_Full_BIT */
27*6285aa6aSThomas Monjalon [4] = 1001, /* ETHTOOL_LINK_MODE_1000baseT_Half_BIT */
28*6285aa6aSThomas Monjalon [5] = 1000, /* ETHTOOL_LINK_MODE_1000baseT_Full_BIT */
29*6285aa6aSThomas Monjalon [12] = 10000, /* ETHTOOL_LINK_MODE_10000baseT_Full_BIT */
30*6285aa6aSThomas Monjalon [15] = 2500, /* ETHTOOL_LINK_MODE_2500baseX_Full_BIT */
31*6285aa6aSThomas Monjalon [17] = 1000, /* ETHTOOL_LINK_MODE_1000baseKX_Full_BIT */
32*6285aa6aSThomas Monjalon [18] = 10000, /* ETHTOOL_LINK_MODE_10000baseKX4_Full_BIT */
33*6285aa6aSThomas Monjalon [19] = 10000, /* ETHTOOL_LINK_MODE_10000baseKR_Full_BIT */
34*6285aa6aSThomas Monjalon [20] = 10000, /* ETHTOOL_LINK_MODE_10000baseR_FEC_BIT */
35*6285aa6aSThomas Monjalon [21] = 20000, /* ETHTOOL_LINK_MODE_20000baseMLD2_Full_BIT */
36*6285aa6aSThomas Monjalon [22] = 20000, /* ETHTOOL_LINK_MODE_20000baseKR2_Full_BIT */
37*6285aa6aSThomas Monjalon [23] = 40000, /* ETHTOOL_LINK_MODE_40000baseKR4_Full_BIT */
38*6285aa6aSThomas Monjalon [24] = 40000, /* ETHTOOL_LINK_MODE_40000baseCR4_Full_BIT */
39*6285aa6aSThomas Monjalon [25] = 40000, /* ETHTOOL_LINK_MODE_40000baseSR4_Full_BIT */
40*6285aa6aSThomas Monjalon [26] = 40000, /* ETHTOOL_LINK_MODE_40000baseLR4_Full_BIT */
41*6285aa6aSThomas Monjalon [27] = 56000, /* ETHTOOL_LINK_MODE_56000baseKR4_Full_BIT */
42*6285aa6aSThomas Monjalon [28] = 56000, /* ETHTOOL_LINK_MODE_56000baseCR4_Full_BIT */
43*6285aa6aSThomas Monjalon [29] = 56000, /* ETHTOOL_LINK_MODE_56000baseSR4_Full_BIT */
44*6285aa6aSThomas Monjalon [30] = 56000, /* ETHTOOL_LINK_MODE_56000baseLR4_Full_BIT */
45*6285aa6aSThomas Monjalon [31] = 25000, /* ETHTOOL_LINK_MODE_25000baseCR_Full_BIT */
46*6285aa6aSThomas Monjalon [32] = 25000, /* ETHTOOL_LINK_MODE_25000baseKR_Full_BIT */
47*6285aa6aSThomas Monjalon [33] = 25000, /* ETHTOOL_LINK_MODE_25000baseSR_Full_BIT */
48*6285aa6aSThomas Monjalon [34] = 50000, /* ETHTOOL_LINK_MODE_50000baseCR2_Full_BIT */
49*6285aa6aSThomas Monjalon [35] = 50000, /* ETHTOOL_LINK_MODE_50000baseKR2_Full_BIT */
50*6285aa6aSThomas Monjalon [36] = 100000, /* ETHTOOL_LINK_MODE_100000baseKR4_Full_BIT */
51*6285aa6aSThomas Monjalon [37] = 100000, /* ETHTOOL_LINK_MODE_100000baseSR4_Full_BIT */
52*6285aa6aSThomas Monjalon [38] = 100000, /* ETHTOOL_LINK_MODE_100000baseCR4_Full_BIT */
53*6285aa6aSThomas Monjalon [39] = 100000, /* ETHTOOL_LINK_MODE_100000baseLR4_ER4_Full_BIT */
54*6285aa6aSThomas Monjalon [40] = 50000, /* ETHTOOL_LINK_MODE_50000baseSR2_Full_BIT */
55*6285aa6aSThomas Monjalon [41] = 1000, /* ETHTOOL_LINK_MODE_1000baseX_Full_BIT */
56*6285aa6aSThomas Monjalon [42] = 10000, /* ETHTOOL_LINK_MODE_10000baseCR_Full_BIT */
57*6285aa6aSThomas Monjalon [43] = 10000, /* ETHTOOL_LINK_MODE_10000baseSR_Full_BIT */
58*6285aa6aSThomas Monjalon [44] = 10000, /* ETHTOOL_LINK_MODE_10000baseLR_Full_BIT */
59*6285aa6aSThomas Monjalon [45] = 10000, /* ETHTOOL_LINK_MODE_10000baseLRM_Full_BIT */
60*6285aa6aSThomas Monjalon [46] = 10000, /* ETHTOOL_LINK_MODE_10000baseER_Full_BIT */
61*6285aa6aSThomas Monjalon [47] = 2500, /* ETHTOOL_LINK_MODE_2500baseT_Full_BIT */
62*6285aa6aSThomas Monjalon [48] = 5000, /* ETHTOOL_LINK_MODE_5000baseT_Full_BIT */
63*6285aa6aSThomas Monjalon [52] = 50000, /* ETHTOOL_LINK_MODE_50000baseKR_Full_BIT */
64*6285aa6aSThomas Monjalon [53] = 50000, /* ETHTOOL_LINK_MODE_50000baseSR_Full_BIT */
65*6285aa6aSThomas Monjalon [54] = 50000, /* ETHTOOL_LINK_MODE_50000baseCR_Full_BIT */
66*6285aa6aSThomas Monjalon [55] = 50000, /* ETHTOOL_LINK_MODE_50000baseLR_ER_FR_Full_BIT */
67*6285aa6aSThomas Monjalon [56] = 50000, /* ETHTOOL_LINK_MODE_50000baseDR_Full_BIT */
68*6285aa6aSThomas Monjalon [57] = 100000, /* ETHTOOL_LINK_MODE_100000baseKR2_Full_BIT */
69*6285aa6aSThomas Monjalon [58] = 100000, /* ETHTOOL_LINK_MODE_100000baseSR2_Full_BIT */
70*6285aa6aSThomas Monjalon [59] = 100000, /* ETHTOOL_LINK_MODE_100000baseCR2_Full_BIT */
71*6285aa6aSThomas Monjalon [60] = 100000, /* ETHTOOL_LINK_MODE_100000baseLR2_ER2_FR2_Full_BIT */
72*6285aa6aSThomas Monjalon [61] = 100000, /* ETHTOOL_LINK_MODE_100000baseDR2_Full_BIT */
73*6285aa6aSThomas Monjalon [62] = 200000, /* ETHTOOL_LINK_MODE_200000baseKR4_Full_BIT */
74*6285aa6aSThomas Monjalon [63] = 200000, /* ETHTOOL_LINK_MODE_200000baseSR4_Full_BIT */
75*6285aa6aSThomas Monjalon [64] = 200000, /* ETHTOOL_LINK_MODE_200000baseLR4_ER4_FR4_Full_BIT */
76*6285aa6aSThomas Monjalon [65] = 200000, /* ETHTOOL_LINK_MODE_200000baseDR4_Full_BIT */
77*6285aa6aSThomas Monjalon [66] = 200000, /* ETHTOOL_LINK_MODE_200000baseCR4_Full_BIT */
78*6285aa6aSThomas Monjalon [67] = 100, /* ETHTOOL_LINK_MODE_100baseT1_Full_BIT */
79*6285aa6aSThomas Monjalon [68] = 1000, /* ETHTOOL_LINK_MODE_1000baseT1_Full_BIT */
80*6285aa6aSThomas Monjalon [69] = 400000, /* ETHTOOL_LINK_MODE_400000baseKR8_Full_BIT */
81*6285aa6aSThomas Monjalon [70] = 400000, /* ETHTOOL_LINK_MODE_400000baseSR8_Full_BIT */
82*6285aa6aSThomas Monjalon [71] = 400000, /* ETHTOOL_LINK_MODE_400000baseLR8_ER8_FR8_Full_BIT */
83*6285aa6aSThomas Monjalon [72] = 400000, /* ETHTOOL_LINK_MODE_400000baseDR8_Full_BIT */
84*6285aa6aSThomas Monjalon [73] = 400000, /* ETHTOOL_LINK_MODE_400000baseCR8_Full_BIT */
85*6285aa6aSThomas Monjalon [75] = 100000, /* ETHTOOL_LINK_MODE_100000baseKR_Full_BIT */
86*6285aa6aSThomas Monjalon [76] = 100000, /* ETHTOOL_LINK_MODE_100000baseSR_Full_BIT */
87*6285aa6aSThomas Monjalon [77] = 100000, /* ETHTOOL_LINK_MODE_100000baseLR_ER_FR_Full_BIT */
88*6285aa6aSThomas Monjalon [78] = 100000, /* ETHTOOL_LINK_MODE_100000baseCR_Full_BIT */
89*6285aa6aSThomas Monjalon [79] = 100000, /* ETHTOOL_LINK_MODE_100000baseDR_Full_BIT */
90*6285aa6aSThomas Monjalon [80] = 200000, /* ETHTOOL_LINK_MODE_200000baseKR2_Full_BIT */
91*6285aa6aSThomas Monjalon [81] = 200000, /* ETHTOOL_LINK_MODE_200000baseSR2_Full_BIT */
92*6285aa6aSThomas Monjalon [82] = 200000, /* ETHTOOL_LINK_MODE_200000baseLR2_ER2_FR2_Full_BIT */
93*6285aa6aSThomas Monjalon [83] = 200000, /* ETHTOOL_LINK_MODE_200000baseDR2_Full_BIT */
94*6285aa6aSThomas Monjalon [84] = 200000, /* ETHTOOL_LINK_MODE_200000baseCR2_Full_BIT */
95*6285aa6aSThomas Monjalon [85] = 400000, /* ETHTOOL_LINK_MODE_400000baseKR4_Full_BIT */
96*6285aa6aSThomas Monjalon [86] = 400000, /* ETHTOOL_LINK_MODE_400000baseSR4_Full_BIT */
97*6285aa6aSThomas Monjalon [87] = 400000, /* ETHTOOL_LINK_MODE_400000baseLR4_ER4_FR4_Full_BIT */
98*6285aa6aSThomas Monjalon [88] = 400000, /* ETHTOOL_LINK_MODE_400000baseDR4_Full_BIT */
99*6285aa6aSThomas Monjalon [89] = 400000, /* ETHTOOL_LINK_MODE_400000baseCR4_Full_BIT */
100*6285aa6aSThomas Monjalon [90] = 101, /* ETHTOOL_LINK_MODE_100baseFX_Half_BIT */
101*6285aa6aSThomas Monjalon [91] = 100, /* ETHTOOL_LINK_MODE_100baseFX_Full_BIT */
102*6285aa6aSThomas Monjalon [92] = 10, /* ETHTOOL_LINK_MODE_10baseT1L_Full_BIT */
103*6285aa6aSThomas Monjalon [93] = 800000, /* ETHTOOL_LINK_MODE_800000baseCR8_Full_BIT */
104*6285aa6aSThomas Monjalon [94] = 800000, /* ETHTOOL_LINK_MODE_800000baseKR8_Full_BIT */
105*6285aa6aSThomas Monjalon [95] = 800000, /* ETHTOOL_LINK_MODE_800000baseDR8_Full_BIT */
106*6285aa6aSThomas Monjalon [96] = 800000, /* ETHTOOL_LINK_MODE_800000baseDR8_2_Full_BIT */
107*6285aa6aSThomas Monjalon [97] = 800000, /* ETHTOOL_LINK_MODE_800000baseSR8_Full_BIT */
108*6285aa6aSThomas Monjalon [98] = 800000, /* ETHTOOL_LINK_MODE_800000baseVR8_Full_BIT */
109*6285aa6aSThomas Monjalon [99] = 10, /* ETHTOOL_LINK_MODE_10baseT1S_Full_BIT */
110*6285aa6aSThomas Monjalon [100] = 11, /* ETHTOOL_LINK_MODE_10baseT1S_Half_BIT */
111*6285aa6aSThomas Monjalon [101] = 11, /* ETHTOOL_LINK_MODE_10baseT1S_P2MP_Half_BIT */
112*6285aa6aSThomas Monjalon };
113*6285aa6aSThomas Monjalon
114*6285aa6aSThomas Monjalon uint32_t
rte_eth_link_speed_ethtool(enum ethtool_link_mode_bit_indices bit)115*6285aa6aSThomas Monjalon rte_eth_link_speed_ethtool(enum ethtool_link_mode_bit_indices bit)
116*6285aa6aSThomas Monjalon {
117*6285aa6aSThomas Monjalon uint32_t speed;
118*6285aa6aSThomas Monjalon int duplex;
119*6285aa6aSThomas Monjalon
120*6285aa6aSThomas Monjalon /* get mode from array */
121*6285aa6aSThomas Monjalon if (bit >= RTE_DIM(link_modes))
122*6285aa6aSThomas Monjalon return RTE_ETH_LINK_SPEED_AUTONEG;
123*6285aa6aSThomas Monjalon speed = link_modes[bit];
124*6285aa6aSThomas Monjalon if (speed == 0)
125*6285aa6aSThomas Monjalon return RTE_ETH_LINK_SPEED_AUTONEG;
126*6285aa6aSThomas Monjalon RTE_BUILD_BUG_ON(RTE_ETH_LINK_SPEED_AUTONEG != 0);
127*6285aa6aSThomas Monjalon
128*6285aa6aSThomas Monjalon /* duplex is LSB */
129*6285aa6aSThomas Monjalon duplex = (speed & 1) ?
130*6285aa6aSThomas Monjalon RTE_ETH_LINK_HALF_DUPLEX :
131*6285aa6aSThomas Monjalon RTE_ETH_LINK_FULL_DUPLEX;
132*6285aa6aSThomas Monjalon speed &= RTE_GENMASK32(31, 1);
133*6285aa6aSThomas Monjalon
134*6285aa6aSThomas Monjalon return rte_eth_speed_bitflag(speed, duplex);
135*6285aa6aSThomas Monjalon }
136*6285aa6aSThomas Monjalon
137*6285aa6aSThomas Monjalon uint32_t
rte_eth_link_speed_glink(const uint32_t * bitmap,int8_t nwords)138*6285aa6aSThomas Monjalon rte_eth_link_speed_glink(const uint32_t *bitmap, int8_t nwords)
139*6285aa6aSThomas Monjalon {
140*6285aa6aSThomas Monjalon uint8_t word, bit;
141*6285aa6aSThomas Monjalon uint32_t ethdev_bitmap = 0;
142*6285aa6aSThomas Monjalon
143*6285aa6aSThomas Monjalon if (nwords < 1)
144*6285aa6aSThomas Monjalon return 0;
145*6285aa6aSThomas Monjalon
146*6285aa6aSThomas Monjalon for (word = 0; word < nwords; word++) {
147*6285aa6aSThomas Monjalon for (bit = 0; bit < 32; bit++) {
148*6285aa6aSThomas Monjalon if ((bitmap[word] & RTE_BIT32(bit)) == 0)
149*6285aa6aSThomas Monjalon continue;
150*6285aa6aSThomas Monjalon ethdev_bitmap |= rte_eth_link_speed_ethtool(word * 32 + bit);
151*6285aa6aSThomas Monjalon }
152*6285aa6aSThomas Monjalon }
153*6285aa6aSThomas Monjalon
154*6285aa6aSThomas Monjalon return ethdev_bitmap;
155*6285aa6aSThomas Monjalon }
156*6285aa6aSThomas Monjalon
157*6285aa6aSThomas Monjalon uint32_t
rte_eth_link_speed_gset(uint32_t legacy_bitmap)158*6285aa6aSThomas Monjalon rte_eth_link_speed_gset(uint32_t legacy_bitmap)
159*6285aa6aSThomas Monjalon {
160*6285aa6aSThomas Monjalon return rte_eth_link_speed_glink(&legacy_bitmap, 1);
161*6285aa6aSThomas Monjalon }
162