xref: /dpdk/drivers/net/mlx5/mlx5_testpmd.c (revision 0f3ba0d4a8df8601bc009f9f1d1e7845f3372723)
1f41a5092SSpike Du /* SPDX-License-Identifier: BSD-3-Clause
2f41a5092SSpike Du  * Copyright 2021 6WIND S.A.
3f41a5092SSpike Du  * Copyright 2021 Mellanox Technologies, Ltd
4f41a5092SSpike Du  */
5f41a5092SSpike Du 
6f41a5092SSpike Du #include <stdint.h>
7f41a5092SSpike Du #include <string.h>
8f41a5092SSpike Du #include <stdlib.h>
985d9252eSMichael Baum #include <unistd.h>
1085d9252eSMichael Baum #ifndef RTE_EXEC_ENV_WINDOWS
1185d9252eSMichael Baum #include <sys/socket.h>
1285d9252eSMichael Baum #include <sys/un.h>
1385d9252eSMichael Baum #endif
14f41a5092SSpike Du 
15f41a5092SSpike Du #include <rte_prefetch.h>
16f41a5092SSpike Du #include <rte_common.h>
17f41a5092SSpike Du #include <rte_branch_prediction.h>
18f41a5092SSpike Du #include <rte_ether.h>
19f41a5092SSpike Du #include <rte_alarm.h>
20f41a5092SSpike Du #include <rte_pmd_mlx5.h>
21f41a5092SSpike Du #include <rte_ethdev.h>
2285d9252eSMichael Baum 
23f41a5092SSpike Du #include "mlx5_testpmd.h"
24f41a5092SSpike Du #include "testpmd.h"
25f41a5092SSpike Du 
26f41a5092SSpike Du static uint8_t host_shaper_avail_thresh_triggered[RTE_MAX_ETHPORTS];
27f41a5092SSpike Du #define SHAPER_DISABLE_DELAY_US 100000 /* 100ms */
28f41a5092SSpike Du 
29f41a5092SSpike Du /**
30f41a5092SSpike Du  * Disable the host shaper and re-arm available descriptor threshold event.
31f41a5092SSpike Du  *
32f41a5092SSpike Du  * @param[in] args
33f41a5092SSpike Du  *   uint32_t integer combining port_id and rxq_id.
34f41a5092SSpike Du  */
35f41a5092SSpike Du static void
36f41a5092SSpike Du mlx5_test_host_shaper_disable(void *args)
37f41a5092SSpike Du {
38f41a5092SSpike Du 	uint32_t port_rxq_id = (uint32_t)(uintptr_t)args;
39f41a5092SSpike Du 	uint16_t port_id = port_rxq_id & 0xffff;
40f41a5092SSpike Du 	uint16_t qid = (port_rxq_id >> 16) & 0xffff;
41f41a5092SSpike Du 	struct rte_eth_rxq_info qinfo;
42*0f3ba0d4SSpike Du 	struct rte_port *port;
43f41a5092SSpike Du 
44*0f3ba0d4SSpike Du 	port = &ports[port_id];
45*0f3ba0d4SSpike Du 	if (port->port_status != RTE_PORT_STARTED) {
46*0f3ba0d4SSpike Du 		printf("%s port_status(%d) is incorrect, stop avail_thresh "
47*0f3ba0d4SSpike Du 		       "event processing.\n",
48*0f3ba0d4SSpike Du 		       __func__, port->port_status);
49*0f3ba0d4SSpike Du 		return;
50*0f3ba0d4SSpike Du 	}
51f41a5092SSpike Du 	printf("%s disable shaper\n", __func__);
52f41a5092SSpike Du 	if (rte_eth_rx_queue_info_get(port_id, qid, &qinfo)) {
53f41a5092SSpike Du 		printf("rx_queue_info_get returns error\n");
54f41a5092SSpike Du 		return;
55f41a5092SSpike Du 	}
56f41a5092SSpike Du 	/* Rearm the available descriptor threshold event. */
57f41a5092SSpike Du 	if (rte_eth_rx_avail_thresh_set(port_id, qid, qinfo.avail_thresh)) {
58f41a5092SSpike Du 		printf("config avail_thresh returns error\n");
59f41a5092SSpike Du 		return;
60f41a5092SSpike Du 	}
61f41a5092SSpike Du 	/* Only disable the shaper when avail_thresh_triggered is set. */
62f41a5092SSpike Du 	if (host_shaper_avail_thresh_triggered[port_id] &&
63f41a5092SSpike Du 	    rte_pmd_mlx5_host_shaper_config(port_id, 0, 0))
64f41a5092SSpike Du 		printf("%s disable shaper returns error\n", __func__);
65f41a5092SSpike Du }
66f41a5092SSpike Du 
67f41a5092SSpike Du void
68f41a5092SSpike Du mlx5_test_avail_thresh_event_handler(uint16_t port_id, uint16_t rxq_id)
69f41a5092SSpike Du {
70f41a5092SSpike Du 	struct rte_eth_dev_info dev_info;
71f41a5092SSpike Du 	uint32_t port_rxq_id = port_id | (rxq_id << 16);
72f41a5092SSpike Du 
73f41a5092SSpike Du 	/* Ensure it's MLX5 port. */
74f41a5092SSpike Du 	if (rte_eth_dev_info_get(port_id, &dev_info) != 0 ||
75f41a5092SSpike Du 	    (strncmp(dev_info.driver_name, "mlx5", 4) != 0))
76f41a5092SSpike Du 		return;
77f41a5092SSpike Du 	rte_eal_alarm_set(SHAPER_DISABLE_DELAY_US,
78f41a5092SSpike Du 			  mlx5_test_host_shaper_disable,
79f41a5092SSpike Du 			  (void *)(uintptr_t)port_rxq_id);
80f41a5092SSpike Du 	printf("%s port_id:%u rxq_id:%u\n", __func__, port_id, rxq_id);
81f41a5092SSpike Du }
82f41a5092SSpike Du 
83f41a5092SSpike Du /**
84f41a5092SSpike Du  * Configure host shaper's avail_thresh_triggered and current rate.
85f41a5092SSpike Du  *
86f41a5092SSpike Du  * @param[in] avail_thresh_triggered
87f41a5092SSpike Du  *   Disable/enable avail_thresh_triggered.
88f41a5092SSpike Du  * @param[in] rate
89f41a5092SSpike Du  *   Configure current host shaper rate.
90f41a5092SSpike Du  * @return
91f41a5092SSpike Du  *   On success, returns 0.
92f41a5092SSpike Du  *   On failure, returns < 0.
93f41a5092SSpike Du  */
94f41a5092SSpike Du static int
95f41a5092SSpike Du mlx5_test_set_port_host_shaper(uint16_t port_id, uint16_t avail_thresh_triggered, uint8_t rate)
96f41a5092SSpike Du {
97f41a5092SSpike Du 	struct rte_eth_link link;
98f41a5092SSpike Du 	bool port_id_valid = false;
99f41a5092SSpike Du 	uint16_t pid;
100f41a5092SSpike Du 	int ret;
101f41a5092SSpike Du 
102f41a5092SSpike Du 	RTE_ETH_FOREACH_DEV(pid)
103f41a5092SSpike Du 		if (port_id == pid) {
104f41a5092SSpike Du 			port_id_valid = true;
105f41a5092SSpike Du 			break;
106f41a5092SSpike Du 		}
107f41a5092SSpike Du 	if (!port_id_valid)
108f41a5092SSpike Du 		return -EINVAL;
109f41a5092SSpike Du 	ret = rte_eth_link_get_nowait(port_id, &link);
110f41a5092SSpike Du 	if (ret < 0)
111f41a5092SSpike Du 		return ret;
112f41a5092SSpike Du 	host_shaper_avail_thresh_triggered[port_id] = avail_thresh_triggered ? 1 : 0;
113f41a5092SSpike Du 	if (!avail_thresh_triggered) {
114f41a5092SSpike Du 		ret = rte_pmd_mlx5_host_shaper_config(port_id, 0,
115f41a5092SSpike Du 		RTE_BIT32(MLX5_HOST_SHAPER_FLAG_AVAIL_THRESH_TRIGGERED));
116f41a5092SSpike Du 	} else {
117f41a5092SSpike Du 		ret = rte_pmd_mlx5_host_shaper_config(port_id, 1,
118f41a5092SSpike Du 		RTE_BIT32(MLX5_HOST_SHAPER_FLAG_AVAIL_THRESH_TRIGGERED));
119f41a5092SSpike Du 	}
120f41a5092SSpike Du 	if (ret)
121f41a5092SSpike Du 		return ret;
122f41a5092SSpike Du 	ret = rte_pmd_mlx5_host_shaper_config(port_id, rate, 0);
123f41a5092SSpike Du 	if (ret)
124f41a5092SSpike Du 		return ret;
125f41a5092SSpike Du 	return 0;
126f41a5092SSpike Du }
127f41a5092SSpike Du 
12885d9252eSMichael Baum #ifndef RTE_EXEC_ENV_WINDOWS
12985d9252eSMichael Baum static const char*
13085d9252eSMichael Baum mlx5_test_get_socket_path(char *extend)
13185d9252eSMichael Baum {
13285d9252eSMichael Baum 	if (strstr(extend, "socket=") == extend) {
13385d9252eSMichael Baum 		const char *socket_path = strchr(extend, '=') + 1;
13485d9252eSMichael Baum 
13585d9252eSMichael Baum 		TESTPMD_LOG(DEBUG, "MLX5 socket path is %s\n", socket_path);
13685d9252eSMichael Baum 		return socket_path;
13785d9252eSMichael Baum 	}
13885d9252eSMichael Baum 
13985d9252eSMichael Baum 	TESTPMD_LOG(ERR, "Failed to extract a valid socket path from %s\n",
14085d9252eSMichael Baum 		    extend);
14185d9252eSMichael Baum 	return NULL;
14285d9252eSMichael Baum }
14385d9252eSMichael Baum 
14485d9252eSMichael Baum static int
14585d9252eSMichael Baum mlx5_test_extend_devargs(char *identifier, char *extend)
14685d9252eSMichael Baum {
14785d9252eSMichael Baum 	struct sockaddr_un un = {
14885d9252eSMichael Baum 		.sun_family = AF_UNIX,
14985d9252eSMichael Baum 	};
15085d9252eSMichael Baum 	int cmd_fd;
15185d9252eSMichael Baum 	int pd_handle;
15285d9252eSMichael Baum 	struct iovec iov = {
15385d9252eSMichael Baum 		.iov_base = &pd_handle,
15485d9252eSMichael Baum 		.iov_len = sizeof(int),
15585d9252eSMichael Baum 	};
15685d9252eSMichael Baum 	union {
15785d9252eSMichael Baum 		char buf[CMSG_SPACE(sizeof(int))];
15885d9252eSMichael Baum 		struct cmsghdr align;
15985d9252eSMichael Baum 	} control;
16085d9252eSMichael Baum 	struct msghdr msgh = {
16185d9252eSMichael Baum 		.msg_iov = NULL,
16285d9252eSMichael Baum 		.msg_iovlen = 0,
16385d9252eSMichael Baum 	};
16485d9252eSMichael Baum 	struct cmsghdr *cmsg;
16585d9252eSMichael Baum 	const char *path = mlx5_test_get_socket_path(extend + 1);
16685d9252eSMichael Baum 	size_t len = 1;
16785d9252eSMichael Baum 	int socket_fd;
16885d9252eSMichael Baum 	int ret;
16985d9252eSMichael Baum 
17085d9252eSMichael Baum 	if (path == NULL) {
17185d9252eSMichael Baum 		TESTPMD_LOG(ERR, "Invalid devargs extension is specified\n");
17285d9252eSMichael Baum 		return -1;
17385d9252eSMichael Baum 	}
17485d9252eSMichael Baum 
17585d9252eSMichael Baum 	/* Initialize IPC channel. */
17685d9252eSMichael Baum 	socket_fd = socket(AF_UNIX, SOCK_SEQPACKET, 0);
17785d9252eSMichael Baum 	if (socket_fd < 0) {
17885d9252eSMichael Baum 		TESTPMD_LOG(ERR, "Failed to create unix socket: %s\n",
17985d9252eSMichael Baum 			    strerror(errno));
18085d9252eSMichael Baum 		return -1;
18185d9252eSMichael Baum 	}
18285d9252eSMichael Baum 	rte_strlcpy(un.sun_path, path, sizeof(un.sun_path));
18385d9252eSMichael Baum 	if (connect(socket_fd, (struct sockaddr *)&un, sizeof(un)) < 0) {
18485d9252eSMichael Baum 		TESTPMD_LOG(ERR, "Failed to connect %s: %s\n", un.sun_path,
18585d9252eSMichael Baum 			    strerror(errno));
18685d9252eSMichael Baum 		close(socket_fd);
18785d9252eSMichael Baum 		return -1;
18885d9252eSMichael Baum 	}
18985d9252eSMichael Baum 
19085d9252eSMichael Baum 	/* Send the request message. */
19185d9252eSMichael Baum 	do {
19285d9252eSMichael Baum 		ret = sendmsg(socket_fd, &msgh, 0);
19385d9252eSMichael Baum 	} while (ret < 0 && errno == EINTR);
19485d9252eSMichael Baum 	if (ret < 0) {
19585d9252eSMichael Baum 		TESTPMD_LOG(ERR, "Failed to send request to (%s): %s\n", path,
19685d9252eSMichael Baum 			    strerror(errno));
19785d9252eSMichael Baum 		close(socket_fd);
19885d9252eSMichael Baum 		return -1;
19985d9252eSMichael Baum 	}
20085d9252eSMichael Baum 
20185d9252eSMichael Baum 	msgh.msg_iov = &iov;
20285d9252eSMichael Baum 	msgh.msg_iovlen = 1;
20385d9252eSMichael Baum 	msgh.msg_control = control.buf;
20485d9252eSMichael Baum 	msgh.msg_controllen = sizeof(control.buf);
20585d9252eSMichael Baum 	do {
20685d9252eSMichael Baum 		ret = recvmsg(socket_fd, &msgh, 0);
20785d9252eSMichael Baum 	} while (ret < 0);
20885d9252eSMichael Baum 	if (ret != sizeof(int) || (msgh.msg_flags & (MSG_TRUNC | MSG_CTRUNC))) {
20985d9252eSMichael Baum 		TESTPMD_LOG(ERR, "truncated msg");
21085d9252eSMichael Baum 		close(socket_fd);
21185d9252eSMichael Baum 		return -1;
21285d9252eSMichael Baum 	}
21385d9252eSMichael Baum 
21485d9252eSMichael Baum 	/* Translate the FD. */
21585d9252eSMichael Baum 	cmsg = CMSG_FIRSTHDR(&msgh);
21685d9252eSMichael Baum 	if (cmsg == NULL || cmsg->cmsg_len != CMSG_LEN(sizeof(int)) ||
21785d9252eSMichael Baum 	    cmsg->cmsg_level != SOL_SOCKET || cmsg->cmsg_type != SCM_RIGHTS) {
21885d9252eSMichael Baum 		TESTPMD_LOG(ERR, "Fail to get FD using SCM_RIGHTS mechanism\n");
21985d9252eSMichael Baum 		close(socket_fd);
22085d9252eSMichael Baum 		unlink(un.sun_path);
22185d9252eSMichael Baum 		return -1;
22285d9252eSMichael Baum 	}
22385d9252eSMichael Baum 	memcpy(&cmd_fd, CMSG_DATA(cmsg), sizeof(int));
22485d9252eSMichael Baum 
22585d9252eSMichael Baum 	TESTPMD_LOG(DEBUG, "Command FD (%d) and PD handle (%d) "
22685d9252eSMichael Baum 		    "are successfully imported from remote process\n",
22785d9252eSMichael Baum 		    cmd_fd, pd_handle);
22885d9252eSMichael Baum 
22985d9252eSMichael Baum 	/* Cleanup IPC channel. */
23085d9252eSMichael Baum 	close(socket_fd);
23185d9252eSMichael Baum 
23285d9252eSMichael Baum 	/* Calculate the new length of devargs string. */
23385d9252eSMichael Baum 	len += snprintf(NULL, 0, ",cmd_fd=%d,pd_handle=%d", cmd_fd, pd_handle);
23485d9252eSMichael Baum 	/* Extend the devargs string. */
23585d9252eSMichael Baum 	snprintf(extend, len, ",cmd_fd=%d,pd_handle=%d", cmd_fd, pd_handle);
23685d9252eSMichael Baum 
23785d9252eSMichael Baum 	TESTPMD_LOG(DEBUG, "Attach port with extra devargs %s\n", identifier);
23885d9252eSMichael Baum 	return 0;
23985d9252eSMichael Baum }
24085d9252eSMichael Baum 
24185d9252eSMichael Baum static bool
24285d9252eSMichael Baum is_delimiter_path_spaces(char *extend)
24385d9252eSMichael Baum {
24485d9252eSMichael Baum 	while (*extend != '\0') {
24585d9252eSMichael Baum 		if (*extend != ' ')
24685d9252eSMichael Baum 			return true;
24785d9252eSMichael Baum 		extend++;
24885d9252eSMichael Baum 	}
24985d9252eSMichael Baum 	return false;
25085d9252eSMichael Baum }
25185d9252eSMichael Baum 
25285d9252eSMichael Baum /*
25385d9252eSMichael Baum  * Extend devargs list with "cmd_fd" and "pd_handle" coming from external
25485d9252eSMichael Baum  * process. It happens only in this format:
25585d9252eSMichael Baum  *  testpmd> mlx5 port attach (identifier) socket=<socket path>
25685d9252eSMichael Baum  * all "(identifier) socket=<socket path>" is in the same string pointed
25785d9252eSMichael Baum  * by the input parameter 'identifier'.
25885d9252eSMichael Baum  *
25985d9252eSMichael Baum  * @param identifier
26085d9252eSMichael Baum  *   Identifier of port attach command line.
26185d9252eSMichael Baum  */
26285d9252eSMichael Baum static void
26385d9252eSMichael Baum mlx5_test_attach_port_extend_devargs(char *identifier)
26485d9252eSMichael Baum {
26585d9252eSMichael Baum 	char *extend;
26685d9252eSMichael Baum 
26785d9252eSMichael Baum 	if (identifier == NULL) {
26885d9252eSMichael Baum 		fprintf(stderr, "Invalid parameters are specified\n");
26985d9252eSMichael Baum 		return;
27085d9252eSMichael Baum 	}
27185d9252eSMichael Baum 
27285d9252eSMichael Baum 	extend = strchr(identifier, ' ');
27385d9252eSMichael Baum 	if (extend != NULL && is_delimiter_path_spaces(extend) &&
27485d9252eSMichael Baum 	    mlx5_test_extend_devargs(identifier, extend) < 0) {
27585d9252eSMichael Baum 		TESTPMD_LOG(ERR, "Failed to extend devargs for port %s\n",
27685d9252eSMichael Baum 			    identifier);
27785d9252eSMichael Baum 		return;
27885d9252eSMichael Baum 	}
27985d9252eSMichael Baum 
28085d9252eSMichael Baum 	attach_port(identifier);
28185d9252eSMichael Baum }
28285d9252eSMichael Baum #endif
28385d9252eSMichael Baum 
284f41a5092SSpike Du /* *** SET HOST_SHAPER FOR A PORT *** */
285f41a5092SSpike Du struct cmd_port_host_shaper_result {
286f41a5092SSpike Du 	cmdline_fixed_string_t mlx5;
287f41a5092SSpike Du 	cmdline_fixed_string_t set;
288f41a5092SSpike Du 	cmdline_fixed_string_t port;
289f41a5092SSpike Du 	uint16_t port_num;
290f41a5092SSpike Du 	cmdline_fixed_string_t host_shaper;
291f41a5092SSpike Du 	cmdline_fixed_string_t avail_thresh_triggered;
292f41a5092SSpike Du 	uint16_t fr;
293f41a5092SSpike Du 	cmdline_fixed_string_t rate;
294f41a5092SSpike Du 	uint8_t rate_num;
295f41a5092SSpike Du };
296f41a5092SSpike Du 
297f41a5092SSpike Du static void cmd_port_host_shaper_parsed(void *parsed_result,
298f41a5092SSpike Du 		__rte_unused struct cmdline *cl,
299f41a5092SSpike Du 		__rte_unused void *data)
300f41a5092SSpike Du {
301f41a5092SSpike Du 	struct cmd_port_host_shaper_result *res = parsed_result;
302f41a5092SSpike Du 	int ret = 0;
303f41a5092SSpike Du 
304f41a5092SSpike Du 	if ((strcmp(res->mlx5, "mlx5") == 0) &&
305f41a5092SSpike Du 	    (strcmp(res->set, "set") == 0) &&
306f41a5092SSpike Du 	    (strcmp(res->port, "port") == 0) &&
307f41a5092SSpike Du 	    (strcmp(res->host_shaper, "host_shaper") == 0) &&
308f41a5092SSpike Du 	    (strcmp(res->avail_thresh_triggered, "avail_thresh_triggered") == 0) &&
309f41a5092SSpike Du 	    (strcmp(res->rate, "rate") == 0))
310f41a5092SSpike Du 		ret = mlx5_test_set_port_host_shaper(res->port_num, res->fr,
311f41a5092SSpike Du 					   res->rate_num);
312f41a5092SSpike Du 	if (ret < 0)
313f41a5092SSpike Du 		printf("cmd_port_host_shaper error: (%s)\n", strerror(-ret));
314f41a5092SSpike Du }
315f41a5092SSpike Du 
316f41a5092SSpike Du static cmdline_parse_token_string_t cmd_port_host_shaper_mlx5 =
317f41a5092SSpike Du 	TOKEN_STRING_INITIALIZER(struct cmd_port_host_shaper_result,
318f41a5092SSpike Du 				mlx5, "mlx5");
319f41a5092SSpike Du static cmdline_parse_token_string_t cmd_port_host_shaper_set =
320f41a5092SSpike Du 	TOKEN_STRING_INITIALIZER(struct cmd_port_host_shaper_result,
321f41a5092SSpike Du 				set, "set");
322f41a5092SSpike Du static cmdline_parse_token_string_t cmd_port_host_shaper_port =
323f41a5092SSpike Du 	TOKEN_STRING_INITIALIZER(struct cmd_port_host_shaper_result,
324f41a5092SSpike Du 				port, "port");
325f41a5092SSpike Du static cmdline_parse_token_num_t cmd_port_host_shaper_portnum =
326f41a5092SSpike Du 	TOKEN_NUM_INITIALIZER(struct cmd_port_host_shaper_result,
327f41a5092SSpike Du 				port_num, RTE_UINT16);
328f41a5092SSpike Du static cmdline_parse_token_string_t cmd_port_host_shaper_host_shaper =
329f41a5092SSpike Du 	TOKEN_STRING_INITIALIZER(struct cmd_port_host_shaper_result,
330f41a5092SSpike Du 				 host_shaper, "host_shaper");
331f41a5092SSpike Du static cmdline_parse_token_string_t cmd_port_host_shaper_avail_thresh_triggered =
332f41a5092SSpike Du 	TOKEN_STRING_INITIALIZER(struct cmd_port_host_shaper_result,
333f41a5092SSpike Du 				 avail_thresh_triggered, "avail_thresh_triggered");
334f41a5092SSpike Du static cmdline_parse_token_num_t cmd_port_host_shaper_fr =
335f41a5092SSpike Du 	TOKEN_NUM_INITIALIZER(struct cmd_port_host_shaper_result,
336f41a5092SSpike Du 			      fr, RTE_UINT16);
337f41a5092SSpike Du static cmdline_parse_token_string_t cmd_port_host_shaper_rate =
338f41a5092SSpike Du 	TOKEN_STRING_INITIALIZER(struct cmd_port_host_shaper_result,
339f41a5092SSpike Du 				 rate, "rate");
340f41a5092SSpike Du static cmdline_parse_token_num_t cmd_port_host_shaper_rate_num =
341f41a5092SSpike Du 	TOKEN_NUM_INITIALIZER(struct cmd_port_host_shaper_result,
342f41a5092SSpike Du 			      rate_num, RTE_UINT8);
343f41a5092SSpike Du static cmdline_parse_inst_t mlx5_test_cmd_port_host_shaper = {
344f41a5092SSpike Du 	.f = cmd_port_host_shaper_parsed,
345f41a5092SSpike Du 	.data = (void *)0,
346f41a5092SSpike Du 	.help_str = "mlx5 set port <port_id> host_shaper avail_thresh_triggered <0|1> "
347f41a5092SSpike Du 	"rate <rate_num>: Set HOST_SHAPER avail_thresh_triggered and rate with port_id",
348f41a5092SSpike Du 	.tokens = {
349f41a5092SSpike Du 		(void *)&cmd_port_host_shaper_mlx5,
350f41a5092SSpike Du 		(void *)&cmd_port_host_shaper_set,
351f41a5092SSpike Du 		(void *)&cmd_port_host_shaper_port,
352f41a5092SSpike Du 		(void *)&cmd_port_host_shaper_portnum,
353f41a5092SSpike Du 		(void *)&cmd_port_host_shaper_host_shaper,
354f41a5092SSpike Du 		(void *)&cmd_port_host_shaper_avail_thresh_triggered,
355f41a5092SSpike Du 		(void *)&cmd_port_host_shaper_fr,
356f41a5092SSpike Du 		(void *)&cmd_port_host_shaper_rate,
357f41a5092SSpike Du 		(void *)&cmd_port_host_shaper_rate_num,
358f41a5092SSpike Du 		NULL,
359f41a5092SSpike Du 	}
360f41a5092SSpike Du };
361f41a5092SSpike Du 
36285d9252eSMichael Baum #ifndef RTE_EXEC_ENV_WINDOWS
36385d9252eSMichael Baum /* *** attach a specified port *** */
36485d9252eSMichael Baum struct mlx5_cmd_operate_attach_port_result {
36585d9252eSMichael Baum 	cmdline_fixed_string_t mlx5;
36685d9252eSMichael Baum 	cmdline_fixed_string_t port;
36785d9252eSMichael Baum 	cmdline_fixed_string_t keyword;
36885d9252eSMichael Baum 	cmdline_multi_string_t identifier;
36985d9252eSMichael Baum };
37085d9252eSMichael Baum 
37185d9252eSMichael Baum static void mlx5_cmd_operate_attach_port_parsed(void *parsed_result,
37285d9252eSMichael Baum 						__rte_unused struct cmdline *cl,
37385d9252eSMichael Baum 						__rte_unused void *data)
37485d9252eSMichael Baum {
37585d9252eSMichael Baum 	struct mlx5_cmd_operate_attach_port_result *res = parsed_result;
37685d9252eSMichael Baum 
37785d9252eSMichael Baum 	if (!strcmp(res->keyword, "attach"))
37885d9252eSMichael Baum 		mlx5_test_attach_port_extend_devargs(res->identifier);
37985d9252eSMichael Baum 	else
38085d9252eSMichael Baum 		fprintf(stderr, "Unknown parameter\n");
38185d9252eSMichael Baum }
38285d9252eSMichael Baum 
38385d9252eSMichael Baum static cmdline_parse_token_string_t mlx5_cmd_operate_attach_port_mlx5 =
38485d9252eSMichael Baum 	TOKEN_STRING_INITIALIZER(struct mlx5_cmd_operate_attach_port_result,
38585d9252eSMichael Baum 				 mlx5, "mlx5");
38685d9252eSMichael Baum static cmdline_parse_token_string_t mlx5_cmd_operate_attach_port_port =
38785d9252eSMichael Baum 	TOKEN_STRING_INITIALIZER(struct mlx5_cmd_operate_attach_port_result,
38885d9252eSMichael Baum 				 port, "port");
38985d9252eSMichael Baum static cmdline_parse_token_string_t mlx5_cmd_operate_attach_port_keyword =
39085d9252eSMichael Baum 	TOKEN_STRING_INITIALIZER(struct mlx5_cmd_operate_attach_port_result,
39185d9252eSMichael Baum 				 keyword, "attach");
39285d9252eSMichael Baum static cmdline_parse_token_string_t mlx5_cmd_operate_attach_port_identifier =
39385d9252eSMichael Baum 	TOKEN_STRING_INITIALIZER(struct mlx5_cmd_operate_attach_port_result,
39485d9252eSMichael Baum 				 identifier, TOKEN_STRING_MULTI);
39585d9252eSMichael Baum 
39685d9252eSMichael Baum static cmdline_parse_inst_t mlx5_cmd_operate_attach_port = {
39785d9252eSMichael Baum 	.f = mlx5_cmd_operate_attach_port_parsed,
39885d9252eSMichael Baum 	.data = NULL,
39985d9252eSMichael Baum 	.help_str = "mlx5 port attach <identifier> socket=<path>: "
40085d9252eSMichael Baum 		"(identifier: pci address or virtual dev name"
40185d9252eSMichael Baum 		", path (optional): socket path to get cmd FD and PD handle)",
40285d9252eSMichael Baum 	.tokens = {
40385d9252eSMichael Baum 		(void *)&mlx5_cmd_operate_attach_port_mlx5,
40485d9252eSMichael Baum 		(void *)&mlx5_cmd_operate_attach_port_port,
40585d9252eSMichael Baum 		(void *)&mlx5_cmd_operate_attach_port_keyword,
40685d9252eSMichael Baum 		(void *)&mlx5_cmd_operate_attach_port_identifier,
40785d9252eSMichael Baum 		NULL,
40885d9252eSMichael Baum 	},
40985d9252eSMichael Baum };
41085d9252eSMichael Baum #endif
41185d9252eSMichael Baum 
412740a2836SMichael Baum /* Map HW queue index to rte queue index. */
413740a2836SMichael Baum struct mlx5_cmd_map_ext_rxq {
414740a2836SMichael Baum 	cmdline_fixed_string_t mlx5;
415740a2836SMichael Baum 	cmdline_fixed_string_t port;
416740a2836SMichael Baum 	portid_t port_id;
417740a2836SMichael Baum 	cmdline_fixed_string_t ext_rxq;
418740a2836SMichael Baum 	cmdline_fixed_string_t map;
419740a2836SMichael Baum 	uint16_t sw_queue_id;
420740a2836SMichael Baum 	uint32_t hw_queue_id;
421740a2836SMichael Baum };
422740a2836SMichael Baum 
423740a2836SMichael Baum cmdline_parse_token_string_t mlx5_cmd_map_ext_rxq_mlx5 =
424740a2836SMichael Baum 	TOKEN_STRING_INITIALIZER(struct mlx5_cmd_map_ext_rxq, mlx5, "mlx5");
425740a2836SMichael Baum cmdline_parse_token_string_t mlx5_cmd_map_ext_rxq_port =
426740a2836SMichael Baum 	TOKEN_STRING_INITIALIZER(struct mlx5_cmd_map_ext_rxq, port, "port");
427740a2836SMichael Baum cmdline_parse_token_num_t mlx5_cmd_map_ext_rxq_port_id =
428740a2836SMichael Baum 	TOKEN_NUM_INITIALIZER(struct mlx5_cmd_map_ext_rxq, port_id, RTE_UINT16);
429740a2836SMichael Baum cmdline_parse_token_string_t mlx5_cmd_map_ext_rxq_ext_rxq =
430740a2836SMichael Baum 	TOKEN_STRING_INITIALIZER(struct mlx5_cmd_map_ext_rxq, ext_rxq,
431740a2836SMichael Baum 				 "ext_rxq");
432740a2836SMichael Baum cmdline_parse_token_string_t mlx5_cmd_map_ext_rxq_map =
433740a2836SMichael Baum 	TOKEN_STRING_INITIALIZER(struct mlx5_cmd_map_ext_rxq, map, "map");
434740a2836SMichael Baum cmdline_parse_token_num_t mlx5_cmd_map_ext_rxq_sw_queue_id =
435740a2836SMichael Baum 	TOKEN_NUM_INITIALIZER(struct mlx5_cmd_map_ext_rxq, sw_queue_id,
436740a2836SMichael Baum 			      RTE_UINT16);
437740a2836SMichael Baum cmdline_parse_token_num_t mlx5_cmd_map_ext_rxq_hw_queue_id =
438740a2836SMichael Baum 	TOKEN_NUM_INITIALIZER(struct mlx5_cmd_map_ext_rxq, hw_queue_id,
439740a2836SMichael Baum 			      RTE_UINT32);
440740a2836SMichael Baum 
441740a2836SMichael Baum static void
442740a2836SMichael Baum mlx5_cmd_map_ext_rxq_parsed(void *parsed_result,
443740a2836SMichael Baum 			    __rte_unused struct cmdline *cl,
444740a2836SMichael Baum 			    __rte_unused void *data)
445740a2836SMichael Baum {
446740a2836SMichael Baum 	struct mlx5_cmd_map_ext_rxq *res = parsed_result;
447740a2836SMichael Baum 	int ret;
448740a2836SMichael Baum 
449740a2836SMichael Baum 	if (port_id_is_invalid(res->port_id, ENABLED_WARN))
450740a2836SMichael Baum 		return;
451740a2836SMichael Baum 	ret = rte_pmd_mlx5_external_rx_queue_id_map(res->port_id,
452740a2836SMichael Baum 						    res->sw_queue_id,
453740a2836SMichael Baum 						    res->hw_queue_id);
454740a2836SMichael Baum 	switch (ret) {
455740a2836SMichael Baum 	case 0:
456740a2836SMichael Baum 		break;
457740a2836SMichael Baum 	case -EINVAL:
458740a2836SMichael Baum 		fprintf(stderr, "invalid ethdev index (%u), out of range\n",
459740a2836SMichael Baum 			res->sw_queue_id);
460740a2836SMichael Baum 		break;
461740a2836SMichael Baum 	case -ENODEV:
462740a2836SMichael Baum 		fprintf(stderr, "invalid port_id %u\n", res->port_id);
463740a2836SMichael Baum 		break;
464740a2836SMichael Baum 	case -ENOTSUP:
465740a2836SMichael Baum 		fprintf(stderr, "function not implemented or supported\n");
466740a2836SMichael Baum 		break;
467740a2836SMichael Baum 	case -EEXIST:
468740a2836SMichael Baum 		fprintf(stderr, "mapping with index %u already exists\n",
469740a2836SMichael Baum 			res->sw_queue_id);
470740a2836SMichael Baum 		break;
471740a2836SMichael Baum 	default:
472740a2836SMichael Baum 		fprintf(stderr, "programming error: (%s)\n", strerror(-ret));
473740a2836SMichael Baum 	}
474740a2836SMichael Baum }
475740a2836SMichael Baum 
476740a2836SMichael Baum cmdline_parse_inst_t mlx5_cmd_map_ext_rxq = {
477740a2836SMichael Baum 	.f = mlx5_cmd_map_ext_rxq_parsed,
478740a2836SMichael Baum 	.data = NULL,
479740a2836SMichael Baum 	.help_str = "mlx5 port <port_id> ext_rxq map <sw_queue_id> <hw_queue_id>",
480740a2836SMichael Baum 	.tokens = {
481740a2836SMichael Baum 		(void *)&mlx5_cmd_map_ext_rxq_mlx5,
482740a2836SMichael Baum 		(void *)&mlx5_cmd_map_ext_rxq_port,
483740a2836SMichael Baum 		(void *)&mlx5_cmd_map_ext_rxq_port_id,
484740a2836SMichael Baum 		(void *)&mlx5_cmd_map_ext_rxq_ext_rxq,
485740a2836SMichael Baum 		(void *)&mlx5_cmd_map_ext_rxq_map,
486740a2836SMichael Baum 		(void *)&mlx5_cmd_map_ext_rxq_sw_queue_id,
487740a2836SMichael Baum 		(void *)&mlx5_cmd_map_ext_rxq_hw_queue_id,
488740a2836SMichael Baum 		NULL,
489740a2836SMichael Baum 	}
490740a2836SMichael Baum };
491740a2836SMichael Baum 
492740a2836SMichael Baum /* Unmap HW queue index to rte queue index. */
493740a2836SMichael Baum struct mlx5_cmd_unmap_ext_rxq {
494740a2836SMichael Baum 	cmdline_fixed_string_t mlx5;
495740a2836SMichael Baum 	cmdline_fixed_string_t port;
496740a2836SMichael Baum 	portid_t port_id;
497740a2836SMichael Baum 	cmdline_fixed_string_t ext_rxq;
498740a2836SMichael Baum 	cmdline_fixed_string_t unmap;
499740a2836SMichael Baum 	uint16_t queue_id;
500740a2836SMichael Baum };
501740a2836SMichael Baum 
502740a2836SMichael Baum cmdline_parse_token_string_t mlx5_cmd_unmap_ext_rxq_mlx5 =
503740a2836SMichael Baum 	TOKEN_STRING_INITIALIZER(struct mlx5_cmd_unmap_ext_rxq, mlx5, "mlx5");
504740a2836SMichael Baum cmdline_parse_token_string_t mlx5_cmd_unmap_ext_rxq_port =
505740a2836SMichael Baum 	TOKEN_STRING_INITIALIZER(struct mlx5_cmd_unmap_ext_rxq, port, "port");
506740a2836SMichael Baum cmdline_parse_token_num_t mlx5_cmd_unmap_ext_rxq_port_id =
507740a2836SMichael Baum 	TOKEN_NUM_INITIALIZER(struct mlx5_cmd_unmap_ext_rxq, port_id,
508740a2836SMichael Baum 			      RTE_UINT16);
509740a2836SMichael Baum cmdline_parse_token_string_t mlx5_cmd_unmap_ext_rxq_ext_rxq =
510740a2836SMichael Baum 	TOKEN_STRING_INITIALIZER(struct mlx5_cmd_unmap_ext_rxq, ext_rxq,
511740a2836SMichael Baum 				 "ext_rxq");
512740a2836SMichael Baum cmdline_parse_token_string_t mlx5_cmd_unmap_ext_rxq_unmap =
513740a2836SMichael Baum 	TOKEN_STRING_INITIALIZER(struct mlx5_cmd_unmap_ext_rxq, unmap, "unmap");
514740a2836SMichael Baum cmdline_parse_token_num_t mlx5_cmd_unmap_ext_rxq_queue_id =
515740a2836SMichael Baum 	TOKEN_NUM_INITIALIZER(struct mlx5_cmd_unmap_ext_rxq, queue_id,
516740a2836SMichael Baum 			      RTE_UINT16);
517740a2836SMichael Baum 
518740a2836SMichael Baum static void
519740a2836SMichael Baum mlx5_cmd_unmap_ext_rxq_parsed(void *parsed_result,
520740a2836SMichael Baum 			      __rte_unused struct cmdline *cl,
521740a2836SMichael Baum 			      __rte_unused void *data)
522740a2836SMichael Baum {
523740a2836SMichael Baum 	struct mlx5_cmd_unmap_ext_rxq *res = parsed_result;
524740a2836SMichael Baum 	int ret;
525740a2836SMichael Baum 
526740a2836SMichael Baum 	if (port_id_is_invalid(res->port_id, ENABLED_WARN))
527740a2836SMichael Baum 		return;
528740a2836SMichael Baum 	ret = rte_pmd_mlx5_external_rx_queue_id_unmap(res->port_id,
529740a2836SMichael Baum 						      res->queue_id);
530740a2836SMichael Baum 	switch (ret) {
531740a2836SMichael Baum 	case 0:
532740a2836SMichael Baum 		break;
533740a2836SMichael Baum 	case -EINVAL:
534740a2836SMichael Baum 		fprintf(stderr, "invalid rte_flow index (%u), "
535740a2836SMichael Baum 			"out of range, doesn't exist or still referenced\n",
536740a2836SMichael Baum 			res->queue_id);
537740a2836SMichael Baum 		break;
538740a2836SMichael Baum 	case -ENODEV:
539740a2836SMichael Baum 		fprintf(stderr, "invalid port_id %u\n", res->port_id);
540740a2836SMichael Baum 		break;
541740a2836SMichael Baum 	case -ENOTSUP:
542740a2836SMichael Baum 		fprintf(stderr, "function not implemented or supported\n");
543740a2836SMichael Baum 		break;
544740a2836SMichael Baum 	default:
545740a2836SMichael Baum 		fprintf(stderr, "programming error: (%s)\n", strerror(-ret));
546740a2836SMichael Baum 	}
547740a2836SMichael Baum }
548740a2836SMichael Baum 
549740a2836SMichael Baum cmdline_parse_inst_t mlx5_cmd_unmap_ext_rxq = {
550740a2836SMichael Baum 	.f = mlx5_cmd_unmap_ext_rxq_parsed,
551740a2836SMichael Baum 	.data = NULL,
552740a2836SMichael Baum 	.help_str = "mlx5 port <port_id> ext_rxq unmap <queue_id>",
553740a2836SMichael Baum 	.tokens = {
554740a2836SMichael Baum 		(void *)&mlx5_cmd_unmap_ext_rxq_mlx5,
555740a2836SMichael Baum 		(void *)&mlx5_cmd_unmap_ext_rxq_port,
556740a2836SMichael Baum 		(void *)&mlx5_cmd_unmap_ext_rxq_port_id,
557740a2836SMichael Baum 		(void *)&mlx5_cmd_unmap_ext_rxq_ext_rxq,
558740a2836SMichael Baum 		(void *)&mlx5_cmd_unmap_ext_rxq_unmap,
559740a2836SMichael Baum 		(void *)&mlx5_cmd_unmap_ext_rxq_queue_id,
560740a2836SMichael Baum 		NULL,
561740a2836SMichael Baum 	}
562740a2836SMichael Baum };
563740a2836SMichael Baum 
564f41a5092SSpike Du static struct testpmd_driver_commands mlx5_driver_cmds = {
565f41a5092SSpike Du 	.commands = {
566f41a5092SSpike Du 		{
567f41a5092SSpike Du 			.ctx = &mlx5_test_cmd_port_host_shaper,
568f41a5092SSpike Du 			.help = "mlx5 set port (port_id) host_shaper avail_thresh_triggered (on|off)"
569f41a5092SSpike Du 				"rate (rate_num):\n"
570f41a5092SSpike Du 				"    Set HOST_SHAPER avail_thresh_triggered and rate with port_id\n\n",
571f41a5092SSpike Du 		},
57285d9252eSMichael Baum #ifndef RTE_EXEC_ENV_WINDOWS
57385d9252eSMichael Baum 		{
57485d9252eSMichael Baum 			.ctx = &mlx5_cmd_operate_attach_port,
57585d9252eSMichael Baum 			.help = "mlx5 port attach (ident) socket=(path)\n"
57685d9252eSMichael Baum 				"    Attach physical or virtual dev by pci address or virtual device name "
57785d9252eSMichael Baum 				"and add \"cmd_fd\" and \"pd_handle\" devargs before attaching\n\n",
57885d9252eSMichael Baum 		},
57985d9252eSMichael Baum #endif
580f41a5092SSpike Du 		{
581740a2836SMichael Baum 			.ctx = &mlx5_cmd_map_ext_rxq,
582740a2836SMichael Baum 			.help = "mlx5 port (port_id) ext_rxq map (sw_queue_id) (hw_queue_id)\n"
583740a2836SMichael Baum 				"    Map HW queue index (32-bit) to ethdev"
584740a2836SMichael Baum 				" queue index (16-bit) for external RxQ\n\n",
585740a2836SMichael Baum 		},
586740a2836SMichael Baum 		{
587740a2836SMichael Baum 			.ctx = &mlx5_cmd_unmap_ext_rxq,
588740a2836SMichael Baum 			.help = "mlx5 port (port_id) ext_rxq unmap (sw_queue_id)\n"
589740a2836SMichael Baum 				"    Unmap external Rx queue ethdev index mapping\n\n",
590740a2836SMichael Baum 		},
591740a2836SMichael Baum 		{
592f41a5092SSpike Du 			.ctx = NULL,
593f41a5092SSpike Du 		},
594f41a5092SSpike Du 	}
595f41a5092SSpike Du };
596f41a5092SSpike Du TESTPMD_ADD_DRIVER_COMMANDS(mlx5_driver_cmds);
597