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