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