1*19c3b8c2SApple OSS Distributions /* -*- Mode: c; tab-width: 8; indent-tabs-mode: 1; c-basic-offset: 8; -*- */
2*19c3b8c2SApple OSS Distributions
3*19c3b8c2SApple OSS Distributions #include <darwintest.h>
4*19c3b8c2SApple OSS Distributions #include <stdio.h>
5*19c3b8c2SApple OSS Distributions #include <unistd.h>
6*19c3b8c2SApple OSS Distributions #include <stdlib.h>
7*19c3b8c2SApple OSS Distributions #include <string.h>
8*19c3b8c2SApple OSS Distributions #include <sys/socket.h>
9*19c3b8c2SApple OSS Distributions #include <netinet/in.h>
10*19c3b8c2SApple OSS Distributions #include <arpa/inet.h>
11*19c3b8c2SApple OSS Distributions #include <errno.h>
12*19c3b8c2SApple OSS Distributions #include <pthread.h>
13*19c3b8c2SApple OSS Distributions #include <stdbool.h>
14*19c3b8c2SApple OSS Distributions #include <TargetConditionals.h>
15*19c3b8c2SApple OSS Distributions
16*19c3b8c2SApple OSS Distributions T_GLOBAL_META(T_META_RUN_CONCURRENTLY(true));
17*19c3b8c2SApple OSS Distributions
18*19c3b8c2SApple OSS Distributions static bool debug;
19*19c3b8c2SApple OSS Distributions
20*19c3b8c2SApple OSS Distributions static int
sock_open_common(int pf,int type)21*19c3b8c2SApple OSS Distributions sock_open_common(int pf, int type)
22*19c3b8c2SApple OSS Distributions {
23*19c3b8c2SApple OSS Distributions int s;
24*19c3b8c2SApple OSS Distributions
25*19c3b8c2SApple OSS Distributions s = socket(pf, type, 0);
26*19c3b8c2SApple OSS Distributions T_QUIET;
27*19c3b8c2SApple OSS Distributions T_ASSERT_POSIX_SUCCESS(s, "socket(%d, %d, 0)", pf, type);
28*19c3b8c2SApple OSS Distributions return s;
29*19c3b8c2SApple OSS Distributions }
30*19c3b8c2SApple OSS Distributions
31*19c3b8c2SApple OSS Distributions static int
sock_open(int type)32*19c3b8c2SApple OSS Distributions sock_open(int type)
33*19c3b8c2SApple OSS Distributions {
34*19c3b8c2SApple OSS Distributions return sock_open_common(PF_INET, type);
35*19c3b8c2SApple OSS Distributions }
36*19c3b8c2SApple OSS Distributions
37*19c3b8c2SApple OSS Distributions static int
sock_bind(int s,int port)38*19c3b8c2SApple OSS Distributions sock_bind(int s, int port)
39*19c3b8c2SApple OSS Distributions {
40*19c3b8c2SApple OSS Distributions struct sockaddr_in sin = {
41*19c3b8c2SApple OSS Distributions .sin_len = sizeof(sin),
42*19c3b8c2SApple OSS Distributions .sin_family = AF_INET,
43*19c3b8c2SApple OSS Distributions };
44*19c3b8c2SApple OSS Distributions
45*19c3b8c2SApple OSS Distributions sin.sin_port = htons(port);
46*19c3b8c2SApple OSS Distributions return bind(s, (const struct sockaddr *)&sin, sizeof(sin));
47*19c3b8c2SApple OSS Distributions }
48*19c3b8c2SApple OSS Distributions
49*19c3b8c2SApple OSS Distributions static int
sockv6_open(int type)50*19c3b8c2SApple OSS Distributions sockv6_open(int type)
51*19c3b8c2SApple OSS Distributions {
52*19c3b8c2SApple OSS Distributions return sock_open_common(PF_INET6, type);
53*19c3b8c2SApple OSS Distributions }
54*19c3b8c2SApple OSS Distributions
55*19c3b8c2SApple OSS Distributions static int
sockv6_bind(int s,int port)56*19c3b8c2SApple OSS Distributions sockv6_bind(int s, int port)
57*19c3b8c2SApple OSS Distributions {
58*19c3b8c2SApple OSS Distributions struct sockaddr_in6 sin6 = {
59*19c3b8c2SApple OSS Distributions .sin6_len = sizeof(sin6),
60*19c3b8c2SApple OSS Distributions .sin6_family = AF_INET6,
61*19c3b8c2SApple OSS Distributions };
62*19c3b8c2SApple OSS Distributions
63*19c3b8c2SApple OSS Distributions sin6.sin6_port = htons(port);
64*19c3b8c2SApple OSS Distributions return bind(s, (const struct sockaddr *)&sin6, sizeof(sin6));
65*19c3b8c2SApple OSS Distributions }
66*19c3b8c2SApple OSS Distributions
67*19c3b8c2SApple OSS Distributions static uint16_t
sock_get_port(int sockfd)68*19c3b8c2SApple OSS Distributions sock_get_port(int sockfd)
69*19c3b8c2SApple OSS Distributions {
70*19c3b8c2SApple OSS Distributions int error;
71*19c3b8c2SApple OSS Distributions uint16_t p;
72*19c3b8c2SApple OSS Distributions union sockaddr_in_4_6 sin;
73*19c3b8c2SApple OSS Distributions socklen_t sin_len;
74*19c3b8c2SApple OSS Distributions
75*19c3b8c2SApple OSS Distributions sin_len = sizeof(sin);
76*19c3b8c2SApple OSS Distributions bzero(&sin, sin_len);
77*19c3b8c2SApple OSS Distributions error = getsockname(sockfd, (struct sockaddr *)&sin, &sin_len);
78*19c3b8c2SApple OSS Distributions T_QUIET;
79*19c3b8c2SApple OSS Distributions T_EXPECT_POSIX_ZERO(error, "getsockname(%d)", sockfd);
80*19c3b8c2SApple OSS Distributions if (error != 0) {
81*19c3b8c2SApple OSS Distributions return 0;
82*19c3b8c2SApple OSS Distributions }
83*19c3b8c2SApple OSS Distributions switch (sin.sa.sa_family) {
84*19c3b8c2SApple OSS Distributions case AF_INET:
85*19c3b8c2SApple OSS Distributions p = sin.sin.sin_port;
86*19c3b8c2SApple OSS Distributions break;
87*19c3b8c2SApple OSS Distributions case AF_INET6:
88*19c3b8c2SApple OSS Distributions p = sin.sin6.sin6_port;
89*19c3b8c2SApple OSS Distributions break;
90*19c3b8c2SApple OSS Distributions default:
91*19c3b8c2SApple OSS Distributions T_ASSERT_FAIL("unknown address family %d\n",
92*19c3b8c2SApple OSS Distributions sin.sa.sa_family);
93*19c3b8c2SApple OSS Distributions p = 0;
94*19c3b8c2SApple OSS Distributions break;
95*19c3b8c2SApple OSS Distributions }
96*19c3b8c2SApple OSS Distributions return p;
97*19c3b8c2SApple OSS Distributions }
98*19c3b8c2SApple OSS Distributions
99*19c3b8c2SApple OSS Distributions typedef struct {
100*19c3b8c2SApple OSS Distributions bool v6;
101*19c3b8c2SApple OSS Distributions int socket_count;
102*19c3b8c2SApple OSS Distributions int * socket_list;
103*19c3b8c2SApple OSS Distributions } SocketInfo, * SocketInfoRef;
104*19c3b8c2SApple OSS Distributions
105*19c3b8c2SApple OSS Distributions static void
bind_sockets(SocketInfoRef info,const char * msg)106*19c3b8c2SApple OSS Distributions bind_sockets(SocketInfoRef info, const char * msg)
107*19c3b8c2SApple OSS Distributions {
108*19c3b8c2SApple OSS Distributions for (int i = 0; i < info->socket_count; i++) {
109*19c3b8c2SApple OSS Distributions int error;
110*19c3b8c2SApple OSS Distributions uint16_t port;
111*19c3b8c2SApple OSS Distributions
112*19c3b8c2SApple OSS Distributions if (info->v6) {
113*19c3b8c2SApple OSS Distributions error = sockv6_bind(info->socket_list[i], 0);
114*19c3b8c2SApple OSS Distributions } else {
115*19c3b8c2SApple OSS Distributions error = sock_bind(info->socket_list[i], 0);
116*19c3b8c2SApple OSS Distributions }
117*19c3b8c2SApple OSS Distributions port = sock_get_port(info->socket_list[i]);
118*19c3b8c2SApple OSS Distributions if (debug) {
119*19c3b8c2SApple OSS Distributions T_LOG( "%s: fd %d port is %d error %d",
120*19c3b8c2SApple OSS Distributions msg, info->socket_list[i], ntohs(port), error);
121*19c3b8c2SApple OSS Distributions }
122*19c3b8c2SApple OSS Distributions }
123*19c3b8c2SApple OSS Distributions return;
124*19c3b8c2SApple OSS Distributions }
125*19c3b8c2SApple OSS Distributions
126*19c3b8c2SApple OSS Distributions static void *
second_thread(void * arg)127*19c3b8c2SApple OSS Distributions second_thread(void * arg)
128*19c3b8c2SApple OSS Distributions {
129*19c3b8c2SApple OSS Distributions SocketInfoRef info = (SocketInfoRef)arg;
130*19c3b8c2SApple OSS Distributions
131*19c3b8c2SApple OSS Distributions bind_sockets(info, "second");
132*19c3b8c2SApple OSS Distributions return NULL;
133*19c3b8c2SApple OSS Distributions }
134*19c3b8c2SApple OSS Distributions
135*19c3b8c2SApple OSS Distributions static void
multithreaded_bind_test(bool v6,int socket_count)136*19c3b8c2SApple OSS Distributions multithreaded_bind_test(bool v6, int socket_count)
137*19c3b8c2SApple OSS Distributions {
138*19c3b8c2SApple OSS Distributions int error;
139*19c3b8c2SApple OSS Distributions SocketInfo info;
140*19c3b8c2SApple OSS Distributions int socket_list[socket_count];
141*19c3b8c2SApple OSS Distributions pthread_t thread;
142*19c3b8c2SApple OSS Distributions
143*19c3b8c2SApple OSS Distributions info.v6 = v6;
144*19c3b8c2SApple OSS Distributions for (int i = 0; i < socket_count; i++) {
145*19c3b8c2SApple OSS Distributions if (v6) {
146*19c3b8c2SApple OSS Distributions socket_list[i] = sockv6_open(SOCK_STREAM);
147*19c3b8c2SApple OSS Distributions } else {
148*19c3b8c2SApple OSS Distributions socket_list[i] = sock_open(SOCK_STREAM);
149*19c3b8c2SApple OSS Distributions }
150*19c3b8c2SApple OSS Distributions }
151*19c3b8c2SApple OSS Distributions info.socket_count = socket_count;
152*19c3b8c2SApple OSS Distributions info.socket_list = socket_list;
153*19c3b8c2SApple OSS Distributions error = pthread_create(&thread, NULL, second_thread, &info);
154*19c3b8c2SApple OSS Distributions T_QUIET;
155*19c3b8c2SApple OSS Distributions T_ASSERT_POSIX_ZERO(error, "pthread_create");
156*19c3b8c2SApple OSS Distributions
157*19c3b8c2SApple OSS Distributions /* compete with second thread */
158*19c3b8c2SApple OSS Distributions bind_sockets(&info, "main");
159*19c3b8c2SApple OSS Distributions error = pthread_join(thread, NULL);
160*19c3b8c2SApple OSS Distributions T_QUIET;
161*19c3b8c2SApple OSS Distributions T_ASSERT_POSIX_ZERO(error, "pthread_join");
162*19c3b8c2SApple OSS Distributions
163*19c3b8c2SApple OSS Distributions for (int i = 0; i < socket_count; i++) {
164*19c3b8c2SApple OSS Distributions error = close(socket_list[i]);
165*19c3b8c2SApple OSS Distributions T_QUIET;
166*19c3b8c2SApple OSS Distributions T_ASSERT_POSIX_ZERO(error, "close socket %d", socket_list[i]);
167*19c3b8c2SApple OSS Distributions }
168*19c3b8c2SApple OSS Distributions }
169*19c3b8c2SApple OSS Distributions
170*19c3b8c2SApple OSS Distributions static void
run_multithreaded_bind_test(int number_of_runs,bool v6,int socket_count)171*19c3b8c2SApple OSS Distributions run_multithreaded_bind_test(int number_of_runs, bool v6, int socket_count)
172*19c3b8c2SApple OSS Distributions {
173*19c3b8c2SApple OSS Distributions #if TARGET_OS_BRIDGE
174*19c3b8c2SApple OSS Distributions T_SKIP("Not enough memory to handle this test");
175*19c3b8c2SApple OSS Distributions #else /* TARGET_OS_BRIDGE */
176*19c3b8c2SApple OSS Distributions for (int i = 0; i < number_of_runs; i++) {
177*19c3b8c2SApple OSS Distributions multithreaded_bind_test(v6, socket_count);
178*19c3b8c2SApple OSS Distributions }
179*19c3b8c2SApple OSS Distributions T_PASS("multithreaded_bind_test %s", v6 ? "IPv6" : "IPv4");
180*19c3b8c2SApple OSS Distributions #endif /* TARGET_OS_BRIDGE */
181*19c3b8c2SApple OSS Distributions }
182*19c3b8c2SApple OSS Distributions
183*19c3b8c2SApple OSS Distributions T_DECL(socket_bind_35685803,
184*19c3b8c2SApple OSS Distributions "multithreaded bind IPv4 socket as root",
185*19c3b8c2SApple OSS Distributions T_META_ASROOT(false),
186*19c3b8c2SApple OSS Distributions T_META_CHECK_LEAKS(false))
187*19c3b8c2SApple OSS Distributions {
188*19c3b8c2SApple OSS Distributions run_multithreaded_bind_test(100, false, 100);
189*19c3b8c2SApple OSS Distributions }
190*19c3b8c2SApple OSS Distributions
191*19c3b8c2SApple OSS Distributions T_DECL(socket_bind_35685803_root,
192*19c3b8c2SApple OSS Distributions "multithreaded bind IPv4 socket",
193*19c3b8c2SApple OSS Distributions T_META_ASROOT(true))
194*19c3b8c2SApple OSS Distributions {
195*19c3b8c2SApple OSS Distributions run_multithreaded_bind_test(100, false, 100);
196*19c3b8c2SApple OSS Distributions }
197*19c3b8c2SApple OSS Distributions
198*19c3b8c2SApple OSS Distributions T_DECL(socket_bind_35685803_v6,
199*19c3b8c2SApple OSS Distributions "multithreaded bind IPv6 socket as root",
200*19c3b8c2SApple OSS Distributions T_META_ASROOT(false),
201*19c3b8c2SApple OSS Distributions T_META_CHECK_LEAKS(false))
202*19c3b8c2SApple OSS Distributions {
203*19c3b8c2SApple OSS Distributions run_multithreaded_bind_test(100, true, 100);
204*19c3b8c2SApple OSS Distributions }
205*19c3b8c2SApple OSS Distributions
206*19c3b8c2SApple OSS Distributions T_DECL(socket_bind_35685803_v6_root,
207*19c3b8c2SApple OSS Distributions "multithreaded bind IPv6 socket",
208*19c3b8c2SApple OSS Distributions T_META_ASROOT(true))
209*19c3b8c2SApple OSS Distributions {
210*19c3b8c2SApple OSS Distributions run_multithreaded_bind_test(100, true, 100);
211*19c3b8c2SApple OSS Distributions }
212