git: 0849f1634a70 - main - tests/netinet: add test for IP_MULTICAST_IF

From: Gleb Smirnoff <glebius_at_FreeBSD.org>
Date: Sat, 22 Mar 2025 23:40:46 UTC
The branch main has been updated by glebius:

URL: https://cgit.FreeBSD.org/src/commit/?id=0849f1634a70099b90256ceece52a598eeb3280e

commit 0849f1634a70099b90256ceece52a598eeb3280e
Author:     Gleb Smirnoff <glebius@FreeBSD.org>
AuthorDate: 2025-03-22 22:44:20 +0000
Commit:     Gleb Smirnoff <glebius@FreeBSD.org>
CommitDate: 2025-03-22 23:39:50 +0000

    tests/netinet: add test for IP_MULTICAST_IF
---
 tests/sys/netinet/Makefile                 |  3 +-
 tests/sys/netinet/multicast.sh             | 61 +++++++++++++++++++++++++++++
 tests/sys/netinet/sendto-IP_MULTICAST_IF.c | 63 ++++++++++++++++++++++++++++++
 3 files changed, 126 insertions(+), 1 deletion(-)

diff --git a/tests/sys/netinet/Makefile b/tests/sys/netinet/Makefile
index bd972bc3b2a0..cc525bf24480 100644
--- a/tests/sys/netinet/Makefile
+++ b/tests/sys/netinet/Makefile
@@ -24,6 +24,7 @@ ATF_TESTS_SH=	arp \
 		fibs_test \
 		forward \
 		lpm \
+		multicast \
 		output \
 		redirect
 
@@ -47,7 +48,7 @@ TEST_METADATA.forward+=	required_programs="python"	\
 TEST_METADATA.output+=	required_programs="python"
 TEST_METADATA.redirect+= required_programs="python"
 
-PROGS=	udp_dontroute tcp_user_cookie
+PROGS=	udp_dontroute tcp_user_cookie sendto-IP_MULTICAST_IF
 
 ${PACKAGE}FILES+=		redirect.py
 
