xref: /xnu-10002.61.3/tests/mcast_group_race_82820812.c (revision 0f4c859e951fba394238ab619495c4e1d54d0f34)
1*0f4c859eSApple OSS Distributions #include <arpa/inet.h>
2*0f4c859eSApple OSS Distributions #include <pthread.h>
3*0f4c859eSApple OSS Distributions #include <unistd.h>
4*0f4c859eSApple OSS Distributions #include <darwintest.h>
5*0f4c859eSApple OSS Distributions #include <TargetConditionals.h>
6*0f4c859eSApple OSS Distributions 
7*0f4c859eSApple OSS Distributions volatile static int lock_a;
8*0f4c859eSApple OSS Distributions volatile static int lock_b;
9*0f4c859eSApple OSS Distributions 
10*0f4c859eSApple OSS Distributions static int fd;
11*0f4c859eSApple OSS Distributions static struct sockaddr_in saddr;
12*0f4c859eSApple OSS Distributions 
13*0f4c859eSApple OSS Distributions static struct ip_mreq filler_group;
14*0f4c859eSApple OSS Distributions static struct ip_mreq group_a;
15*0f4c859eSApple OSS Distributions static struct ip_mreq group_b;
16*0f4c859eSApple OSS Distributions 
17*0f4c859eSApple OSS Distributions #define ITERATIONS_LIMIT 1000
18*0f4c859eSApple OSS Distributions 
19*0f4c859eSApple OSS Distributions static void *
thread_func(__unused void * arg)20*0f4c859eSApple OSS Distributions thread_func(__unused void* arg)
21*0f4c859eSApple OSS Distributions {
22*0f4c859eSApple OSS Distributions 	lock_a = 1;
23*0f4c859eSApple OSS Distributions 	while (lock_b == 0) {
24*0f4c859eSApple OSS Distributions 	}
25*0f4c859eSApple OSS Distributions 
26*0f4c859eSApple OSS Distributions 	setsockopt(fd, IPPROTO_IP, IP_ADD_MEMBERSHIP, &group_a, sizeof(group_a));
27*0f4c859eSApple OSS Distributions 
28*0f4c859eSApple OSS Distributions 	return NULL;
29*0f4c859eSApple OSS Distributions }
30*0f4c859eSApple OSS Distributions 
31*0f4c859eSApple OSS Distributions T_DECL(mcast_group_race_82820812, "Race between multicast group join operations.",
32*0f4c859eSApple OSS Distributions     T_META_ASROOT(true),
33*0f4c859eSApple OSS Distributions     T_META_ENABLED(!TARGET_OS_BRIDGE && !TARGET_OS_SIMULATOR))
34*0f4c859eSApple OSS Distributions {
35*0f4c859eSApple OSS Distributions 	pthread_t th;
36*0f4c859eSApple OSS Distributions 	uint32_t i = 0;
37*0f4c859eSApple OSS Distributions 	uint32_t j = 0;
38*0f4c859eSApple OSS Distributions 
39*0f4c859eSApple OSS Distributions 	saddr.sin_family = AF_INET;
40*0f4c859eSApple OSS Distributions 
41*0f4c859eSApple OSS Distributions 	group_a.imr_multiaddr.s_addr = inet_addr("224.0.0.1");
42*0f4c859eSApple OSS Distributions 	group_b.imr_multiaddr.s_addr = inet_addr("224.0.0.2");
43*0f4c859eSApple OSS Distributions 
44*0f4c859eSApple OSS Distributions 	for (i = 0; i < ITERATIONS_LIMIT; ++i) {
45*0f4c859eSApple OSS Distributions 		T_ASSERT_POSIX_SUCCESS(fd = socket(AF_INET, SOCK_DGRAM, 0), "socket");
46*0f4c859eSApple OSS Distributions 		T_ASSERT_POSIX_SUCCESS(bind(fd, (struct sockaddr *) &saddr, sizeof(saddr)), "bind");
47*0f4c859eSApple OSS Distributions 
48*0f4c859eSApple OSS Distributions 		for (j = 0; j < IP_MIN_MEMBERSHIPS - 1; ++j) {
49*0f4c859eSApple OSS Distributions 			filler_group.imr_multiaddr.s_addr = htonl(ntohl(inet_addr("224.0.0.3")) + j);
50*0f4c859eSApple OSS Distributions 			setsockopt(fd, IPPROTO_IP, IP_ADD_MEMBERSHIP, &filler_group, sizeof(filler_group));
51*0f4c859eSApple OSS Distributions 		}
52*0f4c859eSApple OSS Distributions 
53*0f4c859eSApple OSS Distributions 		T_ASSERT_POSIX_ZERO(pthread_create(&th, NULL, thread_func, NULL), "pthread_create");
54*0f4c859eSApple OSS Distributions 
55*0f4c859eSApple OSS Distributions 		while (lock_a == 0) {
56*0f4c859eSApple OSS Distributions 		}
57*0f4c859eSApple OSS Distributions 		lock_b = 1;
58*0f4c859eSApple OSS Distributions 
59*0f4c859eSApple OSS Distributions 		setsockopt(fd, IPPROTO_IP, IP_ADD_MEMBERSHIP, &group_b, sizeof(group_b));
60*0f4c859eSApple OSS Distributions 
61*0f4c859eSApple OSS Distributions 		T_ASSERT_POSIX_ZERO(pthread_join(th, NULL), "pthread_join");
62*0f4c859eSApple OSS Distributions 		T_ASSERT_POSIX_SUCCESS(close(fd), "close");
63*0f4c859eSApple OSS Distributions 	}
64*0f4c859eSApple OSS Distributions }
65