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