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