Branch data Line data Source code
1 : : /* SPDX-License-Identifier: BSD-3-Clause
2 : : * Copyright (c) 2014-2018 Netronome Systems, Inc.
3 : : * All rights reserved.
4 : : *
5 : : * Small portions derived from code Copyright(c) 2010-2015 Intel Corporation.
6 : : */
7 : :
8 : : #include "nfp_net_common.h"
9 : :
10 : : #include <rte_alarm.h>
11 : :
12 : : #include "flower/nfp_flower_cmsg.h"
13 : : #include "flower/nfp_flower_representor.h"
14 : : #include "nfd3/nfp_nfd3.h"
15 : : #include "nfdk/nfp_nfdk.h"
16 : : #include "nfpcore/nfp_mip.h"
17 : : #include "nfpcore/nfp_nsp.h"
18 : : #include "nfpcore/nfp_rtsym.h"
19 : : #include "nfp_logs.h"
20 : : #include "nfp_net_meta.h"
21 : :
22 : : #define NFP_TX_MAX_SEG UINT8_MAX
23 : : #define NFP_TX_MAX_MTU_SEG 8
24 : :
25 : : #define NFP_NET_LINK_DOWN_CHECK_TIMEOUT 4000 /* ms */
26 : : #define NFP_NET_LINK_UP_CHECK_TIMEOUT 1000 /* ms */
27 : :
28 : : #define DEFAULT_FLBUF_SIZE 9216
29 : : #define NFP_ETH_OVERHEAD \
30 : : (RTE_ETHER_HDR_LEN + RTE_ETHER_CRC_LEN + RTE_VLAN_HLEN * 2)
31 : :
32 : : /* Only show FEC capability supported by the current speed. */
33 : : #define NFP_FEC_CAPA_ENTRY_NUM 1
34 : :
35 : : enum nfp_xstat_group {
36 : : NFP_XSTAT_GROUP_NET,
37 : : NFP_XSTAT_GROUP_MAC
38 : : };
39 : :
40 : : struct nfp_xstat {
41 : : char name[RTE_ETH_XSTATS_NAME_SIZE];
42 : : int offset;
43 : : enum nfp_xstat_group group;
44 : : };
45 : :
46 : : #define NFP_XSTAT_NET(_name, _offset) { \
47 : : .name = _name, \
48 : : .offset = NFP_NET_CFG_STATS_##_offset, \
49 : : .group = NFP_XSTAT_GROUP_NET, \
50 : : }
51 : :
52 : : #define NFP_XSTAT_MAC(_name, _offset) { \
53 : : .name = _name, \
54 : : .offset = NFP_MAC_STATS_##_offset, \
55 : : .group = NFP_XSTAT_GROUP_MAC, \
56 : : }
57 : :
58 : : static const struct nfp_xstat nfp_net_xstats[] = {
59 : : /*
60 : : * Basic xstats available on both VF and PF.
61 : : * Note that in case new statistics of group NFP_XSTAT_GROUP_NET
62 : : * are added to this array, they must appear before any statistics
63 : : * of group NFP_XSTAT_GROUP_MAC.
64 : : */
65 : : NFP_XSTAT_NET("rx_good_packets_mc", RX_MC_FRAMES),
66 : : NFP_XSTAT_NET("tx_good_packets_mc", TX_MC_FRAMES),
67 : : NFP_XSTAT_NET("rx_good_packets_bc", RX_BC_FRAMES),
68 : : NFP_XSTAT_NET("tx_good_packets_bc", TX_BC_FRAMES),
69 : : NFP_XSTAT_NET("rx_good_bytes_uc", RX_UC_OCTETS),
70 : : NFP_XSTAT_NET("tx_good_bytes_uc", TX_UC_OCTETS),
71 : : NFP_XSTAT_NET("rx_good_bytes_mc", RX_MC_OCTETS),
72 : : NFP_XSTAT_NET("tx_good_bytes_mc", TX_MC_OCTETS),
73 : : NFP_XSTAT_NET("rx_good_bytes_bc", RX_BC_OCTETS),
74 : : NFP_XSTAT_NET("tx_good_bytes_bc", TX_BC_OCTETS),
75 : : NFP_XSTAT_NET("tx_missed_erros", TX_DISCARDS),
76 : : NFP_XSTAT_NET("bpf_pass_pkts", APP0_FRAMES),
77 : : NFP_XSTAT_NET("bpf_pass_bytes", APP0_BYTES),
78 : : NFP_XSTAT_NET("bpf_app1_pkts", APP1_FRAMES),
79 : : NFP_XSTAT_NET("bpf_app1_bytes", APP1_BYTES),
80 : : NFP_XSTAT_NET("bpf_app2_pkts", APP2_FRAMES),
81 : : NFP_XSTAT_NET("bpf_app2_bytes", APP2_BYTES),
82 : : NFP_XSTAT_NET("bpf_app3_pkts", APP3_FRAMES),
83 : : NFP_XSTAT_NET("bpf_app3_bytes", APP3_BYTES),
84 : : /*
85 : : * MAC xstats available only on PF. These statistics are not available for VFs as the
86 : : * PF is not initialized when the VF is initialized as it is still bound to the kernel
87 : : * driver. As such, the PMD cannot obtain a CPP handle and access the rtsym_table in order
88 : : * to get the pointer to the start of the MAC statistics counters.
89 : : */
90 : : NFP_XSTAT_MAC("mac.rx_octets", RX_IN_OCTS),
91 : : NFP_XSTAT_MAC("mac.rx_frame_too_long_errors", RX_FRAME_TOO_LONG_ERRORS),
92 : : NFP_XSTAT_MAC("mac.rx_range_length_errors", RX_RANGE_LENGTH_ERRORS),
93 : : NFP_XSTAT_MAC("mac.rx_vlan_received_ok", RX_VLAN_RECEIVED_OK),
94 : : NFP_XSTAT_MAC("mac.rx_errors", RX_IN_ERRORS),
95 : : NFP_XSTAT_MAC("mac.rx_broadcast_pkts", RX_IN_BROADCAST_PKTS),
96 : : NFP_XSTAT_MAC("mac.rx_drop_events", RX_DROP_EVENTS),
97 : : NFP_XSTAT_MAC("mac.rx_alignment_errors", RX_ALIGNMENT_ERRORS),
98 : : NFP_XSTAT_MAC("mac.rx_pause_mac_ctrl_frames", RX_PAUSE_MAC_CTRL_FRAMES),
99 : : NFP_XSTAT_MAC("mac.rx_frames_received_ok", RX_FRAMES_RECEIVED_OK),
100 : : NFP_XSTAT_MAC("mac.rx_frame_check_sequence_errors", RX_FRAME_CHECK_SEQ_ERRORS),
101 : : NFP_XSTAT_MAC("mac.rx_unicast_pkts", RX_UNICAST_PKTS),
102 : : NFP_XSTAT_MAC("mac.rx_multicast_pkts", RX_MULTICAST_PKTS),
103 : : NFP_XSTAT_MAC("mac.rx_pkts", RX_PKTS),
104 : : NFP_XSTAT_MAC("mac.rx_undersize_pkts", RX_UNDERSIZE_PKTS),
105 : : NFP_XSTAT_MAC("mac.rx_pkts_64_octets", RX_PKTS_64_OCTS),
106 : : NFP_XSTAT_MAC("mac.rx_pkts_65_to_127_octets", RX_PKTS_65_TO_127_OCTS),
107 : : NFP_XSTAT_MAC("mac.rx_pkts_128_to_255_octets", RX_PKTS_128_TO_255_OCTS),
108 : : NFP_XSTAT_MAC("mac.rx_pkts_256_to_511_octets", RX_PKTS_256_TO_511_OCTS),
109 : : NFP_XSTAT_MAC("mac.rx_pkts_512_to_1023_octets", RX_PKTS_512_TO_1023_OCTS),
110 : : NFP_XSTAT_MAC("mac.rx_pkts_1024_to_1518_octets", RX_PKTS_1024_TO_1518_OCTS),
111 : : NFP_XSTAT_MAC("mac.rx_pkts_1519_to_max_octets", RX_PKTS_1519_TO_MAX_OCTS),
112 : : NFP_XSTAT_MAC("mac.rx_jabbers", RX_JABBERS),
113 : : NFP_XSTAT_MAC("mac.rx_fragments", RX_FRAGMENTS),
114 : : NFP_XSTAT_MAC("mac.rx_oversize_pkts", RX_OVERSIZE_PKTS),
115 : : NFP_XSTAT_MAC("mac.rx_pause_frames_class0", RX_PAUSE_FRAMES_CLASS0),
116 : : NFP_XSTAT_MAC("mac.rx_pause_frames_class1", RX_PAUSE_FRAMES_CLASS1),
117 : : NFP_XSTAT_MAC("mac.rx_pause_frames_class2", RX_PAUSE_FRAMES_CLASS2),
118 : : NFP_XSTAT_MAC("mac.rx_pause_frames_class3", RX_PAUSE_FRAMES_CLASS3),
119 : : NFP_XSTAT_MAC("mac.rx_pause_frames_class4", RX_PAUSE_FRAMES_CLASS4),
120 : : NFP_XSTAT_MAC("mac.rx_pause_frames_class5", RX_PAUSE_FRAMES_CLASS5),
121 : : NFP_XSTAT_MAC("mac.rx_pause_frames_class6", RX_PAUSE_FRAMES_CLASS6),
122 : : NFP_XSTAT_MAC("mac.rx_pause_frames_class7", RX_PAUSE_FRAMES_CLASS7),
123 : : NFP_XSTAT_MAC("mac.rx_mac_ctrl_frames_received", RX_MAC_CTRL_FRAMES_REC),
124 : : NFP_XSTAT_MAC("mac.rx_mac_head_drop", RX_MAC_HEAD_DROP),
125 : : NFP_XSTAT_MAC("mac.tx_queue_drop", TX_QUEUE_DROP),
126 : : NFP_XSTAT_MAC("mac.tx_octets", TX_OUT_OCTS),
127 : : NFP_XSTAT_MAC("mac.tx_vlan_transmitted_ok", TX_VLAN_TRANSMITTED_OK),
128 : : NFP_XSTAT_MAC("mac.tx_errors", TX_OUT_ERRORS),
129 : : NFP_XSTAT_MAC("mac.tx_broadcast_pkts", TX_BROADCAST_PKTS),
130 : : NFP_XSTAT_MAC("mac.tx_pause_mac_ctrl_frames", TX_PAUSE_MAC_CTRL_FRAMES),
131 : : NFP_XSTAT_MAC("mac.tx_frames_transmitted_ok", TX_FRAMES_TRANSMITTED_OK),
132 : : NFP_XSTAT_MAC("mac.tx_unicast_pkts", TX_UNICAST_PKTS),
133 : : NFP_XSTAT_MAC("mac.tx_multicast_pkts", TX_MULTICAST_PKTS),
134 : : NFP_XSTAT_MAC("mac.tx_pkts_64_octets", TX_PKTS_64_OCTS),
135 : : NFP_XSTAT_MAC("mac.tx_pkts_65_to_127_octets", TX_PKTS_65_TO_127_OCTS),
136 : : NFP_XSTAT_MAC("mac.tx_pkts_128_to_255_octets", TX_PKTS_128_TO_255_OCTS),
137 : : NFP_XSTAT_MAC("mac.tx_pkts_256_to_511_octets", TX_PKTS_256_TO_511_OCTS),
138 : : NFP_XSTAT_MAC("mac.tx_pkts_512_to_1023_octets", TX_PKTS_512_TO_1023_OCTS),
139 : : NFP_XSTAT_MAC("mac.tx_pkts_1024_to_1518_octets", TX_PKTS_1024_TO_1518_OCTS),
140 : : NFP_XSTAT_MAC("mac.tx_pkts_1519_to_max_octets", TX_PKTS_1519_TO_MAX_OCTS),
141 : : NFP_XSTAT_MAC("mac.tx_pause_frames_class0", TX_PAUSE_FRAMES_CLASS0),
142 : : NFP_XSTAT_MAC("mac.tx_pause_frames_class1", TX_PAUSE_FRAMES_CLASS1),
143 : : NFP_XSTAT_MAC("mac.tx_pause_frames_class2", TX_PAUSE_FRAMES_CLASS2),
144 : : NFP_XSTAT_MAC("mac.tx_pause_frames_class3", TX_PAUSE_FRAMES_CLASS3),
145 : : NFP_XSTAT_MAC("mac.tx_pause_frames_class4", TX_PAUSE_FRAMES_CLASS4),
146 : : NFP_XSTAT_MAC("mac.tx_pause_frames_class5", TX_PAUSE_FRAMES_CLASS5),
147 : : NFP_XSTAT_MAC("mac.tx_pause_frames_class6", TX_PAUSE_FRAMES_CLASS6),
148 : : NFP_XSTAT_MAC("mac.tx_pause_frames_class7", TX_PAUSE_FRAMES_CLASS7),
149 : : };
150 : :
151 : : static const uint32_t nfp_net_link_speed_nfp2rte[] = {
152 : : [NFP_NET_CFG_STS_LINK_RATE_UNSUPPORTED] = RTE_ETH_SPEED_NUM_NONE,
153 : : [NFP_NET_CFG_STS_LINK_RATE_UNKNOWN] = RTE_ETH_SPEED_NUM_NONE,
154 : : [NFP_NET_CFG_STS_LINK_RATE_1G] = RTE_ETH_SPEED_NUM_1G,
155 : : [NFP_NET_CFG_STS_LINK_RATE_10G] = RTE_ETH_SPEED_NUM_10G,
156 : : [NFP_NET_CFG_STS_LINK_RATE_25G] = RTE_ETH_SPEED_NUM_25G,
157 : : [NFP_NET_CFG_STS_LINK_RATE_40G] = RTE_ETH_SPEED_NUM_40G,
158 : : [NFP_NET_CFG_STS_LINK_RATE_50G] = RTE_ETH_SPEED_NUM_50G,
159 : : [NFP_NET_CFG_STS_LINK_RATE_100G] = RTE_ETH_SPEED_NUM_100G,
160 : : };
161 : :
162 : : static bool
163 : : nfp_net_is_pf(struct rte_eth_dev *dev)
164 : : {
165 [ # # ]: 0 : if (rte_eth_dev_is_repr(dev))
166 : 0 : return nfp_flower_repr_is_pf(dev);
167 : :
168 : 0 : return ((struct nfp_net_hw_priv *)dev->process_private)->is_pf;
169 : : }
170 : :
171 : : static size_t
172 : : nfp_net_link_speed_rte2nfp(uint32_t speed)
173 : : {
174 : : size_t i;
175 : :
176 [ # # ]: 0 : for (i = 0; i < RTE_DIM(nfp_net_link_speed_nfp2rte); i++) {
177 [ # # ]: 0 : if (speed == nfp_net_link_speed_nfp2rte[i])
178 : : return i;
179 : : }
180 : :
181 : : return NFP_NET_CFG_STS_LINK_RATE_UNKNOWN;
182 : : }
183 : :
184 : : static uint32_t
185 : : nfp_net_link_speed_nfp2rte_check(uint32_t speed)
186 : : {
187 : : size_t i;
188 : :
189 [ # # ]: 0 : for (i = 0; i < RTE_DIM(nfp_net_link_speed_nfp2rte); i++) {
190 [ # # ]: 0 : if (speed == nfp_net_link_speed_nfp2rte[i])
191 : : return nfp_net_link_speed_nfp2rte[i];
192 : : }
193 : :
194 : : return RTE_ETH_SPEED_NUM_NONE;
195 : : }
196 : :
197 : : void
198 : 0 : nfp_net_notify_port_speed(struct nfp_net_hw *hw,
199 : : struct rte_eth_link *link)
200 : : {
201 : : /*
202 : : * Read the link status from NFP_NET_CFG_STS. If the link is down
203 : : * then write the link speed NFP_NET_CFG_STS_LINK_RATE_UNKNOWN to
204 : : * NFP_NET_CFG_STS_NSP_LINK_RATE.
205 : : */
206 [ # # ]: 0 : if (link->link_status == RTE_ETH_LINK_DOWN) {
207 : : nn_cfg_writew(&hw->super, NFP_NET_CFG_STS_NSP_LINK_RATE,
208 : : NFP_NET_CFG_STS_LINK_RATE_UNKNOWN);
209 : 0 : return;
210 : : }
211 : :
212 : : /*
213 : : * Link is up so write the link speed from the eth_table to
214 : : * NFP_NET_CFG_STS_NSP_LINK_RATE.
215 : : */
216 : 0 : nn_cfg_writew(&hw->super, NFP_NET_CFG_STS_NSP_LINK_RATE,
217 : 0 : nfp_net_link_speed_rte2nfp(link->link_speed));
218 : : }
219 : :
220 : : /**
221 : : * Reconfigure the firmware of VF configure
222 : : *
223 : : * @param net_hw
224 : : * Device to reconfigure
225 : : * @param pf_dev
226 : : * Get the Device info
227 : : * @param update
228 : : * The value for the mailbox VF command
229 : : * @param value
230 : : * The value of update
231 : : * @param offset
232 : : * The offset in the VF configure table
233 : : *
234 : : * @return
235 : : * - (0) if OK to reconfigure vf configure.
236 : : * - (-EIO) if I/O err and fail to configure the vf configure
237 : : */
238 : : static int
239 : 0 : nfp_net_vf_reconfig(struct nfp_net_hw *net_hw,
240 : : struct nfp_pf_dev *pf_dev,
241 : : uint16_t update,
242 : : uint8_t value,
243 : : uint32_t offset)
244 : : {
245 : : int ret;
246 : : struct nfp_hw *hw;
247 : :
248 : 0 : hw = &net_hw->super;
249 : 0 : rte_spinlock_lock(&hw->reconfig_lock);
250 : :
251 : : /* Write update info to mailbox in VF config symbol */
252 : 0 : nn_writeb(value, pf_dev->vf_cfg_tbl_bar + offset);
253 : 0 : nn_writew(update, pf_dev->vf_cfg_tbl_bar + NFP_NET_VF_CFG_MB_UPD);
254 : : nn_cfg_writel(hw, NFP_NET_CFG_UPDATE, NFP_NET_CFG_UPDATE_VF);
255 : :
256 : : rte_wmb();
257 : :
258 : 0 : ret = nfp_reconfig_real(hw, NFP_NET_CFG_UPDATE_VF);
259 : :
260 : : rte_spinlock_unlock(&hw->reconfig_lock);
261 : :
262 [ # # ]: 0 : if (ret != 0)
263 : : return -EIO;
264 : :
265 : 0 : return nn_readw(pf_dev->vf_cfg_tbl_bar + NFP_NET_VF_CFG_MB_RET);
266 : : }
267 : :
268 : : /**
269 : : * Reconfigure the firmware via the mailbox
270 : : *
271 : : * @param net_hw
272 : : * Device to reconfigure
273 : : * @param mbox_cmd
274 : : * The value for the mailbox command
275 : : *
276 : : * @return
277 : : * - (0) if OK to reconfigure by the mailbox.
278 : : * - (-EIO) if I/O err and fail to reconfigure by the mailbox
279 : : */
280 : : int
281 : 0 : nfp_net_mbox_reconfig(struct nfp_net_hw *net_hw,
282 : : uint32_t mbox_cmd)
283 : : {
284 : : int ret;
285 : : uint32_t mbox;
286 : :
287 : 0 : mbox = net_hw->tlv_caps.mbox_off;
288 : :
289 : 0 : rte_spinlock_lock(&net_hw->super.reconfig_lock);
290 : :
291 : : nn_cfg_writeq(&net_hw->super, mbox + NFP_NET_CFG_MBOX_SIMPLE_CMD, mbox_cmd);
292 : : nn_cfg_writel(&net_hw->super, NFP_NET_CFG_UPDATE, NFP_NET_CFG_UPDATE_MBOX);
293 : :
294 : : rte_wmb();
295 : :
296 : 0 : ret = nfp_reconfig_real(&net_hw->super, NFP_NET_CFG_UPDATE_MBOX);
297 : :
298 : : rte_spinlock_unlock(&net_hw->super.reconfig_lock);
299 : :
300 [ # # ]: 0 : if (ret != 0) {
301 : 0 : PMD_DRV_LOG(ERR, "Error nft net mailbox reconfig: mbox=%#08x update=%#08x.",
302 : : mbox_cmd, NFP_NET_CFG_UPDATE_MBOX);
303 : 0 : return -EIO;
304 : : }
305 : :
306 : 0 : return nn_cfg_readl(&net_hw->super, mbox + NFP_NET_CFG_MBOX_SIMPLE_RET);
307 : : }
308 : :
309 : : struct nfp_net_hw *
310 [ # # ]: 0 : nfp_net_get_hw(const struct rte_eth_dev *dev)
311 : : {
312 : : struct nfp_net_hw *hw;
313 : :
314 [ # # ]: 0 : if (rte_eth_dev_is_repr(dev)) {
315 : : struct nfp_flower_representor *repr;
316 : 0 : repr = dev->data->dev_private;
317 : 0 : hw = repr->app_fw_flower->pf_hw;
318 : : } else {
319 : 0 : hw = dev->data->dev_private;
320 : : }
321 : :
322 : 0 : return hw;
323 : : }
324 : :
325 : : uint8_t
326 [ # # ]: 0 : nfp_net_get_idx(const struct rte_eth_dev *dev)
327 : : {
328 : : uint8_t idx;
329 : :
330 [ # # ]: 0 : if (rte_eth_dev_is_repr(dev)) {
331 : : struct nfp_flower_representor *repr;
332 : 0 : repr = dev->data->dev_private;
333 : 0 : idx = repr->idx;
334 : : } else {
335 : : struct nfp_net_hw *hw;
336 : 0 : hw = dev->data->dev_private;
337 : 0 : idx = hw->idx;
338 : : }
339 : :
340 : 0 : return idx;
341 : : }
342 : :
343 : : /*
344 : : * Configure an Ethernet device.
345 : : *
346 : : * This function must be invoked first before any other function in the Ethernet API.
347 : : * This function can also be re-invoked when a device is in the stopped state.
348 : : *
349 : : * A DPDK app sends info about how many queues to use and how those queues
350 : : * need to be configured. This is used by the DPDK core and it makes sure no
351 : : * more queues than those advertised by the driver are requested.
352 : : * This function is called after that internal process.
353 : : */
354 : : int
355 : 0 : nfp_net_configure(struct rte_eth_dev *dev)
356 : : {
357 : : struct nfp_net_hw *hw;
358 : : struct rte_eth_conf *dev_conf;
359 : : struct rte_eth_rxmode *rxmode;
360 : : struct rte_eth_txmode *txmode;
361 : :
362 : 0 : hw = nfp_net_get_hw(dev);
363 : 0 : dev_conf = &dev->data->dev_conf;
364 : : rxmode = &dev_conf->rxmode;
365 : : txmode = &dev_conf->txmode;
366 : :
367 [ # # ]: 0 : if ((rxmode->mq_mode & RTE_ETH_MQ_RX_RSS_FLAG) != 0)
368 : 0 : rxmode->offloads |= RTE_ETH_RX_OFFLOAD_RSS_HASH;
369 : :
370 : : /* Checking TX mode */
371 [ # # ]: 0 : if (txmode->mq_mode != RTE_ETH_MQ_TX_NONE) {
372 : 0 : PMD_DRV_LOG(ERR, "TX mq_mode DCB and VMDq not supported.");
373 : 0 : return -EINVAL;
374 : : }
375 : :
376 : : /* Checking RX mode */
377 [ # # ]: 0 : if ((rxmode->mq_mode & RTE_ETH_MQ_RX_RSS_FLAG) != 0 &&
378 [ # # ]: 0 : (hw->super.cap & NFP_NET_CFG_CTRL_RSS_ANY) == 0) {
379 : 0 : PMD_DRV_LOG(ERR, "RSS not supported.");
380 : 0 : return -EINVAL;
381 : : }
382 : :
383 : : /* Checking MTU set */
384 [ # # ]: 0 : if (rxmode->mtu > hw->max_mtu + NFP_ETH_OVERHEAD) {
385 : 0 : PMD_DRV_LOG(ERR, "MTU (%u) larger than the maximum possible frame size (%u).",
386 : : rxmode->mtu, hw->max_mtu + NFP_ETH_OVERHEAD);
387 : 0 : return -ERANGE;
388 : : }
389 : :
390 : : return 0;
391 : : }
392 : :
393 : : void
394 : 0 : nfp_net_log_device_information(const struct nfp_net_hw *hw,
395 : : struct nfp_pf_dev *pf_dev)
396 : : {
397 : 0 : uint32_t cap = hw->super.cap;
398 : 0 : uint32_t cap_ext = hw->super.cap_ext;
399 : 0 : uint32_t cap_rss = hw->super.cap_rss;
400 : :
401 : 0 : PMD_INIT_LOG(INFO, "VER: %u.%u, Maximum supported MTU: %d.",
402 : : pf_dev->ver.major, pf_dev->ver.minor, hw->max_mtu);
403 : :
404 : 0 : PMD_INIT_LOG(INFO, "CAP: %#x.", cap);
405 [ # # # # : 0 : PMD_INIT_LOG(INFO, "%s%s%s%s%s%s%s%s%s%s%s%s%s%s%s%s%s%s%s%s%s%s%s%s%s%s%s%s%s",
# # # # #
# # # # #
# # # # #
# # # # #
# # # # #
# # # # #
# # # # #
# # # # #
# # # # #
# # # # #
# # # # ]
406 : : cap & NFP_NET_CFG_CTRL_ENABLE ? "ENABLE " : "",
407 : : cap & NFP_NET_CFG_CTRL_PROMISC ? "PROMISC " : "",
408 : : cap & NFP_NET_CFG_CTRL_L2BC ? "L2BCFILT " : "",
409 : : cap & NFP_NET_CFG_CTRL_L2MC ? "L2MCFILT " : "",
410 : : cap & NFP_NET_CFG_CTRL_RXCSUM ? "RXCSUM " : "",
411 : : cap & NFP_NET_CFG_CTRL_TXCSUM ? "TXCSUM " : "",
412 : : cap & NFP_NET_CFG_CTRL_RXVLAN ? "RXVLAN " : "",
413 : : cap & NFP_NET_CFG_CTRL_TXVLAN ? "TXVLAN " : "",
414 : : cap & NFP_NET_CFG_CTRL_SCATTER ? "SCATTER " : "",
415 : : cap & NFP_NET_CFG_CTRL_GATHER ? "GATHER " : "",
416 : : cap & NFP_NET_CFG_CTRL_LSO ? "TSO " : "",
417 : : cap & NFP_NET_CFG_CTRL_RXQINQ ? "RXQINQ " : "",
418 : : cap & NFP_NET_CFG_CTRL_RXVLAN_V2 ? "RXVLANv2 " : "",
419 : : cap & NFP_NET_CFG_CTRL_RINGCFG ? "RINGCFG " : "",
420 : : cap & NFP_NET_CFG_CTRL_RSS ? "RSS " : "",
421 : : cap & NFP_NET_CFG_CTRL_IRQMOD ? "IRQMOD " : "",
422 : : cap & NFP_NET_CFG_CTRL_RINGPRIO ? "RINGPRIO " : "",
423 : : cap & NFP_NET_CFG_CTRL_MSIXAUTO ? "MSIXAUTO " : "",
424 : : cap & NFP_NET_CFG_CTRL_TXRWB ? "TXRWB " : "",
425 : : cap & NFP_NET_CFG_CTRL_L2SWITCH ? "L2SWITCH " : "",
426 : : cap & NFP_NET_CFG_CTRL_TXVLAN_V2 ? "TXVLANv2 " : "",
427 : : cap & NFP_NET_CFG_CTRL_VXLAN ? "VXLAN " : "",
428 : : cap & NFP_NET_CFG_CTRL_NVGRE ? "NVGRE " : "",
429 : : cap & NFP_NET_CFG_CTRL_MSIX_TX_OFF ? "MSIX_TX_OFF " : "",
430 : : cap & NFP_NET_CFG_CTRL_LSO2 ? "TSOv2 " : "",
431 : : cap & NFP_NET_CFG_CTRL_RSS2 ? "RSSv2 " : "",
432 : : cap & NFP_NET_CFG_CTRL_CSUM_COMPLETE ? "CSUM " : "",
433 : : cap & NFP_NET_CFG_CTRL_LIVE_ADDR ? "LIVE_ADDR " : "",
434 : : cap & NFP_NET_CFG_CTRL_USO ? "USO" : "");
435 : :
436 : 0 : PMD_INIT_LOG(INFO, "CAP_WORD1: %#x.", cap_ext);
437 [ # # # # : 0 : PMD_INIT_LOG(INFO, "%s%s%s%s%s%s%s",
# # # # #
# # # #
# ]
438 : : cap_ext & NFP_NET_CFG_CTRL_PKT_TYPE ? "PKT_TYPE " : "",
439 : : cap_ext & NFP_NET_CFG_CTRL_IPSEC ? "IPSEC " : "",
440 : : cap_ext & NFP_NET_CFG_CTRL_IPSEC_SM_LOOKUP ? "IPSEC_SM " : "",
441 : : cap_ext & NFP_NET_CFG_CTRL_IPSEC_LM_LOOKUP ? "IPSEC_LM " : "",
442 : : cap_ext & NFP_NET_CFG_CTRL_MULTI_PF ? "MULTI_PF " : "",
443 : : cap_ext & NFP_NET_CFG_CTRL_FLOW_STEER ? "FLOW_STEER " : "",
444 : : cap_ext & NFP_NET_CFG_CTRL_IN_ORDER ? "VIRTIO_IN_ORDER " : "");
445 : :
446 : 0 : PMD_INIT_LOG(INFO, "CAP_RSS: %#x.", cap_rss);
447 [ # # # # : 0 : PMD_INIT_LOG(INFO, "%s%s%s",
# # ]
448 : : cap_rss & NFP_NET_CFG_RSS_TOEPLITZ ? "RSS_TOEPLITZ " : "",
449 : : cap_rss & NFP_NET_CFG_RSS_XOR ? "RSS_XOR " : "",
450 : : cap_rss & NFP_NET_CFG_RSS_CRC32 ? "RSS_CRC32 " : "");
451 : :
452 : 0 : PMD_INIT_LOG(INFO, "The max_rx_queues: %u, max_tx_queues: %u.",
453 : : hw->max_rx_queues, hw->max_tx_queues);
454 : 0 : }
455 : :
456 : : static inline void
457 : : nfp_net_enable_rxvlan_cap(struct nfp_net_hw *hw,
458 : : uint32_t *ctrl)
459 : : {
460 [ # # # # ]: 0 : if ((hw->super.cap & NFP_NET_CFG_CTRL_RXVLAN_V2) != 0)
461 : 0 : *ctrl |= NFP_NET_CFG_CTRL_RXVLAN_V2;
462 [ # # # # ]: 0 : else if ((hw->super.cap & NFP_NET_CFG_CTRL_RXVLAN) != 0)
463 : 0 : *ctrl |= NFP_NET_CFG_CTRL_RXVLAN;
464 : : }
465 : :
466 : : void
467 : 0 : nfp_net_enable_queues(struct rte_eth_dev *dev)
468 : : {
469 : : struct nfp_net_hw *hw;
470 : :
471 : 0 : hw = nfp_net_get_hw(dev);
472 : :
473 : 0 : nfp_enable_queues(&hw->super, dev->data->nb_rx_queues,
474 : 0 : dev->data->nb_tx_queues);
475 : 0 : }
476 : :
477 : : void
478 : 0 : nfp_net_disable_queues(struct rte_eth_dev *dev)
479 : : {
480 : : struct nfp_net_hw *net_hw;
481 : :
482 : 0 : net_hw = nfp_net_get_hw(dev);
483 : :
484 : 0 : nfp_disable_queues(&net_hw->super);
485 : 0 : }
486 : :
487 : : void
488 : 0 : nfp_net_params_setup(struct nfp_net_hw *hw)
489 : : {
490 : 0 : nn_cfg_writel(&hw->super, NFP_NET_CFG_MTU, hw->mtu);
491 : 0 : nn_cfg_writel(&hw->super, NFP_NET_CFG_FLBUFSZ, hw->flbufsz);
492 : 0 : }
493 : :
494 : : void
495 : 0 : nfp_net_cfg_queue_setup(struct nfp_net_hw *hw)
496 : : {
497 : 0 : hw->super.qcp_cfg = hw->tx_bar + NFP_QCP_QUEUE_ADDR_SZ;
498 : 0 : }
499 : :
500 : : int
501 : 0 : nfp_net_set_mac_addr(struct rte_eth_dev *dev,
502 : : struct rte_ether_addr *mac_addr)
503 : : {
504 : : uint32_t update;
505 : : uint32_t new_ctrl;
506 : : struct nfp_hw *hw;
507 : : struct nfp_net_hw *net_hw;
508 : :
509 : 0 : net_hw = nfp_net_get_hw(dev);
510 : 0 : hw = &net_hw->super;
511 [ # # ]: 0 : if ((hw->ctrl & NFP_NET_CFG_CTRL_ENABLE) != 0 &&
512 [ # # ]: 0 : (hw->cap & NFP_NET_CFG_CTRL_LIVE_ADDR) == 0) {
513 : 0 : PMD_DRV_LOG(ERR, "MAC address unable to change when port enabled.");
514 : 0 : return -EBUSY;
515 : : }
516 : :
517 : : if (rte_is_valid_assigned_ether_addr(mac_addr) == 0) {
518 : 0 : PMD_DRV_LOG(ERR, "Invalid MAC address.");
519 : 0 : return -EINVAL;
520 : : }
521 : :
522 : : /* Writing new MAC to the specific port BAR address */
523 : 0 : nfp_write_mac(hw, (uint8_t *)mac_addr);
524 : :
525 : : update = NFP_NET_CFG_UPDATE_MACADDR;
526 : 0 : new_ctrl = hw->ctrl;
527 [ # # ]: 0 : if ((hw->ctrl & NFP_NET_CFG_CTRL_ENABLE) != 0 &&
528 [ # # ]: 0 : (hw->cap & NFP_NET_CFG_CTRL_LIVE_ADDR) != 0)
529 : 0 : new_ctrl |= NFP_NET_CFG_CTRL_LIVE_ADDR;
530 : :
531 : : /* Signal the NIC about the change */
532 [ # # ]: 0 : if (nfp_reconfig(hw, new_ctrl, update) != 0) {
533 : 0 : PMD_DRV_LOG(ERR, "MAC address update failed.");
534 : 0 : return -EIO;
535 : : }
536 : :
537 : 0 : hw->ctrl = new_ctrl;
538 : :
539 : 0 : return 0;
540 : : }
541 : :
542 : : int
543 : 0 : nfp_configure_rx_interrupt(struct rte_eth_dev *dev,
544 : : struct rte_intr_handle *intr_handle)
545 : : {
546 : : uint16_t i;
547 : : struct nfp_net_hw *hw;
548 : :
549 [ # # ]: 0 : if (rte_intr_vec_list_alloc(intr_handle, "intr_vec",
550 : 0 : dev->data->nb_rx_queues) != 0) {
551 : 0 : PMD_DRV_LOG(ERR, "Failed to allocate %d rx_queues intr_vec.",
552 : : dev->data->nb_rx_queues);
553 : 0 : return -ENOMEM;
554 : : }
555 : :
556 : 0 : hw = nfp_net_get_hw(dev);
557 : :
558 [ # # ]: 0 : if (rte_intr_type_get(intr_handle) == RTE_INTR_HANDLE_UIO) {
559 : 0 : PMD_DRV_LOG(INFO, "VF: enabling RX interrupt with UIO.");
560 : : /* UIO just supports one queue and no LSC */
561 : : nn_cfg_writeb(&hw->super, NFP_NET_CFG_RXR_VEC(0), 0);
562 [ # # ]: 0 : if (rte_intr_vec_list_index_set(intr_handle, 0, 0) != 0)
563 : : return -1;
564 : : } else {
565 : 0 : PMD_DRV_LOG(INFO, "VF: enabling RX interrupt with VFIO.");
566 [ # # ]: 0 : for (i = 0; i < dev->data->nb_rx_queues; i++) {
567 : : /*
568 : : * The first msix vector is reserved for non
569 : : * efd interrupts.
570 : : */
571 : 0 : nn_cfg_writeb(&hw->super, NFP_NET_CFG_RXR_VEC(i), i + 1);
572 [ # # ]: 0 : if (rte_intr_vec_list_index_set(intr_handle, i, i + 1) != 0)
573 : : return -1;
574 : : }
575 : : }
576 : :
577 : : /* Avoiding TX interrupts */
578 : 0 : hw->super.ctrl |= NFP_NET_CFG_CTRL_MSIX_TX_OFF;
579 : 0 : return 0;
580 : : }
581 : :
582 : : uint32_t
583 : 0 : nfp_check_offloads(struct rte_eth_dev *dev)
584 : : {
585 : : uint32_t cap;
586 : : uint32_t ctrl = 0;
587 : : uint64_t rx_offload;
588 : : uint64_t tx_offload;
589 : : struct nfp_net_hw *hw;
590 : : struct rte_eth_conf *dev_conf;
591 : :
592 : 0 : hw = nfp_net_get_hw(dev);
593 : 0 : cap = hw->super.cap;
594 : :
595 : 0 : dev_conf = &dev->data->dev_conf;
596 : 0 : rx_offload = dev_conf->rxmode.offloads;
597 : 0 : tx_offload = dev_conf->txmode.offloads;
598 : :
599 [ # # ]: 0 : if ((rx_offload & RTE_ETH_RX_OFFLOAD_IPV4_CKSUM) != 0) {
600 [ # # ]: 0 : if ((cap & NFP_NET_CFG_CTRL_RXCSUM) != 0)
601 : : ctrl |= NFP_NET_CFG_CTRL_RXCSUM;
602 : : }
603 : :
604 [ # # ]: 0 : if ((rx_offload & RTE_ETH_RX_OFFLOAD_VLAN_STRIP) != 0)
605 : : nfp_net_enable_rxvlan_cap(hw, &ctrl);
606 : :
607 [ # # ]: 0 : if ((rx_offload & RTE_ETH_RX_OFFLOAD_QINQ_STRIP) != 0) {
608 [ # # ]: 0 : if ((cap & NFP_NET_CFG_CTRL_RXQINQ) != 0)
609 : 0 : ctrl |= NFP_NET_CFG_CTRL_RXQINQ;
610 : : }
611 : :
612 : 0 : hw->mtu = dev->data->mtu;
613 : :
614 [ # # ]: 0 : if ((tx_offload & RTE_ETH_TX_OFFLOAD_VLAN_INSERT) != 0) {
615 [ # # ]: 0 : if ((cap & NFP_NET_CFG_CTRL_TXVLAN_V2) != 0)
616 : 0 : ctrl |= NFP_NET_CFG_CTRL_TXVLAN_V2;
617 [ # # ]: 0 : else if ((cap & NFP_NET_CFG_CTRL_TXVLAN) != 0)
618 : 0 : ctrl |= NFP_NET_CFG_CTRL_TXVLAN;
619 : : }
620 : :
621 : : /* L2 broadcast */
622 [ # # ]: 0 : if ((cap & NFP_NET_CFG_CTRL_L2BC) != 0)
623 : 0 : ctrl |= NFP_NET_CFG_CTRL_L2BC;
624 : :
625 : : /* L2 multicast */
626 [ # # ]: 0 : if ((cap & NFP_NET_CFG_CTRL_L2MC) != 0)
627 : 0 : ctrl |= NFP_NET_CFG_CTRL_L2MC;
628 : :
629 : : /* TX checksum offload */
630 : 0 : if ((tx_offload & RTE_ETH_TX_OFFLOAD_IPV4_CKSUM) != 0 ||
631 [ # # ]: 0 : (tx_offload & RTE_ETH_TX_OFFLOAD_UDP_CKSUM) != 0 ||
632 : : (tx_offload & RTE_ETH_TX_OFFLOAD_TCP_CKSUM) != 0)
633 : 0 : ctrl |= NFP_NET_CFG_CTRL_TXCSUM;
634 : :
635 : : /* LSO offload */
636 : 0 : if ((tx_offload & RTE_ETH_TX_OFFLOAD_TCP_TSO) != 0 ||
637 [ # # ]: 0 : (tx_offload & RTE_ETH_TX_OFFLOAD_UDP_TSO) != 0 ||
638 : : (tx_offload & RTE_ETH_TX_OFFLOAD_VXLAN_TNL_TSO) != 0) {
639 [ # # ]: 0 : if ((cap & NFP_NET_CFG_CTRL_LSO) != 0)
640 : 0 : ctrl |= NFP_NET_CFG_CTRL_LSO;
641 [ # # ]: 0 : else if ((cap & NFP_NET_CFG_CTRL_LSO2) != 0)
642 : 0 : ctrl |= NFP_NET_CFG_CTRL_LSO2;
643 : : }
644 : :
645 : : /* RX gather */
646 [ # # ]: 0 : if ((tx_offload & RTE_ETH_TX_OFFLOAD_MULTI_SEGS) != 0)
647 : 0 : ctrl |= NFP_NET_CFG_CTRL_GATHER;
648 : :
649 : 0 : return ctrl;
650 : : }
651 : :
652 : : int
653 : 0 : nfp_net_promisc_enable(struct rte_eth_dev *dev)
654 : : {
655 : : int ret;
656 : : uint32_t update;
657 : : uint32_t new_ctrl;
658 : : struct nfp_hw *hw;
659 : : struct nfp_net_hw *net_hw;
660 : :
661 : 0 : net_hw = nfp_net_get_hw(dev);
662 : :
663 : 0 : hw = &net_hw->super;
664 [ # # ]: 0 : if ((hw->cap & NFP_NET_CFG_CTRL_PROMISC) == 0) {
665 : 0 : PMD_DRV_LOG(ERR, "Promiscuous mode not supported.");
666 : 0 : return -ENOTSUP;
667 : : }
668 : :
669 [ # # ]: 0 : if ((hw->ctrl & NFP_NET_CFG_CTRL_PROMISC) != 0) {
670 : 0 : PMD_DRV_LOG(INFO, "Promiscuous mode already enabled.");
671 : 0 : return 0;
672 : : }
673 : :
674 : 0 : new_ctrl = hw->ctrl | NFP_NET_CFG_CTRL_PROMISC;
675 : : update = NFP_NET_CFG_UPDATE_GEN;
676 : :
677 : 0 : ret = nfp_reconfig(hw, new_ctrl, update);
678 [ # # ]: 0 : if (ret != 0)
679 : : return ret;
680 : :
681 : 0 : hw->ctrl = new_ctrl;
682 : :
683 : 0 : return 0;
684 : : }
685 : :
686 : : int
687 : 0 : nfp_net_promisc_disable(struct rte_eth_dev *dev)
688 : : {
689 : : int ret;
690 : : uint32_t update;
691 : : uint32_t new_ctrl;
692 : : struct nfp_hw *hw;
693 : : struct nfp_net_hw *net_hw;
694 : :
695 : 0 : net_hw = nfp_net_get_hw(dev);
696 : 0 : hw = &net_hw->super;
697 : :
698 [ # # ]: 0 : if ((hw->cap & NFP_NET_CFG_CTRL_PROMISC) == 0) {
699 : 0 : PMD_DRV_LOG(ERR, "Promiscuous mode not supported.");
700 : 0 : return -ENOTSUP;
701 : : }
702 : :
703 [ # # ]: 0 : if ((hw->ctrl & NFP_NET_CFG_CTRL_PROMISC) == 0) {
704 : 0 : PMD_DRV_LOG(INFO, "Promiscuous mode already disabled.");
705 : 0 : return 0;
706 : : }
707 : :
708 : 0 : new_ctrl = hw->ctrl & ~NFP_NET_CFG_CTRL_PROMISC;
709 : : update = NFP_NET_CFG_UPDATE_GEN;
710 : :
711 : 0 : ret = nfp_reconfig(hw, new_ctrl, update);
712 [ # # ]: 0 : if (ret != 0)
713 : : return ret;
714 : :
715 : 0 : hw->ctrl = new_ctrl;
716 : :
717 : 0 : return 0;
718 : : }
719 : :
720 : : static int
721 : 0 : nfp_net_set_allmulticast_mode(struct rte_eth_dev *dev,
722 : : bool enable)
723 : : {
724 : : int ret;
725 : : uint32_t update;
726 : : struct nfp_hw *hw;
727 : : uint32_t cap_extend;
728 : : uint32_t ctrl_extend;
729 : : uint32_t new_ctrl_extend;
730 : : struct nfp_net_hw *net_hw;
731 : :
732 : 0 : net_hw = nfp_net_get_hw(dev);
733 : 0 : hw = &net_hw->super;
734 : :
735 : 0 : cap_extend = hw->cap_ext;
736 [ # # ]: 0 : if ((cap_extend & NFP_NET_CFG_CTRL_MCAST_FILTER) == 0) {
737 : 0 : PMD_DRV_LOG(DEBUG, "Allmulticast mode not supported.");
738 : 0 : return -ENOTSUP;
739 : : }
740 : :
741 : : /*
742 : : * Allmulticast mode enabled when NFP_NET_CFG_CTRL_MCAST_FILTER bit is 0.
743 : : * Allmulticast mode disabled when NFP_NET_CFG_CTRL_MCAST_FILTER bit is 1.
744 : : */
745 : 0 : ctrl_extend = hw->ctrl_ext;
746 [ # # ]: 0 : if (enable) {
747 [ # # ]: 0 : if ((ctrl_extend & NFP_NET_CFG_CTRL_MCAST_FILTER) == 0)
748 : : return 0;
749 : :
750 : 0 : new_ctrl_extend = ctrl_extend & ~NFP_NET_CFG_CTRL_MCAST_FILTER;
751 : : } else {
752 [ # # ]: 0 : if ((ctrl_extend & NFP_NET_CFG_CTRL_MCAST_FILTER) != 0)
753 : : return 0;
754 : :
755 : 0 : new_ctrl_extend = ctrl_extend | NFP_NET_CFG_CTRL_MCAST_FILTER;
756 : : }
757 : :
758 : : update = NFP_NET_CFG_UPDATE_GEN;
759 : :
760 : 0 : ret = nfp_ext_reconfig(hw, new_ctrl_extend, update);
761 [ # # ]: 0 : if (ret != 0)
762 : : return ret;
763 : :
764 : 0 : hw->ctrl_ext = new_ctrl_extend;
765 : 0 : return 0;
766 : : }
767 : :
768 : : int
769 : 0 : nfp_net_allmulticast_enable(struct rte_eth_dev *dev)
770 : : {
771 : 0 : return nfp_net_set_allmulticast_mode(dev, true);
772 : : }
773 : :
774 : : int
775 : 0 : nfp_net_allmulticast_disable(struct rte_eth_dev *dev)
776 : : {
777 : 0 : return nfp_net_set_allmulticast_mode(dev, false);
778 : : }
779 : :
780 : : static void
781 : 0 : nfp_net_pf_speed_update(struct rte_eth_dev *dev,
782 : : struct nfp_net_hw_priv *hw_priv,
783 : : struct rte_eth_link *link)
784 : : {
785 : : uint8_t idx;
786 : : enum nfp_eth_aneg aneg;
787 : : struct nfp_pf_dev *pf_dev;
788 : : struct nfp_eth_table *nfp_eth_table;
789 : : struct nfp_eth_table_port *eth_port;
790 : :
791 : 0 : pf_dev = hw_priv->pf_dev;
792 : 0 : idx = nfp_net_get_idx(dev);
793 : 0 : aneg = pf_dev->nfp_eth_table->ports[idx].aneg;
794 : :
795 : : /* Compare whether the current status has changed. */
796 [ # # # # ]: 0 : if (pf_dev->speed_updated || aneg == NFP_ANEG_AUTO) {
797 : 0 : nfp_eth_table = nfp_eth_read_ports(pf_dev->cpp);
798 [ # # ]: 0 : if (nfp_eth_table == NULL) {
799 : 0 : PMD_DRV_LOG(DEBUG, "Failed to get nfp_eth_table.");
800 : : } else {
801 : 0 : pf_dev->nfp_eth_table->ports[idx] = nfp_eth_table->ports[idx];
802 : 0 : free(nfp_eth_table);
803 : 0 : pf_dev->speed_updated = false;
804 : : }
805 : : }
806 : :
807 : 0 : nfp_eth_table = pf_dev->nfp_eth_table;
808 : : eth_port = &nfp_eth_table->ports[idx];
809 : :
810 : 0 : link->link_speed = nfp_net_link_speed_nfp2rte_check(eth_port->speed);
811 : :
812 [ # # ]: 0 : if (dev->data->dev_conf.link_speeds == RTE_ETH_LINK_SPEED_AUTONEG &&
813 [ # # ]: 0 : eth_port->supp_aneg)
814 : 0 : link->link_autoneg = RTE_ETH_LINK_AUTONEG;
815 : 0 : }
816 : :
817 : : static void
818 : : nfp_net_vf_speed_update(struct rte_eth_link *link,
819 : : uint32_t link_status)
820 : : {
821 : : size_t link_rate_index;
822 : :
823 : : /*
824 : : * Shift and mask link_status so that it is effectively the value
825 : : * at offset NFP_NET_CFG_STS_NSP_LINK_RATE.
826 : : */
827 : 0 : link_rate_index = (link_status >> NFP_NET_CFG_STS_LINK_RATE_SHIFT) &
828 : : NFP_NET_CFG_STS_LINK_RATE_MASK;
829 [ # # ]: 0 : if (link_rate_index < RTE_DIM(nfp_net_link_speed_nfp2rte))
830 : 0 : link->link_speed = nfp_net_link_speed_nfp2rte[link_rate_index];
831 : : else
832 : 0 : link->link_speed = RTE_ETH_SPEED_NUM_NONE;
833 : : }
834 : :
835 : : int
836 : 0 : nfp_net_link_update_common(struct rte_eth_dev *dev,
837 : : struct rte_eth_link *link,
838 : : uint32_t link_status)
839 : : {
840 : : int ret;
841 : : struct nfp_net_hw_priv *hw_priv;
842 : :
843 : 0 : hw_priv = dev->process_private;
844 [ # # ]: 0 : if (link->link_status == RTE_ETH_LINK_UP) {
845 [ # # ]: 0 : if (nfp_net_is_pf(dev))
846 : 0 : nfp_net_pf_speed_update(dev, hw_priv, link);
847 : : else
848 : : nfp_net_vf_speed_update(link, link_status);
849 : : }
850 : :
851 : : ret = rte_eth_linkstatus_set(dev, link);
852 : : if (ret == 0) {
853 [ # # ]: 0 : if (link->link_status == RTE_ETH_LINK_UP)
854 : 0 : PMD_DRV_LOG(INFO, "NIC Link is Up.");
855 : : else
856 : 0 : PMD_DRV_LOG(INFO, "NIC Link is Down.");
857 : : }
858 : :
859 : 0 : return ret;
860 : : }
861 : :
862 : : /*
863 : : * Return 0 means link status changed, -1 means not changed
864 : : *
865 : : * Wait to complete is needed as it can take up to 9 seconds to get the Link
866 : : * status.
867 : : */
868 : : int
869 : 0 : nfp_net_link_update(struct rte_eth_dev *dev,
870 : : __rte_unused int wait_to_complete)
871 : : {
872 : : int ret;
873 : : struct nfp_net_hw *hw;
874 : : uint32_t nn_link_status;
875 : : struct rte_eth_link link;
876 : : struct nfp_net_hw_priv *hw_priv;
877 : :
878 : 0 : hw = nfp_net_get_hw(dev);
879 : 0 : hw_priv = dev->process_private;
880 : :
881 : : memset(&link, 0, sizeof(struct rte_eth_link));
882 : :
883 : : /* Read link status */
884 : 0 : nn_link_status = nn_cfg_readw(&hw->super, NFP_NET_CFG_STS);
885 [ # # ]: 0 : if ((nn_link_status & NFP_NET_CFG_STS_LINK) != 0)
886 : 0 : link.link_status = RTE_ETH_LINK_UP;
887 : :
888 : 0 : link.link_duplex = RTE_ETH_LINK_FULL_DUPLEX;
889 : :
890 : 0 : ret = nfp_net_link_update_common(dev, &link, nn_link_status);
891 [ # # ]: 0 : if (ret == -EIO)
892 : : return ret;
893 : :
894 : : /*
895 : : * Notify the port to update the speed value in the CTRL BAR from NSP.
896 : : * Not applicable for VFs as the associated PF is still attached to the
897 : : * kernel driver.
898 : : */
899 [ # # # # ]: 0 : if (hw_priv != NULL && hw_priv->is_pf)
900 : 0 : nfp_net_notify_port_speed(hw, &link);
901 : :
902 : : return ret;
903 : : }
904 : :
905 : : int
906 : 0 : nfp_net_stats_get(struct rte_eth_dev *dev,
907 : : struct rte_eth_stats *stats)
908 : : {
909 : : uint16_t i;
910 : : struct nfp_net_hw *hw;
911 : : struct rte_eth_stats nfp_dev_stats;
912 : :
913 [ # # ]: 0 : if (stats == NULL)
914 : : return -EINVAL;
915 : :
916 : 0 : hw = nfp_net_get_hw(dev);
917 : :
918 : : memset(&nfp_dev_stats, 0, sizeof(nfp_dev_stats));
919 : :
920 : : /* Reading per RX ring stats */
921 [ # # ]: 0 : for (i = 0; i < dev->data->nb_rx_queues; i++) {
922 [ # # ]: 0 : if (i == RTE_ETHDEV_QUEUE_STAT_CNTRS)
923 : : break;
924 : :
925 : : nfp_dev_stats.q_ipackets[i] =
926 : 0 : nn_cfg_readq(&hw->super, NFP_NET_CFG_RXR_STATS(i));
927 : 0 : nfp_dev_stats.q_ipackets[i] -=
928 : 0 : hw->eth_stats_base.q_ipackets[i];
929 : :
930 : : nfp_dev_stats.q_ibytes[i] =
931 : 0 : nn_cfg_readq(&hw->super, NFP_NET_CFG_RXR_STATS(i) + 0x8);
932 : 0 : nfp_dev_stats.q_ibytes[i] -=
933 : 0 : hw->eth_stats_base.q_ibytes[i];
934 : : }
935 : :
936 : : /* Reading per TX ring stats */
937 [ # # ]: 0 : for (i = 0; i < dev->data->nb_tx_queues; i++) {
938 [ # # ]: 0 : if (i == RTE_ETHDEV_QUEUE_STAT_CNTRS)
939 : : break;
940 : :
941 : : nfp_dev_stats.q_opackets[i] =
942 : 0 : nn_cfg_readq(&hw->super, NFP_NET_CFG_TXR_STATS(i));
943 : 0 : nfp_dev_stats.q_opackets[i] -= hw->eth_stats_base.q_opackets[i];
944 : :
945 : : nfp_dev_stats.q_obytes[i] =
946 : 0 : nn_cfg_readq(&hw->super, NFP_NET_CFG_TXR_STATS(i) + 0x8);
947 : 0 : nfp_dev_stats.q_obytes[i] -= hw->eth_stats_base.q_obytes[i];
948 : : }
949 : :
950 : : nfp_dev_stats.ipackets = nn_cfg_readq(&hw->super, NFP_NET_CFG_STATS_RX_FRAMES);
951 : 0 : nfp_dev_stats.ipackets -= hw->eth_stats_base.ipackets;
952 : :
953 : : nfp_dev_stats.ibytes = nn_cfg_readq(&hw->super, NFP_NET_CFG_STATS_RX_OCTETS);
954 : 0 : nfp_dev_stats.ibytes -= hw->eth_stats_base.ibytes;
955 : :
956 : : nfp_dev_stats.opackets =
957 : : nn_cfg_readq(&hw->super, NFP_NET_CFG_STATS_TX_FRAMES);
958 : 0 : nfp_dev_stats.opackets -= hw->eth_stats_base.opackets;
959 : :
960 : : nfp_dev_stats.obytes =
961 : : nn_cfg_readq(&hw->super, NFP_NET_CFG_STATS_TX_OCTETS);
962 : 0 : nfp_dev_stats.obytes -= hw->eth_stats_base.obytes;
963 : :
964 : : /* Reading general device stats */
965 : : nfp_dev_stats.ierrors =
966 : : nn_cfg_readq(&hw->super, NFP_NET_CFG_STATS_RX_ERRORS);
967 : 0 : nfp_dev_stats.ierrors -= hw->eth_stats_base.ierrors;
968 : :
969 : : nfp_dev_stats.oerrors =
970 : : nn_cfg_readq(&hw->super, NFP_NET_CFG_STATS_TX_ERRORS);
971 : 0 : nfp_dev_stats.oerrors -= hw->eth_stats_base.oerrors;
972 : :
973 : : /* RX ring mbuf allocation failures */
974 : 0 : nfp_dev_stats.rx_nombuf = dev->data->rx_mbuf_alloc_failed;
975 : :
976 : : nfp_dev_stats.imissed =
977 : : nn_cfg_readq(&hw->super, NFP_NET_CFG_STATS_RX_DISCARDS);
978 : 0 : nfp_dev_stats.imissed -= hw->eth_stats_base.imissed;
979 : :
980 : : memcpy(stats, &nfp_dev_stats, sizeof(*stats));
981 : 0 : return 0;
982 : : }
983 : :
984 : : /*
985 : : * hw->eth_stats_base records the per counter starting point.
986 : : * Lets update it now.
987 : : */
988 : : int
989 : 0 : nfp_net_stats_reset(struct rte_eth_dev *dev)
990 : : {
991 : : uint16_t i;
992 : : struct nfp_net_hw *hw;
993 : :
994 : 0 : hw = nfp_net_get_hw(dev);
995 : :
996 : : /* Reading per RX ring stats */
997 [ # # ]: 0 : for (i = 0; i < dev->data->nb_rx_queues; i++) {
998 [ # # ]: 0 : if (i == RTE_ETHDEV_QUEUE_STAT_CNTRS)
999 : : break;
1000 : :
1001 : 0 : hw->eth_stats_base.q_ipackets[i] =
1002 : 0 : nn_cfg_readq(&hw->super, NFP_NET_CFG_RXR_STATS(i));
1003 : :
1004 : 0 : hw->eth_stats_base.q_ibytes[i] =
1005 : 0 : nn_cfg_readq(&hw->super, NFP_NET_CFG_RXR_STATS(i) + 0x8);
1006 : : }
1007 : :
1008 : : /* Reading per TX ring stats */
1009 [ # # ]: 0 : for (i = 0; i < dev->data->nb_tx_queues; i++) {
1010 [ # # ]: 0 : if (i == RTE_ETHDEV_QUEUE_STAT_CNTRS)
1011 : : break;
1012 : :
1013 : 0 : hw->eth_stats_base.q_opackets[i] =
1014 : 0 : nn_cfg_readq(&hw->super, NFP_NET_CFG_TXR_STATS(i));
1015 : :
1016 : 0 : hw->eth_stats_base.q_obytes[i] =
1017 : 0 : nn_cfg_readq(&hw->super, NFP_NET_CFG_TXR_STATS(i) + 0x8);
1018 : : }
1019 : :
1020 : 0 : hw->eth_stats_base.ipackets =
1021 : : nn_cfg_readq(&hw->super, NFP_NET_CFG_STATS_RX_FRAMES);
1022 : :
1023 : 0 : hw->eth_stats_base.ibytes =
1024 : : nn_cfg_readq(&hw->super, NFP_NET_CFG_STATS_RX_OCTETS);
1025 : :
1026 : 0 : hw->eth_stats_base.opackets =
1027 : : nn_cfg_readq(&hw->super, NFP_NET_CFG_STATS_TX_FRAMES);
1028 : :
1029 : 0 : hw->eth_stats_base.obytes =
1030 : : nn_cfg_readq(&hw->super, NFP_NET_CFG_STATS_TX_OCTETS);
1031 : :
1032 : : /* Reading general device stats */
1033 : 0 : hw->eth_stats_base.ierrors =
1034 : : nn_cfg_readq(&hw->super, NFP_NET_CFG_STATS_RX_ERRORS);
1035 : :
1036 : 0 : hw->eth_stats_base.oerrors =
1037 : : nn_cfg_readq(&hw->super, NFP_NET_CFG_STATS_TX_ERRORS);
1038 : :
1039 : : /* RX ring mbuf allocation failures */
1040 : 0 : dev->data->rx_mbuf_alloc_failed = 0;
1041 : :
1042 : 0 : hw->eth_stats_base.imissed =
1043 : : nn_cfg_readq(&hw->super, NFP_NET_CFG_STATS_RX_DISCARDS);
1044 : :
1045 : 0 : return 0;
1046 : : }
1047 : :
1048 : : uint32_t
1049 [ # # ]: 0 : nfp_net_xstats_size(const struct rte_eth_dev *dev)
1050 : : {
1051 : : uint32_t count;
1052 : : bool vf_flag = false;
1053 : : struct nfp_net_hw *hw;
1054 : : struct nfp_flower_representor *repr;
1055 : : const uint32_t size = RTE_DIM(nfp_net_xstats);
1056 : :
1057 [ # # ]: 0 : if (rte_eth_dev_is_repr(dev)) {
1058 : 0 : repr = dev->data->dev_private;
1059 [ # # ]: 0 : if (nfp_flower_repr_is_vf(repr))
1060 : : vf_flag = true;
1061 : : } else {
1062 : 0 : hw = dev->data->dev_private;
1063 [ # # ]: 0 : if (hw->mac_stats == NULL)
1064 : : vf_flag = true;
1065 : : }
1066 : :
1067 : : /* If the device is a VF or VF-repr, then there will be no MAC stats */
1068 : : if (vf_flag) {
1069 [ # # ]: 0 : for (count = 0; count < size; count++) {
1070 [ # # ]: 0 : if (nfp_net_xstats[count].group == NFP_XSTAT_GROUP_MAC)
1071 : : break;
1072 : : }
1073 : :
1074 : 0 : return count;
1075 : : }
1076 : :
1077 : : return size;
1078 : : }
1079 : :
1080 : : static const struct nfp_xstat *
1081 : 0 : nfp_net_xstats_info(const struct rte_eth_dev *dev,
1082 : : uint32_t index)
1083 : : {
1084 [ # # ]: 0 : if (index >= nfp_net_xstats_size(dev)) {
1085 : 0 : PMD_DRV_LOG(ERR, "The xstat index out of bounds.");
1086 : 0 : return NULL;
1087 : : }
1088 : :
1089 : 0 : return &nfp_net_xstats[index];
1090 : : }
1091 : :
1092 : : static uint64_t
1093 [ # # ]: 0 : nfp_net_xstats_value(const struct rte_eth_dev *dev,
1094 : : uint32_t index,
1095 : : bool raw)
1096 : : {
1097 : : uint64_t value;
1098 : : uint8_t *mac_stats;
1099 : : struct nfp_net_hw *hw;
1100 : : struct nfp_xstat xstat;
1101 : : struct rte_eth_xstat *xstats_base;
1102 : : struct nfp_flower_representor *repr;
1103 : :
1104 [ # # ]: 0 : if (rte_eth_dev_is_repr(dev)) {
1105 : 0 : repr = dev->data->dev_private;
1106 : 0 : hw = repr->app_fw_flower->pf_hw;
1107 : :
1108 : 0 : mac_stats = repr->mac_stats;
1109 : 0 : xstats_base = repr->repr_xstats_base;
1110 : : } else {
1111 : 0 : hw = dev->data->dev_private;
1112 : :
1113 : 0 : mac_stats = hw->mac_stats;
1114 : 0 : xstats_base = hw->eth_xstats_base;
1115 : : }
1116 : :
1117 : 0 : xstat = nfp_net_xstats[index];
1118 : :
1119 [ # # ]: 0 : if (xstat.group == NFP_XSTAT_GROUP_MAC)
1120 : 0 : value = nn_readq(mac_stats + xstat.offset);
1121 : : else
1122 : 0 : value = nn_cfg_readq(&hw->super, xstat.offset);
1123 : :
1124 [ # # ]: 0 : if (raw)
1125 : : return value;
1126 : :
1127 : : /*
1128 : : * A baseline value of each statistic counter is recorded when stats are "reset".
1129 : : * Thus, the value returned by this function need to be decremented by this
1130 : : * baseline value. The result is the count of this statistic since the last time
1131 : : * it was "reset".
1132 : : */
1133 : 0 : return value - xstats_base[index].value;
1134 : : }
1135 : :
1136 : : /* NOTE: All callers ensure dev is always set. */
1137 : : int
1138 : 0 : nfp_net_xstats_get_names(struct rte_eth_dev *dev,
1139 : : struct rte_eth_xstat_name *xstats_names,
1140 : : unsigned int size)
1141 : : {
1142 : : uint32_t id;
1143 : : uint32_t nfp_size;
1144 : : uint32_t read_size;
1145 : :
1146 : 0 : nfp_size = nfp_net_xstats_size(dev);
1147 : :
1148 [ # # ]: 0 : if (xstats_names == NULL)
1149 : 0 : return nfp_size;
1150 : :
1151 : : /* Read at most NFP xstats number of names. */
1152 : 0 : read_size = RTE_MIN(size, nfp_size);
1153 : :
1154 [ # # ]: 0 : for (id = 0; id < read_size; id++)
1155 : 0 : rte_strlcpy(xstats_names[id].name, nfp_net_xstats[id].name,
1156 : : RTE_ETH_XSTATS_NAME_SIZE);
1157 : :
1158 : 0 : return read_size;
1159 : : }
1160 : :
1161 : : /* NOTE: All callers ensure dev is always set. */
1162 : : int
1163 : 0 : nfp_net_xstats_get(struct rte_eth_dev *dev,
1164 : : struct rte_eth_xstat *xstats,
1165 : : unsigned int n)
1166 : : {
1167 : : uint32_t id;
1168 : : uint32_t nfp_size;
1169 : : uint32_t read_size;
1170 : :
1171 : 0 : nfp_size = nfp_net_xstats_size(dev);
1172 : :
1173 [ # # ]: 0 : if (xstats == NULL)
1174 : 0 : return nfp_size;
1175 : :
1176 : : /* Read at most NFP xstats number of values. */
1177 : 0 : read_size = RTE_MIN(n, nfp_size);
1178 : :
1179 [ # # ]: 0 : for (id = 0; id < read_size; id++) {
1180 : 0 : xstats[id].id = id;
1181 : 0 : xstats[id].value = nfp_net_xstats_value(dev, id, false);
1182 : : }
1183 : :
1184 : 0 : return read_size;
1185 : : }
1186 : :
1187 : : /*
1188 : : * NOTE: The only caller rte_eth_xstats_get_names_by_id() ensures dev,
1189 : : * ids, xstats_names and size are valid, and non-NULL.
1190 : : */
1191 : : int
1192 : 0 : nfp_net_xstats_get_names_by_id(struct rte_eth_dev *dev,
1193 : : const uint64_t *ids,
1194 : : struct rte_eth_xstat_name *xstats_names,
1195 : : unsigned int size)
1196 : : {
1197 : : uint32_t i;
1198 : : uint32_t read_size;
1199 : :
1200 : : /* Read at most NFP xstats number of names. */
1201 : 0 : read_size = RTE_MIN(size, nfp_net_xstats_size(dev));
1202 : :
1203 [ # # ]: 0 : for (i = 0; i < read_size; i++) {
1204 : : const struct nfp_xstat *xstat;
1205 : :
1206 : : /* Make sure ID is valid for device. */
1207 : 0 : xstat = nfp_net_xstats_info(dev, ids[i]);
1208 [ # # ]: 0 : if (xstat == NULL)
1209 : : return -EINVAL;
1210 : :
1211 : 0 : rte_strlcpy(xstats_names[i].name, xstat->name,
1212 : : RTE_ETH_XSTATS_NAME_SIZE);
1213 : : }
1214 : :
1215 : 0 : return read_size;
1216 : : }
1217 : :
1218 : : /*
1219 : : * NOTE: The only caller rte_eth_xstats_get_by_id() ensures dev,
1220 : : * ids, values and n are valid, and non-NULL.
1221 : : */
1222 : : int
1223 : 0 : nfp_net_xstats_get_by_id(struct rte_eth_dev *dev,
1224 : : const uint64_t *ids,
1225 : : uint64_t *values,
1226 : : unsigned int n)
1227 : : {
1228 : : uint32_t i;
1229 : : uint32_t read_size;
1230 : :
1231 : : /* Read at most NFP xstats number of values. */
1232 : 0 : read_size = RTE_MIN(n, nfp_net_xstats_size(dev));
1233 : :
1234 [ # # ]: 0 : for (i = 0; i < read_size; i++) {
1235 : : const struct nfp_xstat *xstat;
1236 : :
1237 : : /* Make sure index is valid for device. */
1238 : 0 : xstat = nfp_net_xstats_info(dev, ids[i]);
1239 [ # # ]: 0 : if (xstat == NULL)
1240 : : return -EINVAL;
1241 : :
1242 : 0 : values[i] = nfp_net_xstats_value(dev, ids[i], false);
1243 : : }
1244 : :
1245 : 0 : return read_size;
1246 : : }
1247 : :
1248 : : int
1249 : 0 : nfp_net_xstats_reset(struct rte_eth_dev *dev)
1250 : : {
1251 : : uint32_t id;
1252 : : uint32_t read_size;
1253 : : struct nfp_net_hw *hw;
1254 : : struct rte_eth_xstat *xstats_base;
1255 : : struct nfp_flower_representor *repr;
1256 : :
1257 : 0 : read_size = nfp_net_xstats_size(dev);
1258 : :
1259 [ # # ]: 0 : if (rte_eth_dev_is_repr(dev)) {
1260 : 0 : repr = dev->data->dev_private;
1261 : 0 : xstats_base = repr->repr_xstats_base;
1262 : : } else {
1263 : 0 : hw = dev->data->dev_private;
1264 : 0 : xstats_base = hw->eth_xstats_base;
1265 : : }
1266 : :
1267 [ # # ]: 0 : for (id = 0; id < read_size; id++) {
1268 : 0 : xstats_base[id].id = id;
1269 : 0 : xstats_base[id].value = nfp_net_xstats_value(dev, id, true);
1270 : : }
1271 : :
1272 : : /* Successfully reset xstats, now call function to reset basic stats. */
1273 [ # # ]: 0 : if (rte_eth_dev_is_repr(dev))
1274 : 0 : return nfp_flower_repr_stats_reset(dev);
1275 : : else
1276 : 0 : return nfp_net_stats_reset(dev);
1277 : : }
1278 : :
1279 : : void
1280 : 0 : nfp_net_rx_desc_limits(struct nfp_net_hw_priv *hw_priv,
1281 : : uint16_t *min_rx_desc,
1282 : : uint16_t *max_rx_desc)
1283 : : {
1284 : 0 : *max_rx_desc = hw_priv->dev_info->max_qc_size;
1285 : 0 : *min_rx_desc = hw_priv->dev_info->min_qc_size;
1286 : 0 : }
1287 : :
1288 : : void
1289 : 0 : nfp_net_tx_desc_limits(struct nfp_net_hw_priv *hw_priv,
1290 : : uint16_t *min_tx_desc,
1291 : : uint16_t *max_tx_desc)
1292 : : {
1293 : : uint16_t tx_dpp;
1294 : :
1295 [ # # ]: 0 : if (hw_priv->pf_dev->ver.extend == NFP_NET_CFG_VERSION_DP_NFD3)
1296 : : tx_dpp = NFD3_TX_DESC_PER_PKT;
1297 : : else
1298 : : tx_dpp = NFDK_TX_DESC_PER_SIMPLE_PKT;
1299 : :
1300 : 0 : *max_tx_desc = hw_priv->dev_info->max_qc_size / tx_dpp;
1301 : 0 : *min_tx_desc = hw_priv->dev_info->min_qc_size / tx_dpp;
1302 : 0 : }
1303 : :
1304 : : int
1305 : 0 : nfp_net_infos_get(struct rte_eth_dev *dev, struct rte_eth_dev_info *dev_info)
1306 : : {
1307 : : uint32_t cap;
1308 : : uint32_t cap_extend;
1309 : : uint16_t min_rx_desc;
1310 : : uint16_t max_rx_desc;
1311 : : uint16_t min_tx_desc;
1312 : : uint16_t max_tx_desc;
1313 : : struct nfp_net_hw *hw;
1314 : : struct nfp_net_hw_priv *hw_priv;
1315 : :
1316 : 0 : hw = nfp_net_get_hw(dev);
1317 : 0 : hw_priv = dev->process_private;
1318 [ # # ]: 0 : if (hw_priv == NULL)
1319 : : return -EINVAL;
1320 : :
1321 : 0 : nfp_net_rx_desc_limits(hw_priv, &min_rx_desc, &max_rx_desc);
1322 : 0 : nfp_net_tx_desc_limits(hw_priv, &min_tx_desc, &max_tx_desc);
1323 : :
1324 : 0 : dev_info->max_rx_queues = (uint16_t)hw->max_rx_queues;
1325 : 0 : dev_info->max_tx_queues = (uint16_t)hw->max_tx_queues;
1326 : 0 : dev_info->min_rx_bufsize = RTE_ETHER_MIN_MTU;
1327 : : /*
1328 : : * The maximum rx packet length is set to the maximum layer 3 MTU,
1329 : : * plus layer 2, CRC and VLAN headers.
1330 : : * The maximum layer 3 MTU (max_mtu) is read from hardware,
1331 : : * which was set by the firmware loaded onto the card.
1332 : : */
1333 : 0 : dev_info->max_rx_pktlen = hw->max_mtu + NFP_ETH_OVERHEAD;
1334 : 0 : dev_info->max_mtu = hw->max_mtu;
1335 : 0 : dev_info->min_mtu = RTE_ETHER_MIN_MTU;
1336 : : /* Next should change when PF support is implemented */
1337 : 0 : dev_info->max_mac_addrs = 1;
1338 : :
1339 : 0 : cap = hw->super.cap;
1340 : :
1341 [ # # ]: 0 : if ((cap & (NFP_NET_CFG_CTRL_RXVLAN | NFP_NET_CFG_CTRL_RXVLAN_V2)) != 0)
1342 : 0 : dev_info->rx_offload_capa = RTE_ETH_RX_OFFLOAD_VLAN_STRIP;
1343 : :
1344 [ # # ]: 0 : if ((cap & NFP_NET_CFG_CTRL_RXQINQ) != 0)
1345 : 0 : dev_info->rx_offload_capa |= RTE_ETH_RX_OFFLOAD_QINQ_STRIP;
1346 : :
1347 [ # # ]: 0 : if ((cap & NFP_NET_CFG_CTRL_RXCSUM) != 0)
1348 : 0 : dev_info->rx_offload_capa |= RTE_ETH_RX_OFFLOAD_IPV4_CKSUM |
1349 : : RTE_ETH_RX_OFFLOAD_UDP_CKSUM |
1350 : : RTE_ETH_RX_OFFLOAD_TCP_CKSUM;
1351 : :
1352 [ # # ]: 0 : if ((cap & (NFP_NET_CFG_CTRL_TXVLAN | NFP_NET_CFG_CTRL_TXVLAN_V2)) != 0)
1353 : 0 : dev_info->tx_offload_capa = RTE_ETH_TX_OFFLOAD_VLAN_INSERT;
1354 : :
1355 [ # # ]: 0 : if ((cap & NFP_NET_CFG_CTRL_TXCSUM) != 0)
1356 : 0 : dev_info->tx_offload_capa |= RTE_ETH_TX_OFFLOAD_IPV4_CKSUM |
1357 : : RTE_ETH_TX_OFFLOAD_UDP_CKSUM |
1358 : : RTE_ETH_TX_OFFLOAD_TCP_CKSUM;
1359 : :
1360 [ # # ]: 0 : if ((cap & NFP_NET_CFG_CTRL_LSO_ANY) != 0) {
1361 : 0 : dev_info->tx_offload_capa |= RTE_ETH_TX_OFFLOAD_TCP_TSO;
1362 [ # # ]: 0 : if ((cap & NFP_NET_CFG_CTRL_USO) != 0)
1363 : 0 : dev_info->tx_offload_capa |= RTE_ETH_TX_OFFLOAD_UDP_TSO;
1364 [ # # ]: 0 : if ((cap & NFP_NET_CFG_CTRL_VXLAN) != 0)
1365 : 0 : dev_info->tx_offload_capa |= RTE_ETH_TX_OFFLOAD_VXLAN_TNL_TSO;
1366 : : }
1367 : :
1368 [ # # ]: 0 : if ((cap & NFP_NET_CFG_CTRL_GATHER) != 0)
1369 : 0 : dev_info->tx_offload_capa |= RTE_ETH_TX_OFFLOAD_MULTI_SEGS;
1370 : :
1371 : 0 : cap_extend = hw->super.cap_ext;
1372 [ # # ]: 0 : if ((cap_extend & NFP_NET_CFG_CTRL_IPSEC) != 0) {
1373 : 0 : dev_info->tx_offload_capa |= RTE_ETH_TX_OFFLOAD_SECURITY;
1374 : 0 : dev_info->rx_offload_capa |= RTE_ETH_RX_OFFLOAD_SECURITY;
1375 : : }
1376 : :
1377 : 0 : dev_info->default_rxconf = (struct rte_eth_rxconf) {
1378 : : .rx_thresh = {
1379 : : .pthresh = DEFAULT_RX_PTHRESH,
1380 : : .hthresh = DEFAULT_RX_HTHRESH,
1381 : : .wthresh = DEFAULT_RX_WTHRESH,
1382 : : },
1383 : : .rx_free_thresh = DEFAULT_RX_FREE_THRESH,
1384 : : .rx_drop_en = 0,
1385 : : };
1386 : :
1387 : 0 : dev_info->default_txconf = (struct rte_eth_txconf) {
1388 : : .tx_thresh = {
1389 : : .pthresh = DEFAULT_TX_PTHRESH,
1390 : : .hthresh = DEFAULT_TX_HTHRESH,
1391 : : .wthresh = DEFAULT_TX_WTHRESH,
1392 : : },
1393 : : .tx_free_thresh = DEFAULT_TX_FREE_THRESH,
1394 : : .tx_rs_thresh = DEFAULT_TX_RSBIT_THRESH,
1395 : : };
1396 : :
1397 : 0 : dev_info->rx_desc_lim = (struct rte_eth_desc_lim) {
1398 : : .nb_max = max_rx_desc,
1399 : : .nb_min = min_rx_desc,
1400 : : .nb_align = NFP_ALIGN_RING_DESC,
1401 : : };
1402 : :
1403 : 0 : dev_info->tx_desc_lim = (struct rte_eth_desc_lim) {
1404 : : .nb_max = max_tx_desc,
1405 : : .nb_min = min_tx_desc,
1406 : : .nb_align = NFP_ALIGN_RING_DESC,
1407 : : .nb_seg_max = NFP_TX_MAX_SEG,
1408 : : .nb_mtu_seg_max = NFP_TX_MAX_MTU_SEG,
1409 : : };
1410 : :
1411 [ # # ]: 0 : if ((cap & NFP_NET_CFG_CTRL_RSS_ANY) != 0) {
1412 : 0 : dev_info->rx_offload_capa |= RTE_ETH_RX_OFFLOAD_RSS_HASH;
1413 : 0 : dev_info->flow_type_rss_offloads = NFP_NET_RSS_CAP;
1414 : 0 : dev_info->reta_size = NFP_NET_CFG_RSS_ITBL_SZ;
1415 : 0 : dev_info->hash_key_size = NFP_NET_CFG_RSS_KEY_SZ;
1416 : 0 : nfp_net_rss_algo_capa_get(hw, dev_info);
1417 : : }
1418 : :
1419 : : /* Only PF supports getting speed capability. */
1420 [ # # ]: 0 : if (hw_priv->is_pf)
1421 : 0 : dev_info->speed_capa = hw_priv->pf_dev->speed_capa;
1422 : :
1423 : : return 0;
1424 : : }
1425 : :
1426 : : int
1427 : 0 : nfp_net_common_init(struct nfp_pf_dev *pf_dev,
1428 : : struct nfp_net_hw *hw)
1429 : : {
1430 : : const int stride = 4;
1431 : : struct rte_pci_device *pci_dev;
1432 : :
1433 : 0 : pci_dev = pf_dev->pci_dev;
1434 : 0 : hw->device_id = pci_dev->id.device_id;
1435 : 0 : hw->vendor_id = pci_dev->id.vendor_id;
1436 : 0 : hw->subsystem_device_id = pci_dev->id.subsystem_device_id;
1437 : 0 : hw->subsystem_vendor_id = pci_dev->id.subsystem_vendor_id;
1438 : :
1439 : 0 : hw->max_rx_queues = nn_cfg_readl(&hw->super, NFP_NET_CFG_MAX_RXRINGS);
1440 : 0 : hw->max_tx_queues = nn_cfg_readl(&hw->super, NFP_NET_CFG_MAX_TXRINGS);
1441 : 0 : hw->super.cap_rss = nn_cfg_readl(&hw->super, NFP_NET_CFG_RSS_CAP);
1442 [ # # # # ]: 0 : if (hw->max_rx_queues == 0 || hw->max_tx_queues == 0) {
1443 : 0 : PMD_INIT_LOG(ERR, "Device %s can not be used, there are no valid queue "
1444 : : "pairs for use.", pci_dev->name);
1445 : 0 : return -ENODEV;
1446 : : }
1447 : :
1448 [ # # ]: 0 : if (nfp_net_check_dma_mask(pf_dev, pci_dev->name) != 0)
1449 : : return -ENODEV;
1450 : :
1451 : : /* Get some of the read-only fields from the config BAR */
1452 : 0 : hw->super.cap = nn_cfg_readl(&hw->super, NFP_NET_CFG_CAP);
1453 : 0 : hw->super.cap_ext = nn_cfg_readl(&hw->super, NFP_NET_CFG_CAP_WORD1);
1454 : 0 : hw->max_mtu = nn_cfg_readl(&hw->super, NFP_NET_CFG_MAX_MTU);
1455 : 0 : hw->flbufsz = DEFAULT_FLBUF_SIZE;
1456 : :
1457 : 0 : nfp_net_meta_init_format(hw, pf_dev);
1458 : :
1459 : : /* Read the Rx offset configured from firmware */
1460 [ # # ]: 0 : if (pf_dev->ver.major < 2)
1461 : 0 : hw->rx_offset = NFP_NET_RX_OFFSET;
1462 : : else
1463 : 0 : hw->rx_offset = nn_cfg_readl(&hw->super, NFP_NET_CFG_RX_OFFSET);
1464 : :
1465 : 0 : hw->super.ctrl = 0;
1466 : 0 : hw->stride_rx = stride;
1467 : 0 : hw->stride_tx = stride;
1468 : :
1469 : 0 : return 0;
1470 : : }
1471 : :
1472 : : const uint32_t *
1473 : 0 : nfp_net_supported_ptypes_get(struct rte_eth_dev *dev, size_t *no_of_elements)
1474 : : {
1475 : : struct nfp_net_hw *net_hw;
1476 : : static const uint32_t ptypes[] = {
1477 : : RTE_PTYPE_L2_ETHER,
1478 : : RTE_PTYPE_L3_IPV4,
1479 : : RTE_PTYPE_L3_IPV4_EXT,
1480 : : RTE_PTYPE_L3_IPV6,
1481 : : RTE_PTYPE_L3_IPV6_EXT,
1482 : : RTE_PTYPE_L3_IPV4_EXT_UNKNOWN,
1483 : : RTE_PTYPE_L3_IPV6_EXT_UNKNOWN,
1484 : : RTE_PTYPE_L4_TCP,
1485 : : RTE_PTYPE_L4_UDP,
1486 : : RTE_PTYPE_L4_FRAG,
1487 : : RTE_PTYPE_L4_NONFRAG,
1488 : : RTE_PTYPE_L4_ICMP,
1489 : : RTE_PTYPE_L4_SCTP,
1490 : : RTE_PTYPE_TUNNEL_VXLAN,
1491 : : RTE_PTYPE_TUNNEL_NVGRE,
1492 : : RTE_PTYPE_TUNNEL_GENEVE,
1493 : : RTE_PTYPE_INNER_L2_ETHER,
1494 : : RTE_PTYPE_INNER_L3_IPV4,
1495 : : RTE_PTYPE_INNER_L3_IPV4_EXT,
1496 : : RTE_PTYPE_INNER_L3_IPV6,
1497 : : RTE_PTYPE_INNER_L3_IPV6_EXT,
1498 : : RTE_PTYPE_INNER_L3_IPV4_EXT_UNKNOWN,
1499 : : RTE_PTYPE_INNER_L3_IPV6_EXT_UNKNOWN,
1500 : : RTE_PTYPE_INNER_L4_TCP,
1501 : : RTE_PTYPE_INNER_L4_UDP,
1502 : : RTE_PTYPE_INNER_L4_FRAG,
1503 : : RTE_PTYPE_INNER_L4_NONFRAG,
1504 : : RTE_PTYPE_INNER_L4_ICMP,
1505 : : RTE_PTYPE_INNER_L4_SCTP,
1506 : : };
1507 : :
1508 [ # # ]: 0 : if (dev->rx_pkt_burst == NULL)
1509 : : return NULL;
1510 : :
1511 : 0 : net_hw = dev->data->dev_private;
1512 [ # # ]: 0 : if ((net_hw->super.cap_ext & NFP_NET_CFG_CTRL_PKT_TYPE) == 0)
1513 : : return NULL;
1514 : :
1515 : 0 : *no_of_elements = RTE_DIM(ptypes);
1516 : 0 : return ptypes;
1517 : : }
1518 : :
1519 : : int
1520 : 0 : nfp_net_ptypes_set(struct rte_eth_dev *dev,
1521 : : uint32_t ptype_mask)
1522 : : {
1523 : : int ret;
1524 : : uint32_t update;
1525 : : uint32_t ctrl_ext;
1526 : : struct nfp_hw *hw;
1527 : : struct nfp_net_hw *net_hw;
1528 : :
1529 : 0 : net_hw = dev->data->dev_private;
1530 : 0 : hw = &net_hw->super;
1531 : :
1532 [ # # ]: 0 : if ((hw->cap_ext & NFP_NET_CFG_CTRL_PKT_TYPE) == 0)
1533 : : return -ENOTSUP;
1534 : :
1535 : 0 : ctrl_ext = hw->ctrl_ext;
1536 [ # # ]: 0 : if (ptype_mask == 0) {
1537 [ # # ]: 0 : if ((ctrl_ext & NFP_NET_CFG_CTRL_PKT_TYPE) == 0)
1538 : : return 0;
1539 : :
1540 : 0 : ctrl_ext &= ~NFP_NET_CFG_CTRL_PKT_TYPE;
1541 : : } else {
1542 [ # # ]: 0 : if ((ctrl_ext & NFP_NET_CFG_CTRL_PKT_TYPE) != 0)
1543 : : return 0;
1544 : :
1545 : 0 : ctrl_ext |= NFP_NET_CFG_CTRL_PKT_TYPE;
1546 : : }
1547 : :
1548 : : update = NFP_NET_CFG_UPDATE_GEN;
1549 : :
1550 : 0 : ret = nfp_ext_reconfig(hw, ctrl_ext, update);
1551 [ # # ]: 0 : if (ret != 0)
1552 : : return ret;
1553 : :
1554 : 0 : hw->ctrl_ext = ctrl_ext;
1555 : :
1556 : 0 : return 0;
1557 : : }
1558 : :
1559 : : int
1560 : 0 : nfp_rx_queue_intr_enable(struct rte_eth_dev *dev,
1561 : : uint16_t queue_id)
1562 : : {
1563 : : uint16_t base = 0;
1564 : : struct nfp_net_hw *hw;
1565 : : struct rte_pci_device *pci_dev;
1566 : :
1567 : 0 : pci_dev = RTE_ETH_DEV_TO_PCI(dev);
1568 [ # # ]: 0 : if (rte_intr_type_get(pci_dev->intr_handle) != RTE_INTR_HANDLE_UIO)
1569 : : base = 1;
1570 : :
1571 : : /* Make sure all updates are written before un-masking */
1572 : : rte_wmb();
1573 : :
1574 : 0 : hw = nfp_net_get_hw(dev);
1575 : 0 : nn_cfg_writeb(&hw->super, NFP_NET_CFG_ICR(base + queue_id),
1576 : : NFP_NET_CFG_ICR_UNMASKED);
1577 : 0 : return 0;
1578 : : }
1579 : :
1580 : : int
1581 : 0 : nfp_rx_queue_intr_disable(struct rte_eth_dev *dev,
1582 : : uint16_t queue_id)
1583 : : {
1584 : : uint16_t base = 0;
1585 : : struct nfp_net_hw *hw;
1586 : : struct rte_pci_device *pci_dev;
1587 : :
1588 : 0 : pci_dev = RTE_ETH_DEV_TO_PCI(dev);
1589 [ # # ]: 0 : if (rte_intr_type_get(pci_dev->intr_handle) != RTE_INTR_HANDLE_UIO)
1590 : : base = 1;
1591 : :
1592 : : /* Make sure all updates are written before un-masking */
1593 : : rte_wmb();
1594 : :
1595 : 0 : hw = nfp_net_get_hw(dev);
1596 : 0 : nn_cfg_writeb(&hw->super, NFP_NET_CFG_ICR(base + queue_id), NFP_NET_CFG_ICR_RXTX);
1597 : :
1598 : 0 : return 0;
1599 : : }
1600 : :
1601 : : static void
1602 : 0 : nfp_net_dev_link_status_print(struct rte_eth_dev *dev)
1603 : : {
1604 : : struct rte_eth_link link;
1605 : 0 : struct rte_pci_device *pci_dev = RTE_ETH_DEV_TO_PCI(dev);
1606 : :
1607 : 0 : rte_eth_linkstatus_get(dev, &link);
1608 [ # # ]: 0 : if (link.link_status != 0)
1609 [ # # ]: 0 : PMD_DRV_LOG(INFO, "Port %d: Link Up - speed %u Mbps - %s.",
1610 : : dev->data->port_id, link.link_speed,
1611 : : link.link_duplex == RTE_ETH_LINK_FULL_DUPLEX ?
1612 : : "full-duplex" : "half-duplex");
1613 : : else
1614 : 0 : PMD_DRV_LOG(INFO, " Port %d: Link Down.", dev->data->port_id);
1615 : :
1616 : 0 : PMD_DRV_LOG(INFO, "PCI Address: " PCI_PRI_FMT,
1617 : : pci_dev->addr.domain, pci_dev->addr.bus,
1618 : : pci_dev->addr.devid, pci_dev->addr.function);
1619 : 0 : }
1620 : :
1621 : : /*
1622 : : * Unmask an interrupt
1623 : : *
1624 : : * If MSI-X auto-masking is enabled clear the mask bit, otherwise
1625 : : * clear the ICR for the entry.
1626 : : */
1627 : : void
1628 : 0 : nfp_net_irq_unmask(struct rte_eth_dev *dev)
1629 : : {
1630 : : struct nfp_net_hw *hw;
1631 : : struct rte_pci_device *pci_dev;
1632 : :
1633 : 0 : hw = nfp_net_get_hw(dev);
1634 : 0 : pci_dev = RTE_ETH_DEV_TO_PCI(dev);
1635 : :
1636 : : /* Make sure all updates are written before un-masking */
1637 : : rte_wmb();
1638 : :
1639 [ # # ]: 0 : if ((hw->super.ctrl & NFP_NET_CFG_CTRL_MSIXAUTO) != 0) {
1640 : : /* If MSI-X auto-masking is used, clear the entry */
1641 : 0 : rte_intr_ack(pci_dev->intr_handle);
1642 : : } else {
1643 : : nn_cfg_writeb(&hw->super, NFP_NET_CFG_ICR(NFP_NET_IRQ_LSC_IDX),
1644 : : NFP_NET_CFG_ICR_UNMASKED);
1645 : : }
1646 : 0 : }
1647 : :
1648 : : /**
1649 : : * Interrupt handler which shall be registered for alarm callback for delayed
1650 : : * handling specific interrupt to wait for the stable nic state. As the NIC
1651 : : * interrupt state is not stable for nfp after link is just down, it needs
1652 : : * to wait 4 seconds to get the stable status.
1653 : : *
1654 : : * @param param
1655 : : * The address of parameter (struct rte_eth_dev *)
1656 : : */
1657 : : void
1658 : 0 : nfp_net_dev_interrupt_delayed_handler(void *param)
1659 : : {
1660 : : struct rte_eth_dev *dev = param;
1661 : :
1662 : 0 : nfp_net_link_update(dev, 0);
1663 : 0 : rte_eth_dev_callback_process(dev, RTE_ETH_EVENT_INTR_LSC, NULL);
1664 : :
1665 : 0 : nfp_net_dev_link_status_print(dev);
1666 : :
1667 : : /* Unmasking */
1668 : 0 : nfp_net_irq_unmask(dev);
1669 : 0 : }
1670 : :
1671 : : void
1672 : 0 : nfp_net_dev_interrupt_handler(void *param)
1673 : : {
1674 : : int64_t timeout;
1675 : : struct rte_eth_link link;
1676 : : struct rte_eth_dev *dev = param;
1677 : :
1678 : 0 : PMD_DRV_LOG(DEBUG, "We got a LSC interrupt!!!");
1679 : :
1680 : 0 : rte_eth_linkstatus_get(dev, &link);
1681 : :
1682 : 0 : nfp_net_link_update(dev, 0);
1683 : :
1684 : : /* Likely to up */
1685 [ # # ]: 0 : if (link.link_status == 0) {
1686 : : /* Handle it 1 sec later, wait it being stable */
1687 : : timeout = NFP_NET_LINK_UP_CHECK_TIMEOUT;
1688 : : } else { /* Likely to down */
1689 : : /* Handle it 4 sec later, wait it being stable */
1690 : : timeout = NFP_NET_LINK_DOWN_CHECK_TIMEOUT;
1691 : : }
1692 : :
1693 [ # # ]: 0 : if (rte_eal_alarm_set(timeout * 1000,
1694 : : nfp_net_dev_interrupt_delayed_handler,
1695 : : (void *)dev) != 0) {
1696 : 0 : PMD_INIT_LOG(ERR, "Error setting alarm.");
1697 : : /* Unmasking */
1698 : 0 : nfp_net_irq_unmask(dev);
1699 : : }
1700 : 0 : }
1701 : :
1702 : : int
1703 : 0 : nfp_net_dev_mtu_set(struct rte_eth_dev *dev,
1704 : : uint16_t mtu)
1705 : : {
1706 : : struct nfp_net_hw *hw;
1707 : :
1708 : 0 : hw = nfp_net_get_hw(dev);
1709 : :
1710 : : /* MTU setting is forbidden if port is started */
1711 [ # # ]: 0 : if (dev->data->dev_started) {
1712 : 0 : PMD_DRV_LOG(ERR, "Port %d must be stopped before configuration.",
1713 : : dev->data->port_id);
1714 : 0 : return -EBUSY;
1715 : : }
1716 : :
1717 : : /* MTU larger than current mbufsize not supported */
1718 [ # # ]: 0 : if (mtu > hw->flbufsz) {
1719 : 0 : PMD_DRV_LOG(ERR, "MTU (%u) larger than current mbufsize (%u) not supported.",
1720 : : mtu, hw->flbufsz);
1721 : 0 : return -ERANGE;
1722 : : }
1723 : :
1724 : : /* Writing to configuration space */
1725 : 0 : nn_cfg_writel(&hw->super, NFP_NET_CFG_MTU, mtu);
1726 : :
1727 : 0 : hw->mtu = mtu;
1728 : :
1729 : 0 : return 0;
1730 : : }
1731 : :
1732 : : int
1733 : 0 : nfp_net_vlan_offload_set(struct rte_eth_dev *dev,
1734 : : int mask)
1735 : : {
1736 : : int ret;
1737 : : uint32_t update;
1738 : : uint32_t new_ctrl;
1739 : : struct nfp_hw *hw;
1740 : : uint64_t rx_offload;
1741 : : struct nfp_net_hw *net_hw;
1742 : : uint32_t rxvlan_ctrl = 0;
1743 : :
1744 : 0 : net_hw = nfp_net_get_hw(dev);
1745 : 0 : hw = &net_hw->super;
1746 : 0 : rx_offload = dev->data->dev_conf.rxmode.offloads;
1747 : 0 : new_ctrl = hw->ctrl;
1748 : :
1749 : : /* VLAN stripping setting */
1750 [ # # ]: 0 : if ((mask & RTE_ETH_VLAN_STRIP_MASK) != 0) {
1751 : : nfp_net_enable_rxvlan_cap(net_hw, &rxvlan_ctrl);
1752 [ # # ]: 0 : if ((rx_offload & RTE_ETH_RX_OFFLOAD_VLAN_STRIP) != 0)
1753 : 0 : new_ctrl |= rxvlan_ctrl;
1754 : : else
1755 : 0 : new_ctrl &= ~rxvlan_ctrl;
1756 : : }
1757 : :
1758 : : /* QinQ stripping setting */
1759 [ # # ]: 0 : if ((mask & RTE_ETH_QINQ_STRIP_MASK) != 0) {
1760 [ # # ]: 0 : if ((rx_offload & RTE_ETH_RX_OFFLOAD_QINQ_STRIP) != 0)
1761 : 0 : new_ctrl |= NFP_NET_CFG_CTRL_RXQINQ;
1762 : : else
1763 : 0 : new_ctrl &= ~NFP_NET_CFG_CTRL_RXQINQ;
1764 : : }
1765 : :
1766 [ # # ]: 0 : if (new_ctrl == hw->ctrl)
1767 : : return 0;
1768 : :
1769 : : update = NFP_NET_CFG_UPDATE_GEN;
1770 : :
1771 : 0 : ret = nfp_reconfig(hw, new_ctrl, update);
1772 [ # # ]: 0 : if (ret != 0)
1773 : : return ret;
1774 : :
1775 : 0 : hw->ctrl = new_ctrl;
1776 : :
1777 : 0 : return 0;
1778 : : }
1779 : :
1780 : : static int
1781 : 0 : nfp_net_rss_reta_write(struct rte_eth_dev *dev,
1782 : : struct rte_eth_rss_reta_entry64 *reta_conf,
1783 : : uint16_t reta_size)
1784 : : {
1785 : : uint16_t i;
1786 : : uint16_t j;
1787 : : uint16_t idx;
1788 : : uint8_t mask;
1789 : : uint32_t reta;
1790 : : uint16_t shift;
1791 : : struct nfp_hw *hw;
1792 : : struct nfp_net_hw *net_hw;
1793 : :
1794 : 0 : net_hw = nfp_net_get_hw(dev);
1795 : : hw = &net_hw->super;
1796 : :
1797 [ # # ]: 0 : if (reta_size != NFP_NET_CFG_RSS_ITBL_SZ) {
1798 : 0 : PMD_DRV_LOG(ERR, "The size of hash lookup table configured (%hu)"
1799 : : " does not match hardware can supported (%d).",
1800 : : reta_size, NFP_NET_CFG_RSS_ITBL_SZ);
1801 : 0 : return -EINVAL;
1802 : : }
1803 : :
1804 : : /*
1805 : : * Update Redirection Table. There are 128 8bit-entries which can be
1806 : : * manage as 32 32bit-entries.
1807 : : */
1808 [ # # ]: 0 : for (i = 0; i < reta_size; i += 4) {
1809 : : /* Handling 4 RSS entries per loop */
1810 : 0 : idx = i / RTE_ETH_RETA_GROUP_SIZE;
1811 : 0 : shift = i % RTE_ETH_RETA_GROUP_SIZE;
1812 : 0 : mask = (uint8_t)((reta_conf[idx].mask >> shift) & 0xF);
1813 [ # # ]: 0 : if (mask == 0)
1814 : 0 : continue;
1815 : :
1816 : : reta = 0;
1817 : :
1818 : : /* If all 4 entries were set, don't need read RETA register */
1819 [ # # ]: 0 : if (mask != 0xF)
1820 : 0 : reta = nn_cfg_readl(hw, NFP_NET_CFG_RSS_ITBL + i);
1821 : :
1822 [ # # ]: 0 : for (j = 0; j < 4; j++) {
1823 [ # # ]: 0 : if ((mask & (0x1 << j)) == 0)
1824 : 0 : continue;
1825 : :
1826 : : /* Clearing the entry bits */
1827 [ # # ]: 0 : if (mask != 0xF)
1828 : 0 : reta &= ~(0xFF << (8 * j));
1829 : :
1830 : 0 : reta |= reta_conf[idx].reta[shift + j] << (8 * j);
1831 : : }
1832 : :
1833 : 0 : nn_cfg_writel(hw, NFP_NET_CFG_RSS_ITBL + (idx * 64) + shift, reta);
1834 : : }
1835 : :
1836 : : return 0;
1837 : : }
1838 : :
1839 : : /* Update Redirection Table(RETA) of Receive Side Scaling of Ethernet device */
1840 : : int
1841 : 0 : nfp_net_reta_update(struct rte_eth_dev *dev,
1842 : : struct rte_eth_rss_reta_entry64 *reta_conf,
1843 : : uint16_t reta_size)
1844 : : {
1845 : : int ret;
1846 : : uint32_t update;
1847 : : struct nfp_hw *hw;
1848 : : struct nfp_net_hw *net_hw;
1849 : :
1850 : 0 : net_hw = nfp_net_get_hw(dev);
1851 : 0 : hw = &net_hw->super;
1852 : :
1853 [ # # ]: 0 : if ((hw->ctrl & NFP_NET_CFG_CTRL_RSS_ANY) == 0)
1854 : : return -EINVAL;
1855 : :
1856 : 0 : ret = nfp_net_rss_reta_write(dev, reta_conf, reta_size);
1857 [ # # ]: 0 : if (ret != 0)
1858 : : return ret;
1859 : :
1860 : : update = NFP_NET_CFG_UPDATE_RSS;
1861 : :
1862 [ # # ]: 0 : if (nfp_reconfig(hw, hw->ctrl, update) != 0)
1863 : 0 : return -EIO;
1864 : :
1865 : : return 0;
1866 : : }
1867 : :
1868 : : /* Query Redirection Table(RETA) of Receive Side Scaling of Ethernet device. */
1869 : : int
1870 : 0 : nfp_net_reta_query(struct rte_eth_dev *dev,
1871 : : struct rte_eth_rss_reta_entry64 *reta_conf,
1872 : : uint16_t reta_size)
1873 : : {
1874 : : uint16_t i;
1875 : : uint16_t j;
1876 : : uint16_t idx;
1877 : : uint8_t mask;
1878 : : uint32_t reta;
1879 : : uint16_t shift;
1880 : : struct nfp_hw *hw;
1881 : : struct nfp_net_hw *net_hw;
1882 : :
1883 : 0 : net_hw = nfp_net_get_hw(dev);
1884 : : hw = &net_hw->super;
1885 : :
1886 [ # # ]: 0 : if ((hw->ctrl & NFP_NET_CFG_CTRL_RSS_ANY) == 0)
1887 : : return -EINVAL;
1888 : :
1889 [ # # ]: 0 : if (reta_size != NFP_NET_CFG_RSS_ITBL_SZ) {
1890 : 0 : PMD_DRV_LOG(ERR, "The size of hash lookup table configured (%d)"
1891 : : " does not match hardware can supported (%d).",
1892 : : reta_size, NFP_NET_CFG_RSS_ITBL_SZ);
1893 : 0 : return -EINVAL;
1894 : : }
1895 : :
1896 : : /*
1897 : : * Reading Redirection Table. There are 128 8bit-entries which can be
1898 : : * manage as 32 32bit-entries.
1899 : : */
1900 [ # # ]: 0 : for (i = 0; i < reta_size; i += 4) {
1901 : : /* Handling 4 RSS entries per loop */
1902 : 0 : idx = i / RTE_ETH_RETA_GROUP_SIZE;
1903 : 0 : shift = i % RTE_ETH_RETA_GROUP_SIZE;
1904 : 0 : mask = (reta_conf[idx].mask >> shift) & 0xF;
1905 : :
1906 [ # # ]: 0 : if (mask == 0)
1907 : 0 : continue;
1908 : :
1909 : 0 : reta = nn_cfg_readl(hw, NFP_NET_CFG_RSS_ITBL + (idx * 64) + shift);
1910 [ # # ]: 0 : for (j = 0; j < 4; j++) {
1911 [ # # ]: 0 : if ((mask & (0x1 << j)) == 0)
1912 : 0 : continue;
1913 : :
1914 : 0 : reta_conf[idx].reta[shift + j] =
1915 : 0 : (uint8_t)((reta >> (8 * j)) & 0xF);
1916 : : }
1917 : : }
1918 : :
1919 : : return 0;
1920 : : }
1921 : :
1922 : : static void
1923 : 0 : nfp_net_rss_hf_get(uint32_t cfg_rss_ctrl,
1924 : : struct rte_eth_rss_conf *rss_conf)
1925 : : {
1926 : : uint64_t rss_hf = 0;
1927 : :
1928 [ # # ]: 0 : if ((cfg_rss_ctrl & NFP_NET_CFG_RSS_IPV4) != 0)
1929 : : rss_hf |= RTE_ETH_RSS_IPV4;
1930 : :
1931 [ # # ]: 0 : if ((cfg_rss_ctrl & NFP_NET_CFG_RSS_IPV4_TCP) != 0)
1932 : 0 : rss_hf |= RTE_ETH_RSS_NONFRAG_IPV4_TCP;
1933 : :
1934 [ # # ]: 0 : if ((cfg_rss_ctrl & NFP_NET_CFG_RSS_IPV6_TCP) != 0)
1935 : 0 : rss_hf |= RTE_ETH_RSS_NONFRAG_IPV6_TCP;
1936 : :
1937 [ # # ]: 0 : if ((cfg_rss_ctrl & NFP_NET_CFG_RSS_IPV4_UDP) != 0)
1938 : 0 : rss_hf |= RTE_ETH_RSS_NONFRAG_IPV4_UDP;
1939 : :
1940 [ # # ]: 0 : if ((cfg_rss_ctrl & NFP_NET_CFG_RSS_IPV6_UDP) != 0)
1941 : 0 : rss_hf |= RTE_ETH_RSS_NONFRAG_IPV6_UDP;
1942 : :
1943 [ # # ]: 0 : if ((cfg_rss_ctrl & NFP_NET_CFG_RSS_IPV6) != 0)
1944 : 0 : rss_hf |= RTE_ETH_RSS_IPV6;
1945 : :
1946 [ # # ]: 0 : if ((cfg_rss_ctrl & NFP_NET_CFG_RSS_IPV4_SCTP) != 0)
1947 : 0 : rss_hf |= RTE_ETH_RSS_NONFRAG_IPV4_SCTP;
1948 : :
1949 [ # # ]: 0 : if ((cfg_rss_ctrl & NFP_NET_CFG_RSS_IPV6_SCTP) != 0)
1950 : 0 : rss_hf |= RTE_ETH_RSS_NONFRAG_IPV6_SCTP;
1951 : :
1952 : 0 : rss_conf->rss_hf = rss_hf;
1953 : 0 : }
1954 : :
1955 : : static void
1956 : 0 : nfp_net_rss_hf_set(uint32_t *cfg_rss_ctrl,
1957 : : struct rte_eth_rss_conf *rss_conf)
1958 : : {
1959 : 0 : uint64_t rss_hf = rss_conf->rss_hf;
1960 : :
1961 [ # # ]: 0 : if ((rss_hf & RTE_ETH_RSS_IPV4) != 0)
1962 : 0 : *cfg_rss_ctrl |= NFP_NET_CFG_RSS_IPV4;
1963 : :
1964 [ # # ]: 0 : if ((rss_hf & RTE_ETH_RSS_NONFRAG_IPV4_TCP) != 0)
1965 : 0 : *cfg_rss_ctrl |= NFP_NET_CFG_RSS_IPV4_TCP;
1966 : :
1967 [ # # ]: 0 : if ((rss_hf & RTE_ETH_RSS_NONFRAG_IPV4_UDP) != 0)
1968 : 0 : *cfg_rss_ctrl |= NFP_NET_CFG_RSS_IPV4_UDP;
1969 : :
1970 [ # # ]: 0 : if ((rss_hf & RTE_ETH_RSS_NONFRAG_IPV4_SCTP) != 0)
1971 : 0 : *cfg_rss_ctrl |= NFP_NET_CFG_RSS_IPV4_SCTP;
1972 : :
1973 [ # # ]: 0 : if ((rss_hf & RTE_ETH_RSS_IPV6) != 0)
1974 : 0 : *cfg_rss_ctrl |= NFP_NET_CFG_RSS_IPV6;
1975 : :
1976 [ # # ]: 0 : if ((rss_hf & RTE_ETH_RSS_NONFRAG_IPV6_TCP) != 0)
1977 : 0 : *cfg_rss_ctrl |= NFP_NET_CFG_RSS_IPV6_TCP;
1978 : :
1979 [ # # ]: 0 : if ((rss_hf & RTE_ETH_RSS_NONFRAG_IPV6_UDP) != 0)
1980 : 0 : *cfg_rss_ctrl |= NFP_NET_CFG_RSS_IPV6_UDP;
1981 : :
1982 [ # # ]: 0 : if ((rss_hf & RTE_ETH_RSS_NONFRAG_IPV6_SCTP) != 0)
1983 : 0 : *cfg_rss_ctrl |= NFP_NET_CFG_RSS_IPV6_SCTP;
1984 : 0 : }
1985 : :
1986 : : static int
1987 : : nfp_net_rss_algo_conf_get(uint32_t cfg_rss_ctrl,
1988 : : struct rte_eth_rss_conf *rss_conf)
1989 : : {
1990 [ # # ]: 0 : if ((cfg_rss_ctrl & NFP_NET_CFG_RSS_TOEPLITZ) != 0)
1991 : 0 : rss_conf->algorithm = RTE_ETH_HASH_FUNCTION_TOEPLITZ;
1992 [ # # ]: 0 : else if ((cfg_rss_ctrl & NFP_NET_CFG_RSS_XOR) != 0)
1993 : 0 : rss_conf->algorithm = RTE_ETH_HASH_FUNCTION_SIMPLE_XOR;
1994 [ # # ]: 0 : else if ((cfg_rss_ctrl & NFP_NET_CFG_RSS_CRC32) != 0)
1995 : 0 : rss_conf->algorithm = RTE_ETH_HASH_FUNCTION_DEFAULT;
1996 : : else
1997 : : return -EIO;
1998 : :
1999 : : return 0;
2000 : : }
2001 : :
2002 : : static int
2003 : : nfp_net_rss_algo_conf_set(uint32_t *cfg_rss_ctrl,
2004 : : struct rte_eth_rss_conf *rss_conf)
2005 : : {
2006 : 0 : enum rte_eth_hash_function algorithm = rss_conf->algorithm;
2007 : :
2008 : 0 : if (algorithm == RTE_ETH_HASH_FUNCTION_TOEPLITZ)
2009 : 0 : *cfg_rss_ctrl |= NFP_NET_CFG_RSS_TOEPLITZ;
2010 [ # # ]: 0 : else if (algorithm == RTE_ETH_HASH_FUNCTION_SIMPLE_XOR)
2011 : 0 : *cfg_rss_ctrl |= NFP_NET_CFG_RSS_XOR;
2012 [ # # ]: 0 : else if (algorithm == RTE_ETH_HASH_FUNCTION_DEFAULT)
2013 : 0 : *cfg_rss_ctrl |= NFP_NET_CFG_RSS_CRC32;
2014 : : else
2015 : : return -ENOTSUP;
2016 : :
2017 : : return 0;
2018 : : }
2019 : :
2020 : : static int
2021 : 0 : nfp_net_rss_hash_write(struct rte_eth_dev *dev,
2022 : : struct rte_eth_rss_conf *rss_conf)
2023 : : {
2024 : : int ret;
2025 : : uint8_t i;
2026 : : uint8_t key;
2027 : : struct nfp_hw *hw;
2028 : : struct nfp_net_hw *net_hw;
2029 : 0 : uint32_t cfg_rss_ctrl = 0;
2030 : :
2031 : 0 : net_hw = nfp_net_get_hw(dev);
2032 : : hw = &net_hw->super;
2033 : :
2034 : : /* Writing the key byte by byte */
2035 [ # # ]: 0 : for (i = 0; i < rss_conf->rss_key_len; i++) {
2036 : 0 : memcpy(&key, &rss_conf->rss_key[i], 1);
2037 : 0 : nn_cfg_writeb(hw, NFP_NET_CFG_RSS_KEY + i, key);
2038 : : }
2039 : :
2040 : 0 : nfp_net_rss_hf_set(&cfg_rss_ctrl, rss_conf);
2041 : :
2042 [ # # ]: 0 : cfg_rss_ctrl |= NFP_NET_CFG_RSS_MASK;
2043 : :
2044 : : ret = nfp_net_rss_algo_conf_set(&cfg_rss_ctrl, rss_conf);
2045 : : if (ret != 0) {
2046 : 0 : PMD_DRV_LOG(ERR, "Fail to set rss algorithm configuration.");
2047 : 0 : return ret;
2048 : : }
2049 : :
2050 : : /* Configuring where to apply the RSS hash */
2051 : 0 : nn_cfg_writel(hw, NFP_NET_CFG_RSS_CTRL, cfg_rss_ctrl);
2052 : :
2053 : 0 : return 0;
2054 : : }
2055 : :
2056 : : int
2057 : 0 : nfp_net_rss_hash_update(struct rte_eth_dev *dev,
2058 : : struct rte_eth_rss_conf *rss_conf)
2059 : : {
2060 : : int ret;
2061 : : uint32_t update;
2062 : : struct nfp_hw *hw;
2063 : : struct nfp_net_hw *net_hw;
2064 : :
2065 : 0 : net_hw = nfp_net_get_hw(dev);
2066 : 0 : hw = &net_hw->super;
2067 : :
2068 : : /* Checking if RSS is enabled */
2069 [ # # ]: 0 : if ((hw->ctrl & NFP_NET_CFG_CTRL_RSS_ANY) == 0) {
2070 : 0 : PMD_DRV_LOG(ERR, "RSS unsupported.");
2071 : 0 : return -EINVAL;
2072 : : }
2073 : :
2074 [ # # ]: 0 : if (rss_conf->rss_key_len > NFP_NET_CFG_RSS_KEY_SZ) {
2075 : 0 : PMD_DRV_LOG(ERR, "RSS hash key too long.");
2076 : 0 : return -EINVAL;
2077 : : }
2078 : :
2079 : 0 : ret = nfp_net_rss_hash_write(dev, rss_conf);
2080 [ # # ]: 0 : if (ret != 0) {
2081 : 0 : PMD_DRV_LOG(ERR, "RSS write failed.");
2082 : 0 : return ret;
2083 : : }
2084 : :
2085 : : update = NFP_NET_CFG_UPDATE_RSS;
2086 : :
2087 [ # # ]: 0 : if (nfp_reconfig(hw, hw->ctrl, update) != 0)
2088 : 0 : return -EIO;
2089 : :
2090 : : return 0;
2091 : : }
2092 : :
2093 : : int
2094 : 0 : nfp_net_rss_hash_conf_get(struct rte_eth_dev *dev,
2095 : : struct rte_eth_rss_conf *rss_conf)
2096 : : {
2097 : : int ret;
2098 : : uint8_t i;
2099 : : uint8_t key;
2100 : : struct nfp_hw *hw;
2101 : : uint32_t cfg_rss_ctrl;
2102 : : struct nfp_net_hw *net_hw;
2103 : :
2104 : 0 : net_hw = nfp_net_get_hw(dev);
2105 : : hw = &net_hw->super;
2106 : :
2107 [ # # ]: 0 : if ((hw->ctrl & NFP_NET_CFG_CTRL_RSS_ANY) == 0)
2108 : : return -EINVAL;
2109 : :
2110 : : cfg_rss_ctrl = nn_cfg_readl(hw, NFP_NET_CFG_RSS_CTRL);
2111 : :
2112 : : /* Propagate current RSS hash functions to caller */
2113 : 0 : nfp_net_rss_hf_get(cfg_rss_ctrl, rss_conf);
2114 : :
2115 : : /* Reading the key size */
2116 : 0 : rss_conf->rss_key_len = NFP_NET_CFG_RSS_KEY_SZ;
2117 : :
2118 : : /* Reading the key byte a byte */
2119 [ # # ]: 0 : if (rss_conf->rss_key != NULL) {
2120 [ # # ]: 0 : for (i = 0; i < rss_conf->rss_key_len; i++) {
2121 : 0 : key = nn_cfg_readb(hw, NFP_NET_CFG_RSS_KEY + i);
2122 : 0 : memcpy(&rss_conf->rss_key[i], &key, 1);
2123 : : }
2124 : : }
2125 : :
2126 : : ret = nfp_net_rss_algo_conf_get(cfg_rss_ctrl, rss_conf);
2127 : : if (ret != 0) {
2128 : 0 : PMD_DRV_LOG(ERR, "Fail to get rss algorithm configuration.");
2129 : 0 : return ret;
2130 : : }
2131 : :
2132 : : return 0;
2133 : : }
2134 : :
2135 : : int
2136 : 0 : nfp_net_rss_config_default(struct rte_eth_dev *dev)
2137 : : {
2138 : : int ret;
2139 : : uint8_t i;
2140 : : uint8_t j;
2141 : : uint16_t queue = 0;
2142 : 0 : uint8_t default_key[] = {
2143 : : 0x6d, 0x5a, 0x56, 0xda,
2144 : : 0x25, 0x5b, 0x0e, 0xc2,
2145 : : 0x41, 0x67, 0x25, 0x3d,
2146 : : 0x43, 0xa3, 0x8f, 0xb0,
2147 : : 0xd0, 0xca, 0x2b, 0xcb,
2148 : : 0xae, 0x7b, 0x30, 0xb4,
2149 : : 0x77, 0xcb, 0x2d, 0xa3,
2150 : : 0x80, 0x30, 0xf2, 0x0c,
2151 : : 0x6a, 0x42, 0xb7, 0x3b,
2152 : : 0xbe, 0xac, 0x01, 0xfa,
2153 : : };
2154 : : struct rte_eth_conf *dev_conf;
2155 : : struct rte_eth_rss_conf rss_conf;
2156 : 0 : uint16_t rx_queues = dev->data->nb_rx_queues;
2157 : : struct rte_eth_rss_reta_entry64 nfp_reta_conf[2];
2158 : :
2159 : 0 : nfp_reta_conf[0].mask = ~0x0;
2160 : 0 : nfp_reta_conf[1].mask = ~0x0;
2161 : :
2162 [ # # ]: 0 : for (i = 0; i < 0x40; i += 8) {
2163 [ # # ]: 0 : for (j = i; j < (i + 8); j++) {
2164 : 0 : nfp_reta_conf[0].reta[j] = queue;
2165 : 0 : nfp_reta_conf[1].reta[j] = queue++;
2166 : 0 : queue %= rx_queues;
2167 : : }
2168 : : }
2169 : :
2170 : 0 : ret = nfp_net_rss_reta_write(dev, nfp_reta_conf, 0x80);
2171 [ # # ]: 0 : if (ret != 0)
2172 : : return ret;
2173 : :
2174 : 0 : dev_conf = &dev->data->dev_conf;
2175 : : if (dev_conf == NULL) {
2176 : : PMD_DRV_LOG(ERR, "Wrong rss conf.");
2177 : : return -EINVAL;
2178 : : }
2179 : :
2180 : 0 : rss_conf = dev_conf->rx_adv_conf.rss_conf;
2181 [ # # ]: 0 : if (rss_conf.rss_key_len == 0) {
2182 : 0 : rss_conf.rss_key = default_key;
2183 : 0 : rss_conf.rss_key_len = NFP_NET_CFG_RSS_KEY_SZ;
2184 : : }
2185 : :
2186 : 0 : ret = nfp_net_rss_hash_write(dev, &rss_conf);
2187 : :
2188 : 0 : return ret;
2189 : : }
2190 : :
2191 : : void
2192 : 0 : nfp_net_stop_rx_queue(struct rte_eth_dev *dev)
2193 : : {
2194 : : uint16_t i;
2195 : : struct nfp_net_rxq *this_rx_q;
2196 : :
2197 [ # # ]: 0 : for (i = 0; i < dev->data->nb_rx_queues; i++) {
2198 : 0 : this_rx_q = dev->data->rx_queues[i];
2199 : 0 : nfp_net_reset_rx_queue(this_rx_q);
2200 : 0 : dev->data->rx_queue_state[i] = RTE_ETH_QUEUE_STATE_STOPPED;
2201 : : }
2202 : 0 : }
2203 : :
2204 : : void
2205 : 0 : nfp_net_close_rx_queue(struct rte_eth_dev *dev)
2206 : : {
2207 : : uint16_t i;
2208 : : struct nfp_net_rxq *this_rx_q;
2209 : :
2210 [ # # ]: 0 : for (i = 0; i < dev->data->nb_rx_queues; i++) {
2211 : 0 : this_rx_q = dev->data->rx_queues[i];
2212 : 0 : nfp_net_reset_rx_queue(this_rx_q);
2213 : 0 : nfp_net_rx_queue_release(dev, i);
2214 : : }
2215 : 0 : }
2216 : :
2217 : : void
2218 : 0 : nfp_net_stop_tx_queue(struct rte_eth_dev *dev)
2219 : : {
2220 : : uint16_t i;
2221 : : struct nfp_net_txq *this_tx_q;
2222 : :
2223 [ # # ]: 0 : for (i = 0; i < dev->data->nb_tx_queues; i++) {
2224 : 0 : this_tx_q = dev->data->tx_queues[i];
2225 : 0 : nfp_net_reset_tx_queue(this_tx_q);
2226 : 0 : dev->data->tx_queue_state[i] = RTE_ETH_QUEUE_STATE_STOPPED;
2227 : : }
2228 : 0 : }
2229 : :
2230 : : void
2231 : 0 : nfp_net_close_tx_queue(struct rte_eth_dev *dev)
2232 : : {
2233 : : uint16_t i;
2234 : : struct nfp_net_txq *this_tx_q;
2235 : :
2236 [ # # ]: 0 : for (i = 0; i < dev->data->nb_tx_queues; i++) {
2237 : 0 : this_tx_q = dev->data->tx_queues[i];
2238 : 0 : nfp_net_reset_tx_queue(this_tx_q);
2239 : 0 : nfp_net_tx_queue_release(dev, i);
2240 : : }
2241 : 0 : }
2242 : :
2243 : : int
2244 : 0 : nfp_net_set_vxlan_port(struct nfp_net_hw *net_hw,
2245 : : size_t idx,
2246 : : uint16_t port,
2247 : : uint32_t ctrl)
2248 : : {
2249 : : uint32_t i;
2250 : 0 : struct nfp_hw *hw = &net_hw->super;
2251 : :
2252 [ # # ]: 0 : if (idx >= NFP_NET_N_VXLAN_PORTS) {
2253 : 0 : PMD_DRV_LOG(ERR, "The idx value is out of range.");
2254 : 0 : return -ERANGE;
2255 : : }
2256 : :
2257 : 0 : net_hw->vxlan_ports[idx] = port;
2258 : :
2259 [ # # ]: 0 : for (i = 0; i < NFP_NET_N_VXLAN_PORTS; i += 2) {
2260 : 0 : nn_cfg_writel(hw, NFP_NET_CFG_VXLAN_PORT + i * sizeof(port),
2261 : 0 : (net_hw->vxlan_ports[i + 1] << 16) | net_hw->vxlan_ports[i]);
2262 : : }
2263 : :
2264 : 0 : return nfp_reconfig(hw, ctrl, NFP_NET_CFG_UPDATE_VXLAN);
2265 : : }
2266 : :
2267 : : /*
2268 : : * The firmware with NFD3 can not handle DMA address requiring more
2269 : : * than 40 bits.
2270 : : */
2271 : : int
2272 : 0 : nfp_net_check_dma_mask(struct nfp_pf_dev *pf_dev,
2273 : : char *name)
2274 : : {
2275 [ # # # # ]: 0 : if (pf_dev->ver.extend == NFP_NET_CFG_VERSION_DP_NFD3 &&
2276 : 0 : rte_mem_check_dma_mask(40) != 0) {
2277 : 0 : PMD_DRV_LOG(ERR, "Device %s can not be used: restricted dma mask to 40 bits!",
2278 : : name);
2279 : 0 : return -ENODEV;
2280 : : }
2281 : :
2282 : : return 0;
2283 : : }
2284 : :
2285 : : int
2286 : 0 : nfp_net_txrwb_alloc(struct rte_eth_dev *eth_dev)
2287 : : {
2288 : : struct nfp_net_hw *net_hw;
2289 : : char mz_name[RTE_MEMZONE_NAMESIZE];
2290 : :
2291 : 0 : net_hw = nfp_net_get_hw(eth_dev);
2292 : 0 : snprintf(mz_name, sizeof(mz_name), "%s_TXRWB", eth_dev->data->name);
2293 : 0 : net_hw->txrwb_mz = rte_memzone_reserve_aligned(mz_name,
2294 : 0 : net_hw->max_tx_queues * sizeof(uint64_t),
2295 : 0 : rte_socket_id(),
2296 : : RTE_MEMZONE_IOVA_CONTIG, RTE_CACHE_LINE_SIZE);
2297 [ # # ]: 0 : if (net_hw->txrwb_mz == NULL) {
2298 : 0 : PMD_INIT_LOG(ERR, "Failed to alloc %s for TX ring write back.",
2299 : : mz_name);
2300 : 0 : return -ENOMEM;
2301 : : }
2302 : :
2303 : : return 0;
2304 : : }
2305 : :
2306 : : void
2307 : 0 : nfp_net_txrwb_free(struct rte_eth_dev *eth_dev)
2308 : : {
2309 : : struct nfp_net_hw *net_hw;
2310 : :
2311 : 0 : net_hw = nfp_net_get_hw(eth_dev);
2312 [ # # ]: 0 : if (net_hw->txrwb_mz == NULL)
2313 : : return;
2314 : :
2315 : 0 : rte_memzone_free(net_hw->txrwb_mz);
2316 : 0 : net_hw->txrwb_mz = NULL;
2317 : : }
2318 : :
2319 : : static void
2320 : : nfp_net_cfg_read_version(struct nfp_hw *hw,
2321 : : struct nfp_pf_dev *pf_dev)
2322 : : {
2323 : : union {
2324 : : uint32_t whole;
2325 : : struct nfp_net_fw_ver split;
2326 : : } version;
2327 : :
2328 : : version.whole = nn_cfg_readl(hw, NFP_NET_CFG_VERSION);
2329 : 0 : pf_dev->ver = version.split;
2330 : : }
2331 : :
2332 : : bool
2333 : 0 : nfp_net_version_check(struct nfp_hw *hw,
2334 : : struct nfp_pf_dev *pf_dev)
2335 : : {
2336 : : nfp_net_cfg_read_version(hw, pf_dev);
2337 [ # # ]: 0 : if (!nfp_net_is_valid_nfd_version(pf_dev->ver))
2338 : : return false;
2339 : :
2340 [ # # ]: 0 : if (!nfp_net_is_valid_version_class(pf_dev->ver))
2341 : 0 : return false;
2342 : :
2343 : : return true;
2344 : : }
2345 : :
2346 : : static void
2347 : 0 : nfp_net_get_nsp_info(struct nfp_net_hw_priv *hw_priv,
2348 : : char *nsp_version)
2349 : : {
2350 : : struct nfp_nsp *nsp;
2351 : :
2352 : 0 : nsp = nfp_nsp_open(hw_priv->pf_dev->cpp);
2353 [ # # ]: 0 : if (nsp == NULL)
2354 : : return;
2355 : :
2356 : 0 : snprintf(nsp_version, FW_VER_LEN, "%hu.%hu",
2357 : 0 : nfp_nsp_get_abi_ver_major(nsp),
2358 : 0 : nfp_nsp_get_abi_ver_minor(nsp));
2359 : :
2360 : 0 : nfp_nsp_close(nsp);
2361 : : }
2362 : :
2363 : : void
2364 : 0 : nfp_net_get_fw_version(struct nfp_cpp *cpp,
2365 : : uint32_t *mip_version)
2366 : : {
2367 : : struct nfp_mip *mip;
2368 : :
2369 : 0 : mip = nfp_mip_open(cpp);
2370 [ # # ]: 0 : if (mip == NULL) {
2371 : 0 : *mip_version = 0;
2372 : 0 : return;
2373 : : }
2374 : :
2375 : 0 : *mip_version = nfp_mip_fw_version(mip);
2376 : :
2377 : 0 : nfp_mip_close(mip);
2378 : : }
2379 : :
2380 : : static void
2381 : 0 : nfp_net_get_mip_name(struct nfp_net_hw_priv *hw_priv,
2382 : : char *mip_name)
2383 : : {
2384 : : struct nfp_mip *mip;
2385 : :
2386 : 0 : mip = nfp_mip_open(hw_priv->pf_dev->cpp);
2387 [ # # ]: 0 : if (mip == NULL)
2388 : : return;
2389 : :
2390 : 0 : strlcpy(mip_name, nfp_mip_name(mip), FW_VER_LEN);
2391 : :
2392 : 0 : nfp_mip_close(mip);
2393 : : }
2394 : :
2395 : : static void
2396 : 0 : nfp_net_get_app_name(struct nfp_net_hw_priv *hw_priv,
2397 : : char *app_name)
2398 : : {
2399 [ # # # ]: 0 : switch (hw_priv->pf_dev->app_fw_id) {
2400 : : case NFP_APP_FW_CORE_NIC:
2401 : : strlcpy(app_name, "nic", FW_VER_LEN);
2402 : : break;
2403 : : case NFP_APP_FW_FLOWER_NIC:
2404 : : strlcpy(app_name, "flower", FW_VER_LEN);
2405 : : break;
2406 : : default:
2407 : : strlcpy(app_name, "unknown", FW_VER_LEN);
2408 : : break;
2409 : : }
2410 : 0 : }
2411 : :
2412 : : int
2413 : 0 : nfp_net_firmware_version_get(struct rte_eth_dev *dev,
2414 : : char *fw_version,
2415 : : size_t fw_size)
2416 : : {
2417 : : struct nfp_net_hw *hw;
2418 : : struct nfp_pf_dev *pf_dev;
2419 : : struct nfp_net_hw_priv *hw_priv;
2420 : 0 : char app_name[FW_VER_LEN] = {0};
2421 : 0 : char mip_name[FW_VER_LEN] = {0};
2422 : 0 : char nsp_version[FW_VER_LEN] = {0};
2423 : 0 : char vnic_version[FW_VER_LEN] = {0};
2424 : :
2425 [ # # ]: 0 : if (fw_size < FW_VER_LEN)
2426 : : return FW_VER_LEN;
2427 : :
2428 : 0 : hw = nfp_net_get_hw(dev);
2429 : 0 : hw_priv = dev->process_private;
2430 : 0 : pf_dev = hw_priv->pf_dev;
2431 : :
2432 [ # # ]: 0 : if (hw->fw_version[0] != 0) {
2433 : 0 : snprintf(fw_version, FW_VER_LEN, "%s", hw->fw_version);
2434 : 0 : return 0;
2435 : : }
2436 : :
2437 [ # # ]: 0 : if (!rte_eth_dev_is_repr(dev)) {
2438 : 0 : snprintf(vnic_version, FW_VER_LEN, "%d.%d.%d.%d",
2439 : 0 : pf_dev->ver.extend, pf_dev->ver.class,
2440 : 0 : pf_dev->ver.major, pf_dev->ver.minor);
2441 : : } else {
2442 : : snprintf(vnic_version, FW_VER_LEN, "*");
2443 : : }
2444 : :
2445 : 0 : nfp_net_get_nsp_info(hw_priv, nsp_version);
2446 : 0 : nfp_net_get_mip_name(hw_priv, mip_name);
2447 : 0 : nfp_net_get_app_name(hw_priv, app_name);
2448 : :
2449 [ # # # # ]: 0 : if (nsp_version[0] == 0 || mip_name[0] == 0) {
2450 : : snprintf(fw_version, FW_VER_LEN, "%s %s %s %s",
2451 : : vnic_version, nsp_version, mip_name, app_name);
2452 : 0 : return 0;
2453 : : }
2454 : :
2455 : 0 : snprintf(hw->fw_version, FW_VER_LEN, "%s %s %s %s",
2456 : : vnic_version, nsp_version, mip_name, app_name);
2457 : :
2458 : : snprintf(fw_version, FW_VER_LEN, "%s", hw->fw_version);
2459 : :
2460 : 0 : return 0;
2461 : : }
2462 : :
2463 : : bool
2464 : 0 : nfp_net_is_valid_nfd_version(struct nfp_net_fw_ver version)
2465 : : {
2466 : 0 : uint8_t nfd_version = version.extend;
2467 : :
2468 [ # # ]: 0 : if (nfd_version == NFP_NET_CFG_VERSION_DP_NFD3)
2469 : : return true;
2470 : :
2471 [ # # ]: 0 : if (nfd_version == NFP_NET_CFG_VERSION_DP_NFDK) {
2472 [ # # ]: 0 : if (version.major < 5) {
2473 : 0 : PMD_INIT_LOG(ERR, "NFDK must use ABI 5 or newer, found: %d.",
2474 : : version.major);
2475 : 0 : return false;
2476 : : }
2477 : :
2478 : : return true;
2479 : : }
2480 : :
2481 : : return false;
2482 : : }
2483 : :
2484 : : bool
2485 : 0 : nfp_net_is_valid_version_class(struct nfp_net_fw_ver version)
2486 : : {
2487 [ # # ]: 0 : switch (version.class) {
2488 : : case NFP_NET_CFG_VERSION_CLASS_GENERIC:
2489 : : return true;
2490 : : case NFP_NET_CFG_VERSION_CLASS_NO_EMEM:
2491 : : return true;
2492 : 0 : default:
2493 : 0 : return false;
2494 : : }
2495 : : }
2496 : :
2497 : : void
2498 : 0 : nfp_net_ctrl_bar_size_set(struct nfp_pf_dev *pf_dev)
2499 : : {
2500 [ # # ]: 0 : if (pf_dev->ver.class == NFP_NET_CFG_VERSION_CLASS_GENERIC)
2501 : 0 : pf_dev->ctrl_bar_size = NFP_NET_CFG_BAR_SZ_32K;
2502 : : else
2503 : 0 : pf_dev->ctrl_bar_size = NFP_NET_CFG_BAR_SZ_8K;
2504 : 0 : }
2505 : :
2506 : : /* Disable rx and tx functions to allow for reconfiguring. */
2507 : : int
2508 : 0 : nfp_net_stop(struct rte_eth_dev *dev)
2509 : : {
2510 : : int ret;
2511 : : struct nfp_net_hw *hw;
2512 : : struct nfp_net_hw_priv *hw_priv;
2513 : :
2514 : 0 : hw = nfp_net_get_hw(dev);
2515 : 0 : hw_priv = dev->process_private;
2516 : :
2517 : 0 : nfp_net_disable_queues(dev);
2518 : :
2519 : : /* Clear queues */
2520 : 0 : nfp_net_stop_tx_queue(dev);
2521 : 0 : nfp_net_stop_rx_queue(dev);
2522 : :
2523 : 0 : ret = nfp_eth_set_configured(hw_priv->pf_dev->cpp, hw->nfp_idx, 0);
2524 : : if (ret < 0)
2525 : : return ret;
2526 : :
2527 : : return 0;
2528 : : }
2529 : :
2530 : : static enum rte_eth_fc_mode
2531 : : nfp_net_get_pause_mode(struct nfp_eth_table_port *eth_port)
2532 : : {
2533 : : enum rte_eth_fc_mode mode;
2534 : :
2535 : 0 : if (eth_port->rx_pause_enabled) {
2536 [ # # # # ]: 0 : if (eth_port->tx_pause_enabled)
2537 : : mode = RTE_ETH_FC_FULL;
2538 : : else
2539 : : mode = RTE_ETH_FC_RX_PAUSE;
2540 : : } else {
2541 [ # # # # ]: 0 : if (eth_port->tx_pause_enabled)
2542 : : mode = RTE_ETH_FC_TX_PAUSE;
2543 : : else
2544 : : mode = RTE_ETH_FC_NONE;
2545 : : }
2546 : :
2547 : : return mode;
2548 : : }
2549 : :
2550 : : int
2551 : 0 : nfp_net_flow_ctrl_get(struct rte_eth_dev *dev,
2552 : : struct rte_eth_fc_conf *fc_conf)
2553 : : {
2554 : : struct nfp_net_hw_priv *hw_priv;
2555 : : struct nfp_eth_table *nfp_eth_table;
2556 : : struct nfp_eth_table_port *eth_port;
2557 : :
2558 : 0 : hw_priv = dev->process_private;
2559 [ # # # # ]: 0 : if (hw_priv == NULL || hw_priv->pf_dev == NULL)
2560 : : return -EINVAL;
2561 : :
2562 : 0 : nfp_eth_table = hw_priv->pf_dev->nfp_eth_table;
2563 [ # # ]: 0 : eth_port = &nfp_eth_table->ports[dev->data->port_id];
2564 : :
2565 : : /* Currently only RX/TX switch are supported */
2566 : 0 : fc_conf->mode = nfp_net_get_pause_mode(eth_port);
2567 : :
2568 : 0 : return 0;
2569 : : }
2570 : :
2571 : : static int
2572 : 0 : nfp_net_pause_frame_set(struct nfp_net_hw_priv *hw_priv,
2573 : : struct nfp_eth_table_port *eth_port,
2574 : : enum rte_eth_fc_mode mode)
2575 : : {
2576 : : int err;
2577 : : bool flag;
2578 : : struct nfp_nsp *nsp;
2579 : :
2580 : 0 : nsp = nfp_eth_config_start(hw_priv->pf_dev->cpp, eth_port->index);
2581 [ # # ]: 0 : if (nsp == NULL) {
2582 : 0 : PMD_DRV_LOG(ERR, "NFP error when obtaining NSP handle.");
2583 : 0 : return -EIO;
2584 : : }
2585 : :
2586 : 0 : flag = (mode & RTE_ETH_FC_TX_PAUSE) == 0 ? false : true;
2587 : 0 : err = nfp_eth_set_tx_pause(nsp, flag);
2588 [ # # ]: 0 : if (err != 0) {
2589 : 0 : PMD_DRV_LOG(ERR, "Failed to configure TX pause frame.");
2590 : 0 : nfp_eth_config_cleanup_end(nsp);
2591 : 0 : return err;
2592 : : }
2593 : :
2594 : 0 : flag = (mode & RTE_ETH_FC_RX_PAUSE) == 0 ? false : true;
2595 : 0 : err = nfp_eth_set_rx_pause(nsp, flag);
2596 [ # # ]: 0 : if (err != 0) {
2597 : 0 : PMD_DRV_LOG(ERR, "Failed to configure RX pause frame.");
2598 : 0 : nfp_eth_config_cleanup_end(nsp);
2599 : 0 : return err;
2600 : : }
2601 : :
2602 : 0 : err = nfp_eth_config_commit_end(nsp);
2603 [ # # ]: 0 : if (err < 0) {
2604 : 0 : PMD_DRV_LOG(ERR, "Failed to configure pause frame.");
2605 : 0 : return err;
2606 : : }
2607 : :
2608 : : return 0;
2609 : : }
2610 : :
2611 : : int
2612 : 0 : nfp_net_flow_ctrl_set(struct rte_eth_dev *dev,
2613 : : struct rte_eth_fc_conf *fc_conf)
2614 : : {
2615 : : int ret;
2616 : : uint8_t idx;
2617 : : enum rte_eth_fc_mode set_mode;
2618 : : struct nfp_net_hw_priv *hw_priv;
2619 : : enum rte_eth_fc_mode original_mode;
2620 : : struct nfp_eth_table *nfp_eth_table;
2621 : : struct nfp_eth_table_port *eth_port;
2622 : :
2623 : 0 : idx = nfp_net_get_idx(dev);
2624 : 0 : hw_priv = dev->process_private;
2625 [ # # # # ]: 0 : if (hw_priv == NULL || hw_priv->pf_dev == NULL)
2626 : : return -EINVAL;
2627 : :
2628 : 0 : nfp_eth_table = hw_priv->pf_dev->nfp_eth_table;
2629 [ # # ]: 0 : eth_port = &nfp_eth_table->ports[idx];
2630 : :
2631 : : original_mode = nfp_net_get_pause_mode(eth_port);
2632 : 0 : set_mode = fc_conf->mode;
2633 : :
2634 [ # # ]: 0 : if (set_mode == original_mode)
2635 : : return 0;
2636 : :
2637 : 0 : ret = nfp_net_pause_frame_set(hw_priv, eth_port, set_mode);
2638 [ # # ]: 0 : if (ret != 0)
2639 : : return ret;
2640 : :
2641 : : /* Update eth_table after modifying RX/TX pause frame mode. */
2642 : 0 : eth_port->tx_pause_enabled = (set_mode & RTE_ETH_FC_TX_PAUSE) == 0 ? false : true;
2643 : 0 : eth_port->rx_pause_enabled = (set_mode & RTE_ETH_FC_RX_PAUSE) == 0 ? false : true;
2644 : :
2645 : 0 : return 0;
2646 : : }
2647 : :
2648 : : int
2649 : 0 : nfp_net_fec_get_capability(struct rte_eth_dev *dev,
2650 : : struct rte_eth_fec_capa *speed_fec_capa,
2651 : : __rte_unused unsigned int num)
2652 : : {
2653 : : uint8_t idx;
2654 : : uint16_t speed;
2655 : : uint32_t supported_fec;
2656 : : struct nfp_net_hw_priv *hw_priv;
2657 : : struct nfp_eth_table *nfp_eth_table;
2658 : : struct nfp_eth_table_port *eth_port;
2659 : :
2660 : 0 : idx = nfp_net_get_idx(dev);
2661 : 0 : hw_priv = dev->process_private;
2662 [ # # # # ]: 0 : if (hw_priv == NULL || hw_priv->pf_dev == NULL)
2663 : : return -EINVAL;
2664 : :
2665 : 0 : nfp_eth_table = hw_priv->pf_dev->nfp_eth_table;
2666 : 0 : eth_port = &nfp_eth_table->ports[idx];
2667 : :
2668 [ # # ]: 0 : speed = eth_port->speed;
2669 : : supported_fec = nfp_eth_supported_fec_modes(eth_port);
2670 [ # # ]: 0 : if (speed == 0 || supported_fec == 0) {
2671 : 0 : PMD_DRV_LOG(ERR, "FEC modes supported or Speed is invalid.");
2672 : 0 : return -EINVAL;
2673 : : }
2674 : :
2675 [ # # ]: 0 : if (speed_fec_capa == NULL)
2676 : : return NFP_FEC_CAPA_ENTRY_NUM;
2677 : :
2678 : 0 : speed_fec_capa->speed = speed;
2679 : :
2680 [ # # ]: 0 : if ((supported_fec & NFP_FEC_AUTO) != 0)
2681 : 0 : speed_fec_capa->capa |= RTE_ETH_FEC_MODE_CAPA_MASK(AUTO);
2682 [ # # ]: 0 : if ((supported_fec & NFP_FEC_BASER) != 0)
2683 : 0 : speed_fec_capa->capa |= RTE_ETH_FEC_MODE_CAPA_MASK(BASER);
2684 [ # # ]: 0 : if ((supported_fec & NFP_FEC_REED_SOLOMON) != 0)
2685 : 0 : speed_fec_capa->capa |= RTE_ETH_FEC_MODE_CAPA_MASK(RS);
2686 [ # # ]: 0 : if ((supported_fec & NFP_FEC_DISABLED) != 0)
2687 : 0 : speed_fec_capa->capa |= RTE_ETH_FEC_MODE_CAPA_MASK(NOFEC);
2688 : :
2689 : : return NFP_FEC_CAPA_ENTRY_NUM;
2690 : : }
2691 : :
2692 : : static uint32_t
2693 [ # # ]: 0 : nfp_net_fec_nfp_to_rte(enum nfp_eth_fec fec)
2694 : : {
2695 : : switch (fec) {
2696 : : case NFP_FEC_AUTO_BIT:
2697 : : return RTE_ETH_FEC_MODE_CAPA_MASK(AUTO);
2698 : : case NFP_FEC_BASER_BIT:
2699 : : return RTE_ETH_FEC_MODE_CAPA_MASK(BASER);
2700 : : case NFP_FEC_REED_SOLOMON_BIT:
2701 : : return RTE_ETH_FEC_MODE_CAPA_MASK(RS);
2702 : : case NFP_FEC_DISABLED_BIT:
2703 : : return RTE_ETH_FEC_MODE_CAPA_MASK(NOFEC);
2704 : 0 : default:
2705 : 0 : PMD_DRV_LOG(ERR, "FEC mode is invalid.");
2706 : 0 : return 0;
2707 : : }
2708 : : }
2709 : :
2710 : : int
2711 : 0 : nfp_net_fec_get(struct rte_eth_dev *dev,
2712 : : uint32_t *fec_capa)
2713 : : {
2714 : : uint8_t idx;
2715 : : struct nfp_net_hw_priv *hw_priv;
2716 : : struct nfp_eth_table *nfp_eth_table;
2717 : : struct nfp_eth_table_port *eth_port;
2718 : :
2719 : 0 : idx = nfp_net_get_idx(dev);
2720 : 0 : hw_priv = dev->process_private;
2721 [ # # # # ]: 0 : if (hw_priv == NULL || hw_priv->pf_dev == NULL)
2722 : : return -EINVAL;
2723 : :
2724 [ # # ]: 0 : if (dev->data->dev_link.link_status == RTE_ETH_LINK_DOWN) {
2725 : 0 : nfp_eth_table = nfp_eth_read_ports(hw_priv->pf_dev->cpp);
2726 : 0 : hw_priv->pf_dev->nfp_eth_table->ports[idx] = nfp_eth_table->ports[idx];
2727 : 0 : free(nfp_eth_table);
2728 : : }
2729 : :
2730 : 0 : nfp_eth_table = hw_priv->pf_dev->nfp_eth_table;
2731 [ # # ]: 0 : eth_port = &nfp_eth_table->ports[idx];
2732 : :
2733 [ # # ]: 0 : if (!nfp_eth_can_support_fec(eth_port)) {
2734 : 0 : PMD_DRV_LOG(ERR, "NFP can not support FEC.");
2735 : 0 : return -ENOTSUP;
2736 : : }
2737 : :
2738 : : /*
2739 : : * If link is down and AUTO is enabled, AUTO is returned, otherwise,
2740 : : * configured FEC mode is returned.
2741 : : * If link is up, current FEC mode is returned.
2742 : : */
2743 [ # # ]: 0 : if (dev->data->dev_link.link_status == RTE_ETH_LINK_DOWN)
2744 : 0 : *fec_capa = nfp_net_fec_nfp_to_rte(eth_port->fec);
2745 : : else
2746 : 0 : *fec_capa = nfp_net_fec_nfp_to_rte(eth_port->act_fec);
2747 : :
2748 [ # # ]: 0 : if (*fec_capa == 0)
2749 : 0 : return -EINVAL;
2750 : :
2751 : : return 0;
2752 : : }
2753 : :
2754 : : static enum nfp_eth_fec
2755 : : nfp_net_fec_rte_to_nfp(uint32_t fec)
2756 : : {
2757 : : switch (fec) {
2758 : : case RTE_BIT32(RTE_ETH_FEC_AUTO):
2759 : : return NFP_FEC_AUTO_BIT;
2760 : : case RTE_BIT32(RTE_ETH_FEC_NOFEC):
2761 : : return NFP_FEC_DISABLED_BIT;
2762 : : case RTE_BIT32(RTE_ETH_FEC_RS):
2763 : : return NFP_FEC_REED_SOLOMON_BIT;
2764 : : case RTE_BIT32(RTE_ETH_FEC_BASER):
2765 : : return NFP_FEC_BASER_BIT;
2766 : : default:
2767 : : return NFP_FEC_INVALID_BIT;
2768 : : }
2769 : : }
2770 : :
2771 : : int
2772 : 0 : nfp_net_fec_set(struct rte_eth_dev *dev,
2773 : : uint32_t fec_capa)
2774 : : {
2775 : : int ret;
2776 : : uint8_t idx;
2777 : : enum nfp_eth_fec fec;
2778 : : uint32_t supported_fec;
2779 : : struct nfp_net_hw_priv *hw_priv;
2780 : : struct nfp_eth_table *nfp_eth_table;
2781 : : struct nfp_eth_table_port *eth_port;
2782 : :
2783 : 0 : idx = nfp_net_get_idx(dev);
2784 : 0 : hw_priv = dev->process_private;
2785 [ # # # # ]: 0 : if (hw_priv == NULL || hw_priv->pf_dev == NULL)
2786 : : return -EINVAL;
2787 : :
2788 : 0 : nfp_eth_table = hw_priv->pf_dev->nfp_eth_table;
2789 [ # # ]: 0 : eth_port = &nfp_eth_table->ports[idx];
2790 : :
2791 : : supported_fec = nfp_eth_supported_fec_modes(eth_port);
2792 [ # # ]: 0 : if (supported_fec == 0) {
2793 : 0 : PMD_DRV_LOG(ERR, "NFP can not support FEC.");
2794 : 0 : return -ENOTSUP;
2795 : : }
2796 : :
2797 : : fec = nfp_net_fec_rte_to_nfp(fec_capa);
2798 [ # # ]: 0 : if (fec == NFP_FEC_INVALID_BIT) {
2799 : 0 : PMD_DRV_LOG(ERR, "FEC modes is invalid.");
2800 : 0 : return -EINVAL;
2801 : : }
2802 : :
2803 [ # # ]: 0 : if ((RTE_BIT32(fec) & supported_fec) == 0) {
2804 : 0 : PMD_DRV_LOG(ERR, "Unsupported FEC mode is set.");
2805 : 0 : return -EIO;
2806 : : }
2807 : :
2808 : 0 : ret = nfp_eth_set_fec(hw_priv->pf_dev->cpp, eth_port->index, fec);
2809 [ # # ]: 0 : if (ret < 0) {
2810 : 0 : PMD_DRV_LOG(ERR, "NFP set FEC mode failed.");
2811 : 0 : return ret;
2812 : : }
2813 : :
2814 : : return 0;
2815 : : }
2816 : :
2817 : : uint32_t
2818 : 0 : nfp_net_get_phyports_from_nsp(struct nfp_pf_dev *pf_dev)
2819 : : {
2820 [ # # ]: 0 : if (pf_dev->multi_pf.enabled)
2821 : : return 1;
2822 : : else
2823 : 0 : return pf_dev->nfp_eth_table->count;
2824 : : }
2825 : :
2826 : : uint32_t
2827 : 0 : nfp_net_get_phyports_from_fw(struct nfp_pf_dev *pf_dev)
2828 : : {
2829 : 0 : int ret = 0;
2830 : : uint8_t total_phyports;
2831 : : char pf_name[RTE_ETH_NAME_MAX_LEN];
2832 : :
2833 : : /* Read the number of vNIC's created for the PF */
2834 : 0 : snprintf(pf_name, sizeof(pf_name), "nfd_cfg_pf%u_num_ports",
2835 : 0 : pf_dev->multi_pf.function_id);
2836 : 0 : total_phyports = nfp_rtsym_read_le(pf_dev->sym_tbl, pf_name, &ret);
2837 [ # # # # ]: 0 : if (ret != 0 || total_phyports == 0 || total_phyports > 8) {
2838 : 0 : PMD_INIT_LOG(ERR, "%s symbol with wrong value", pf_name);
2839 : 0 : return 0;
2840 : : }
2841 : :
2842 : 0 : return total_phyports;
2843 : : }
2844 : :
2845 : : uint8_t
2846 : 0 : nfp_function_id_get(const struct nfp_pf_dev *pf_dev,
2847 : : uint8_t port_id)
2848 : : {
2849 [ # # ]: 0 : if (pf_dev->multi_pf.enabled)
2850 : 0 : return pf_dev->multi_pf.function_id;
2851 : :
2852 : : return port_id;
2853 : : }
2854 : :
2855 : : static int
2856 : : nfp_net_sriov_check(struct nfp_pf_dev *pf_dev,
2857 : : uint16_t cap)
2858 : : {
2859 : : uint16_t cap_vf;
2860 : :
2861 : 0 : cap_vf = nn_readw(pf_dev->vf_cfg_tbl_bar + NFP_NET_VF_CFG_MB_CAP);
2862 [ # # # # ]: 0 : if ((cap_vf & cap) != cap)
2863 : : return -ENOTSUP;
2864 : :
2865 : : return 0;
2866 : : }
2867 : :
2868 : : static int
2869 : 0 : nfp_net_sriov_update(struct nfp_net_hw *net_hw,
2870 : : struct nfp_pf_dev *pf_dev,
2871 : : uint16_t update)
2872 : : {
2873 : : int ret;
2874 : :
2875 : : /* Reuse NFP_NET_VF_CFG_MB_VF_NUM to pass vf_base_id to FW. */
2876 : 0 : ret = nfp_net_vf_reconfig(net_hw, pf_dev, update, pf_dev->vf_base_id,
2877 : : NFP_NET_VF_CFG_MB_VF_NUM);
2878 [ # # ]: 0 : if (ret != 0) {
2879 : 0 : PMD_INIT_LOG(ERR, "Error nfp VF reconfig.");
2880 : 0 : return ret;
2881 : : }
2882 : :
2883 : : return 0;
2884 : : }
2885 : :
2886 : : static int
2887 : 0 : nfp_net_vf_queues_config(struct nfp_net_hw *net_hw,
2888 : : struct nfp_pf_dev *pf_dev)
2889 : : {
2890 : : int ret;
2891 : : uint32_t i;
2892 : : uint32_t offset;
2893 : :
2894 : : ret = nfp_net_sriov_check(pf_dev, NFP_NET_VF_CFG_MB_CAP_QUEUE_CONFIG);
2895 : : if (ret != 0) {
2896 : : if (ret == -ENOTSUP) {
2897 : 0 : PMD_INIT_LOG(DEBUG, "Set VF max queue not supported.");
2898 : 0 : return 0;
2899 : : }
2900 : :
2901 : : PMD_INIT_LOG(ERR, "Set VF max queue failed.");
2902 : : return ret;
2903 : : }
2904 : :
2905 : 0 : offset = NFP_NET_VF_CFG_MB_SZ + pf_dev->max_vfs * NFP_NET_VF_CFG_SZ;
2906 [ # # ]: 0 : for (i = 0; i < pf_dev->sriov_vf; i++) {
2907 : 0 : ret = nfp_net_vf_reconfig(net_hw, pf_dev, NFP_NET_VF_CFG_MB_UPD_QUEUE_CONFIG,
2908 : 0 : pf_dev->queue_per_vf, pf_dev->vf_base_id + offset + i);
2909 [ # # ]: 0 : if (ret != 0) {
2910 : 0 : PMD_INIT_LOG(ERR, "Set VF max_queue failed.");
2911 : 0 : return ret;
2912 : : }
2913 : : }
2914 : :
2915 : : return 0;
2916 : : }
2917 : :
2918 : : static int
2919 : 0 : nfp_net_sriov_init(struct nfp_net_hw *net_hw,
2920 : : struct nfp_pf_dev *pf_dev)
2921 : : {
2922 : : int ret;
2923 : :
2924 : : ret = nfp_net_sriov_check(pf_dev, NFP_NET_VF_CFG_MB_CAP_SPLIT);
2925 : : if (ret != 0) {
2926 : : if (ret == -ENOTSUP) {
2927 : 0 : PMD_INIT_LOG(DEBUG, "Set VF split not supported.");
2928 : 0 : return 0;
2929 : : }
2930 : :
2931 : : PMD_INIT_LOG(ERR, "Set VF split failed.");
2932 : : return ret;
2933 : : }
2934 : :
2935 : 0 : nn_writeb(pf_dev->sriov_vf, pf_dev->vf_cfg_tbl_bar + NFP_NET_VF_CFG_MB_VF_CNT);
2936 : :
2937 : 0 : ret = nfp_net_sriov_update(net_hw, pf_dev, NFP_NET_VF_CFG_MB_UPD_SPLIT);
2938 [ # # ]: 0 : if (ret != 0) {
2939 : 0 : PMD_INIT_LOG(ERR, "The nfp sriov update spilt failed.");
2940 : 0 : return ret;
2941 : : }
2942 : :
2943 : : return 0;
2944 : : }
2945 : :
2946 : : int
2947 : 0 : nfp_net_vf_config_app_init(struct nfp_net_hw *net_hw,
2948 : : struct nfp_pf_dev *pf_dev)
2949 : : {
2950 : : int ret;
2951 : :
2952 [ # # ]: 0 : if (pf_dev->sriov_vf == 0)
2953 : : return 0;
2954 : :
2955 : 0 : ret = nfp_net_sriov_init(net_hw, pf_dev);
2956 [ # # ]: 0 : if (ret != 0) {
2957 : 0 : PMD_INIT_LOG(ERR, "Failed to init sriov module.");
2958 : 0 : return ret;
2959 : : }
2960 : :
2961 : 0 : ret = nfp_net_vf_queues_config(net_hw, pf_dev);
2962 [ # # ]: 0 : if (ret != 0) {
2963 : 0 : PMD_INIT_LOG(ERR, "Failed to config vf queue.");
2964 : 0 : return ret;
2965 : : }
2966 : :
2967 : : return 0;
2968 : : }
2969 : :
2970 : : static inline bool
2971 : 0 : nfp_net_meta_has_no_port_type(__rte_unused struct nfp_net_meta_parsed *meta)
2972 : : {
2973 : 0 : return true;
2974 : : }
2975 : :
2976 : : static inline bool
2977 : 0 : nfp_net_meta_is_not_pf_port(__rte_unused struct nfp_net_meta_parsed *meta)
2978 : : {
2979 : 0 : return false;
2980 : : }
2981 : :
2982 : : static inline bool
2983 : 0 : nfp_net_meta_is_pf_port(struct nfp_net_meta_parsed *meta)
2984 : : {
2985 : 0 : return nfp_flower_port_is_phy_port(meta->port_id);
2986 : : }
2987 : :
2988 : : bool
2989 : 0 : nfp_net_recv_pkt_meta_check_register(struct nfp_net_hw_priv *hw_priv)
2990 : : {
2991 : : struct nfp_pf_dev *pf_dev;
2992 : :
2993 : 0 : pf_dev = hw_priv->pf_dev;
2994 [ # # ]: 0 : if (!hw_priv->is_pf) {
2995 : 0 : pf_dev->recv_pkt_meta_check_t = nfp_net_meta_has_no_port_type;
2996 : 0 : return true;
2997 : : }
2998 : :
2999 [ # # # ]: 0 : switch (pf_dev->app_fw_id) {
3000 : 0 : case NFP_APP_FW_CORE_NIC:
3001 : 0 : pf_dev->recv_pkt_meta_check_t = nfp_net_meta_has_no_port_type;
3002 : 0 : break;
3003 : 0 : case NFP_APP_FW_FLOWER_NIC:
3004 [ # # ]: 0 : if (pf_dev->multi_pf.enabled)
3005 : 0 : pf_dev->recv_pkt_meta_check_t = nfp_net_meta_is_pf_port;
3006 : : else
3007 : 0 : pf_dev->recv_pkt_meta_check_t = nfp_net_meta_is_not_pf_port;
3008 : : break;
3009 : 0 : default:
3010 : 0 : PMD_INIT_LOG(ERR, "Unsupported Firmware loaded.");
3011 : 0 : return false;
3012 : : }
3013 : :
3014 : : return true;
3015 : : }
3016 : :
3017 : : static int
3018 : : nfp_net_get_nfp_index(struct rte_eth_dev *dev)
3019 : : {
3020 : : int nfp_idx;
3021 : :
3022 [ # # # # : 0 : if (rte_eth_dev_is_repr(dev)) {
# # ]
3023 : : struct nfp_flower_representor *repr;
3024 : 0 : repr = dev->data->dev_private;
3025 : 0 : nfp_idx = repr->nfp_idx;
3026 : : } else {
3027 : : struct nfp_net_hw *net_hw;
3028 : 0 : net_hw = dev->data->dev_private;
3029 : 0 : nfp_idx = net_hw->nfp_idx;
3030 : : }
3031 : :
3032 : : return nfp_idx;
3033 : : }
3034 : :
3035 : : int
3036 : 0 : nfp_net_get_eeprom_len(__rte_unused struct rte_eth_dev *dev)
3037 : : {
3038 : 0 : return RTE_ETHER_ADDR_LEN;
3039 : : }
3040 : :
3041 : : static int
3042 : 0 : nfp_net_get_port_mac_hwinfo(struct nfp_net_hw_priv *hw_priv,
3043 : : uint32_t index,
3044 : : struct rte_ether_addr *mac_addr)
3045 : : {
3046 : : int ret;
3047 : : char hwinfo[32];
3048 : : struct nfp_nsp *nsp;
3049 : :
3050 : : snprintf(hwinfo, sizeof(hwinfo), "eth%u.mac", index);
3051 : :
3052 : 0 : nsp = nfp_nsp_open(hw_priv->pf_dev->cpp);
3053 [ # # ]: 0 : if (nsp == NULL)
3054 : : return -EOPNOTSUPP;
3055 : :
3056 : 0 : ret = nfp_nsp_hwinfo_lookup(nsp, hwinfo, sizeof(hwinfo));
3057 : 0 : nfp_nsp_close(nsp);
3058 : :
3059 [ # # ]: 0 : if (ret != 0) {
3060 : 0 : PMD_DRV_LOG(ERR, "Read persistent MAC address failed for eth_index %u.", index);
3061 : 0 : return ret;
3062 : : }
3063 : :
3064 : 0 : ret = rte_ether_unformat_addr(hwinfo, mac_addr);
3065 [ # # ]: 0 : if (ret != 0) {
3066 : 0 : PMD_DRV_LOG(ERR, "Can not parse persistent MAC address.");
3067 : 0 : return -EOPNOTSUPP;
3068 : : }
3069 : :
3070 : : return 0;
3071 : : }
3072 : :
3073 : : static int
3074 : 0 : nfp_net_set_port_mac_hwinfo(struct nfp_net_hw_priv *hw_priv,
3075 : : uint32_t index,
3076 : : struct rte_ether_addr *mac_addr)
3077 : : {
3078 : : int ret;
3079 : : char hwinfo_mac[32];
3080 : : struct nfp_nsp *nsp;
3081 : : char buf[RTE_ETHER_ADDR_FMT_SIZE];
3082 : :
3083 : 0 : rte_ether_format_addr(buf, RTE_ETHER_ADDR_FMT_SIZE, mac_addr);
3084 : : snprintf(hwinfo_mac, sizeof(hwinfo_mac), "eth%u.mac=%s", index, buf);
3085 : :
3086 : 0 : nsp = nfp_nsp_open(hw_priv->pf_dev->cpp);
3087 [ # # ]: 0 : if (nsp == NULL)
3088 : : return -EOPNOTSUPP;
3089 : :
3090 : 0 : ret = nfp_nsp_hwinfo_set(nsp, hwinfo_mac, sizeof(hwinfo_mac));
3091 : 0 : nfp_nsp_close(nsp);
3092 : :
3093 [ # # ]: 0 : if (ret != 0) {
3094 : 0 : PMD_DRV_LOG(ERR, "HWinfo set failed: %d.", ret);
3095 : 0 : return ret;
3096 : : }
3097 : :
3098 : : return 0;
3099 : : }
3100 : :
3101 : : int
3102 : 0 : nfp_net_get_eeprom(struct rte_eth_dev *dev,
3103 : : struct rte_dev_eeprom_info *eeprom)
3104 : : {
3105 : : int ret;
3106 : : uint32_t nfp_idx;
3107 : : struct nfp_net_hw *net_hw;
3108 : : struct rte_ether_addr mac_addr;
3109 : : struct nfp_net_hw_priv *hw_priv;
3110 : :
3111 [ # # ]: 0 : if (eeprom->length == 0)
3112 : : return -EINVAL;
3113 : :
3114 [ # # ]: 0 : hw_priv = dev->process_private;
3115 : 0 : nfp_idx = nfp_net_get_nfp_index(dev);
3116 : :
3117 : 0 : ret = nfp_net_get_port_mac_hwinfo(hw_priv, nfp_idx, &mac_addr);
3118 [ # # ]: 0 : if (ret != 0)
3119 : : return -EOPNOTSUPP;
3120 : :
3121 : 0 : net_hw = nfp_net_get_hw(dev);
3122 : 0 : eeprom->magic = net_hw->vendor_id | (net_hw->device_id << 16);
3123 : 0 : memcpy(eeprom->data, mac_addr.addr_bytes + eeprom->offset, eeprom->length);
3124 : :
3125 : 0 : return 0;
3126 : : }
3127 : :
3128 : : int
3129 : 0 : nfp_net_set_eeprom(struct rte_eth_dev *dev,
3130 : : struct rte_dev_eeprom_info *eeprom)
3131 : : {
3132 : : int ret;
3133 : : uint32_t nfp_idx;
3134 : : struct nfp_net_hw *net_hw;
3135 : : struct rte_ether_addr mac_addr;
3136 : : struct nfp_net_hw_priv *hw_priv;
3137 : :
3138 [ # # ]: 0 : if (eeprom->length == 0)
3139 : : return -EINVAL;
3140 : :
3141 : 0 : net_hw = nfp_net_get_hw(dev);
3142 [ # # ]: 0 : if (eeprom->magic != (uint32_t)(net_hw->vendor_id | (net_hw->device_id << 16)))
3143 : : return -EINVAL;
3144 : :
3145 [ # # ]: 0 : hw_priv = dev->process_private;
3146 : 0 : nfp_idx = nfp_net_get_nfp_index(dev);
3147 : 0 : ret = nfp_net_get_port_mac_hwinfo(hw_priv, nfp_idx, &mac_addr);
3148 [ # # ]: 0 : if (ret != 0)
3149 : : return -EOPNOTSUPP;
3150 : :
3151 : 0 : memcpy(mac_addr.addr_bytes + eeprom->offset, eeprom->data, eeprom->length);
3152 : 0 : ret = nfp_net_set_port_mac_hwinfo(hw_priv, nfp_idx, &mac_addr);
3153 [ # # ]: 0 : if (ret != 0)
3154 : 0 : return -EOPNOTSUPP;
3155 : :
3156 : : return 0;
3157 : : }
3158 : :
3159 : : int
3160 : 0 : nfp_net_get_module_info(struct rte_eth_dev *dev,
3161 : : struct rte_eth_dev_module_info *info)
3162 : : {
3163 : : int ret = 0;
3164 : : uint8_t data;
3165 : : uint32_t idx;
3166 : : uint32_t read_len;
3167 : : struct nfp_nsp *nsp;
3168 : : struct nfp_net_hw_priv *hw_priv;
3169 : : struct nfp_eth_table_port *eth_port;
3170 : :
3171 : 0 : hw_priv = dev->process_private;
3172 : 0 : nsp = nfp_nsp_open(hw_priv->pf_dev->cpp);
3173 [ # # ]: 0 : if (nsp == NULL) {
3174 : 0 : PMD_DRV_LOG(ERR, "Unable to open NSP.");
3175 : 0 : return -EIO;
3176 : : }
3177 : :
3178 [ # # ]: 0 : if (!nfp_nsp_has_read_module_eeprom(nsp)) {
3179 : 0 : PMD_DRV_LOG(ERR, "Read module eeprom not supported. Please update flash.");
3180 : : ret = -EOPNOTSUPP;
3181 : 0 : goto exit_close_nsp;
3182 : : }
3183 : :
3184 : 0 : idx = nfp_net_get_idx(dev);
3185 : 0 : eth_port = &hw_priv->pf_dev->nfp_eth_table->ports[idx];
3186 [ # # # # ]: 0 : switch (eth_port->interface) {
3187 : 0 : case NFP_INTERFACE_SFP:
3188 : : /* FALLTHROUGH */
3189 : : case NFP_INTERFACE_SFP28:
3190 : : /* Read which revision the transceiver compiles with */
3191 : 0 : ret = nfp_nsp_read_module_eeprom(nsp, eth_port->eth_index,
3192 : : SFP_SFF8472_COMPLIANCE, &data, 1, &read_len);
3193 [ # # ]: 0 : if (ret != 0)
3194 : 0 : goto exit_close_nsp;
3195 : :
3196 [ # # ]: 0 : if (data == 0) {
3197 : 0 : info->type = RTE_ETH_MODULE_SFF_8079;
3198 : 0 : info->eeprom_len = RTE_ETH_MODULE_SFF_8079_LEN;
3199 : : } else {
3200 : 0 : info->type = RTE_ETH_MODULE_SFF_8472;
3201 : 0 : info->eeprom_len = RTE_ETH_MODULE_SFF_8472_LEN;
3202 : : }
3203 : : break;
3204 : 0 : case NFP_INTERFACE_QSFP:
3205 : : /* Read which revision the transceiver compiles with */
3206 : 0 : ret = nfp_nsp_read_module_eeprom(nsp, eth_port->eth_index,
3207 : : SFP_SFF_REV_COMPLIANCE, &data, 1, &read_len);
3208 [ # # ]: 0 : if (ret != 0)
3209 : 0 : goto exit_close_nsp;
3210 : :
3211 [ # # ]: 0 : if (data == 0) {
3212 : 0 : info->type = RTE_ETH_MODULE_SFF_8436;
3213 : 0 : info->eeprom_len = RTE_ETH_MODULE_SFF_8436_MAX_LEN;
3214 : : } else {
3215 : 0 : info->type = RTE_ETH_MODULE_SFF_8636;
3216 : 0 : info->eeprom_len = RTE_ETH_MODULE_SFF_8636_MAX_LEN;
3217 : : }
3218 : : break;
3219 : 0 : case NFP_INTERFACE_QSFP28:
3220 : 0 : info->type = RTE_ETH_MODULE_SFF_8636;
3221 : 0 : info->eeprom_len = RTE_ETH_MODULE_SFF_8636_MAX_LEN;
3222 : 0 : break;
3223 : 0 : default:
3224 : 0 : PMD_DRV_LOG(ERR, "Unsupported module %#x detected.",
3225 : : eth_port->interface);
3226 : : ret = -EINVAL;
3227 : : }
3228 : :
3229 : 0 : exit_close_nsp:
3230 : 0 : nfp_nsp_close(nsp);
3231 : 0 : return ret;
3232 : : }
3233 : :
3234 : : int
3235 : 0 : nfp_net_get_module_eeprom(struct rte_eth_dev *dev,
3236 : : struct rte_dev_eeprom_info *info)
3237 : : {
3238 : : int ret = 0;
3239 : : uint32_t idx;
3240 : : struct nfp_nsp *nsp;
3241 : : struct nfp_net_hw_priv *hw_priv;
3242 : : struct nfp_eth_table_port *eth_port;
3243 : :
3244 : 0 : hw_priv = dev->process_private;
3245 : 0 : nsp = nfp_nsp_open(hw_priv->pf_dev->cpp);
3246 [ # # ]: 0 : if (nsp == NULL) {
3247 : 0 : PMD_DRV_LOG(ERR, "Unable to open NSP.");
3248 : 0 : return -EIO;
3249 : : }
3250 : :
3251 [ # # ]: 0 : if (!nfp_nsp_has_read_module_eeprom(nsp)) {
3252 : 0 : PMD_DRV_LOG(ERR, "Read module eeprom not supported. Please update flash.");
3253 : : ret = -EOPNOTSUPP;
3254 : 0 : goto exit_close_nsp;
3255 : : }
3256 : :
3257 : 0 : idx = nfp_net_get_idx(dev);
3258 : 0 : eth_port = &hw_priv->pf_dev->nfp_eth_table->ports[idx];
3259 : 0 : ret = nfp_nsp_read_module_eeprom(nsp, eth_port->eth_index, info->offset,
3260 : : info->data, info->length, &info->length);
3261 [ # # ]: 0 : if (ret != 0) {
3262 [ # # ]: 0 : if (info->length)
3263 : 0 : PMD_DRV_LOG(ERR, "Incomplete read from module EEPROM: %d.", ret);
3264 : : else
3265 : 0 : PMD_DRV_LOG(ERR, "Read from module EEPROM failed: %d.", ret);
3266 : : }
3267 : :
3268 : 0 : exit_close_nsp:
3269 : 0 : nfp_nsp_close(nsp);
3270 : 0 : return ret;
3271 : : }
3272 : :
3273 : : static int
3274 : 0 : nfp_net_led_control(struct rte_eth_dev *dev,
3275 : : bool is_on)
3276 : : {
3277 : : int ret;
3278 : : uint32_t nfp_idx;
3279 : : struct nfp_net_hw_priv *hw_priv;
3280 : :
3281 [ # # ]: 0 : hw_priv = dev->process_private;
3282 : 0 : nfp_idx = nfp_net_get_nfp_index(dev);
3283 : :
3284 : 0 : ret = nfp_eth_set_idmode(hw_priv->pf_dev->cpp, nfp_idx, is_on);
3285 [ # # ]: 0 : if (ret < 0) {
3286 : 0 : PMD_DRV_LOG(ERR, "Set nfp idmode failed.");
3287 : 0 : return ret;
3288 : : }
3289 : :
3290 : : return 0;
3291 : : }
3292 : :
3293 : : int
3294 : 0 : nfp_net_led_on(struct rte_eth_dev *dev)
3295 : : {
3296 : 0 : return nfp_net_led_control(dev, true);
3297 : : }
3298 : :
3299 : : int
3300 : 0 : nfp_net_led_off(struct rte_eth_dev *dev)
3301 : : {
3302 : 0 : return nfp_net_led_control(dev, false);
3303 : : }
3304 : :
3305 : : void
3306 : 0 : nfp_net_rss_algo_capa_get(struct nfp_net_hw *hw,
3307 : : struct rte_eth_dev_info *dev_info)
3308 : : {
3309 : : uint32_t cap_rss;
3310 : :
3311 : 0 : cap_rss = hw->super.cap_rss;
3312 [ # # ]: 0 : if ((cap_rss & NFP_NET_CFG_RSS_TOEPLITZ) != 0)
3313 : 0 : dev_info->rss_algo_capa |= RTE_ETH_HASH_ALGO_CAPA_MASK(TOEPLITZ);
3314 : :
3315 [ # # ]: 0 : if ((cap_rss & NFP_NET_CFG_RSS_XOR) != 0)
3316 : 0 : dev_info->rss_algo_capa |= RTE_ETH_HASH_ALGO_CAPA_MASK(SIMPLE_XOR);
3317 : :
3318 [ # # ]: 0 : if ((cap_rss & NFP_NET_CFG_RSS_CRC32) != 0)
3319 : 0 : dev_info->rss_algo_capa |= RTE_ETH_HASH_ALGO_CAPA_MASK(DEFAULT);
3320 : 0 : }
|