diff options
Diffstat (limited to 'lib')
| -rw-r--r-- | lib/geom/part/gpart.8 | 22 | ||||
| -rw-r--r-- | lib/libarchive/tests/Makefile | 6 | ||||
| -rw-r--r-- | lib/libc/inet/inet_net_ntop.c | 206 | ||||
| -rw-r--r-- | lib/libc/inet/inet_net_pton.c | 363 | ||||
| -rw-r--r-- | lib/libc/tests/net/Makefile | 2 | ||||
| -rw-r--r-- | lib/libc/tests/net/inet_net_test.cc | 333 | ||||
| -rw-r--r-- | lib/libmd/Makefile | 9 | ||||
| -rw-r--r-- | lib/libmd/aarch64/md5block.S | 206 | ||||
| -rw-r--r-- | lib/libmd/aarch64/sha1block.S | 2 | ||||
| -rw-r--r-- | lib/libmd/aarch64/sha1dispatch.c | 2 | ||||
| -rw-r--r-- | lib/libmd/amd64/md5block.S | 363 | ||||
| -rw-r--r-- | lib/libmd/amd64/md5dispatch.c | 41 | ||||
| -rw-r--r-- | lib/libmd/amd64/sha1block.S | 2 | ||||
| -rw-r--r-- | lib/libmd/amd64/sha1dispatch.c | 2 | ||||
| -rw-r--r-- | lib/libmd/sha1c.c | 2 | ||||
| -rw-r--r-- | lib/libsys/clock_gettime.2 | 3 | ||||
| -rw-r--r-- | lib/libsys/getrlimitusage.2 | 4 | ||||
| -rw-r--r-- | lib/libunbound/config.h | 4 | ||||
| -rw-r--r-- | lib/libutil/mntopts.c | 53 | 
19 files changed, 1184 insertions, 441 deletions
| diff --git a/lib/geom/part/gpart.8 b/lib/geom/part/gpart.8 index f76c1d9d5d6c..2e11417f8494 100644 --- a/lib/geom/part/gpart.8 +++ b/lib/geom/part/gpart.8 @@ -22,7 +22,7 @@  .\" OUT OF THE USE OF THIS SOFTWARE, EVEN IF ADVISED OF THE POSSIBILITY OF  .\" SUCH DAMAGE.  .\" -.Dd February 11, 2025 +.Dd October 24, 2025  .Dt GPART 8  .Os  .Sh NAME @@ -1497,6 +1497,26 @@ and  .Bd -literal -offset indent  /sbin/gpart backup ada0 | /sbin/gpart restore -F ada1 ada2  .Ed +.Sh DIAGNOSTICS +.Bl -diag +.It gpart: arg0 '%s': Invalid argument +The provided +.Ar geom +argument +is not a GEOM provider. +Not every device in +.Xr devfs 4 +is a GEOM provider. +For example, a +.Xr zfs 4 +zvol will show up as a GEOM provider only if its +.Sy volmode +is set properly +.Po refer to +.Xr zfsprops 8 +for details +.Pc . +.El  .Sh SEE ALSO  .Xr geom 4 ,  .Xr boot0cfg 8 , diff --git a/lib/libarchive/tests/Makefile b/lib/libarchive/tests/Makefile index 07c5fe24dd30..930250d974c5 100644 --- a/lib/libarchive/tests/Makefile +++ b/lib/libarchive/tests/Makefile @@ -185,6 +185,7 @@ TESTS_SRCS= \  	test_read_format_rar_overflow.c		\  	test_read_format_raw.c			\  	test_read_format_tar.c			\ +	test_read_format_tar_V_negative_size.c	\  	test_read_format_tar_concatenated.c	\  	test_read_format_tar_empty_filename.c	\  	test_read_format_tar_empty_pax.c	\ @@ -607,6 +608,7 @@ ${PACKAGE}FILES+=	test_read_format_rar5_decode_number_out_of_bounds_read.rar.uu  ${PACKAGE}FILES+=	test_read_format_rar5_different_solid_window_size.rar.uu  ${PACKAGE}FILES+=	test_read_format_rar5_different_window_size.rar.uu  ${PACKAGE}FILES+=	test_read_format_rar5_different_winsize_on_merge.rar.uu +${PACKAGE}FILES+=	test_read_format_rar5_dirdata.rar.uu  ${PACKAGE}FILES+=	test_read_format_rar5_distance_overflow.rar.uu  ${PACKAGE}FILES+=	test_read_format_rar5_encrypted.rar.uu  ${PACKAGE}FILES+=	test_read_format_rar5_encrypted_filenames.rar.uu @@ -616,6 +618,7 @@ ${PACKAGE}FILES+=	test_read_format_rar5_extra_field_version.rar.uu  ${PACKAGE}FILES+=	test_read_format_rar5_fileattr.rar.uu  ${PACKAGE}FILES+=	test_read_format_rar5_hardlink.rar.uu  ${PACKAGE}FILES+=	test_read_format_rar5_invalid_dict_reference.rar.uu +${PACKAGE}FILES+=	test_read_format_rar5_invalid_hash_valid_htime_exfld.rar.uu  ${PACKAGE}FILES+=	test_read_format_rar5_leftshift1.rar.uu  ${PACKAGE}FILES+=	test_read_format_rar5_leftshift2.rar.uu  ${PACKAGE}FILES+=	test_read_format_rar5_multiarchive.part01.rar.uu @@ -633,6 +636,7 @@ ${PACKAGE}FILES+=	test_read_format_rar5_multiarchive_solid.part04.rar.uu  ${PACKAGE}FILES+=	test_read_format_rar5_multiple_files.rar.uu  ${PACKAGE}FILES+=	test_read_format_rar5_multiple_files_solid.rar.uu  ${PACKAGE}FILES+=	test_read_format_rar5_nonempty_dir_stream.rar.uu +${PACKAGE}FILES+=	test_read_format_rar5_only_crypt_exfld.rar.uu  ${PACKAGE}FILES+=	test_read_format_rar5_owner.rar.uu  ${PACKAGE}FILES+=	test_read_format_rar5_readtables_overflow.rar.uu  ${PACKAGE}FILES+=	test_read_format_rar5_sfx.exe.uu @@ -642,12 +646,14 @@ ${PACKAGE}FILES+=	test_read_format_rar5_stored_manyfiles.rar.uu  ${PACKAGE}FILES+=	test_read_format_rar5_symlink.rar.uu  ${PACKAGE}FILES+=	test_read_format_rar5_truncated_huff.rar.uu  ${PACKAGE}FILES+=	test_read_format_rar5_unicode.rar.uu +${PACKAGE}FILES+=	test_read_format_rar5_unsupported_exfld.rar.uu  ${PACKAGE}FILES+=	test_read_format_rar5_win32.rar.uu  ${PACKAGE}FILES+=	test_read_format_rar5_window_buf_and_size_desync.rar.uu  ${PACKAGE}FILES+=	test_read_format_raw.bufr.uu  ${PACKAGE}FILES+=	test_read_format_raw.data.Z.uu  ${PACKAGE}FILES+=	test_read_format_raw.data.gz.uu  ${PACKAGE}FILES+=	test_read_format_raw.data.uu +${PACKAGE}FILES+=	test_read_format_tar_V_negative_size.tar.uu  ${PACKAGE}FILES+=	test_read_format_tar_concatenated.tar.uu  ${PACKAGE}FILES+=	test_read_format_tar_empty_filename.tar.uu  ${PACKAGE}FILES+=	test_read_format_tar_empty_with_gnulabel.tar.uu diff --git a/lib/libc/inet/inet_net_ntop.c b/lib/libc/inet/inet_net_ntop.c index 9d98dbb5ca99..30dd5c0571f2 100644 --- a/lib/libc/inet/inet_net_ntop.c +++ b/lib/libc/inet/inet_net_ntop.c @@ -1,20 +1,23 @@ -/*- - * SPDX-License-Identifier: ISC +/*	$OpenBSD: inet_net_ntop.c,v 1.9 2019/07/03 03:24:04 deraadt Exp $	*/ + +/* + * Copyright (c) 2012 by Gilles Chehade <gilles@openbsd.org> + * Copyright (c) 1996 by Internet Software Consortium.   * - * Copyright (c) 2004 by Internet Systems Consortium, Inc. ("ISC") - * Copyright (c) 1996,1999 by Internet Software Consortium. + * SPDX-License-Identifier: ISC   *   * Permission to use, copy, modify, and distribute this software for any   * purpose with or without fee is hereby granted, provided that the above   * copyright notice and this permission notice appear in all copies.   * - * THE SOFTWARE IS PROVIDED "AS IS" AND ISC DISCLAIMS ALL WARRANTIES - * WITH REGARD TO THIS SOFTWARE INCLUDING ALL IMPLIED WARRANTIES OF - * MERCHANTABILITY AND FITNESS.  IN NO EVENT SHALL ISC BE LIABLE FOR - * ANY SPECIAL, DIRECT, INDIRECT, OR CONSEQUENTIAL DAMAGES OR ANY DAMAGES - * WHATSOEVER RESULTING FROM LOSS OF USE, DATA OR PROFITS, WHETHER IN AN - * ACTION OF CONTRACT, NEGLIGENCE OR OTHER TORTIOUS ACTION, ARISING OUT - * OF OR IN CONNECTION WITH THE USE OR PERFORMANCE OF THIS SOFTWARE. + * THE SOFTWARE IS PROVIDED "AS IS" AND INTERNET SOFTWARE CONSORTIUM DISCLAIMS + * ALL WARRANTIES WITH REGARD TO THIS SOFTWARE INCLUDING ALL IMPLIED WARRANTIES + * OF MERCHANTABILITY AND FITNESS. IN NO EVENT SHALL INTERNET SOFTWARE + * CONSORTIUM BE LIABLE FOR ANY SPECIAL, DIRECT, INDIRECT, OR CONSEQUENTIAL + * DAMAGES OR ANY DAMAGES WHATSOEVER RESULTING FROM LOSS OF USE, DATA OR + * PROFITS, WHETHER IN AN ACTION OF CONTRACT, NEGLIGENCE OR OTHER TORTIOUS + * ACTION, ARISING OUT OF OR IN CONNECTION WITH THE USE OR PERFORMANCE OF THIS + * SOFTWARE.   */  #include "port_before.h" @@ -31,18 +34,10 @@  #include "port_after.h" -#ifdef SPRINTF_CHAR -# define SPRINTF(x) strlen(sprintf/**/x) -#else -# define SPRINTF(x) ((size_t)sprintf x) -#endif - -static char *	inet_net_ntop_ipv4(const u_char *src, int bits, char *dst, -		    size_t size); -static char *	inet_net_ntop_ipv6(const u_char *src, int bits, char *dst, -		    size_t size); +static char *inet_net_ntop_ipv4(const u_char *, int, char *, size_t); +static char *inet_net_ntop_ipv6(const u_char *, int, char *, size_t); -/*% +/*   * char *   * inet_net_ntop(af, src, bits, dst, size)   *	convert network number from network to presentation format. @@ -66,7 +61,7 @@ inet_net_ntop(int af, const void *src, int bits, char *dst, size_t size)  	}  } -/*% +/*   * static char *   * inet_net_ntop_ipv4(src, bits, dst, size)   *	convert IPv4 network number from network to presentation format. @@ -83,53 +78,63 @@ static char *  inet_net_ntop_ipv4(const u_char *src, int bits, char *dst, size_t size)  {  	char *odst = dst; -	char *t;  	u_int m;  	int b; +	char *ep; +	int advance; + +	ep = dst + size; +	if (ep <= dst) +		goto emsgsize;  	if (bits < 0 || bits > 32) {  		errno = EINVAL;  		return (NULL);  	} -  	if (bits == 0) { -		if (size < sizeof "0") +		if (ep - dst < sizeof "0")  			goto emsgsize;  		*dst++ = '0'; -		size--;  		*dst = '\0';  	}  	/* Format whole octets. */  	for (b = bits / 8; b > 0; b--) { -		if (size <= sizeof "255.") +		if (ep - dst < sizeof "255.") +			goto emsgsize; +		advance = snprintf(dst, ep - dst, "%u", *src++); +		if (advance <= 0 || advance >= ep - dst)  			goto emsgsize; -		t = dst; -		dst += SPRINTF((dst, "%u", *src++)); +		dst += advance;  		if (b > 1) { +			if (dst + 1 >= ep) +				goto emsgsize;  			*dst++ = '.';  			*dst = '\0';  		} -		size -= (size_t)(dst - t);  	}  	/* Format partial octet. */  	b = bits % 8;  	if (b > 0) { -		if (size <= sizeof ".255") +		if (ep - dst < sizeof ".255")  			goto emsgsize; -		t = dst;  		if (dst != odst)  			*dst++ = '.';  		m = ((1 << b) - 1) << (8 - b); -		dst += SPRINTF((dst, "%u", *src & m)); -		size -= (size_t)(dst - t); +		advance = snprintf(dst, ep - dst, "%u", *src & m); +		if (advance <= 0 || advance >= ep - dst) +			goto emsgsize; +		dst += advance;  	}  	/* Format CIDR /width. */ -	if (size <= sizeof "/32") +	if (ep - dst < sizeof "/32") +		goto emsgsize; +	advance = snprintf(dst, ep - dst, "/%u", bits); +	if (advance <= 0 || advance >= ep - dst)  		goto emsgsize; -	dst += SPRINTF((dst, "/%u", bits)); +	dst += advance;  	return (odst);   emsgsize: @@ -137,132 +142,27 @@ inet_net_ntop_ipv4(const u_char *src, int bits, char *dst, size_t size)  	return (NULL);  } -/*% - * static char * - * inet_net_ntop_ipv6(src, bits, fakebits, dst, size) - *	convert IPv6 network number from network to presentation format. - *	generates CIDR style result always. Picks the shortest representation - *	unless the IP is really IPv4. - *	always prints specified number of bits (bits). - * return: - *	pointer to dst, or NULL if an error occurred (check errno). - * note: - *	network byte order assumed.  this means 192.5.5.240/28 has - *	0b11110000 in its fourth octet. - * author: - *	Vadim Kogan (UCB), June 2001 - *  Original version (IPv4) by Paul Vixie (ISC), July 1996 - */ -  static char * -inet_net_ntop_ipv6(const u_char *src, int bits, char *dst, size_t size) { -	u_int	m; -	int	b; -	int	p; -	int	zero_s, zero_l, tmp_zero_s, tmp_zero_l; -	int	i; -	int	is_ipv4 = 0; -	unsigned char inbuf[16]; -	char outbuf[sizeof("xxxx:xxxx:xxxx:xxxx:xxxx:xxxx:255.255.255.255/128")]; -	char	*cp; -	int	words; -	u_char	*s; +inet_net_ntop_ipv6(const u_char *src, int bits, char *dst, size_t size) +{ +	int	ret; +	char	buf[sizeof("xxxx:xxxx:xxxx:xxxx:xxxx:xxxx:255:255:255:255/128")];  	if (bits < 0 || bits > 128) {  		errno = EINVAL;  		return (NULL);  	} -	cp = outbuf; - -	if (bits == 0) { -		*cp++ = ':'; -		*cp++ = ':'; -		*cp = '\0'; -	} else { -		/* Copy src to private buffer.  Zero host part. */ -		p = (bits + 7) / 8; -		memcpy(inbuf, src, p); -		memset(inbuf + p, 0, 16 - p); -		b = bits % 8; -		if (b != 0) { -			m = ~0 << (8 - b); -			inbuf[p-1] &= m; -		} - -		s = inbuf; - -		/* how many words need to be displayed in output */ -		words = (bits + 15) / 16; -		if (words == 1) -			words = 2; - -		/* Find the longest substring of zero's */ -		zero_s = zero_l = tmp_zero_s = tmp_zero_l = 0; -		for (i = 0; i < (words * 2); i += 2) { -			if ((s[i] | s[i+1]) == 0) { -				if (tmp_zero_l == 0) -					tmp_zero_s = i / 2; -				tmp_zero_l++; -			} else { -				if (tmp_zero_l && zero_l < tmp_zero_l) { -					zero_s = tmp_zero_s; -					zero_l = tmp_zero_l; -					tmp_zero_l = 0; -				} -			} -		} - -		if (tmp_zero_l && zero_l < tmp_zero_l) { -			zero_s = tmp_zero_s; -			zero_l = tmp_zero_l; -		} - -		if (zero_l != words && zero_s == 0 && ((zero_l == 6) || -		    ((zero_l == 5 && s[10] == 0xff && s[11] == 0xff) || -		    ((zero_l == 7 && s[14] != 0 && s[15] != 1))))) -			is_ipv4 = 1; - -		/* Format whole words. */ -		for (p = 0; p < words; p++) { -			if (zero_l != 0 && p >= zero_s && p < zero_s + zero_l) { -				/* Time to skip some zeros */ -				if (p == zero_s) -					*cp++ = ':'; -				if (p == words - 1) -					*cp++ = ':'; -				s++; -				s++; -				continue; -			} +	if (inet_ntop(AF_INET6, src, buf, size) == NULL) +		return (NULL); -			if (is_ipv4 && p > 5 ) { -				*cp++ = (p == 6) ? ':' : '.'; -				cp += SPRINTF((cp, "%u", *s++)); -				/* we can potentially drop the last octet */ -				if (p != 7 || bits > 120) { -					*cp++ = '.'; -					cp += SPRINTF((cp, "%u", *s++)); -				} -			} else { -				if (cp != outbuf) -					*cp++ = ':'; -				cp += SPRINTF((cp, "%x", *s * 256 + s[1])); -				s += 2; -			} -		} +	ret = snprintf(dst, size, "%s/%d", buf, bits); +	if (ret < 0 || ret >= size) { +		errno = EMSGSIZE; +		return (NULL);  	} -	/* Format CIDR /width. */ -	sprintf(cp, "/%u", bits); -	if (strlen(outbuf) + 1 > size) -		goto emsgsize; -	strcpy(dst, outbuf);  	return (dst); - -emsgsize: -	errno = EMSGSIZE; -	return (NULL);  }  /* @@ -271,5 +171,3 @@ emsgsize:   */  #undef inet_net_ntop  __weak_reference(__inet_net_ntop, inet_net_ntop); - -/*! \file */ diff --git a/lib/libc/inet/inet_net_pton.c b/lib/libc/inet/inet_net_pton.c index d566a0e1d3c3..14c88eb72014 100644 --- a/lib/libc/inet/inet_net_pton.c +++ b/lib/libc/inet/inet_net_pton.c @@ -1,20 +1,23 @@ -/*- - * SPDX-License-Identifier: ISC +/*	$OpenBSD: inet_net_pton.c,v 1.14 2022/12/27 17:10:06 jmc Exp $	*/ + +/* + * Copyright (c) 2012 by Gilles Chehade <gilles@openbsd.org> + * Copyright (c) 1996,1999 by Internet Software Consortium.   * - * Copyright (C) 2004, 2005, 2008  Internet Systems Consortium, Inc. ("ISC") - * Copyright (C) 1996, 1998, 1999, 2001, 2003  Internet Software Consortium. + * SPDX-License-Identifier: ISC   * - * Permission to use, copy, modify, and/or distribute this software for any + * Permission to use, copy, modify, and distribute this software for any   * purpose with or without fee is hereby granted, provided that the above   * copyright notice and this permission notice appear in all copies.   * - * THE SOFTWARE IS PROVIDED "AS IS" AND ISC DISCLAIMS ALL WARRANTIES WITH - * REGARD TO THIS SOFTWARE INCLUDING ALL IMPLIED WARRANTIES OF MERCHANTABILITY - * AND FITNESS.  IN NO EVENT SHALL ISC BE LIABLE FOR ANY SPECIAL, DIRECT, - * INDIRECT, OR CONSEQUENTIAL DAMAGES OR ANY DAMAGES WHATSOEVER RESULTING FROM - * LOSS OF USE, DATA OR PROFITS, WHETHER IN AN ACTION OF CONTRACT, NEGLIGENCE - * OR OTHER TORTIOUS ACTION, ARISING OUT OF OR IN CONNECTION WITH THE USE OR - * PERFORMANCE OF THIS SOFTWARE. + * THE SOFTWARE IS PROVIDED "AS IS" AND INTERNET SOFTWARE CONSORTIUM DISCLAIMS + * ALL WARRANTIES WITH REGARD TO THIS SOFTWARE INCLUDING ALL IMPLIED WARRANTIES + * OF MERCHANTABILITY AND FITNESS. IN NO EVENT SHALL INTERNET SOFTWARE + * CONSORTIUM BE LIABLE FOR ANY SPECIAL, DIRECT, INDIRECT, OR CONSEQUENTIAL + * DAMAGES OR ANY DAMAGES WHATSOEVER RESULTING FROM LOSS OF USE, DATA OR + * PROFITS, WHETHER IN AN ACTION OF CONTRACT, NEGLIGENCE OR OTHER TORTIOUS + * ACTION, ARISING OUT OF OR IN CONNECTION WITH THE USE OR PERFORMANCE OF THIS + * SOFTWARE.   */  #include "port_before.h" @@ -22,7 +25,6 @@  #include <sys/types.h>  #include <sys/socket.h>  #include <netinet/in.h> -#include <arpa/nameser.h>  #include <arpa/inet.h>  #include <assert.h> @@ -34,13 +36,37 @@  #include "port_after.h" -#ifdef SPRINTF_CHAR -# define SPRINTF(x) strlen(sprintf/**/x) -#else -# define SPRINTF(x) ((size_t)sprintf x) -#endif +static int	inet_net_pton_ipv4(const char *, u_char *, size_t); +static int	inet_net_pton_ipv6(const char *, u_char *, size_t); -/*% +/* + * static int + * inet_net_pton(af, src, dst, size) + *	convert network number from presentation to network format. + *	accepts hex octets, hex strings, decimal octets, and /CIDR. + *	"size" is in bytes and describes "dst". + * return: + *	number of bits, either imputed classfully or specified with /CIDR, + *	or -1 if some failure occurred (check errno).  ENOENT means it was + *	not a valid network specification. + * author: + *	Paul Vixie (ISC), June 1996 + */ +int +inet_net_pton(int af, const char *src, void *dst, size_t size) +{ +	switch (af) { +	case AF_INET: +		return (inet_net_pton_ipv4(src, dst, size)); +	case AF_INET6: +		return (inet_net_pton_ipv6(src, dst, size)); +	default: +		errno = EAFNOSUPPORT; +		return (-1); +	} +} + +/*   * static int   * inet_net_pton_ipv4(src, dst, size)   *	convert IPv4 network number from presentation to network format. @@ -57,22 +83,24 @@   *	Paul Vixie (ISC), June 1996   */  static int -inet_net_pton_ipv4(const char *src, u_char *dst, size_t size) { -	static const char xdigits[] = "0123456789abcdef"; -	static const char digits[] = "0123456789"; -	int n, ch, tmp = 0, dirty, bits; +inet_net_pton_ipv4(const char *src, u_char *dst, size_t size) +{ +	static const char +		xdigits[] = "0123456789abcdef", +		digits[] = "0123456789"; +	int n, ch, tmp, dirty, bits;  	const u_char *odst = dst; -	ch = *src++; +	ch = (unsigned char)*src++;  	if (ch == '0' && (src[0] == 'x' || src[0] == 'X') -	    && isascii((unsigned char)(src[1])) -	    && isxdigit((unsigned char)(src[1]))) { +	    && isascii((unsigned char)src[1]) && isxdigit((unsigned char)src[1])) {  		/* Hexadecimal: Eat nybble string. */ -		if (size <= 0U) +		if (size == 0)  			goto emsgsize; -		dirty = 0; -		src++;	/*%< skip x or X. */ -		while ((ch = *src++) != '\0' && isascii(ch) && isxdigit(ch)) { +		tmp = 0, dirty = 0; +		src++;	/* skip x or X. */ +		while ((ch = (unsigned char)*src++) != '\0' && +		    isascii(ch) && isxdigit(ch)) {  			if (isupper(ch))  				ch = tolower(ch);  			n = strchr(xdigits, ch) - xdigits; @@ -82,14 +110,14 @@ inet_net_pton_ipv4(const char *src, u_char *dst, size_t size) {  			else  				tmp = (tmp << 4) | n;  			if (++dirty == 2) { -				if (size-- <= 0U) +				if (size-- == 0)  					goto emsgsize;  				*dst++ = (u_char) tmp;  				dirty = 0;  			}  		} -		if (dirty) {  /*%< Odd trailing nybble? */ -			if (size-- <= 0U) +		if (dirty) {  /* Odd trailing nybble? */ +			if (size-- == 0)  				goto emsgsize;  			*dst++ = (u_char) (tmp << 4);  		} @@ -104,16 +132,16 @@ inet_net_pton_ipv4(const char *src, u_char *dst, size_t size) {  				tmp += n;  				if (tmp > 255)  					goto enoent; -			} while ((ch = *src++) != '\0' && +			} while ((ch = (unsigned char)*src++) != '\0' &&  				 isascii(ch) && isdigit(ch)); -			if (size-- <= 0U) +			if (size-- == 0)  				goto emsgsize;  			*dst++ = (u_char) tmp;  			if (ch == '\0' || ch == '/')  				break;  			if (ch != '.')  				goto enoent; -			ch = *src++; +			ch = (unsigned char)*src++;  			if (!isascii(ch) || !isdigit(ch))  				goto enoent;  		} @@ -121,10 +149,10 @@ inet_net_pton_ipv4(const char *src, u_char *dst, size_t size) {  		goto enoent;  	bits = -1; -	if (ch == '/' && isascii((unsigned char)(src[0])) && -	    isdigit((unsigned char)(src[0])) && dst > odst) { +	if (ch == '/' && isascii((unsigned char)src[0]) && +	    isdigit((unsigned char)src[0]) && dst > odst) {  		/* CIDR width specifier.  Nothing can follow it. */ -		ch = *src++;	/*%< Skip over the /. */ +		ch = (unsigned char)*src++;	/* Skip over the /. */  		bits = 0;  		do {  			n = strchr(digits, ch) - digits; @@ -132,8 +160,9 @@ inet_net_pton_ipv4(const char *src, u_char *dst, size_t size) {  			bits *= 10;  			bits += n;  			if (bits > 32) -				goto enoent; -		} while ((ch = *src++) != '\0' && isascii(ch) && isdigit(ch)); +				goto emsgsize; +		} while ((ch = (unsigned char)*src++) != '\0' && +			 isascii(ch) && isdigit(ch));  		if (ch != '\0')  			goto enoent;  	} @@ -147,29 +176,23 @@ inet_net_pton_ipv4(const char *src, u_char *dst, size_t size) {  		goto enoent;  	/* If no CIDR spec was given, infer width from net class. */  	if (bits == -1) { -		if (*odst >= 240)	/*%< Class E */ +		if (*odst >= 240)	/* Class E */  			bits = 32; -		else if (*odst >= 224)	/*%< Class D */ -			bits = 8; -		else if (*odst >= 192)	/*%< Class C */ +		else if (*odst >= 224)	/* Class D */ +			bits = 4; +		else if (*odst >= 192)	/* Class C */  			bits = 24; -		else if (*odst >= 128)	/*%< Class B */ +		else if (*odst >= 128)	/* Class B */  			bits = 16; -		else			/*%< Class A */ +		else			/* Class A */  			bits = 8;  		/* If imputed mask is narrower than specified octets, widen. */  		if (bits < ((dst - odst) * 8))  			bits = (dst - odst) * 8; -		/* -		 * If there are no additional bits specified for a class D -		 * address adjust bits to 4. -		 */ -		if (bits == 8 && *odst == 224) -			bits = 4;  	}  	/* Extend network to cover the actual mask. */  	while (bits > ((dst - odst) * 8)) { -		if (size-- <= 0U) +		if (size-- == 0)  			goto emsgsize;  		*dst++ = '\0';  	} @@ -184,222 +207,48 @@ inet_net_pton_ipv4(const char *src, u_char *dst, size_t size) {  	return (-1);  } -static int -getbits(const char *src, int *bitsp) { -	static const char digits[] = "0123456789"; -	int n; -	int val; -	char ch; - -	val = 0; -	n = 0; -	while ((ch = *src++) != '\0') { -		const char *pch; - -		pch = strchr(digits, ch); -		if (pch != NULL) { -			if (n++ != 0 && val == 0)	/*%< no leading zeros */ -				return (0); -			val *= 10; -			val += (pch - digits); -			if (val > 128)			/*%< range */ -				return (0); -			continue; -		} -		return (0); -	} -	if (n == 0) -		return (0); -	*bitsp = val; -	return (1); -}  static int -getv4(const char *src, u_char *dst, int *bitsp) { -	static const char digits[] = "0123456789"; -	u_char *odst = dst; -	int n; -	u_int val; -	char ch; - -	val = 0; -	n = 0; -	while ((ch = *src++) != '\0') { -		const char *pch; +inet_net_pton_ipv6(const char *src, u_char *dst, size_t size) +{ +	struct in6_addr	 in6; +	int		 ret; +	int		 bits; +	size_t		 bytes; +	char		 buf[INET6_ADDRSTRLEN + sizeof("/128")]; +	char		*sep; +	const char	*errstr; -		pch = strchr(digits, ch); -		if (pch != NULL) { -			if (n++ != 0 && val == 0)	/*%< no leading zeros */ -				return (0); -			val *= 10; -			val += (pch - digits); -			if (val > 255)			/*%< range */ -				return (0); -			continue; -		} -		if (ch == '.' || ch == '/') { -			if (dst - odst > 3)		/*%< too many octets? */ -				return (0); -			*dst++ = val; -			if (ch == '/') -				return (getbits(src, bitsp)); -			val = 0; -			n = 0; -			continue; -		} -		return (0); +	if (strlcpy(buf, src, sizeof buf) >= sizeof buf) { +		errno = EMSGSIZE; +		return (-1);  	} -	if (n == 0) -		return (0); -	if (dst - odst > 3)		/*%< too many octets? */ -		return (0); -	*dst++ = val; -	return (1); -} -static int -inet_net_pton_ipv6(const char *src, u_char *dst, size_t size) { -	static const char xdigits_l[] = "0123456789abcdef", -			  xdigits_u[] = "0123456789ABCDEF"; -	u_char tmp[NS_IN6ADDRSZ], *tp, *endp, *colonp; -	const char *xdigits, *curtok; -	int ch, saw_xdigit; -	u_int val; -	int digits; -	int bits; -	size_t bytes; -	int words; -	int ipv4; +	sep = strchr(buf, '/'); +	if (sep != NULL) +		*sep++ = '\0'; -	memset((tp = tmp), '\0', NS_IN6ADDRSZ); -	endp = tp + NS_IN6ADDRSZ; -	colonp = NULL; -	/* Leading :: requires some special handling. */ -	if (*src == ':') -		if (*++src != ':') -			goto enoent; -	curtok = src; -	saw_xdigit = 0; -	val = 0; -	digits = 0; -	bits = -1; -	ipv4 = 0; -	while ((ch = *src++) != '\0') { -		const char *pch; +	ret = inet_pton(AF_INET6, buf, &in6); +	if (ret != 1) +		return (-1); -		if ((pch = strchr((xdigits = xdigits_l), ch)) == NULL) -			pch = strchr((xdigits = xdigits_u), ch); -		if (pch != NULL) { -			val <<= 4; -			val |= (pch - xdigits); -			if (++digits > 4) -				goto enoent; -			saw_xdigit = 1; -			continue; -		} -		if (ch == ':') { -			curtok = src; -			if (!saw_xdigit) { -				if (colonp) -					goto enoent; -				colonp = tp; -				continue; -			} else if (*src == '\0') -				goto enoent; -			if (tp + NS_INT16SZ > endp) -				return (0); -			*tp++ = (u_char) (val >> 8) & 0xff; -			*tp++ = (u_char) val & 0xff; -			saw_xdigit = 0; -			digits = 0; -			val = 0; -			continue; -		} -		if (ch == '.' && ((tp + NS_INADDRSZ) <= endp) && -		     getv4(curtok, tp, &bits) > 0) { -			tp += NS_INADDRSZ; -			saw_xdigit = 0; -			ipv4 = 1; -			break;	/*%< '\\0' was seen by inet_pton4(). */ -		} -		if (ch == '/' && getbits(src, &bits) > 0) -			break; -		goto enoent; -	} -	if (saw_xdigit) { -		if (tp + NS_INT16SZ > endp) -			goto enoent; -		*tp++ = (u_char) (val >> 8) & 0xff; -		*tp++ = (u_char) val & 0xff; -	} -	if (bits == -1) +	if (sep == NULL)  		bits = 128; - -	words = (bits + 15) / 16; -	if (words < 2) -		words = 2; -	if (ipv4) -		words = 8; -	endp =  tmp + 2 * words; - -	if (colonp != NULL) { -		/* -		 * Since some memmove()'s erroneously fail to handle -		 * overlapping regions, we'll do the shift by hand. -		 */ -		const int n = tp - colonp; -		int i; - -		if (tp == endp) -			goto enoent; -		for (i = 1; i <= n; i++) { -			endp[- i] = colonp[n - i]; -			colonp[n - i] = 0; +	else { +		bits = strtonum(sep, 0, 128, &errstr); +		if (errstr) { +			errno = EINVAL; +			return (-1);  		} -		tp = endp;  	} -	if (tp != endp) -		goto enoent;  	bytes = (bits + 7) / 8; -	if (bytes > size) -		goto emsgsize; -	memcpy(dst, tmp, bytes); -	return (bits); - - enoent: -	errno = ENOENT; -	return (-1); - - emsgsize: -	errno = EMSGSIZE; -	return (-1); -} - -/*% - * int - * inet_net_pton(af, src, dst, size) - *	convert network number from presentation to network format. - *	accepts hex octets, hex strings, decimal octets, and /CIDR. - *	"size" is in bytes and describes "dst". - * return: - *	number of bits, either imputed classfully or specified with /CIDR, - *	or -1 if some failure occurred (check errno).  ENOENT means it was - *	not a valid network specification. - * author: - *	Paul Vixie (ISC), June 1996 - */ -int -inet_net_pton(int af, const char *src, void *dst, size_t size) { -	switch (af) { -	case AF_INET: -		return (inet_net_pton_ipv4(src, dst, size)); -	case AF_INET6: -		return (inet_net_pton_ipv6(src, dst, size)); -	default: -		errno = EAFNOSUPPORT; +	if (bytes > size) { +		errno = EMSGSIZE;  		return (-1);  	} +	memcpy(dst, &in6.s6_addr, bytes); +	return (bits);  }  /* @@ -408,5 +257,3 @@ inet_net_pton(int af, const char *src, void *dst, size_t size) {   */  #undef inet_net_pton  __weak_reference(__inet_net_pton, inet_net_pton); - -/*! \file */ diff --git a/lib/libc/tests/net/Makefile b/lib/libc/tests/net/Makefile index 24cff61e8d24..ec0668633508 100644 --- a/lib/libc/tests/net/Makefile +++ b/lib/libc/tests/net/Makefile @@ -4,8 +4,10 @@ ATF_TESTS_C+=	ether_test  ATF_TESTS_C+=	eui64_aton_test  ATF_TESTS_C+=	eui64_ntoa_test  ATF_TESTS_CXX+=	link_addr_test +ATF_TESTS_CXX+=	inet_net_test  CXXSTD.link_addr_test=	c++20 +CXXSTD.inet_net_test=	c++20  CFLAGS+=	-I${.CURDIR} diff --git a/lib/libc/tests/net/inet_net_test.cc b/lib/libc/tests/net/inet_net_test.cc new file mode 100644 index 000000000000..60b60b152eca --- /dev/null +++ b/lib/libc/tests/net/inet_net_test.cc @@ -0,0 +1,333 @@ +/* + * SPDX-License-Identifier: ISC + * + * Copyright (c) 2025 Lexi Winter <ivy@FreeBSD.org> + * + * Permission to use, copy, modify, and distribute this software for any + * purpose with or without fee is hereby granted, provided that the above + * copyright notice and this permission notice appear in all copies. + * + * THE SOFTWARE IS PROVIDED "AS IS" AND THE AUTHOR DISCLAIMS ALL WARRANTIES + * WITH REGARD TO THIS SOFTWARE INCLUDING ALL IMPLIED WARRANTIES OF + * MERCHANTABILITY AND FITNESS. IN NO EVENT SHALL THE AUTHOR BE LIABLE FOR + * ANY SPECIAL, DIRECT, INDIRECT, OR CONSEQUENTIAL DAMAGES OR ANY DAMAGES + * WHATSOEVER RESULTING FROM LOSS OF USE, DATA OR PROFITS, WHETHER IN AN + * ACTION OF CONTRACT, NEGLIGENCE OR OTHER TORTIOUS ACTION, ARISING OUT OF + * OR IN CONNECTION WITH THE USE OR PERFORMANCE OF THIS SOFTWARE. + */ + +/* + * Tests for inet_net_pton() and inet_net_ntop(). + */ + +#include <sys/types.h> +#include <sys/socket.h> + +#include <netinet/in.h> +#include <arpa/inet.h> + +#include <ranges> +#include <string> +#include <vector> + +#include <atf-c++.hpp> + +using namespace std::literals; + +/* + * inet_net_ntop() and inet_net_pton() for IPv4. + */ +ATF_TEST_CASE_WITHOUT_HEAD(inet_net_inet4) +ATF_TEST_CASE_BODY(inet_net_inet4) +{ +	/* +	 * Define a list of addresses we want to check.  Each address is passed +	 * to inet_net_pton() to convert it to an in_addr, then we convert the +	 * in_addr back to a string and compare it with the expected value.  We +	 * want to test over-long prefixes here (such as 10.0.0.1/8), so we also +	 * specify what the result is expected to be. +	 */ + +	struct test_addr { +		std::string input; +		int bits; +		std::string output; +	}; + +	auto test_addrs = std::vector<test_addr>{ +		// Simple prefixes that fall on octet boundaries. +		{ "10.0.0.0/8",		8,	"10/8" }, +		{ "10.1.0.0/16",	16,	"10.1/16" }, +		{ "10.1.2.0/24",	24,	"10.1.2/24" }, +		{ "10.1.2.3/32",	32,	"10.1.2.3/32" }, + +		// Simple prefixes with the short-form address. +		{ "10/8",		8,	"10/8" }, +		{ "10.1/16",		16,	"10.1/16" }, +		{ "10.1.2/24",		24,	"10.1.2/24" }, + +		// A prefix that doesn't fall on an octet boundary. +		{ "10.1.64/18",		18,	"10.1.64/18" }, + +		// An overlong prefix with bits that aren't part of the prefix. +		{ "10.0.0.1/8",		8,	"10/8" }, +	}; + +	for (auto const &addr: test_addrs) { +		/* +		 * Convert the input string to an in_addr + bits, and make +		 * sure the result produces the number of bits we expected. +		 */ + +		auto in = in_addr{}; +		auto bits = inet_net_pton(AF_INET, addr.input.c_str(), +		    &in, sizeof(in)); +		ATF_REQUIRE(bits != -1); +		ATF_REQUIRE_EQ(bits, addr.bits); + +		/* +		 * Convert the in_addr back to a string +		 */ + +		/* +		 * XXX: Should there be a constant for the size of the result +		 * buffer?  For now, use ADDRSTRLEN + 3 ("/32") + 1 (NUL). +		 * +		 * Fill the buffer with 'Z', so we can check the result was +		 * properly terminated. +		 */ +		auto strbuf = std::vector<char>(INET_ADDRSTRLEN + 3 + 1, 'Z'); +		auto ret = inet_net_ntop(AF_INET, &in, bits, +		    strbuf.data(), strbuf.size()); +		ATF_REQUIRE(ret != NULL); +		ATF_REQUIRE_EQ(ret, strbuf.data()); + +		/* Make sure the result was NUL-terminated and find the NUL */ +		ATF_REQUIRE(strbuf.size() >= 1); +		auto end = std::ranges::find(strbuf, '\0'); +		ATF_REQUIRE(end != strbuf.end()); + +		/* +		 * Check the result matches what we expect.  Use a temporary +		 * string here instead of std::ranges::equal because this +		 * means ATF can print the mismatch. +		 */ +		auto str = std::string(std::ranges::begin(strbuf), end); +		ATF_REQUIRE_EQ(str, addr.output); +	} +} + +/* + * inet_net_ntop() and inet_net_pton() for IPv6. + */ +ATF_TEST_CASE_WITHOUT_HEAD(inet_net_inet6) +ATF_TEST_CASE_BODY(inet_net_inet6) +{ +	/* +	 * Define a list of addresses we want to check.  Each address is +	 * passed to inet_net_pton() to convert it to an in6_addr, then we +	 * convert the in6_addr back to a string and compare it with the +	 * expected value.  We want to test over-long prefixes here (such +	 * as 2001:db8::1/32), so we also specify what the result is +	 * expected to be. +	 */ + +	struct test_addr { +		std::string input; +		int bits; +		std::string output; +	}; + +	auto test_addrs = std::vector<test_addr>{ +		// A prefix with a trailing :: +		{ "2001:db8::/32",	32,	"2001:db8::/32" }, + +		// A prefix with a leading ::.  Note that the output is +		// different from the input because inet_ntop() renders +		// this prefix with an IPv4 suffix for legacy reasons. +		{ "::ffff:0:0/96",	96,	"::ffff:0.0.0.0/96" }, + +		// The same prefix but with the IPv4 legacy form as input. +		{ "::ffff:0.0.0.0/96",	96,	"::ffff:0.0.0.0/96" }, + +		// A prefix with an infix ::. +		{ "2001:db8::1/128",	128,	"2001:db8::1/128" }, + +		// A prefix with bits set which are outside the prefix; +		// these should be silently ignored. +		{ "2001:db8:1:1:1:1:1:1/32", 32, "2001:db8::/32" }, + +		// As above but with infix ::. +		{ "2001:db8::1/32",	32,	"2001:db8::/32" }, + +		// A prefix with only ::, commonly used to represent the +		// entire address space. +		{ "::/0",		0,	"::/0" }, + +		// A single address with no ::. +		{ "2001:db8:1:1:1:1:1:1/128", 128, "2001:db8:1:1:1:1:1:1/128" }, + +		// A prefix with no ::. +		{ "2001:db8:1:1:0:0:0:0/64", 64, "2001:db8:1:1::/64" }, + +		// A prefix which isn't on a 16-bit boundary. +		{ "2001:db8:c000::/56",	56,	"2001:db8:c000::/56" }, + +		// A prefix which isn't on a nibble boundary. +		{ "2001:db8:c100::/57",	57,	"2001:db8:c100::/57" }, + +		// An address without a prefix length, which should be treated +		// as a /128. +		{ "2001:db8::",		128,	"2001:db8::/128" }, +		{ "2001:db8::1",	128,	"2001:db8::1/128" }, + +		// Test vectors provided in PR bin/289198. +		{ "fe80::1/64",		64,	"fe80::/64" }, +		{ "fe80::f000:74ff:fe54:bed2/64", +					64,	"fe80::/64" }, +		{ "ffff:ffff:ffff:ffff:ffff:ffff:ffff:ffff/64", +					64,	"ffff:ffff:ffff:ffff::/64" }, +		{ "ffff:ffff:ffff:ffff:ffff:ffff:ffff:ffff", 128, +		    "ffff:ffff:ffff:ffff:ffff:ffff:ffff:ffff/128" }, +	}; + +	for (auto const &addr: test_addrs) { +		/* +		 * Convert the input string to an in6_addr + bits, and make +		 * sure the result produces the number of bits we expected. +		 */ + +		auto in6 = in6_addr{}; +		errno = 0; +		auto bits = inet_net_pton(AF_INET6, addr.input.c_str(), +		    &in6, sizeof(in6)); +		ATF_REQUIRE(bits != -1); +		ATF_REQUIRE_EQ(bits, addr.bits); + +		/* +		 * Convert the in6_addr back to a string +		 */ + +		/* +		 * XXX: Should there be a constant for the size of the result +		 * buffer?  For now, use ADDRSTRLEN + 4 ("/128") + 1 (NUL). +		 * +		 * Fill the buffer with 'Z', so we can check the result was +		 * properly terminated. +		 */ +		auto strbuf = std::vector<char>(INET6_ADDRSTRLEN + 4 + 1, 'Z'); +		auto ret = inet_net_ntop(AF_INET6, &in6, bits, +		    strbuf.data(), strbuf.size()); +		ATF_REQUIRE(ret != NULL); +		ATF_REQUIRE_EQ(ret, strbuf.data()); + +		/* Make sure the result was NUL-terminated and find the NUL */ +		ATF_REQUIRE(strbuf.size() >= 1); +		auto end = std::ranges::find(strbuf, '\0'); +		ATF_REQUIRE(end != strbuf.end()); + +		/* +		 * Check the result matches what we expect.  Use a temporary +		 * string here instead of std::ranges::equal because this +		 * means ATF can print the mismatch. +		 */ +		auto str = std::string(std::ranges::begin(strbuf), end); +		ATF_REQUIRE_EQ(str, addr.output); +	} +} + +ATF_TEST_CASE_WITHOUT_HEAD(inet_net_pton_invalid) +ATF_TEST_CASE_BODY(inet_net_pton_invalid) +{ +	auto ret = int{}; +	auto addr4 = in_addr{}; +	auto str4 = "10.0.0.0"s; +	auto addr6 = in6_addr{}; +	auto str6 = "2001:db8::"s; + +	/* Passing an address which is too short should be an error */ +	ret = inet_net_pton(AF_INET6, str6.c_str(), &addr6, sizeof(addr6) - 1); +	ATF_REQUIRE_EQ(ret, -1); + +	ret = inet_net_pton(AF_INET, str4.c_str(), &addr4, sizeof(addr4) - 1); +	ATF_REQUIRE_EQ(ret, -1); + +	/* Test some generally invalid addresses. */ +	auto invalid4 = std::vector<std::string>{ +		// Prefix length too big +		"10.0.0.0/33", +		// Prefix length is negative +		"10.0.0.0/-1", +		// Prefix length is not a number +		"10.0.0.0/foo", +		// Input is not a network prefix +		"this is not an IP address", +	}; + +	for (auto const &addr: invalid4) { +		auto ret = inet_net_pton(AF_INET, addr.c_str(), &addr4, +		    sizeof(addr4)); +		ATF_REQUIRE_EQ(ret, -1); +	} + +	auto invalid6 = std::vector<std::string>{ +		// Prefix length too big +		"2001:db8::/129", +		// Prefix length is negative +		"2001:db8::/-1", +		// Prefix length is not a number +		"2001:db8::/foo", +		// Input is not a network prefix +		"this is not an IP address", +	}; + +	for (auto const &addr: invalid6) { +		auto ret = inet_net_pton(AF_INET6, addr.c_str(), &addr6, +		    sizeof(addr6)); +		ATF_REQUIRE_EQ(ret, -1); +	} +} + +ATF_TEST_CASE_WITHOUT_HEAD(inet_net_ntop_invalid) +ATF_TEST_CASE_BODY(inet_net_ntop_invalid) +{ +	auto addr4 = in_addr{}; +	auto addr6 = in6_addr{}; +	auto strbuf = std::vector<char>(INET6_ADDRSTRLEN + 4 + 1); + +	/* +	 * Passing a buffer which is too small should not overrun the buffer. +	 * Test this by initialising the buffer to 'Z', and only providing +	 * part of it to the function. +	 */ + +	std::ranges::fill(strbuf, 'Z'); +	auto ret = inet_net_ntop(AF_INET6, &addr6, 128, strbuf.data(), 1); +	ATF_REQUIRE_EQ(ret, nullptr); +	ATF_REQUIRE_EQ(strbuf[1], 'Z'); + +	std::ranges::fill(strbuf, 'Z'); +	ret = inet_net_ntop(AF_INET, &addr4, 32, strbuf.data(), 1); +	ATF_REQUIRE_EQ(ret, nullptr); +	ATF_REQUIRE_EQ(strbuf[1], 'Z'); + +	/* Check that invalid prefix lengths return an error */ + +	ret = inet_net_ntop(AF_INET6, &addr6, 129, strbuf.data(), strbuf.size()); +	ATF_REQUIRE_EQ(ret, nullptr); +	ret = inet_net_ntop(AF_INET6, &addr6, -1, strbuf.data(), strbuf.size()); +	ATF_REQUIRE_EQ(ret, nullptr); + +	ret = inet_net_ntop(AF_INET, &addr4, 33, strbuf.data(), strbuf.size()); +	ATF_REQUIRE_EQ(ret, nullptr); +	ret = inet_net_ntop(AF_INET, &addr4, -1, strbuf.data(), strbuf.size()); +	ATF_REQUIRE_EQ(ret, nullptr); +} + +ATF_INIT_TEST_CASES(tcs) +{ +	ATF_ADD_TEST_CASE(tcs, inet_net_inet4); +	ATF_ADD_TEST_CASE(tcs, inet_net_inet6); +	ATF_ADD_TEST_CASE(tcs, inet_net_pton_invalid); +	ATF_ADD_TEST_CASE(tcs, inet_net_ntop_invalid); +} diff --git a/lib/libmd/Makefile b/lib/libmd/Makefile index 547a134fc440..c4ab767c8b2f 100644 --- a/lib/libmd/Makefile +++ b/lib/libmd/Makefile @@ -108,7 +108,7 @@ CFLAGS+= -DWEAK_REFS  CFLAGS.skein_block.c+= -DSKEIN_LOOP=995  .PATH: ${.CURDIR}/${MACHINE_ARCH} ${SRCTOP}/sys/crypto/sha2  .PATH: ${SRCTOP}/sys/crypto/skein ${SRCTOP}/sys/crypto/skein/${MACHINE_ARCH} -.PATH: ${SRCTOP}/sys/kern +.PATH: ${SRCTOP}/sys/crypto  USE_ASM_SOURCES?=1  .if defined(BOOTSTRAPPING) || ${MK_MACHDEP_OPTIMIZATIONS} == no @@ -117,6 +117,13 @@ USE_ASM_SOURCES:=0  .endif  .if ${USE_ASM_SOURCES} != 0 +.if exists(${MACHINE_ARCH}/md5block.S) +SRCS+=	md5block.S +CFLAGS+= -DMD5_ASM +.if exists(${MACHINE_ARCH}/md5dispatch.c) +SRCS+=  md5dispatch.c +.endif +.endif  .if exists(${MACHINE_ARCH}/sha1block.S)  SRCS+=	sha1block.S  CFLAGS+= -DSHA1_ASM diff --git a/lib/libmd/aarch64/md5block.S b/lib/libmd/aarch64/md5block.S new file mode 100644 index 000000000000..b928c8dd795a --- /dev/null +++ b/lib/libmd/aarch64/md5block.S @@ -0,0 +1,206 @@ +/*- + * Copyright (c) 2024 Robert Clausecker <fuz@FreeBSD.org> + * + * SPDX-License-Identifier: BSD-2-Clause + */ + +#include <sys/elf_common.h> +#include <machine/asm.h> + +# optimal instruction sequence for k = \key + \m +.macro	addkm	key, m +.if 0x100000000 - \key > 0x00ffffff +	movz	k, #\key & 0xffff +	movk	k, #\key >> 16, lsl #16 +	add	k, k, \m +.elseif 0x100000000 - \key > 0x0000ffff +	sub	k, \m, #(0x100000000 - \key) & 0xfff000 +	sub	k, k, #(0x100000000 - \key) & 0xfff +.else +	movz	k, #0x100000000 - \key +	sub	k, \m, k +.endif +.endm + +.macro	round	a, b, c, d, f, key, m, s +	\f	f, \b, \c, \d +	addkm	\key, \m		// k[i] + m[g] +	add	\a, \a, k		// k[i] + m[g] + a +	add	\a, \a, f		// k[i] + m[g] + a + f +	ror	\a, \a, #32-\s +	add	\a, \a, \b +.endm + +	/* f = b ? c : d */ +.macro	f0	f, b, c, d +	eor	\f, \c, \d +	and	\f, \f, \b +	eor	\f, \f, \d +.endm + +	/* +	 * special cased round 1 function +	 * f1 = d ? b : c = (d & b) + (~d & c) +	 */ +.macro	round1	a, b, c, d, key, m, s +	bic	tmp, \c, \d		// ~d & c +	addkm	\key, \m		// k[i] + m[g] +	add	\a, \a, k		// k[i] + m[g] + a +	and	f, \b, \d		// d & b +	add	\a, \a, tmp		// k[i] + m[g] + a + (~d & c) +	add	\a, \a, f		// k[i] + m[g] + a + (~d & c) + (d & b) +	ror	\a, \a, #32-\s +	add	\a, \a, \b +.endm + +	/* f = b ^ c ^ d */ +.macro	f2	f, b, c, d +	eor	\f, \c, \d +	eor	\f, \f, \b +.endm + +	/* f = c ^ (b | ~d) */ +.macro	f3	f, b, c, d +	orn	\f, \b, \d +	eor	\f, \f, \c +.endm + +	/* do 4 rounds */ +.macro	rounds	f, m0, m1, m2, m3, s0, s1, s2, s3, k0, k1, k2, k3 +	round	a, b, c, d, \f, \k0, \m0, \s0 +	round	d, a, b, c, \f, \k1, \m1, \s1 +	round	c, d, a, b, \f, \k2, \m2, \s2 +	round	b, c, d, a, \f, \k3, \m3, \s3 +.endm + +	/* do 4 rounds with f0, f1, f2, f3 */ +.macro	rounds0	m0, m1, m2, m3, k0, k1, k2, k3 +	rounds	f0, \m0, \m1, \m2, \m3, 7, 12, 17, 22, \k0, \k1, \k2, \k3 +.endm + +.macro	rounds1	m0, m1, m2, m3, k0, k1, k2, k3 +	round1	a, b, c, d, \k0, \m0,  5 +	round1	d, a, b, c, \k1, \m1,  9 +	round1	c, d, a, b, \k2, \m2, 14 +	round1	b, c, d, a, \k3, \m3, 20 +.endm + +.macro	rounds2	m0, m1, m2, m3, k0, k1, k2, k3 +	rounds	f2, \m0, \m1, \m2, \m3, 4, 11, 16, 23, \k0, \k1, \k2, \k3 +.endm + +.macro	rounds3	m0, m1, m2, m3, k0, k1, k2, k3 +	rounds	f3, \m0, \m1, \m2, \m3, 6, 10, 15, 21, \k0, \k1, \k2, \k3 +.endm + +	/* md5block(MD5_CTX, buf, len) */ +ENTRY(_libmd_md5block) +ctx	.req	x0 +buf	.req	x1 +len	.req	x2 +end	.req	x2			// aliases len +a	.req	w3 +b	.req	w4 +c	.req	w5 +d	.req	w6 +f	.req	w7 +tmp	.req	w8 +k	.req	w9 +m0	.req	w10 +m1	.req	w11 +m2	.req	w12 +m3	.req	w13 +m4	.req	w14 +m5	.req	w15 +m6	.req	w16 +m7	.req	w17 +					// x18 is the platform register +m8	.req	w19 +m9	.req	w20 +m10	.req	w21 +m11	.req	w22 +m12	.req	w23 +m13	.req	w24 +m14	.req	w25 +m15	.req	w26 + +a_	.req	m0 +b_	.req	m7 +c_	.req	m14 +d_	.req	m5 + +	stp	x19, x20, [sp, #-0x40]! +	stp	x21, x22, [sp, #0x10] +	stp	x23, x24, [sp, #0x20] +	stp	x25, x26, [sp, #0x30] + +	bics	len, len, #63		// length in blocks +	add	end, buf, len		// end pointer + +	beq	.Lend			// was len == 0 after BICS? + +	ldp	a, b, [ctx, #0] +	ldp	c, d, [ctx, #8] + +	/* first eight rounds interleaved with data loads */ +.Lloop:	ldp	m0, m1, [buf, #0] +	round	a, b, c, d, f0, 0xd76aa478, m0,  7 +	ldp	m2, m3, [buf, #8] +	round	d, a, b, c, f0, 0xe8c7b756, m1, 12 +	ldp	m4, m5, [buf, #16] +	round	c, d, a, b, f0, 0x242070db, m2, 17 +	ldp	m6, m7, [buf, #24] +	round	b, c, d, a, f0, 0xc1bdceee, m3, 22 + +	ldp	m8, m9, [buf, #32] +	round	a, b, c, d, f0, 0xf57c0faf, m4,  7 +	ldp	m10, m11, [buf, #40] +	round	d, a, b, c, f0, 0x4787c62a, m5, 12 +	ldp	m12, m13, [buf, #48] +	round	c, d, a, b, f0, 0xa8304613, m6, 17 +	ldp	m14, m15, [buf, #56] +	round	b, c, d, a, f0, 0xfd469501, m7, 22 + +	/* remaining rounds use the roundsX macros */ +	rounds0	 m8,  m9, m10, m11, 0x698098d8, 0x8b44f7af, 0xffff5bb1, 0x895cd7be +	rounds0	m12, m13, m14, m15, 0x6b901122, 0xfd987193, 0xa679438e, 0x49b40821 + +	rounds1	 m1,  m6, m11,  m0, 0xf61e2562, 0xc040b340, 0x265e5a51, 0xe9b6c7aa +	rounds1	 m5, m10, m15,  m4, 0xd62f105d, 0x02441453, 0xd8a1e681, 0xe7d3fbc8 +	rounds1	 m9, m14,  m3,  m8, 0x21e1cde6, 0xc33707d6, 0xf4d50d87, 0x455a14ed +	rounds1	m13,  m2,  m7, m12, 0xa9e3e905, 0xfcefa3f8, 0x676f02d9, 0x8d2a4c8a + +	rounds2	 m5,  m8, m11, m14, 0xfffa3942, 0x8771f681, 0x6d9d6122, 0xfde5380c +	rounds2	 m1,  m4,  m7, m10, 0xa4beea44, 0x4bdecfa9, 0xf6bb4b60, 0xbebfbc70 +	rounds2	m13,  m0,  m3,  m6, 0x289b7ec6, 0xeaa127fa, 0xd4ef3085, 0x04881d05 +	rounds2	 m9, m12, m15,  m2, 0xd9d4d039, 0xe6db99e5, 0x1fa27cf8, 0xc4ac5665 + +	rounds3	 m0,  m7, m14,  m5, 0xf4292244, 0x432aff97, 0xab9423a7, 0xfc93a039 +	rounds3	m12,  m3, m10,  m1, 0x655b59c3, 0x8f0ccc92, 0xffeff47d, 0x85845dd1 +	rounds3	 m8, m15,  m6, m13, 0x6fa87e4f, 0xfe2ce6e0, 0xa3014314, 0x4e0811a1 +	rounds3	 m4, m11,  m2,  m9, 0xf7537e82, 0xbd3af235, 0x2ad7d2bb, 0xeb86d391 + +	ldp	a_, b_, [ctx, #0] +	ldp	c_, d_, [ctx, #8] +	add	a, a, a_ +	add	b, b, b_ +	add	c, c, c_ +	add	d, d, d_ +	stp	a, b, [ctx, #0] +	stp	c, d, [ctx, #8] + +	add	buf, buf, #64 +	cmp	buf, end +	bne	.Lloop + +.Lend:	ldp	x25, x26, [sp, #0x30] +	ldp	x23, x24, [sp, #0x20] +	ldp	x21, x22, [sp, #0x10] +	ldp	x19, x20, [sp], #0x40 + +	ret +END(_libmd_md5block) + +GNU_PROPERTY_AARCH64_FEATURE_1_NOTE(GNU_PROPERTY_AARCH64_FEATURE_1_VAL) + +	.section .note.GNU-stack,"",%progbits diff --git a/lib/libmd/aarch64/sha1block.S b/lib/libmd/aarch64/sha1block.S index 56a0297efadd..e16fb36342fd 100644 --- a/lib/libmd/aarch64/sha1block.S +++ b/lib/libmd/aarch64/sha1block.S @@ -1,5 +1,5 @@  /*- - * Copyright (c) 2024 Robert Clausecker <fuz@freebsd.org> + * Copyright (c) 2024 Robert Clausecker <fuz@FreeBSD.org>   *   * SPDX-License-Identifier: BSD-2-Clause   * diff --git a/lib/libmd/aarch64/sha1dispatch.c b/lib/libmd/aarch64/sha1dispatch.c index e34bf0a1a344..045527044320 100644 --- a/lib/libmd/aarch64/sha1dispatch.c +++ b/lib/libmd/aarch64/sha1dispatch.c @@ -1,5 +1,5 @@  /*- - * Copyright (c) 2024 Robert Clausecker <fuz@freebsd.org> + * Copyright (c) 2024 Robert Clausecker <fuz@FreeBSD.org>   *   * SPDX-License-Identifier: BSD-2-Clause   */ diff --git a/lib/libmd/amd64/md5block.S b/lib/libmd/amd64/md5block.S new file mode 100644 index 000000000000..0dd594dd5dc2 --- /dev/null +++ b/lib/libmd/amd64/md5block.S @@ -0,0 +1,363 @@ +/*- + * Copyright (c) 2024, 2025 Robert Clausecker <fuz@FreeBSD.org> + * + * SPDX-License-Identifier: BSD-2-Clause + */ + +#include <machine/asm.h> + +/* apply the round keys to the four round functions */ +.macro	allrounds	rfn0, rfn1, rfn2, rfn3 +	\rfn0	 0, 0xd76aa478, 0xe8c7b756, 0x242070db, 0xc1bdceee +	\rfn0	 4, 0xf57c0faf, 0x4787c62a, 0xa8304613, 0xfd469501 +	\rfn0	 8, 0x698098d8, 0x8b44f7af, 0xffff5bb1, 0x895cd7be +	\rfn0	12, 0x6b901122, 0xfd987193, 0xa679438e, 0x49b40821 + +	\rfn1	16, 0xf61e2562, 0xc040b340, 0x265e5a51, 0xe9b6c7aa +	\rfn1	20, 0xd62f105d, 0x02441453, 0xd8a1e681, 0xe7d3fbc8 +	\rfn1	24, 0x21e1cde6, 0xc33707d6, 0xf4d50d87, 0x455a14ed +	\rfn1	28, 0xa9e3e905, 0xfcefa3f8, 0x676f02d9, 0x8d2a4c8a + +	\rfn2	32, 0xfffa3942, 0x8771f681, 0x6d9d6122, 0xfde5380c +	\rfn2	36, 0xa4beea44, 0x4bdecfa9, 0xf6bb4b60, 0xbebfbc70 +	\rfn2	40, 0x289b7ec6, 0xeaa127fa, 0xd4ef3085, 0x04881d05 +	\rfn2	44, 0xd9d4d039, 0xe6db99e5, 0x1fa27cf8, 0xc4ac5665 + +	\rfn3	48, 0xf4292244, 0x432aff97, 0xab9423a7, 0xfc93a039 +	\rfn3	52, 0x655b59c3, 0x8f0ccc92, 0xffeff47d, 0x85845dd1 +	\rfn3	56, 0x6fa87e4f, 0xfe2ce6e0, 0xa3014314, 0x4e0811a1 +	\rfn3	60, 0xf7537e82, 0xbd3af235, 0x2ad7d2bb, 0xeb86d391 +.endm + +	// md5block(MD5_CTX, buf, len) +ENTRY(_libmd_md5block_baseline) +.macro	round	a, b, c, d, f, k, m, s +	\f	%ebp, \b, \c, \d +	add	$\k, \a			// a + k[i] +	add	((\m)%16*4)(%rsi), \a	// a + k[i] + m[g] +	add	%ebp, \a		// a + k[i] + m[g] + f +	rol	$\s, \a +	add	\b, \a +.endm + +	// f = b ? c : d +.macro	f0	f, b, c, d +	mov	\c, \f +	xor	\d, \f +	and	\b, \f +	xor	\d, \f +.endm + +	// f = d ? b : c +.macro	f1	f, b, c, d +	mov	\c, \f +	xor	\b, \f +	and	\d, \f +	xor	\c, \f +.endm + +	// f = b ^ c ^ d +.macro	f2	f, b, c, d +	mov	\c, \f +	xor	\d, \f +	xor	\b, \f +.endm + +	// f = c ^ (b | ~d) +.macro	f3	f, b, c, d +	mov	$-1, \f +	xor	\d, \f +	or	\b, \f +	xor	\c, \f +.endm + +	// do 4 rounds +.macro	rounds	f, p, q, s0, s1, s2, s3, k0, k1, k2, k3 +	round	%eax, %ebx, %ecx, %edx, \f, \k0, \p*0+\q, \s0 +	round	%edx, %eax, %ebx, %ecx, \f, \k1, \p*1+\q, \s1 +	round	%ecx, %edx, %eax, %ebx, \f, \k2, \p*2+\q, \s2 +	round	%ebx, %ecx, %edx, %eax, \f, \k3, \p*3+\q, \s3 +.endm + +	// do 4 rounds with f0, f1, f2, f3 +.macro	rounds0	i, k0, k1, k2, k3 +	rounds	f0, 1, \i, 7, 12, 17, 22, \k0, \k1, \k2, \k3 +.endm + +.macro	rounds1	i, k0, k1, k2, k3 +	rounds	f1, 5, 5*\i+1, 5, 9, 14, 20, \k0, \k1, \k2, \k3 +.endm + +.macro	rounds2	i, k0, k1, k2, k3 +	rounds	f2, 3, 3*\i+5, 4, 11, 16, 23, \k0, \k1, \k2, \k3 +.endm + +.macro	rounds3	i, k0, k1, k2, k3 +	rounds	f3, 7, 7*\i, 6, 10, 15, 21, \k0, \k1, \k2, \k3 +.endm + +	push	%rbx +	push	%rbp +	push	%r12 + +	and	$~63, %rdx		// length in blocks +	lea	(%rsi, %rdx, 1), %r12	// end pointer + +	mov	(%rdi), %eax		// a +	mov	4(%rdi), %ebx		// b +	mov	8(%rdi), %ecx		// c +	mov	12(%rdi), %edx		// d + +	cmp	%rsi, %r12		// any data to process? +	je	.Lend + +	.balign	16 +.Lloop:	mov	%eax, %r8d +	mov	%ebx, %r9d +	mov	%ecx, %r10d +	mov	%edx, %r11d + +	allrounds	rounds0, rounds1, rounds2, rounds3 + +	add	%r8d, %eax +	add	%r9d, %ebx +	add	%r10d, %ecx +	add	%r11d, %edx + +	add	$64, %rsi +	cmp	%rsi, %r12 +	jne	.Lloop + +	mov	%eax, (%rdi) +	mov	%ebx, 4(%rdi) +	mov	%ecx, 8(%rdi) +	mov	%edx, 12(%rdi) + +.Lend:	pop	%r12 +	pop	%rbp +	pop	%rbx +	ret +END(_libmd_md5block_baseline) + +	/* +	 * An implementation leveraging the ANDN instruction +	 * from BMI1 to shorten some dependency chains. +	 */ +ENTRY(_libmd_md5block_bmi1) +	// special-cased round 1 +	// f1 = d ? b : c = (d & b) + (~d & c) +.macro	round1	a, b, c, d, k, m, s +	andn	\c, \d, %edi		// ~d & c +	add	$\k, \a			// a + k[i] +	mov	\d, %ebp +	add	((\m)%16*4)(%rsi), \a	// a + k[i] + m[g] +	and	\b, %ebp		// d & b +	add	%edi, \a		// a + k[i] + m[g] + (~d & c) +	add	%ebp, \a		// a + k[i] + m[g] + (~d & c) + (d & b) +	rol	$\s, \a +	add	\b, \a +.endm + +	// special-cased round 3 +	// f3 = c ^ (b | ~d) = ~(c ^ ~b & d) = -1 - (c ^ ~b & d) +.macro	round3	a, b, c, d, k, m, s +	andn	\d, \b, %ebp +	add	$\k - 1, \a		// a + k[i] - 1 +	add	((\m)%16*4)(%rsi), \a	// a + k[i] + m[g] +	xor	\c, %ebp +	sub	%ebp, \a		// a + k[i] + m[g] + f +	rol	$\s, \a +	add	\b, \a +.endm + +	.purgem	rounds1 +.macro	rounds1	i, k0, k1, k2, k3 +	round1	%eax, %ebx, %ecx, %edx, \k0, 5*\i+ 1,  5 +	round1	%edx, %eax, %ebx, %ecx, \k1, 5*\i+ 6,  9 +	round1	%ecx, %edx, %eax, %ebx, \k2, 5*\i+11, 14 +	round1	%ebx, %ecx, %edx, %eax, \k3, 5*\i+16, 20 +.endm + +	.purgem	rounds3 +.macro	rounds3	i, k0, k1, k2, k3 +	round3	%eax, %ebx, %ecx, %edx, \k0, 7*\i+ 0,  6 +	round3	%edx, %eax, %ebx, %ecx, \k1, 7*\i+ 7, 10 +	round3	%ecx, %edx, %eax, %ebx, \k2, 7*\i+14, 15 +	round3	%ebx, %ecx, %edx, %eax, \k3, 7*\i+21, 21 +.endm + +	push	%rbx +	push	%rbp +	push	%r12 + +	and	$~63, %rdx		// length in blocks +	lea	(%rsi, %rdx, 1), %r12	// end pointer + +	mov	(%rdi), %eax		// a +	mov	4(%rdi), %ebx		// b +	mov	8(%rdi), %ecx		// c +	mov	12(%rdi), %edx		// d + +	cmp	%rsi, %r12		// any data to process? +	je	0f + +	push	%rdi + +	.balign	16 +1:	mov	%eax, %r8d +	mov	%ebx, %r9d +	mov	%ecx, %r10d +	mov	%edx, %r11d + +	allrounds	rounds0, rounds1, rounds2, rounds3 + +	add	%r8d, %eax +	add	%r9d, %ebx +	add	%r10d, %ecx +	add	%r11d, %edx + +	add	$64, %rsi +	cmp	%rsi, %r12 +	jne	1b + +	pop	%rdi +	mov	%eax, (%rdi) +	mov	%ebx, 4(%rdi) +	mov	%ecx, 8(%rdi) +	mov	%edx, 12(%rdi) + +0:	pop	%r12 +	pop	%rbp +	pop	%rbx +	ret +END(_libmd_md5block_bmi1) + +#ifndef _KERNEL +	/* +	 * An implementation leveraging AVX-512 for its VPTERNLOGD +	 * instruction.  We're using only XMM registers here, +	 * avoiding costly thermal licensing. +	 */ +ENTRY(_libmd_md5block_avx512) +.macro	vround		a, b, c, d, f, i, m, mi, s +	vmovdqa		\b, %xmm4 +	vpternlogd	$\f, \d, \c, %xmm4 +	vpaddd		4*(\i)(%rax){1to4}, \m, %xmm5 // m[g] + k[i] +.if	\mi != 0 +	vpshufd		$0x55 * \mi, %xmm5, %xmm5	// broadcast to each dword +.endif +	vpaddd		%xmm5, \a, \a		// a + k[i] + m[g] +	vpaddd		%xmm4, \a, \a		// a + k[i] + m[g] + f +	vprold		$\s, \a, \a +	vpaddd		\b, \a, \a +.endm + +.macro	vrounds		f, i, m0, i0, m1, i1, m2, i2, m3, i3, s0, s1, s2, s3 +	vround		%xmm0, %xmm1, %xmm2, %xmm3, \f, \i+0, \m0, \i0, \s0 +	vround		%xmm3, %xmm0, %xmm1, %xmm2, \f, \i+1, \m1, \i1, \s1 +	vround		%xmm2, %xmm3, %xmm0, %xmm1, \f, \i+2, \m2, \i2, \s2 +	vround		%xmm1, %xmm2, %xmm3, %xmm0, \f, \i+3, \m3, \i3, \s3 +.endm + +/* + * d c b f0 f1 f2 f3 + * 0 0 0  0  0  0  1 + * 1 0 0  1  0  1  0 + * 0 1 0  0  1  1  0 + * 1 1 0  1  0  0  1 + * 0 0 1  0  0  1  1 + * 1 0 1  0  1  0  1 + * 0 1 1  1  1  0  0 + * 1 1 1  1  1  1  0 + */ + +.macro	vrounds0	i, m +	vrounds		0xca, \i, \m, 0, \m, 1, \m, 2, \m, 3, 7, 12, 17, 22 +.endm + +.macro	vrounds1	i, m0, i0, m1, i1, m2, i2, m3, i3 +	vrounds		0xe4, \i, \m0, \i0, \m1, \i1, \m2, \i2, \m3, \i3, 5, 9, 14, 20 +.endm + +.macro	vrounds2	i, m0, i0, m1, i1, m2, i2, m3, i3 +	vrounds		0x96, \i, \m0, \i0, \m1, \i1, \m2, \i2, \m3, \i3, 4, 11, 16, 23 +.endm + +.macro	vrounds3	i, m0, i0, m1, i1, m2, i2, m3, i3 +	vrounds		0x39, \i, \m0, \i0, \m1, \i1, \m2, \i2, \m3, \i3, 6, 10, 15, 21 +.endm + +	and		$~63, %rdx		// length in blocks +	add		%rsi, %rdx		// end pointer + +	vmovd		(%rdi), %xmm0		// a +	vmovd		4(%rdi), %xmm1		// b +	vmovd		8(%rdi), %xmm2		// c +	vmovd		12(%rdi), %xmm3		// d + +	lea		keys(%rip), %rax + +	cmp		%rsi, %rdx		// any data to process? +	je		0f + +	.balign		16 +1:	vmovdqu		0*4(%rsi), %xmm8	// message words +	vmovdqu		4*4(%rsi), %xmm9 +	vmovdqu		8*4(%rsi), %xmm10 +	vmovdqu		12*4(%rsi), %xmm11 + +	vmovdqa		%xmm0, %xmm12		// stash old state variables +	vmovdqa		%xmm1, %xmm13 +	vmovdqa		%xmm2, %xmm14 +	vmovdqa		%xmm3, %xmm15 + +	vrounds0	 0, %xmm8 +	vrounds0	 4, %xmm9 +	vrounds0	 8, %xmm10 +	vrounds0	12, %xmm11 + +	vrounds1	16,  %xmm8, 1,  %xmm9, 2, %xmm10, 3,  %xmm8, 0 +	vrounds1	20,  %xmm9, 1, %xmm10, 2, %xmm11, 3,  %xmm9, 0 +	vrounds1	24, %xmm10, 1, %xmm11, 2,  %xmm8, 3, %xmm10, 0 +	vrounds1	28, %xmm11, 1,  %xmm8, 2,  %xmm9, 3, %xmm11, 0 + +	vrounds2	32,  %xmm9, 1, %xmm10, 0, %xmm10, 3, %xmm11, 2 +	vrounds2	36,  %xmm8, 1,  %xmm9, 0,  %xmm9, 3, %xmm10, 2 +	vrounds2	40, %xmm11, 1,  %xmm8, 0,  %xmm8, 3,  %xmm9, 2 +	vrounds2	44  %xmm10, 1, %xmm11, 0, %xmm11, 3,  %xmm8, 2 + +	vrounds3	48,  %xmm8, 0,  %xmm9, 3, %xmm11, 2,  %xmm9, 1 +	vrounds3	52, %xmm11, 0,  %xmm8, 3, %xmm10, 2,  %xmm8, 1 +	vrounds3	56, %xmm10, 0, %xmm11, 3,  %xmm9, 2, %xmm11, 1 +	vrounds3	60,  %xmm9, 0, %xmm10, 3,  %xmm8, 2, %xmm10, 1 + +	vpaddd		%xmm12, %xmm0, %xmm0 +	vpaddd		%xmm13, %xmm1, %xmm1 +	vpaddd		%xmm14, %xmm2, %xmm2 +	vpaddd		%xmm15, %xmm3, %xmm3 + +	add		$64, %rsi +	cmp		%rsi, %rdx +	jne		1b + +	vmovd		%xmm0, (%rdi) +	vmovd		%xmm1, 4(%rdi) +	vmovd		%xmm2, 8(%rdi) +	vmovd		%xmm3, 12(%rdi) + +0:	ret +END(_libmd_md5block_avx512) + +	// round keys, for use in md5block_avx512 +	.section	.rodata +	.balign		16 + +.macro	putkeys		i, a, b, c, d +	.4byte		\a, \b, \c, \d +.endm + +keys:	allrounds	putkeys, putkeys, putkeys, putkeys +	.size		keys, .-keys +#endif /* !defined(_KERNEL) */ + +	.section .note.GNU-stack,"",%progbits diff --git a/lib/libmd/amd64/md5dispatch.c b/lib/libmd/amd64/md5dispatch.c new file mode 100644 index 000000000000..dd2131c5a57c --- /dev/null +++ b/lib/libmd/amd64/md5dispatch.c @@ -0,0 +1,41 @@ +/*- + * Copyright (c) 2024 Robert Clausecker <fuz@FreeBSD.org> + * + * SPDX-License-Identifier: BSD-2-Clause + */ + +#include <sys/types.h> +#include <sys/md5.h> + +#include <machine/cpufunc.h> +#include <machine/specialreg.h> +#include <stdint.h> +#include <string.h> +#include <x86/ifunc.h> + +extern void _libmd_md5block_baseline(MD5_CTX *, const void *, size_t); +extern void _libmd_md5block_bmi1(MD5_CTX *, const void *, size_t); +extern void _libmd_md5block_avx512(MD5_CTX *, const void *, size_t); + +DEFINE_UIFUNC(, void, _libmd_md5block, (MD5_CTX *, const void *, size_t)) +{ +	if ((cpu_stdext_feature & (CPUID_STDEXT_AVX512F | CPUID_STDEXT_AVX512VL)) +	    == (CPUID_STDEXT_AVX512F | CPUID_STDEXT_AVX512VL)) { +		u_int regs[4]; +		char cpu_vendor[12]; + +		do_cpuid(0, regs); +		((u_int *)&cpu_vendor)[0] = regs[1]; +		((u_int *)&cpu_vendor)[1] = regs[3]; +		((u_int *)&cpu_vendor)[2] = regs[2]; + +		/* the AVX-512 kernel performs poorly on AMD */ +		if (memcmp(cpu_vendor, AMD_VENDOR_ID, sizeof(cpu_vendor)) != 0) +			return (_libmd_md5block_avx512); +	} + +	if (cpu_stdext_feature & CPUID_STDEXT_BMI1) +		return (_libmd_md5block_bmi1); +	else +		return (_libmd_md5block_baseline); +} diff --git a/lib/libmd/amd64/sha1block.S b/lib/libmd/amd64/sha1block.S index f1291ef2647a..6ef083178abc 100644 --- a/lib/libmd/amd64/sha1block.S +++ b/lib/libmd/amd64/sha1block.S @@ -1,6 +1,6 @@  /*-   * Copyright (c) 2013 The Go Authors. All rights reserved. - * Copyright (c) 2024 Robert Clausecker <fuz@freebsd.org> + * Copyright (c) 2024 Robert Clausecker <fuz@FreeBSD.org>   *   * Adapted from Go's crypto/sha1/sha1block_amd64.s.   * diff --git a/lib/libmd/amd64/sha1dispatch.c b/lib/libmd/amd64/sha1dispatch.c index 86509195d56e..c82a60334739 100644 --- a/lib/libmd/amd64/sha1dispatch.c +++ b/lib/libmd/amd64/sha1dispatch.c @@ -1,6 +1,6 @@  /*-   * Copyright (c) 2016 The Go Authors. All rights reserved. - * Copyright (c) 2024 Robert Clausecker <fuz@freebsd.org> + * Copyright (c) 2024 Robert Clausecker <fuz@FreeBSD.org>   *   * Adapted from Go's crypto/sha1/sha1block_amd64.go.   * diff --git a/lib/libmd/sha1c.c b/lib/libmd/sha1c.c index 128e0b991742..02132d720dac 100644 --- a/lib/libmd/sha1c.c +++ b/lib/libmd/sha1c.c @@ -1,6 +1,6 @@  /*-   * Copyright (c) 2009 The Go Authors. All rights reserved. - * Copyright (c) 2024 Robert Clausecker <fuz@freebsd.org> + * Copyright (c) 2024 Robert Clausecker <fuz@FreeBSD.org>   *   * Adapted from Go's crypto/sha1/sha1.go.   * diff --git a/lib/libsys/clock_gettime.2 b/lib/libsys/clock_gettime.2 index 89551d0f720b..841673648dfc 100644 --- a/lib/libsys/clock_gettime.2 +++ b/lib/libsys/clock_gettime.2 @@ -27,7 +27,7 @@  .\" OUT OF THE USE OF THIS SOFTWARE, EVEN IF ADVISED OF THE POSSIBILITY OF  .\" SUCH DAMAGE.  .\" -.Dd August 10, 2024 +.Dd October 15, 2025  .Dt CLOCK_GETTIME 2  .Os  .Sh NAME @@ -80,7 +80,6 @@ Behavior during a leap second is not defined by and POSIX standard.  Increments in SI seconds, even while the system is suspended.  Its epoch is unspecified.  The count is not adjusted by leap seconds. -.Fx implements  .It Dv CLOCK_UPTIME  .It Dv CLOCK_UPTIME_PRECISE  .It Dv CLOCK_UPTIME_FAST diff --git a/lib/libsys/getrlimitusage.2 b/lib/libsys/getrlimitusage.2 index e2114def56c2..d0e92d7f88b4 100644 --- a/lib/libsys/getrlimitusage.2 +++ b/lib/libsys/getrlimitusage.2 @@ -25,7 +25,7 @@  .\" OUT OF THE USE OF THIS SOFTWARE, EVEN IF ADVISED OF THE POSSIBILITY OF  .\" SUCH DAMAGE.  .\" -.Dd September 27, 2024 +.Dd October 24, 2025  .Dt GETRLIMITUSAGE 2  .Os  .Sh NAME @@ -97,4 +97,4 @@ and  The  .Fn getrlimitusage  system call appeared in -.Bx 15.0 . +.Fx 14.2 . diff --git a/lib/libunbound/config.h b/lib/libunbound/config.h index 1cedf5b4de36..1b939b7f1074 100644 --- a/lib/libunbound/config.h +++ b/lib/libunbound/config.h @@ -884,7 +884,7 @@  #define PACKAGE_NAME "unbound"  /* Define to the full name and version of this package. */ -#define PACKAGE_STRING "unbound 1.24.0" +#define PACKAGE_STRING "unbound 1.24.1"  /* Define to the one symbol short name of this package. */  #define PACKAGE_TARNAME "unbound" @@ -893,7 +893,7 @@  #define PACKAGE_URL ""  /* Define to the version of this package. */ -#define PACKAGE_VERSION "1.24.0" +#define PACKAGE_VERSION "1.24.1"  /* default pidfile location */  #define PIDFILE "/var/unbound/unbound.pid" diff --git a/lib/libutil/mntopts.c b/lib/libutil/mntopts.c index 07d3dd6d98a3..4a064a086fd5 100644 --- a/lib/libutil/mntopts.c +++ b/lib/libutil/mntopts.c @@ -145,6 +145,18 @@ checkpath_allow_file(const char *path, char *resolved)  	return (0);  } +static char * +prependdevtopath(const char *path, char *buf, u_long buflen) +{ +	u_long len; + +	if ((len = strlen(_PATH_DEV) + strlen(path) + 1) > buflen) +		return NULL; +	strncpy(buf, _PATH_DEV, len); +	strncat(buf, path, len - sizeof(_PATH_DEV)); +	return (buf); +} +  /*   * Get the mount point information for name. Name may be mount point name   * or device name (with or without /dev/ preprended). @@ -153,19 +165,27 @@ struct statfs *  getmntpoint(const char *name)  {  	struct stat devstat, mntdevstat; -	char device[sizeof(_PATH_DEV) - 1 + MNAMELEN]; -	char *ddevname; +	char *devname;  	struct statfs *mntbuf, *statfsp; -	int i, mntsize, isdev; -	u_long len; +	int i, len, isdev, mntsize, mntfromnamesize; +	char device[sizeof(_PATH_DEV) - 1 + MNAMELEN]; +	u_long devlen; -	if (stat(name, &devstat) != 0) +	devlen = sizeof(device); +	/* +	 * Note that stat(NULL, &statbuf) returns -1 (EBADF) which will +	 * cause us to return NULL if prependdevtopath() returns NULL. +	 */ +	if (stat(name, &devstat) != 0 && +	    (name[0] != '/' && +	     stat(prependdevtopath(name, device, devlen), &devstat) != 0))  		return (NULL);  	if (S_ISCHR(devstat.st_mode) || S_ISBLK(devstat.st_mode))  		isdev = 1;  	else  		isdev = 0;  	mntsize = getmntinfo(&mntbuf, MNT_NOWAIT); +	mntfromnamesize = sizeof(statfsp->f_mntfromname);  	for (i = 0; i < mntsize; i++) {  		statfsp = &mntbuf[i];  		if (isdev == 0) { @@ -173,19 +193,20 @@ getmntpoint(const char *name)  				continue;  			return (statfsp);  		} -		ddevname = statfsp->f_mntfromname; -		if (*ddevname != '/') { -			if ((len = strlen(_PATH_DEV) + strlen(ddevname) + 1) > -			    sizeof(statfsp->f_mntfromname) || -			    len > sizeof(device)) +		devname = statfsp->f_mntfromname; +		if (*devname == '/') { +			if (stat(devname, &mntdevstat) != 0) +				continue; +		} else { +			devname = prependdevtopath(devname, device, devlen); +			if (devname == NULL || +			    (len = strlen(devname)) > mntfromnamesize) +				continue; +			if (stat(devname, &mntdevstat) != 0)  				continue; -			strncpy(device, _PATH_DEV, len); -			strncat(device, ddevname, len); -			if (stat(device, &mntdevstat) == 0) -				strncpy(statfsp->f_mntfromname, device, len); +			strncpy(statfsp->f_mntfromname, devname, len);  		} -		if (stat(ddevname, &mntdevstat) == 0 && -		    S_ISCHR(mntdevstat.st_mode) && +		if (S_ISCHR(mntdevstat.st_mode) &&  		    mntdevstat.st_rdev == devstat.st_rdev)  			return (statfsp);  	} | 
