xref: /xnu-11417.121.6/tests/kqueue_in_select.c (revision a1e26a70f38d1d7daa7b49b258e2f8538ad81650)
1*a1e26a70SApple OSS Distributions #include <unistd.h>
2*a1e26a70SApple OSS Distributions #include <errno.h>
3*a1e26a70SApple OSS Distributions #include <sys/event.h>
4*a1e26a70SApple OSS Distributions #include <sys/select.h>
5*a1e26a70SApple OSS Distributions #include <Block.h>
6*a1e26a70SApple OSS Distributions #include <darwintest.h>
7*a1e26a70SApple OSS Distributions 
8*a1e26a70SApple OSS Distributions T_GLOBAL_META(
9*a1e26a70SApple OSS Distributions 	T_META_NAMESPACE("xnu.kevent"),
10*a1e26a70SApple OSS Distributions 	T_META_RADAR_COMPONENT_NAME("xnu"),
11*a1e26a70SApple OSS Distributions 	T_META_RADAR_COMPONENT_VERSION("kevent"),
12*a1e26a70SApple OSS Distributions 	T_META_RUN_CONCURRENTLY(true)
13*a1e26a70SApple OSS Distributions 	);
14*a1e26a70SApple OSS Distributions 
15*a1e26a70SApple OSS Distributions #define TV(s) (struct timeval){ .tv_sec = s }
16*a1e26a70SApple OSS Distributions 
17*a1e26a70SApple OSS Distributions static void *
pthread_async_do(void * arg)18*a1e26a70SApple OSS Distributions pthread_async_do(void *arg)
19*a1e26a70SApple OSS Distributions {
20*a1e26a70SApple OSS Distributions 	void (^block)(void) = arg;
21*a1e26a70SApple OSS Distributions 	block();
22*a1e26a70SApple OSS Distributions 	Block_release(block);
23*a1e26a70SApple OSS Distributions 	pthread_detach(pthread_self());
24*a1e26a70SApple OSS Distributions 	return NULL;
25*a1e26a70SApple OSS Distributions }
26*a1e26a70SApple OSS Distributions 
27*a1e26a70SApple OSS Distributions static void
28*a1e26a70SApple OSS Distributions pthread_async(void (^block)(void))
29*a1e26a70SApple OSS Distributions {
30*a1e26a70SApple OSS Distributions 	pthread_t th;
31*a1e26a70SApple OSS Distributions 	int rc;
32*a1e26a70SApple OSS Distributions 
33*a1e26a70SApple OSS Distributions 	rc = pthread_create(&th, NULL, pthread_async_do, Block_copy(block));
34*a1e26a70SApple OSS Distributions 	T_QUIET; T_ASSERT_POSIX_SUCCESS(rc, "pthread_create");
35*a1e26a70SApple OSS Distributions }
36*a1e26a70SApple OSS Distributions 
37*a1e26a70SApple OSS Distributions T_DECL(kqueue_in_select, "make sure kqueue in select works", T_META_TAG_VM_PREFERRED)
38*a1e26a70SApple OSS Distributions {
39*a1e26a70SApple OSS Distributions 	fd_set rd_set;
40*a1e26a70SApple OSS Distributions 	int kq_fd, ret, nfd;
41*a1e26a70SApple OSS Distributions 	struct kevent ret_kev;
42*a1e26a70SApple OSS Distributions 	const struct kevent kev = {
43*a1e26a70SApple OSS Distributions 		.ident = 1,
44*a1e26a70SApple OSS Distributions 		.filter = EVFILT_USER,
45*a1e26a70SApple OSS Distributions 		.flags = EV_ADD | EV_CLEAR,
46*a1e26a70SApple OSS Distributions 	};
47*a1e26a70SApple OSS Distributions 
48*a1e26a70SApple OSS Distributions 	T_ASSERT_POSIX_SUCCESS((kq_fd = kqueue()), NULL);
49*a1e26a70SApple OSS Distributions 	ret = kevent(kq_fd, &kev, 1, NULL, 0, NULL);
50*a1e26a70SApple OSS Distributions 	T_ASSERT_POSIX_SUCCESS(ret, "kevent");
51*a1e26a70SApple OSS Distributions 
52*a1e26a70SApple OSS Distributions 	FD_ZERO(&rd_set);
53*a1e26a70SApple OSS Distributions 	FD_SET(kq_fd, &rd_set);
54*a1e26a70SApple OSS Distributions 	nfd = select(kq_fd + 1, &rd_set, NULL, NULL, &TV(1));
55*a1e26a70SApple OSS Distributions 	T_EXPECT_EQ(nfd, 0, "no trigger");
56*a1e26a70SApple OSS Distributions 
57*a1e26a70SApple OSS Distributions 	pthread_async(^{
58*a1e26a70SApple OSS Distributions 		sleep(1);
59*a1e26a70SApple OSS Distributions 		const struct kevent k = {
60*a1e26a70SApple OSS Distributions 		        .ident = 1,
61*a1e26a70SApple OSS Distributions 		        .filter = EVFILT_USER,
62*a1e26a70SApple OSS Distributions 		        .flags = EV_ADD | EV_CLEAR,
63*a1e26a70SApple OSS Distributions 		        .fflags = NOTE_TRIGGER,
64*a1e26a70SApple OSS Distributions 		};
65*a1e26a70SApple OSS Distributions 
66*a1e26a70SApple OSS Distributions 		T_ASSERT_POSIX_SUCCESS(kevent(kq_fd, &k, 1, NULL, 0, NULL), "trigger");
67*a1e26a70SApple OSS Distributions 	});
68*a1e26a70SApple OSS Distributions 
69*a1e26a70SApple OSS Distributions 	FD_ZERO(&rd_set);
70*a1e26a70SApple OSS Distributions 	FD_SET(kq_fd, &rd_set);
71*a1e26a70SApple OSS Distributions 	nfd = select(kq_fd + 1, &rd_set, NULL, NULL, &TV(5));
72*a1e26a70SApple OSS Distributions 	T_EXPECT_EQ(nfd, 1, "kqueue triggered");
73*a1e26a70SApple OSS Distributions 
74*a1e26a70SApple OSS Distributions 	FD_ZERO(&rd_set);
75*a1e26a70SApple OSS Distributions 	FD_SET(kq_fd, &rd_set);
76*a1e26a70SApple OSS Distributions 	nfd = select(kq_fd + 1, &rd_set, NULL, NULL, &TV(1));
77*a1e26a70SApple OSS Distributions 	T_EXPECT_EQ(nfd, 1, "kqueue is still triggered");
78*a1e26a70SApple OSS Distributions 
79*a1e26a70SApple OSS Distributions 	T_EXPECT_EQ(kevent(kq_fd, NULL, 0, &ret_kev, 1, NULL), 1, "pump event");
80*a1e26a70SApple OSS Distributions 
81*a1e26a70SApple OSS Distributions 	FD_ZERO(&rd_set);
82*a1e26a70SApple OSS Distributions 	FD_SET(kq_fd, &rd_set);
83*a1e26a70SApple OSS Distributions 	nfd = select(kq_fd + 1, &rd_set, NULL, NULL, &TV(1));
84*a1e26a70SApple OSS Distributions 	T_EXPECT_EQ(nfd, 0, "no trigger");
85*a1e26a70SApple OSS Distributions }
86