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