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