1*699cd480SApple OSS Distributions #include <darwintest.h>
2*699cd480SApple OSS Distributions #include <arpa/inet.h>
3*699cd480SApple OSS Distributions #include <sys/types.h>
4*699cd480SApple OSS Distributions #include <sys/event.h>
5*699cd480SApple OSS Distributions #include <sys/time.h>
6*699cd480SApple OSS Distributions #include <sys/socket.h>
7*699cd480SApple OSS Distributions #include <netinet/in.h>
8*699cd480SApple OSS Distributions #include <string.h>
9*699cd480SApple OSS Distributions #include <stdio.h>
10*699cd480SApple OSS Distributions #include <unistd.h>
11*699cd480SApple OSS Distributions
12*699cd480SApple OSS Distributions T_GLOBAL_META(T_META_RUN_CONCURRENTLY(true));
13*699cd480SApple OSS Distributions
14*699cd480SApple OSS Distributions #define TEST_ADDR "127.0.0.1"
15*699cd480SApple OSS Distributions #define TEST_PORT 4242
16*699cd480SApple OSS Distributions
17*699cd480SApple OSS Distributions static struct {
18*699cd480SApple OSS Distributions int fd;
19*699cd480SApple OSS Distributions struct sockaddr_in addr;
20*699cd480SApple OSS Distributions } server;
21*699cd480SApple OSS Distributions
22*699cd480SApple OSS Distributions static void
server_listen(void)23*699cd480SApple OSS Distributions server_listen(void)
24*699cd480SApple OSS Distributions {
25*699cd480SApple OSS Distributions int r;
26*699cd480SApple OSS Distributions
27*699cd480SApple OSS Distributions server.fd = socket(AF_INET, SOCK_DGRAM, IPPROTO_UDP);
28*699cd480SApple OSS Distributions T_ASSERT_POSIX_SUCCESS(server.fd, "socket");
29*699cd480SApple OSS Distributions
30*699cd480SApple OSS Distributions memset(&server.addr, 0, sizeof(server.addr));
31*699cd480SApple OSS Distributions server.addr.sin_family = AF_INET;
32*699cd480SApple OSS Distributions server.addr.sin_port = htons(TEST_PORT);
33*699cd480SApple OSS Distributions
34*699cd480SApple OSS Distributions inet_pton(AF_INET, TEST_ADDR, &server.addr.sin_addr);
35*699cd480SApple OSS Distributions
36*699cd480SApple OSS Distributions r = bind(server.fd, (struct sockaddr*) &server.addr, sizeof(server.addr));
37*699cd480SApple OSS Distributions T_ASSERT_POSIX_SUCCESS(r, "bind");
38*699cd480SApple OSS Distributions }
39*699cd480SApple OSS Distributions
40*699cd480SApple OSS Distributions static void
send_message(void)41*699cd480SApple OSS Distributions send_message(void)
42*699cd480SApple OSS Distributions {
43*699cd480SApple OSS Distributions int fd;
44*699cd480SApple OSS Distributions struct msghdr msg;
45*699cd480SApple OSS Distributions struct iovec iov;
46*699cd480SApple OSS Distributions
47*699cd480SApple OSS Distributions fd = socket(AF_INET, SOCK_DGRAM, IPPROTO_UDP);
48*699cd480SApple OSS Distributions T_ASSERT_POSIX_SUCCESS(fd, "socket");
49*699cd480SApple OSS Distributions
50*699cd480SApple OSS Distributions memset(&msg, 0, sizeof(msg));
51*699cd480SApple OSS Distributions
52*699cd480SApple OSS Distributions msg.msg_name = &server.addr;
53*699cd480SApple OSS Distributions msg.msg_namelen = sizeof(server.addr);
54*699cd480SApple OSS Distributions
55*699cd480SApple OSS Distributions iov.iov_base = "";
56*699cd480SApple OSS Distributions iov.iov_len = 0;
57*699cd480SApple OSS Distributions msg.msg_iov = &iov;
58*699cd480SApple OSS Distributions msg.msg_iovlen = 1;
59*699cd480SApple OSS Distributions
60*699cd480SApple OSS Distributions ssize_t r = sendmsg(fd, &msg, 0);
61*699cd480SApple OSS Distributions T_ASSERT_EQ(r, (ssize_t)iov.iov_len, "sendmsg");
62*699cd480SApple OSS Distributions
63*699cd480SApple OSS Distributions close(fd);
64*699cd480SApple OSS Distributions }
65*699cd480SApple OSS Distributions
66*699cd480SApple OSS Distributions static void
server_poll(void)67*699cd480SApple OSS Distributions server_poll(void)
68*699cd480SApple OSS Distributions {
69*699cd480SApple OSS Distributions int kq;
70*699cd480SApple OSS Distributions struct kevent event = {
71*699cd480SApple OSS Distributions .flags = EV_ADD,
72*699cd480SApple OSS Distributions .filter = EVFILT_READ,
73*699cd480SApple OSS Distributions .ident = (unsigned long)server.fd,
74*699cd480SApple OSS Distributions };
75*699cd480SApple OSS Distributions int r;
76*699cd480SApple OSS Distributions
77*699cd480SApple OSS Distributions kq = kqueue();
78*699cd480SApple OSS Distributions T_ASSERT_POSIX_SUCCESS(kq, "kqueue");
79*699cd480SApple OSS Distributions
80*699cd480SApple OSS Distributions /* Add and poll */
81*699cd480SApple OSS Distributions r = kevent(kq, &event, 1, &event, 1, NULL);
82*699cd480SApple OSS Distributions T_EXPECT_EQ(r, 1, "should return an event");
83*699cd480SApple OSS Distributions
84*699cd480SApple OSS Distributions close(kq);
85*699cd480SApple OSS Distributions }
86*699cd480SApple OSS Distributions
87*699cd480SApple OSS Distributions T_DECL(socket_0byte_udp_poll_58140856,
88*699cd480SApple OSS Distributions "Tests that 0-sized UDP packets wake up kevent")
89*699cd480SApple OSS Distributions {
90*699cd480SApple OSS Distributions T_LOG("Starting...\n");
91*699cd480SApple OSS Distributions
92*699cd480SApple OSS Distributions /* Listen on UDP port */
93*699cd480SApple OSS Distributions server_listen();
94*699cd480SApple OSS Distributions
95*699cd480SApple OSS Distributions T_LOG("Server bound to [%s]:%d\n", TEST_ADDR, TEST_PORT);
96*699cd480SApple OSS Distributions
97*699cd480SApple OSS Distributions /* Send 0-UDP packet to that port */
98*699cd480SApple OSS Distributions send_message();
99*699cd480SApple OSS Distributions
100*699cd480SApple OSS Distributions T_LOG("Sent message to server\n");
101*699cd480SApple OSS Distributions
102*699cd480SApple OSS Distributions /* Poll kqueue events */
103*699cd480SApple OSS Distributions server_poll();
104*699cd480SApple OSS Distributions
105*699cd480SApple OSS Distributions T_LOG("Got kqueue event\n");
106*699cd480SApple OSS Distributions
107*699cd480SApple OSS Distributions close(server.fd);
108*699cd480SApple OSS Distributions }
109