xref: /freebsd/usr.sbin/bhyve/net_utils.c (revision 4d65a7c6951cea0333f1a0c1b32c38489cdfa6c5)
14f7c3b7bSVincenzo Maffione /*-
2*4d846d26SWarner Losh  * SPDX-License-Identifier: BSD-2-Clause
390db4ba9SVincenzo Maffione  *
44f7c3b7bSVincenzo Maffione  * Copyright (c) 2011 NetApp, Inc.
54f7c3b7bSVincenzo Maffione  *
64f7c3b7bSVincenzo Maffione  * Redistribution and use in source and binary forms, with or without
74f7c3b7bSVincenzo Maffione  * modification, are permitted provided that the following conditions
84f7c3b7bSVincenzo Maffione  * are met:
94f7c3b7bSVincenzo Maffione  * 1. Redistributions of source code must retain the above copyright
104f7c3b7bSVincenzo Maffione  *    notice, this list of conditions and the following disclaimer.
114f7c3b7bSVincenzo Maffione  * 2. Redistributions in binary form must reproduce the above copyright
124f7c3b7bSVincenzo Maffione  *    notice, this list of conditions and the following disclaimer in the
134f7c3b7bSVincenzo Maffione  *    documentation and/or other materials provided with the distribution.
144f7c3b7bSVincenzo Maffione  *
154f7c3b7bSVincenzo Maffione  * THIS SOFTWARE IS PROVIDED BY THE AUTHOR AND CONTRIBUTORS``AS IS'' AND
164f7c3b7bSVincenzo Maffione  * ANY EXPRESS OR IMPLIED WARRANTIES, INCLUDING, BUT NOT LIMITED TO, THE
174f7c3b7bSVincenzo Maffione  * IMPLIED WARRANTIES OF MERCHANTABILITY AND FITNESS FOR A PARTICULAR
184f7c3b7bSVincenzo Maffione  * PURPOSE ARE DISCLAIMED. IN NO EVENT SHALL THE AUTHOR OR CONTRIBUTORS
194f7c3b7bSVincenzo Maffione  * BE LIABLE FOR ANY DIRECT, INDIRECT, INCIDENTAL, SPECIAL, EXEMPLARY,
204f7c3b7bSVincenzo Maffione  * OR CONSEQUENTIAL DAMAGES (INCLUDING, BUT NOT LIMITED TO, PROCUREMENT
214f7c3b7bSVincenzo Maffione  * OF SUBSTITUTE GOODS OR SERVICES; LOSS OF USE, DATA, OR PROFITS; OR
224f7c3b7bSVincenzo Maffione  * BUSINESS INTERRUPTION) HOWEVER CAUSED AND ON ANY THEORY OF LIABILITY,
234f7c3b7bSVincenzo Maffione  * WHETHER IN CONTRACT, STRICT LIABILITY, OR TORT (INCLUDING NEGLIGENCE
244f7c3b7bSVincenzo Maffione  * OR OTHERWISE) ARISING IN ANY WAY OUT OF THE USE OF THIS SOFTWARE,
254f7c3b7bSVincenzo Maffione  * EVEN IF ADVISED OF THE POSSIBILITY OF SUCH DAMAGE.
264f7c3b7bSVincenzo Maffione  */
274f7c3b7bSVincenzo Maffione 
287aa24c60SJohn Baldwin #include <sys/types.h>
294f7c3b7bSVincenzo Maffione #include <net/ethernet.h>
307aa24c60SJohn Baldwin 
311ff57e3aSAleksandr Fedorov #include <assert.h>
324f7c3b7bSVincenzo Maffione #include <errno.h>
331ff57e3aSAleksandr Fedorov #include <limits.h>
347aa24c60SJohn Baldwin #include <md5.h>
357aa24c60SJohn Baldwin #include <stdio.h>
361ff57e3aSAleksandr Fedorov #include <stdlib.h>
377aa24c60SJohn Baldwin #include <string.h>
387aa24c60SJohn Baldwin 
397aa24c60SJohn Baldwin #include "bhyverun.h"
40621b5090SJohn Baldwin #include "config.h"
41332eff95SVincenzo Maffione #include "debug.h"
427aa24c60SJohn Baldwin #include "net_utils.h"
434f7c3b7bSVincenzo Maffione 
444f7c3b7bSVincenzo Maffione int
net_parsemac(const char * mac_str,uint8_t * mac_addr)45621b5090SJohn Baldwin net_parsemac(const char *mac_str, uint8_t *mac_addr)
464f7c3b7bSVincenzo Maffione {
474f7c3b7bSVincenzo Maffione         struct ether_addr *ea;
484f7c3b7bSVincenzo Maffione         char zero_addr[ETHER_ADDR_LEN] = { 0, 0, 0, 0, 0, 0 };
494f7c3b7bSVincenzo Maffione 
5066c662b0SVincenzo Maffione 	if (mac_str == NULL)
5166c662b0SVincenzo Maffione 		return (EINVAL);
524f7c3b7bSVincenzo Maffione 
534f7c3b7bSVincenzo Maffione 	ea = ether_aton(mac_str);
544f7c3b7bSVincenzo Maffione 
554f7c3b7bSVincenzo Maffione 	if (ea == NULL || ETHER_IS_MULTICAST(ea->octet) ||
564f7c3b7bSVincenzo Maffione 	    memcmp(ea->octet, zero_addr, ETHER_ADDR_LEN) == 0) {
57332eff95SVincenzo Maffione 		EPRINTLN("Invalid MAC %s", mac_str);
584f7c3b7bSVincenzo Maffione 		return (EINVAL);
594f7c3b7bSVincenzo Maffione 	} else
604f7c3b7bSVincenzo Maffione 		memcpy(mac_addr, ea->octet, ETHER_ADDR_LEN);
614f7c3b7bSVincenzo Maffione 
624f7c3b7bSVincenzo Maffione         return (0);
634f7c3b7bSVincenzo Maffione }
644f7c3b7bSVincenzo Maffione 
651ff57e3aSAleksandr Fedorov int
net_parsemtu(const char * mtu_str,unsigned long * mtu)661ff57e3aSAleksandr Fedorov net_parsemtu(const char *mtu_str, unsigned long *mtu)
671ff57e3aSAleksandr Fedorov {
681ff57e3aSAleksandr Fedorov 	char *end;
691ff57e3aSAleksandr Fedorov 	unsigned long val;
701ff57e3aSAleksandr Fedorov 
711ff57e3aSAleksandr Fedorov 	assert(mtu_str != NULL);
721ff57e3aSAleksandr Fedorov 
731ff57e3aSAleksandr Fedorov 	if (*mtu_str == '-')
741ff57e3aSAleksandr Fedorov 		goto err;
751ff57e3aSAleksandr Fedorov 
761ff57e3aSAleksandr Fedorov 	val = strtoul(mtu_str, &end, 0);
771ff57e3aSAleksandr Fedorov 
781ff57e3aSAleksandr Fedorov 	if (*end != '\0')
791ff57e3aSAleksandr Fedorov 		goto err;
801ff57e3aSAleksandr Fedorov 
811ff57e3aSAleksandr Fedorov 	if (val == ULONG_MAX)
821ff57e3aSAleksandr Fedorov 		return (ERANGE);
831ff57e3aSAleksandr Fedorov 
841ff57e3aSAleksandr Fedorov 	if (val == 0 && errno == EINVAL)
851ff57e3aSAleksandr Fedorov 		return (EINVAL);
861ff57e3aSAleksandr Fedorov 
871ff57e3aSAleksandr Fedorov 	*mtu = val;
881ff57e3aSAleksandr Fedorov 
891ff57e3aSAleksandr Fedorov 	return (0);
901ff57e3aSAleksandr Fedorov 
911ff57e3aSAleksandr Fedorov err:
921ff57e3aSAleksandr Fedorov 	errno = EINVAL;
931ff57e3aSAleksandr Fedorov 	return (EINVAL);
941ff57e3aSAleksandr Fedorov }
951ff57e3aSAleksandr Fedorov 
964f7c3b7bSVincenzo Maffione void
net_genmac(struct pci_devinst * pi,uint8_t * macaddr)974f7c3b7bSVincenzo Maffione net_genmac(struct pci_devinst *pi, uint8_t *macaddr)
984f7c3b7bSVincenzo Maffione {
994f7c3b7bSVincenzo Maffione 	/*
1004f7c3b7bSVincenzo Maffione 	 * The default MAC address is the standard NetApp OUI of 00-a0-98,
1014f7c3b7bSVincenzo Maffione 	 * followed by an MD5 of the PCI slot/func number and dev name
1024f7c3b7bSVincenzo Maffione 	 */
1034f7c3b7bSVincenzo Maffione 	MD5_CTX mdctx;
1044f7c3b7bSVincenzo Maffione 	unsigned char digest[16];
1054f7c3b7bSVincenzo Maffione 	char nstr[80];
1064f7c3b7bSVincenzo Maffione 
1074f7c3b7bSVincenzo Maffione 	snprintf(nstr, sizeof(nstr), "%d-%d-%s", pi->pi_slot,
108621b5090SJohn Baldwin 	    pi->pi_func, get_config_value("name"));
1094f7c3b7bSVincenzo Maffione 
1104f7c3b7bSVincenzo Maffione 	MD5Init(&mdctx);
1114f7c3b7bSVincenzo Maffione 	MD5Update(&mdctx, nstr, (unsigned int)strlen(nstr));
1124f7c3b7bSVincenzo Maffione 	MD5Final(digest, &mdctx);
1134f7c3b7bSVincenzo Maffione 
1144f7c3b7bSVincenzo Maffione 	macaddr[0] = 0x00;
1154f7c3b7bSVincenzo Maffione 	macaddr[1] = 0xa0;
1164f7c3b7bSVincenzo Maffione 	macaddr[2] = 0x98;
1174f7c3b7bSVincenzo Maffione 	macaddr[3] = digest[0];
1184f7c3b7bSVincenzo Maffione 	macaddr[4] = digest[1];
1194f7c3b7bSVincenzo Maffione 	macaddr[5] = digest[2];
1204f7c3b7bSVincenzo Maffione }
121