diff --git a/tests/sys/netinet/multicast.sh b/tests/sys/netinet/multicast.sh
new file mode 100644
index 000000000000..eb2b962dac70
--- /dev/null
+++ b/tests/sys/netinet/multicast.sh
@@ -0,0 +1,61 @@
+#
+# SPDX-License-Identifier: BSD-2-Clause
+#
+# Copyright (c) 2025 Gleb Smirnoff <glebius@FreeBSD.org>
+#
+# Redistribution and use in source and binary forms, with or without
+# modification, are permitted provided that the following conditions
+# are met:
+# 1. Redistributions of source code must retain the above copyright
+#    notice, this list of conditions and the following disclaimer.
+# 2. Redistributions in binary form must reproduce the above copyright
+#    notice, this list of conditions and the following disclaimer in the
+#    documentation and/or other materials provided with the distribution.
+#
+# THIS SOFTWARE IS PROVIDED BY THE AUTHOR AND CONTRIBUTORS ``AS IS'' AND
+# ANY EXPRESS OR IMPLIED WARRANTIES, INCLUDING, BUT NOT LIMITED TO, THE
+# IMPLIED WARRANTIES OF MERCHANTABILITY AND FITNESS FOR A PARTICULAR PURPOSE
+# ARE DISCLAIMED.  IN NO EVENT SHALL THE AUTHOR OR CONTRIBUTORS BE LIABLE
+# FOR ANY DIRECT, INDIRECT, INCIDENTAL, SPECIAL, EXEMPLARY, OR CONSEQUENTIAL
+# DAMAGES (INCLUDING, BUT NOT LIMITED TO, PROCUREMENT OF SUBSTITUTE GOODS
+# OR SERVICES; LOSS OF USE, DATA, OR PROFITS; OR BUSINESS INTERRUPTION)
+# HOWEVER CAUSED AND ON ANY THEORY OF LIABILITY, WHETHER IN CONTRACT, STRICT
+# LIABILITY, OR TORT (INCLUDING NEGLIGENCE OR OTHERWISE) ARISING IN ANY WAY
+# OUT OF THE USE OF THIS SOFTWARE, EVEN IF ADVISED OF THE POSSIBILITY OF
+# SUCH DAMAGE.
+
+. $(atf_get_srcdir)/../common/vnet.subr
+
+# See regression fixed in baad45c9c12028964acd0b58096f3aaa0fb22859
+atf_test_case "IP_MULTICAST_IF" "cleanup"
+IP_MULTICAST_IF_head()
+{
+	atf_set descr \
+	    'sendto() for IP_MULTICAST_IF socket does not do routing lookup'
+	atf_set require.user root
+
+}
+
+IP_MULTICAST_IF_body()
+{
+	local epair mjail
+
+	vnet_init
+	# The test doesn't use our half of epair
+	epair=$(vnet_mkepair)
+	vnet_mkjail mjail ${epair}a
+	jexec mjail ifconfig ${epair}a up
+	jexec mjail ifconfig ${epair}a 192.0.2.1/24
+	atf_check -s exit:0 -o empty \
+	    jexec mjail $(atf_get_srcdir)/sendto-IP_MULTICAST_IF 192.0.2.1
+}
+
+IP_MULTICAST_IF_cleanup()
+{
+	vnet_cleanup
+}
+
+atf_init_test_cases()
+{
+	atf_add_test_case "IP_MULTICAST_IF"
+}
diff --git a/tests/sys/netinet/sendto-IP_MULTICAST_IF.c b/tests/sys/netinet/sendto-IP_MULTICAST_IF.c
new file mode 100644
index 000000000000..d478e4da0b3b
--- /dev/null
+++ b/tests/sys/netinet/sendto-IP_MULTICAST_IF.c
@@ -0,0 +1,63 @@
+/*-
+ * SPDX-License-Identifier: BSD-2-Clause
+ *
+ * Copyright (c) 2025 Gleb Smirnoff <glebius@FreeBSD.org>
+ *
+ * Redistribution and use in source and binary forms, with or without
+ * modification, are permitted provided that the following conditions
+ * are met:
+ * 1. Redistributions of source code must retain the above copyright
+ *    notice, this list of conditions and the following disclaimer.
+ * 2. Redistributions in binary form must reproduce the above copyright
+ *    notice, this list of conditions and the following disclaimer in the
+ *    documentation and/or other materials provided with the distribution.
+ *
+ * THIS SOFTWARE IS PROVIDED BY THE AUTHOR AND CONTRIBUTORS ``AS IS'' AND
+ * ANY EXPRESS OR IMPLIED WARRANTIES, INCLUDING, BUT NOT LIMITED TO, THE
+ * IMPLIED WARRANTIES OF MERCHANTABILITY AND FITNESS FOR A PARTICULAR PURPOSE
+ * ARE DISCLAIMED.  IN NO EVENT SHALL THE AUTHOR OR CONTRIBUTORS BE LIABLE
+ * FOR ANY DIRECT, INDIRECT, INCIDENTAL, SPECIAL, EXEMPLARY, OR CONSEQUENTIAL
+ * DAMAGES (INCLUDING, BUT NOT LIMITED TO, PROCUREMENT OF SUBSTITUTE GOODS
+ * OR SERVICES; LOSS OF USE, DATA, OR PROFITS; OR BUSINESS INTERRUPTION)
+ * HOWEVER CAUSED AND ON ANY THEORY OF LIABILITY, WHETHER IN CONTRACT, STRICT
+ * LIABILITY, OR TORT (INCLUDING NEGLIGENCE OR OTHERWISE) ARISING IN ANY WAY
+ * OUT OF THE USE OF THIS SOFTWARE, EVEN IF ADVISED OF THE POSSIBILITY OF
+ * SUCH DAMAGE.
+ */
+
+#include <sys/socket.h>
+#include <netinet/in.h>
+#include <arpa/inet.h>
+#include <assert.h>
+#include <err.h>
+
+int
+main(int argc, char *argv[])
+{
+	struct sockaddr_in sin = {
+		.sin_family = AF_INET,
+		.sin_len = sizeof(struct sockaddr_in),
+	};
+	struct in_addr in;
+	int s, rv;
+
+	if (argc < 2)
+		errx(1, "Usage: %s IPv4-address", argv[0]);
+
+	if (inet_pton(AF_INET, argv[1], &in) != 1)
+		err(1, "inet_pton(%s) failed", argv[1]);
+
+	assert((s = socket(PF_INET, SOCK_DGRAM, 0)) > 0);
+	assert(bind(s, (struct sockaddr *)&sin, sizeof(sin)) == 0);
+	assert(setsockopt(s, IPPROTO_IP, IP_MULTICAST_IF, &in, sizeof(in))
+	  == 0);
+	/* RFC 6676 */
+	assert(inet_pton(AF_INET, "233.252.0.1", &sin.sin_addr) == 1);
+	sin.sin_port = htons(6676);
+	rv = sendto(s, &sin, sizeof(sin), 0,
+	    (struct sockaddr *)&sin, sizeof(sin));
+	if (rv != sizeof(sin))
+		err(1, "sendto failed");
+
+	return (0);
+}