xref: /xnu-12377.41.6/tests/socket_bind_35243417.c (revision bbb1b6f9e71b8cdde6e5cd6f4841f207dee3d828)
1*bbb1b6f9SApple OSS Distributions /* -*- Mode: c; tab-width: 8; indent-tabs-mode: 1; c-basic-offset: 8; -*- */
2*bbb1b6f9SApple OSS Distributions 
3*bbb1b6f9SApple OSS Distributions #include <darwintest.h>
4*bbb1b6f9SApple OSS Distributions #include <poll.h>
5*bbb1b6f9SApple OSS Distributions #include <sys/socket.h>
6*bbb1b6f9SApple OSS Distributions #include <unistd.h>
7*bbb1b6f9SApple OSS Distributions #include <netinet/in.h>
8*bbb1b6f9SApple OSS Distributions #include <arpa/inet.h>
9*bbb1b6f9SApple OSS Distributions #include <errno.h>
10*bbb1b6f9SApple OSS Distributions 
11*bbb1b6f9SApple OSS Distributions #include "net_test_lib.h"
12*bbb1b6f9SApple OSS Distributions 
13*bbb1b6f9SApple OSS Distributions static int
sockv6_open(void)14*bbb1b6f9SApple OSS Distributions sockv6_open(void)
15*bbb1b6f9SApple OSS Distributions {
16*bbb1b6f9SApple OSS Distributions 	int     s;
17*bbb1b6f9SApple OSS Distributions 
18*bbb1b6f9SApple OSS Distributions 	s = socket(AF_INET6, SOCK_DGRAM, 0);
19*bbb1b6f9SApple OSS Distributions 	T_QUIET;
20*bbb1b6f9SApple OSS Distributions 	T_ASSERT_POSIX_SUCCESS(s, "socket(AF_INET6, SOCK_DGRAM, 0)");
21*bbb1b6f9SApple OSS Distributions 	return s;
22*bbb1b6f9SApple OSS Distributions }
23*bbb1b6f9SApple OSS Distributions 
24*bbb1b6f9SApple OSS Distributions static int
sockv6_bind(int s,in_port_t port)25*bbb1b6f9SApple OSS Distributions sockv6_bind(int s, in_port_t port)
26*bbb1b6f9SApple OSS Distributions {
27*bbb1b6f9SApple OSS Distributions 	struct sockaddr_in6     sin6;
28*bbb1b6f9SApple OSS Distributions 
29*bbb1b6f9SApple OSS Distributions 	bzero(&sin6, sizeof(sin6));
30*bbb1b6f9SApple OSS Distributions 	sin6.sin6_len = sizeof(sin6);
31*bbb1b6f9SApple OSS Distributions 	sin6.sin6_family = AF_INET6;
32*bbb1b6f9SApple OSS Distributions 	sin6.sin6_port = port;
33*bbb1b6f9SApple OSS Distributions 	return bind(s, (const struct sockaddr *)&sin6, sizeof(sin6));
34*bbb1b6f9SApple OSS Distributions }
35*bbb1b6f9SApple OSS Distributions 
36*bbb1b6f9SApple OSS Distributions static void
sockv6_set_v6only(int s)37*bbb1b6f9SApple OSS Distributions sockv6_set_v6only(int s)
38*bbb1b6f9SApple OSS Distributions {
39*bbb1b6f9SApple OSS Distributions 	int             on = 1;
40*bbb1b6f9SApple OSS Distributions 	int             ret;
41*bbb1b6f9SApple OSS Distributions 
42*bbb1b6f9SApple OSS Distributions 	ret = setsockopt(s, IPPROTO_IPV6, IPV6_V6ONLY, &on, sizeof(on));
43*bbb1b6f9SApple OSS Distributions 	T_QUIET;
44*bbb1b6f9SApple OSS Distributions 	T_ASSERT_POSIX_SUCCESS(ret, "setsockopt(%d, IPV6_ONLY)", s);
45*bbb1b6f9SApple OSS Distributions }
46*bbb1b6f9SApple OSS Distributions 
47*bbb1b6f9SApple OSS Distributions static bool
alloc_and_bind_ports(in_port_t port_start,in_port_t port_end,int bind_attempts)48*bbb1b6f9SApple OSS Distributions alloc_and_bind_ports(in_port_t port_start, in_port_t port_end,
49*bbb1b6f9SApple OSS Distributions     int bind_attempts)
50*bbb1b6f9SApple OSS Distributions {
51*bbb1b6f9SApple OSS Distributions 	int     bound_count = 0;
52*bbb1b6f9SApple OSS Distributions 	bool    success = true;
53*bbb1b6f9SApple OSS Distributions 
54*bbb1b6f9SApple OSS Distributions 	for (in_port_t i = port_start; success && i <= port_end; i++) {
55*bbb1b6f9SApple OSS Distributions 		int     s6 = -1;
56*bbb1b6f9SApple OSS Distributions 		int     s6_other = -1;
57*bbb1b6f9SApple OSS Distributions 		int     ret;
58*bbb1b6f9SApple OSS Distributions 
59*bbb1b6f9SApple OSS Distributions 		s6 = sockv6_open();
60*bbb1b6f9SApple OSS Distributions 		sockv6_set_v6only(s6);
61*bbb1b6f9SApple OSS Distributions 		if (sockv6_bind(s6, i) != 0) {
62*bbb1b6f9SApple OSS Distributions 			/* find the next available port */
63*bbb1b6f9SApple OSS Distributions 			goto loop_done;
64*bbb1b6f9SApple OSS Distributions 		}
65*bbb1b6f9SApple OSS Distributions 		s6_other = sockv6_open();
66*bbb1b6f9SApple OSS Distributions 		ret = sockv6_bind(s6_other, i);
67*bbb1b6f9SApple OSS Distributions 		T_WITH_ERRNO;
68*bbb1b6f9SApple OSS Distributions 		T_QUIET;
69*bbb1b6f9SApple OSS Distributions 		T_ASSERT_TRUE(ret != 0, "socket %d bind %d", s6_other, i);
70*bbb1b6f9SApple OSS Distributions 		/*
71*bbb1b6f9SApple OSS Distributions 		 * After bind fails, try binding to a different port.
72*bbb1b6f9SApple OSS Distributions 		 * For non-root user, this will panic without the fix for
73*bbb1b6f9SApple OSS Distributions 		 * <rdar://problem/35243417>.
74*bbb1b6f9SApple OSS Distributions 		 */
75*bbb1b6f9SApple OSS Distributions 		if (sockv6_bind(s6_other, i + 1) == 0) {
76*bbb1b6f9SApple OSS Distributions 			bound_count++;
77*bbb1b6f9SApple OSS Distributions 			if (bound_count >= bind_attempts) {
78*bbb1b6f9SApple OSS Distributions 				break;
79*bbb1b6f9SApple OSS Distributions 			}
80*bbb1b6f9SApple OSS Distributions 		}
81*bbb1b6f9SApple OSS Distributions loop_done:
82*bbb1b6f9SApple OSS Distributions 		if (s6 >= 0) {
83*bbb1b6f9SApple OSS Distributions 			close(s6);
84*bbb1b6f9SApple OSS Distributions 		}
85*bbb1b6f9SApple OSS Distributions 		if (s6_other >= 0) {
86*bbb1b6f9SApple OSS Distributions 			close(s6_other);
87*bbb1b6f9SApple OSS Distributions 		}
88*bbb1b6f9SApple OSS Distributions 	}
89*bbb1b6f9SApple OSS Distributions 	T_ASSERT_TRUE(bound_count == bind_attempts,
90*bbb1b6f9SApple OSS Distributions 	    "number of successful binds %d (out of %d)",
91*bbb1b6f9SApple OSS Distributions 	    bound_count, bind_attempts);
92*bbb1b6f9SApple OSS Distributions 
93*bbb1b6f9SApple OSS Distributions 	force_zone_gc();
94*bbb1b6f9SApple OSS Distributions 
95*bbb1b6f9SApple OSS Distributions 	return success;
96*bbb1b6f9SApple OSS Distributions }
97*bbb1b6f9SApple OSS Distributions 
98*bbb1b6f9SApple OSS Distributions 
99*bbb1b6f9SApple OSS Distributions T_DECL(socket_bind_35243417,
100*bbb1b6f9SApple OSS Distributions     "bind IPv6 only UDP socket, then bind IPv6 socket.",
101*bbb1b6f9SApple OSS Distributions     T_META_ASROOT(false),
102*bbb1b6f9SApple OSS Distributions     T_META_CHECK_LEAKS(false),
103*bbb1b6f9SApple OSS Distributions     T_META_TAG_VM_PREFERRED)
104*bbb1b6f9SApple OSS Distributions {
105*bbb1b6f9SApple OSS Distributions 	alloc_and_bind_ports(1, 65534, 10);
106*bbb1b6f9SApple OSS Distributions }
107*bbb1b6f9SApple OSS Distributions 
108*bbb1b6f9SApple OSS Distributions T_DECL(socket_bind_35243417_root,
109*bbb1b6f9SApple OSS Distributions     "bind IPv6 only UDP socket, then bind IPv6 socket.",
110*bbb1b6f9SApple OSS Distributions     T_META_ASROOT(true),
111*bbb1b6f9SApple OSS Distributions     T_META_TAG_VM_PREFERRED)
112*bbb1b6f9SApple OSS Distributions {
113*bbb1b6f9SApple OSS Distributions 	alloc_and_bind_ports(1, 65534, 10);
114*bbb1b6f9SApple OSS Distributions }
115