xref: /dpdk/lib/net/rte_ether.c (revision df6e6ddecff1a9d7f95df3afd9627c9a958f67ff)
199a2dd95SBruce Richardson /* SPDX-License-Identifier: BSD-3-Clause
299a2dd95SBruce Richardson  * Copyright(c) 2010-2014 Intel Corporation
399a2dd95SBruce Richardson  */
499a2dd95SBruce Richardson 
599a2dd95SBruce Richardson #include <stdbool.h>
699a2dd95SBruce Richardson 
799a2dd95SBruce Richardson #include <rte_ether.h>
899a2dd95SBruce Richardson #include <rte_errno.h>
999a2dd95SBruce Richardson 
1099a2dd95SBruce Richardson void
rte_eth_random_addr(uint8_t * addr)1199a2dd95SBruce Richardson rte_eth_random_addr(uint8_t *addr)
1299a2dd95SBruce Richardson {
1399a2dd95SBruce Richardson 	uint64_t rand = rte_rand();
1499a2dd95SBruce Richardson 	uint8_t *p = (uint8_t *)&rand;
1599a2dd95SBruce Richardson 
1699a2dd95SBruce Richardson 	rte_memcpy(addr, p, RTE_ETHER_ADDR_LEN);
1799a2dd95SBruce Richardson 	addr[0] &= (uint8_t)~RTE_ETHER_GROUP_ADDR;	/* clear multicast bit */
1899a2dd95SBruce Richardson 	addr[0] |= RTE_ETHER_LOCAL_ADMIN_ADDR;	/* set local assignment bit */
1999a2dd95SBruce Richardson }
2099a2dd95SBruce Richardson 
2199a2dd95SBruce Richardson void
rte_ether_format_addr(char * buf,uint16_t size,const struct rte_ether_addr * eth_addr)2299a2dd95SBruce Richardson rte_ether_format_addr(char *buf, uint16_t size,
2399a2dd95SBruce Richardson 		      const struct rte_ether_addr *eth_addr)
2499a2dd95SBruce Richardson {
25c2c4f87bSAman Deep Singh 	snprintf(buf, size, RTE_ETHER_ADDR_PRT_FMT,
26a7db3afcSAman Deep Singh 		RTE_ETHER_ADDR_BYTES(eth_addr));
2799a2dd95SBruce Richardson }
2899a2dd95SBruce Richardson 
get_xdigit(char ch)2999a2dd95SBruce Richardson static int8_t get_xdigit(char ch)
3099a2dd95SBruce Richardson {
3199a2dd95SBruce Richardson 	if (ch >= '0' && ch <= '9')
3299a2dd95SBruce Richardson 		return ch - '0';
3399a2dd95SBruce Richardson 	if (ch >= 'a' && ch <= 'f')
3499a2dd95SBruce Richardson 		return ch - 'a' + 10;
3599a2dd95SBruce Richardson 	if (ch >= 'A' && ch <= 'F')
3699a2dd95SBruce Richardson 		return ch - 'A' + 10;
3799a2dd95SBruce Richardson 	return -1;
3899a2dd95SBruce Richardson }
3999a2dd95SBruce Richardson 
4099a2dd95SBruce Richardson /* Convert 00:11:22:33:44:55 to ethernet address */
get_ether_addr6(const char * s0,struct rte_ether_addr * ea,const char sep)41*df6e6ddeSStephen Hemminger static bool get_ether_addr6(const char *s0, struct rte_ether_addr *ea,
42*df6e6ddeSStephen Hemminger 			    const char sep)
4399a2dd95SBruce Richardson {
4499a2dd95SBruce Richardson 	const char *s = s0;
4599a2dd95SBruce Richardson 	int i;
4699a2dd95SBruce Richardson 
4799a2dd95SBruce Richardson 	for (i = 0; i < RTE_ETHER_ADDR_LEN; i++) {
4899a2dd95SBruce Richardson 		int8_t x;
4999a2dd95SBruce Richardson 
5099a2dd95SBruce Richardson 		x = get_xdigit(*s++);
5199a2dd95SBruce Richardson 		if (x < 0)
52*df6e6ddeSStephen Hemminger 			return false;	/* not a hex digit */
5399a2dd95SBruce Richardson 
54*df6e6ddeSStephen Hemminger 		ea->addr_bytes[i] = x;
55*df6e6ddeSStephen Hemminger 		if (*s != sep && *s != '\0') {
5699a2dd95SBruce Richardson 			x = get_xdigit(*s++);
5799a2dd95SBruce Richardson 			if (x < 0)
58*df6e6ddeSStephen Hemminger 				return false;	/* not a hex digit */
59*df6e6ddeSStephen Hemminger 			ea->addr_bytes[i] <<= 4;
6099a2dd95SBruce Richardson 			ea->addr_bytes[i] |= x;
6199a2dd95SBruce Richardson 		}
6299a2dd95SBruce Richardson 
63*df6e6ddeSStephen Hemminger 		if (i < RTE_ETHER_ADDR_LEN - 1 &&
64*df6e6ddeSStephen Hemminger 		    *s++ != sep)
65*df6e6ddeSStephen Hemminger 			return false;	/* premature end of string */
66*df6e6ddeSStephen Hemminger 	}
67*df6e6ddeSStephen Hemminger 
68*df6e6ddeSStephen Hemminger 	/* return true if no trailing characters */
6999a2dd95SBruce Richardson 	return *s == '\0';
7099a2dd95SBruce Richardson }
7199a2dd95SBruce Richardson 
7299a2dd95SBruce Richardson /* Convert 0011:2233:4455 to ethernet address */
get_ether_addr3(const char * s,struct rte_ether_addr * ea,const char sep)73*df6e6ddeSStephen Hemminger static bool get_ether_addr3(const char *s, struct rte_ether_addr *ea,
74*df6e6ddeSStephen Hemminger 			    const char sep)
7599a2dd95SBruce Richardson {
7699a2dd95SBruce Richardson 	int i, j;
7799a2dd95SBruce Richardson 
7899a2dd95SBruce Richardson 	for (i = 0; i < RTE_ETHER_ADDR_LEN; i += 2) {
7999a2dd95SBruce Richardson 		uint16_t w = 0;
8099a2dd95SBruce Richardson 
8199a2dd95SBruce Richardson 		for (j = 0; j < 4; j++) {
8299a2dd95SBruce Richardson 			int8_t x;
8399a2dd95SBruce Richardson 
8499a2dd95SBruce Richardson 			x = get_xdigit(*s++);
8599a2dd95SBruce Richardson 			if (x < 0)
86*df6e6ddeSStephen Hemminger 				return false;	/* not a hex digit */
8799a2dd95SBruce Richardson 			w = (w << 4) | x;
8899a2dd95SBruce Richardson 		}
89*df6e6ddeSStephen Hemminger 
9099a2dd95SBruce Richardson 		ea->addr_bytes[i] = w >> 8;
9199a2dd95SBruce Richardson 		ea->addr_bytes[i + 1] = w & 0xff;
9299a2dd95SBruce Richardson 
9399a2dd95SBruce Richardson 		if (i < RTE_ETHER_ADDR_LEN - 2 &&
94*df6e6ddeSStephen Hemminger 		    *s++ != sep)
9599a2dd95SBruce Richardson 			return false;
9699a2dd95SBruce Richardson 	}
9799a2dd95SBruce Richardson 
9899a2dd95SBruce Richardson 	return *s == '\0';
9999a2dd95SBruce Richardson }
10099a2dd95SBruce Richardson 
10199a2dd95SBruce Richardson /*
102*df6e6ddeSStephen Hemminger  * Scan input to see if separated by dash, colon or period
103*df6e6ddeSStephen Hemminger  * Returns separator and number of matches
104*df6e6ddeSStephen Hemminger  * If separators are mixed will return
105*df6e6ddeSStephen Hemminger  */
get_ether_sep(const char * s,char * sep)106*df6e6ddeSStephen Hemminger static unsigned int get_ether_sep(const char *s, char *sep)
107*df6e6ddeSStephen Hemminger {
108*df6e6ddeSStephen Hemminger 	static const char separators[] = "-:.";
109*df6e6ddeSStephen Hemminger 	unsigned int count = 0;
110*df6e6ddeSStephen Hemminger 	const char *cp;
111*df6e6ddeSStephen Hemminger 
112*df6e6ddeSStephen Hemminger 	cp = strpbrk(s, separators);
113*df6e6ddeSStephen Hemminger 	if (cp == NULL)
114*df6e6ddeSStephen Hemminger 		return 0;	/* no separator found */
115*df6e6ddeSStephen Hemminger 
116*df6e6ddeSStephen Hemminger 	*sep = *cp;		/* return the separator */
117*df6e6ddeSStephen Hemminger 	do {
118*df6e6ddeSStephen Hemminger 		++count;
119*df6e6ddeSStephen Hemminger 		/* find next instance of separator */
120*df6e6ddeSStephen Hemminger 		cp = strchr(cp + 1, *sep);
121*df6e6ddeSStephen Hemminger 	} while (cp != NULL);
122*df6e6ddeSStephen Hemminger 
123*df6e6ddeSStephen Hemminger 	return count;
124*df6e6ddeSStephen Hemminger }
125*df6e6ddeSStephen Hemminger 
126*df6e6ddeSStephen Hemminger /*
127*df6e6ddeSStephen Hemminger  * Be liberal in accepting a wide variety of notational formats
128*df6e6ddeSStephen Hemminger  * for MAC address including:
129*df6e6ddeSStephen Hemminger  *  - Linux format six groups of hexadecimal digits separated by colon
130*df6e6ddeSStephen Hemminger  *  - Windows format six groups separated by hyphen
131*df6e6ddeSStephen Hemminger  *  - two groups hexadecimal digits
13299a2dd95SBruce Richardson  */
13399a2dd95SBruce Richardson int
rte_ether_unformat_addr(const char * s,struct rte_ether_addr * ea)13499a2dd95SBruce Richardson rte_ether_unformat_addr(const char *s, struct rte_ether_addr *ea)
13599a2dd95SBruce Richardson {
136*df6e6ddeSStephen Hemminger 	unsigned int count;
137*df6e6ddeSStephen Hemminger 	char sep = '\0';
138*df6e6ddeSStephen Hemminger 
139*df6e6ddeSStephen Hemminger 	count = get_ether_sep(s, &sep);
140*df6e6ddeSStephen Hemminger 	switch (count) {
141*df6e6ddeSStephen Hemminger 	case 5:	/* i.e 01:23:45:67:89:AB */
142*df6e6ddeSStephen Hemminger 		if (get_ether_addr6(s, ea, sep))
14399a2dd95SBruce Richardson 			return 0;
144*df6e6ddeSStephen Hemminger 		break;
145*df6e6ddeSStephen Hemminger 	case 2: /* i.e 0123.4567.89AB */
146*df6e6ddeSStephen Hemminger 		if (get_ether_addr3(s, ea, sep))
14799a2dd95SBruce Richardson 			return 0;
148*df6e6ddeSStephen Hemminger 		break;
149*df6e6ddeSStephen Hemminger 	default:
150*df6e6ddeSStephen Hemminger 		break;
151*df6e6ddeSStephen Hemminger 	}
15299a2dd95SBruce Richardson 
15399a2dd95SBruce Richardson 	rte_errno = EINVAL;
15499a2dd95SBruce Richardson 	return -1;
15599a2dd95SBruce Richardson }
156