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