1*d30ea906Sjfb8856606 /* SPDX-License-Identifier: BSD-3-Clause
2*d30ea906Sjfb8856606  * Copyright(c) 2018 Gaëtan Rivet
3*d30ea906Sjfb8856606  */
4*d30ea906Sjfb8856606 
5*d30ea906Sjfb8856606 #include "rte_ethdev.h"
6*d30ea906Sjfb8856606 #include "rte_ethdev_driver.h"
7*d30ea906Sjfb8856606 #include "ethdev_private.h"
8*d30ea906Sjfb8856606 
9*d30ea906Sjfb8856606 uint16_t
eth_dev_to_id(const struct rte_eth_dev * dev)10*d30ea906Sjfb8856606 eth_dev_to_id(const struct rte_eth_dev *dev)
11*d30ea906Sjfb8856606 {
12*d30ea906Sjfb8856606 	if (dev == NULL)
13*d30ea906Sjfb8856606 		return RTE_MAX_ETHPORTS;
14*d30ea906Sjfb8856606 	return dev - rte_eth_devices;
15*d30ea906Sjfb8856606 }
16*d30ea906Sjfb8856606 
17*d30ea906Sjfb8856606 struct rte_eth_dev *
eth_find_device(const struct rte_eth_dev * start,rte_eth_cmp_t cmp,const void * data)18*d30ea906Sjfb8856606 eth_find_device(const struct rte_eth_dev *start, rte_eth_cmp_t cmp,
19*d30ea906Sjfb8856606 		const void *data)
20*d30ea906Sjfb8856606 {
21*d30ea906Sjfb8856606 	struct rte_eth_dev *edev;
22*d30ea906Sjfb8856606 	ptrdiff_t idx;
23*d30ea906Sjfb8856606 
24*d30ea906Sjfb8856606 	/* Avoid Undefined Behaviour */
25*d30ea906Sjfb8856606 	if (start != NULL &&
26*d30ea906Sjfb8856606 	    (start < &rte_eth_devices[0] ||
27*d30ea906Sjfb8856606 	     start > &rte_eth_devices[RTE_MAX_ETHPORTS]))
28*d30ea906Sjfb8856606 		return NULL;
29*d30ea906Sjfb8856606 	if (start != NULL)
30*d30ea906Sjfb8856606 		idx = eth_dev_to_id(start) + 1;
31*d30ea906Sjfb8856606 	else
32*d30ea906Sjfb8856606 		idx = 0;
33*d30ea906Sjfb8856606 	for (; idx < RTE_MAX_ETHPORTS; idx++) {
34*d30ea906Sjfb8856606 		edev = &rte_eth_devices[idx];
35*d30ea906Sjfb8856606 		if (cmp(edev, data) == 0)
36*d30ea906Sjfb8856606 			return edev;
37*d30ea906Sjfb8856606 	}
38*d30ea906Sjfb8856606 	return NULL;
39*d30ea906Sjfb8856606 }
40*d30ea906Sjfb8856606 
41*d30ea906Sjfb8856606 int
rte_eth_devargs_parse_list(char * str,rte_eth_devargs_callback_t callback,void * data)42*d30ea906Sjfb8856606 rte_eth_devargs_parse_list(char *str, rte_eth_devargs_callback_t callback,
43*d30ea906Sjfb8856606 	void *data)
44*d30ea906Sjfb8856606 {
45*d30ea906Sjfb8856606 	char *str_start;
46*d30ea906Sjfb8856606 	int state;
47*d30ea906Sjfb8856606 	int result;
48*d30ea906Sjfb8856606 
49*d30ea906Sjfb8856606 	if (*str != '[')
50*d30ea906Sjfb8856606 		/* Single element, not a list */
51*d30ea906Sjfb8856606 		return callback(str, data);
52*d30ea906Sjfb8856606 
53*d30ea906Sjfb8856606 	/* Sanity check, then strip the brackets */
54*d30ea906Sjfb8856606 	str_start = &str[strlen(str) - 1];
55*d30ea906Sjfb8856606 	if (*str_start != ']') {
56*d30ea906Sjfb8856606 		RTE_LOG(ERR, EAL, "(%s): List does not end with ']'\n", str);
57*d30ea906Sjfb8856606 		return -EINVAL;
58*d30ea906Sjfb8856606 	}
59*d30ea906Sjfb8856606 	str++;
60*d30ea906Sjfb8856606 	*str_start = '\0';
61*d30ea906Sjfb8856606 
62*d30ea906Sjfb8856606 	/* Process list elements */
63*d30ea906Sjfb8856606 	state = 0;
64*d30ea906Sjfb8856606 	while (1) {
65*d30ea906Sjfb8856606 		if (state == 0) {
66*d30ea906Sjfb8856606 			if (*str == '\0')
67*d30ea906Sjfb8856606 				break;
68*d30ea906Sjfb8856606 			if (*str != ',') {
69*d30ea906Sjfb8856606 				str_start = str;
70*d30ea906Sjfb8856606 				state = 1;
71*d30ea906Sjfb8856606 			}
72*d30ea906Sjfb8856606 		} else if (state == 1) {
73*d30ea906Sjfb8856606 			if (*str == ',' || *str == '\0') {
74*d30ea906Sjfb8856606 				if (str > str_start) {
75*d30ea906Sjfb8856606 					/* Non-empty string fragment */
76*d30ea906Sjfb8856606 					*str = '\0';
77*d30ea906Sjfb8856606 					result = callback(str_start, data);
78*d30ea906Sjfb8856606 					if (result < 0)
79*d30ea906Sjfb8856606 						return result;
80*d30ea906Sjfb8856606 				}
81*d30ea906Sjfb8856606 				state = 0;
82*d30ea906Sjfb8856606 			}
83*d30ea906Sjfb8856606 		}
84*d30ea906Sjfb8856606 		str++;
85*d30ea906Sjfb8856606 	}
86*d30ea906Sjfb8856606 	return 0;
87*d30ea906Sjfb8856606 }
88*d30ea906Sjfb8856606 
89*d30ea906Sjfb8856606 static int
rte_eth_devargs_process_range(char * str,uint16_t * list,uint16_t * len_list,const uint16_t max_list)90*d30ea906Sjfb8856606 rte_eth_devargs_process_range(char *str, uint16_t *list, uint16_t *len_list,
91*d30ea906Sjfb8856606 	const uint16_t max_list)
92*d30ea906Sjfb8856606 {
93*d30ea906Sjfb8856606 	uint16_t lo, hi, val;
94*d30ea906Sjfb8856606 	int result;
95*d30ea906Sjfb8856606 
96*d30ea906Sjfb8856606 	result = sscanf(str, "%hu-%hu", &lo, &hi);
97*d30ea906Sjfb8856606 	if (result == 1) {
98*d30ea906Sjfb8856606 		if (*len_list >= max_list)
99*d30ea906Sjfb8856606 			return -ENOMEM;
100*d30ea906Sjfb8856606 		list[(*len_list)++] = lo;
101*d30ea906Sjfb8856606 	} else if (result == 2) {
102*d30ea906Sjfb8856606 		if (lo >= hi || lo > RTE_MAX_ETHPORTS || hi > RTE_MAX_ETHPORTS)
103*d30ea906Sjfb8856606 			return -EINVAL;
104*d30ea906Sjfb8856606 		for (val = lo; val <= hi; val++) {
105*d30ea906Sjfb8856606 			if (*len_list >= max_list)
106*d30ea906Sjfb8856606 				return -ENOMEM;
107*d30ea906Sjfb8856606 			list[(*len_list)++] = val;
108*d30ea906Sjfb8856606 		}
109*d30ea906Sjfb8856606 	} else
110*d30ea906Sjfb8856606 		return -EINVAL;
111*d30ea906Sjfb8856606 	return 0;
112*d30ea906Sjfb8856606 }
113*d30ea906Sjfb8856606 
114*d30ea906Sjfb8856606 int
rte_eth_devargs_parse_representor_ports(char * str,void * data)115*d30ea906Sjfb8856606 rte_eth_devargs_parse_representor_ports(char *str, void *data)
116*d30ea906Sjfb8856606 {
117*d30ea906Sjfb8856606 	struct rte_eth_devargs *eth_da = data;
118*d30ea906Sjfb8856606 
119*d30ea906Sjfb8856606 	return rte_eth_devargs_process_range(str, eth_da->representor_ports,
120*d30ea906Sjfb8856606 		&eth_da->nb_representor_ports, RTE_MAX_ETHPORTS);
121*d30ea906Sjfb8856606 }
122