xref: /xnu-12377.61.12/tests/flow_div_doubleconnect_55917185.c (revision 4d495c6e23c53686cf65f45067f79024cf5dcee8)
1*4d495c6eSApple OSS Distributions #include <stdlib.h>
2*4d495c6eSApple OSS Distributions 
3*4d495c6eSApple OSS Distributions #include <pthread.h>
4*4d495c6eSApple OSS Distributions #include <sys/kern_control.h>
5*4d495c6eSApple OSS Distributions #include <sys/socket.h>
6*4d495c6eSApple OSS Distributions #include <sys/sys_domain.h>
7*4d495c6eSApple OSS Distributions #include <unistd.h>
8*4d495c6eSApple OSS Distributions 
9*4d495c6eSApple OSS Distributions #include <darwintest.h>
10*4d495c6eSApple OSS Distributions 
11*4d495c6eSApple OSS Distributions #include "net_test_lib.h"
12*4d495c6eSApple OSS Distributions 
13*4d495c6eSApple OSS Distributions /* we should win the race in this window: */
14*4d495c6eSApple OSS Distributions #define NTRIES 200000
15*4d495c6eSApple OSS Distributions 
16*4d495c6eSApple OSS Distributions static void *
connect_race(void * data)17*4d495c6eSApple OSS Distributions connect_race(void *data)
18*4d495c6eSApple OSS Distributions {
19*4d495c6eSApple OSS Distributions 	int *ps = data;
20*4d495c6eSApple OSS Distributions 	struct sockaddr_ctl sc = {
21*4d495c6eSApple OSS Distributions 		.sc_id = 1 /* com.apple.flow-divert */
22*4d495c6eSApple OSS Distributions 	};
23*4d495c6eSApple OSS Distributions 	int n;
24*4d495c6eSApple OSS Distributions 
25*4d495c6eSApple OSS Distributions 	for (n = 0; n < NTRIES; ++n) {
26*4d495c6eSApple OSS Distributions 		connect(*ps, (const struct sockaddr *)&sc, sizeof(sc));
27*4d495c6eSApple OSS Distributions 	}
28*4d495c6eSApple OSS Distributions 
29*4d495c6eSApple OSS Distributions 	return NULL;
30*4d495c6eSApple OSS Distributions }
31*4d495c6eSApple OSS Distributions 
32*4d495c6eSApple OSS Distributions T_DECL(flow_div_doubleconnect_55917185, "Bad error path in double-connect for flow_divert_kctl_connect", T_META_TAG_VM_PREFERRED)
33*4d495c6eSApple OSS Distributions {
34*4d495c6eSApple OSS Distributions 	int s = -1;
35*4d495c6eSApple OSS Distributions 	int tmp_s;
36*4d495c6eSApple OSS Distributions 	struct sockaddr_ctl sc = {
37*4d495c6eSApple OSS Distributions 		.sc_id = 1 /* com.apple.flow-divert */
38*4d495c6eSApple OSS Distributions 	};
39*4d495c6eSApple OSS Distributions 	pthread_t t;
40*4d495c6eSApple OSS Distributions 	int n;
41*4d495c6eSApple OSS Distributions 
42*4d495c6eSApple OSS Distributions 	T_SETUPBEGIN;
43*4d495c6eSApple OSS Distributions 	T_ASSERT_POSIX_ZERO(pthread_create(&t, NULL, connect_race, &s), NULL);
44*4d495c6eSApple OSS Distributions 	T_SETUPEND;
45*4d495c6eSApple OSS Distributions 
46*4d495c6eSApple OSS Distributions 	for (n = 0; n < NTRIES; ++n) {
47*4d495c6eSApple OSS Distributions 		T_ASSERT_POSIX_SUCCESS(tmp_s = socket(AF_SYSTEM, SOCK_DGRAM, SYSPROTO_CONTROL), NULL);
48*4d495c6eSApple OSS Distributions 
49*4d495c6eSApple OSS Distributions 		/*
50*4d495c6eSApple OSS Distributions 		 * this bind will fail, but that's ok because it initialises
51*4d495c6eSApple OSS Distributions 		 * kctl:
52*4d495c6eSApple OSS Distributions 		 */
53*4d495c6eSApple OSS Distributions 		bind(tmp_s, (const struct sockaddr *)&sc, sizeof(sc));
54*4d495c6eSApple OSS Distributions 
55*4d495c6eSApple OSS Distributions 		/* this is what we're racing the other thread for: */
56*4d495c6eSApple OSS Distributions 		s = tmp_s;
57*4d495c6eSApple OSS Distributions 		connect(s, (const struct sockaddr *)&sc, sizeof(sc));
58*4d495c6eSApple OSS Distributions 
59*4d495c6eSApple OSS Distributions 		T_ASSERT_POSIX_SUCCESS(close(s), NULL);
60*4d495c6eSApple OSS Distributions 		s = -1;
61*4d495c6eSApple OSS Distributions 	}
62*4d495c6eSApple OSS Distributions 
63*4d495c6eSApple OSS Distributions 
64*4d495c6eSApple OSS Distributions 	T_ASSERT_POSIX_ZERO(pthread_join(t, NULL), NULL);
65*4d495c6eSApple OSS Distributions 	T_PASS("flow_divert_kctl_connect race didn't trigger panic");
66*4d495c6eSApple OSS Distributions 
67*4d495c6eSApple OSS Distributions 	force_zone_gc();
68*4d495c6eSApple OSS Distributions }
69