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