1*663afb9bSAndroid Build Coastguard Worker /*
2*663afb9bSAndroid Build Coastguard Worker * Copyright (c) 2009-2012 Niels Provos and Nick Mathewson
3*663afb9bSAndroid Build Coastguard Worker *
4*663afb9bSAndroid Build Coastguard Worker * Redistribution and use in source and binary forms, with or without
5*663afb9bSAndroid Build Coastguard Worker * modification, are permitted provided that the following conditions
6*663afb9bSAndroid Build Coastguard Worker * are met:
7*663afb9bSAndroid Build Coastguard Worker * 1. Redistributions of source code must retain the above copyright
8*663afb9bSAndroid Build Coastguard Worker * notice, this list of conditions and the following disclaimer.
9*663afb9bSAndroid Build Coastguard Worker * 2. Redistributions in binary form must reproduce the above copyright
10*663afb9bSAndroid Build Coastguard Worker * notice, this list of conditions and the following disclaimer in the
11*663afb9bSAndroid Build Coastguard Worker * documentation and/or other materials provided with the distribution.
12*663afb9bSAndroid Build Coastguard Worker * 3. The name of the author may not be used to endorse or promote products
13*663afb9bSAndroid Build Coastguard Worker * derived from this software without specific prior written permission.
14*663afb9bSAndroid Build Coastguard Worker *
15*663afb9bSAndroid Build Coastguard Worker * THIS SOFTWARE IS PROVIDED BY THE AUTHOR ``AS IS'' AND ANY EXPRESS OR
16*663afb9bSAndroid Build Coastguard Worker * IMPLIED WARRANTIES, INCLUDING, BUT NOT LIMITED TO, THE IMPLIED WARRANTIES
17*663afb9bSAndroid Build Coastguard Worker * OF MERCHANTABILITY AND FITNESS FOR A PARTICULAR PURPOSE ARE DISCLAIMED.
18*663afb9bSAndroid Build Coastguard Worker * IN NO EVENT SHALL THE AUTHOR BE LIABLE FOR ANY DIRECT, INDIRECT,
19*663afb9bSAndroid Build Coastguard Worker * INCIDENTAL, SPECIAL, EXEMPLARY, OR CONSEQUENTIAL DAMAGES (INCLUDING, BUT
20*663afb9bSAndroid Build Coastguard Worker * NOT LIMITED TO, PROCUREMENT OF SUBSTITUTE GOODS OR SERVICES; LOSS OF USE,
21*663afb9bSAndroid Build Coastguard Worker * DATA, OR PROFITS; OR BUSINESS INTERRUPTION) HOWEVER CAUSED AND ON ANY
22*663afb9bSAndroid Build Coastguard Worker * THEORY OF LIABILITY, WHETHER IN CONTRACT, STRICT LIABILITY, OR TORT
23*663afb9bSAndroid Build Coastguard Worker * (INCLUDING NEGLIGENCE OR OTHERWISE) ARISING IN ANY WAY OUT OF THE USE OF
24*663afb9bSAndroid Build Coastguard Worker * THIS SOFTWARE, EVEN IF ADVISED OF THE POSSIBILITY OF SUCH DAMAGE.
25*663afb9bSAndroid Build Coastguard Worker */
26*663afb9bSAndroid Build Coastguard Worker #include "../util-internal.h"
27*663afb9bSAndroid Build Coastguard Worker #include "event2/event-config.h"
28*663afb9bSAndroid Build Coastguard Worker
29*663afb9bSAndroid Build Coastguard Worker #ifdef _WIN32
30*663afb9bSAndroid Build Coastguard Worker #include <winsock2.h>
31*663afb9bSAndroid Build Coastguard Worker #endif
32*663afb9bSAndroid Build Coastguard Worker #include <sys/types.h>
33*663afb9bSAndroid Build Coastguard Worker #include <sys/stat.h>
34*663afb9bSAndroid Build Coastguard Worker #ifdef EVENT__HAVE_SYS_SOCKET_H
35*663afb9bSAndroid Build Coastguard Worker #include <sys/socket.h>
36*663afb9bSAndroid Build Coastguard Worker #endif
37*663afb9bSAndroid Build Coastguard Worker #include <fcntl.h>
38*663afb9bSAndroid Build Coastguard Worker #include <stdlib.h>
39*663afb9bSAndroid Build Coastguard Worker #include <stdio.h>
40*663afb9bSAndroid Build Coastguard Worker #include <string.h>
41*663afb9bSAndroid Build Coastguard Worker #ifndef _WIN32
42*663afb9bSAndroid Build Coastguard Worker #include <sys/time.h>
43*663afb9bSAndroid Build Coastguard Worker #include <unistd.h>
44*663afb9bSAndroid Build Coastguard Worker #endif
45*663afb9bSAndroid Build Coastguard Worker #include <errno.h>
46*663afb9bSAndroid Build Coastguard Worker
47*663afb9bSAndroid Build Coastguard Worker #include "event2/event.h"
48*663afb9bSAndroid Build Coastguard Worker #include "event2/util.h"
49*663afb9bSAndroid Build Coastguard Worker
50*663afb9bSAndroid Build Coastguard Worker #include "regress.h"
51*663afb9bSAndroid Build Coastguard Worker
52*663afb9bSAndroid Build Coastguard Worker static int was_et = 0;
53*663afb9bSAndroid Build Coastguard Worker
base_supports_et(struct event_base * base)54*663afb9bSAndroid Build Coastguard Worker static int base_supports_et(struct event_base *base)
55*663afb9bSAndroid Build Coastguard Worker {
56*663afb9bSAndroid Build Coastguard Worker return
57*663afb9bSAndroid Build Coastguard Worker (!strcmp(event_base_get_method(base), "epoll") ||
58*663afb9bSAndroid Build Coastguard Worker !strcmp(event_base_get_method(base), "epoll (with changelist)") ||
59*663afb9bSAndroid Build Coastguard Worker !strcmp(event_base_get_method(base), "kqueue"));
60*663afb9bSAndroid Build Coastguard Worker }
61*663afb9bSAndroid Build Coastguard Worker
62*663afb9bSAndroid Build Coastguard Worker static void
read_cb(evutil_socket_t fd,short event,void * arg)63*663afb9bSAndroid Build Coastguard Worker read_cb(evutil_socket_t fd, short event, void *arg)
64*663afb9bSAndroid Build Coastguard Worker {
65*663afb9bSAndroid Build Coastguard Worker char buf;
66*663afb9bSAndroid Build Coastguard Worker int len;
67*663afb9bSAndroid Build Coastguard Worker
68*663afb9bSAndroid Build Coastguard Worker len = recv(fd, &buf, sizeof(buf), 0);
69*663afb9bSAndroid Build Coastguard Worker
70*663afb9bSAndroid Build Coastguard Worker called++;
71*663afb9bSAndroid Build Coastguard Worker if (event & EV_ET)
72*663afb9bSAndroid Build Coastguard Worker was_et = 1;
73*663afb9bSAndroid Build Coastguard Worker
74*663afb9bSAndroid Build Coastguard Worker if (!len)
75*663afb9bSAndroid Build Coastguard Worker event_del(arg);
76*663afb9bSAndroid Build Coastguard Worker }
77*663afb9bSAndroid Build Coastguard Worker
78*663afb9bSAndroid Build Coastguard Worker static void
test_edgetriggered(void * data_)79*663afb9bSAndroid Build Coastguard Worker test_edgetriggered(void *data_)
80*663afb9bSAndroid Build Coastguard Worker {
81*663afb9bSAndroid Build Coastguard Worker struct basic_test_data *data = data_;
82*663afb9bSAndroid Build Coastguard Worker struct event_base *base = data->base;
83*663afb9bSAndroid Build Coastguard Worker evutil_socket_t *pair = data->pair;
84*663afb9bSAndroid Build Coastguard Worker struct event *ev = NULL;
85*663afb9bSAndroid Build Coastguard Worker const char *test = "test string";
86*663afb9bSAndroid Build Coastguard Worker int supports_et;
87*663afb9bSAndroid Build Coastguard Worker
88*663afb9bSAndroid Build Coastguard Worker /* On Linux 3.2.1 (at least, as patched by Fedora and tested by Nick),
89*663afb9bSAndroid Build Coastguard Worker * doing a "recv" on an AF_UNIX socket resets the readability of the
90*663afb9bSAndroid Build Coastguard Worker * socket, even though there is no state change, so we don't actually
91*663afb9bSAndroid Build Coastguard Worker * get edge-triggered behavior. Yuck! Linux 3.1.9 didn't have this
92*663afb9bSAndroid Build Coastguard Worker * problem.
93*663afb9bSAndroid Build Coastguard Worker */
94*663afb9bSAndroid Build Coastguard Worker
95*663afb9bSAndroid Build Coastguard Worker called = was_et = 0;
96*663afb9bSAndroid Build Coastguard Worker
97*663afb9bSAndroid Build Coastguard Worker tt_int_op(send(pair[0], test, (int)strlen(test)+1, 0), >, 0);
98*663afb9bSAndroid Build Coastguard Worker tt_int_op(shutdown(pair[0], EVUTIL_SHUT_WR), ==, 0);
99*663afb9bSAndroid Build Coastguard Worker
100*663afb9bSAndroid Build Coastguard Worker supports_et = base_supports_et(base);
101*663afb9bSAndroid Build Coastguard Worker TT_BLATHER(("Checking for edge-triggered events with %s, which should %s"
102*663afb9bSAndroid Build Coastguard Worker "support edge-triggering", event_base_get_method(base),
103*663afb9bSAndroid Build Coastguard Worker supports_et?"":"not "));
104*663afb9bSAndroid Build Coastguard Worker
105*663afb9bSAndroid Build Coastguard Worker /* Initialize one event */
106*663afb9bSAndroid Build Coastguard Worker ev = event_new(base, pair[1], EV_READ|EV_ET|EV_PERSIST, read_cb, &ev);
107*663afb9bSAndroid Build Coastguard Worker tt_assert(ev != NULL);
108*663afb9bSAndroid Build Coastguard Worker tt_int_op(event_add(ev, NULL), ==, 0);
109*663afb9bSAndroid Build Coastguard Worker
110*663afb9bSAndroid Build Coastguard Worker /* We're going to call the dispatch function twice. The first invocation
111*663afb9bSAndroid Build Coastguard Worker * will read a single byte from pair[1] in either case. If we're edge
112*663afb9bSAndroid Build Coastguard Worker * triggered, we'll only see the event once (since we only see transitions
113*663afb9bSAndroid Build Coastguard Worker * from no data to data), so the second invocation of event_base_loop will
114*663afb9bSAndroid Build Coastguard Worker * do nothing. If we're level triggered, the second invocation of
115*663afb9bSAndroid Build Coastguard Worker * event_base_loop will also activate the event (because there's still
116*663afb9bSAndroid Build Coastguard Worker * data to read). */
117*663afb9bSAndroid Build Coastguard Worker tt_int_op(event_base_loop(base,EVLOOP_NONBLOCK|EVLOOP_ONCE), ==, 0);
118*663afb9bSAndroid Build Coastguard Worker tt_int_op(event_base_loop(base,EVLOOP_NONBLOCK|EVLOOP_ONCE), ==, 0);
119*663afb9bSAndroid Build Coastguard Worker
120*663afb9bSAndroid Build Coastguard Worker if (supports_et) {
121*663afb9bSAndroid Build Coastguard Worker tt_int_op(called, ==, 1);
122*663afb9bSAndroid Build Coastguard Worker tt_assert(was_et);
123*663afb9bSAndroid Build Coastguard Worker } else {
124*663afb9bSAndroid Build Coastguard Worker tt_int_op(called, ==, 2);
125*663afb9bSAndroid Build Coastguard Worker tt_assert(!was_et);
126*663afb9bSAndroid Build Coastguard Worker }
127*663afb9bSAndroid Build Coastguard Worker
128*663afb9bSAndroid Build Coastguard Worker end:
129*663afb9bSAndroid Build Coastguard Worker if (ev) {
130*663afb9bSAndroid Build Coastguard Worker event_del(ev);
131*663afb9bSAndroid Build Coastguard Worker event_free(ev);
132*663afb9bSAndroid Build Coastguard Worker }
133*663afb9bSAndroid Build Coastguard Worker }
134*663afb9bSAndroid Build Coastguard Worker
135*663afb9bSAndroid Build Coastguard Worker static void
test_edgetriggered_mix_error(void * data_)136*663afb9bSAndroid Build Coastguard Worker test_edgetriggered_mix_error(void *data_)
137*663afb9bSAndroid Build Coastguard Worker {
138*663afb9bSAndroid Build Coastguard Worker struct basic_test_data *data = data_;
139*663afb9bSAndroid Build Coastguard Worker struct event_base *base = NULL;
140*663afb9bSAndroid Build Coastguard Worker struct event *ev_et=NULL, *ev_lt=NULL;
141*663afb9bSAndroid Build Coastguard Worker
142*663afb9bSAndroid Build Coastguard Worker #ifdef EVENT__DISABLE_DEBUG_MODE
143*663afb9bSAndroid Build Coastguard Worker if (1)
144*663afb9bSAndroid Build Coastguard Worker tt_skip();
145*663afb9bSAndroid Build Coastguard Worker #endif
146*663afb9bSAndroid Build Coastguard Worker
147*663afb9bSAndroid Build Coastguard Worker if (!libevent_tests_running_in_debug_mode)
148*663afb9bSAndroid Build Coastguard Worker event_enable_debug_mode();
149*663afb9bSAndroid Build Coastguard Worker
150*663afb9bSAndroid Build Coastguard Worker base = event_base_new();
151*663afb9bSAndroid Build Coastguard Worker
152*663afb9bSAndroid Build Coastguard Worker /* try mixing edge-triggered and level-triggered to make sure it fails*/
153*663afb9bSAndroid Build Coastguard Worker ev_et = event_new(base, data->pair[0], EV_READ|EV_ET, read_cb, ev_et);
154*663afb9bSAndroid Build Coastguard Worker tt_assert(ev_et);
155*663afb9bSAndroid Build Coastguard Worker ev_lt = event_new(base, data->pair[0], EV_READ, read_cb, ev_lt);
156*663afb9bSAndroid Build Coastguard Worker tt_assert(ev_lt);
157*663afb9bSAndroid Build Coastguard Worker
158*663afb9bSAndroid Build Coastguard Worker /* Add edge-triggered, then level-triggered. Get an error. */
159*663afb9bSAndroid Build Coastguard Worker tt_int_op(0, ==, event_add(ev_et, NULL));
160*663afb9bSAndroid Build Coastguard Worker tt_int_op(-1, ==, event_add(ev_lt, NULL));
161*663afb9bSAndroid Build Coastguard Worker tt_int_op(EV_READ, ==, event_pending(ev_et, EV_READ, NULL));
162*663afb9bSAndroid Build Coastguard Worker tt_int_op(0, ==, event_pending(ev_lt, EV_READ, NULL));
163*663afb9bSAndroid Build Coastguard Worker
164*663afb9bSAndroid Build Coastguard Worker tt_int_op(0, ==, event_del(ev_et));
165*663afb9bSAndroid Build Coastguard Worker /* Add level-triggered, then edge-triggered. Get an error. */
166*663afb9bSAndroid Build Coastguard Worker tt_int_op(0, ==, event_add(ev_lt, NULL));
167*663afb9bSAndroid Build Coastguard Worker tt_int_op(-1, ==, event_add(ev_et, NULL));
168*663afb9bSAndroid Build Coastguard Worker tt_int_op(EV_READ, ==, event_pending(ev_lt, EV_READ, NULL));
169*663afb9bSAndroid Build Coastguard Worker tt_int_op(0, ==, event_pending(ev_et, EV_READ, NULL));
170*663afb9bSAndroid Build Coastguard Worker
171*663afb9bSAndroid Build Coastguard Worker end:
172*663afb9bSAndroid Build Coastguard Worker if (ev_et)
173*663afb9bSAndroid Build Coastguard Worker event_free(ev_et);
174*663afb9bSAndroid Build Coastguard Worker if (ev_lt)
175*663afb9bSAndroid Build Coastguard Worker event_free(ev_lt);
176*663afb9bSAndroid Build Coastguard Worker if (base)
177*663afb9bSAndroid Build Coastguard Worker event_base_free(base);
178*663afb9bSAndroid Build Coastguard Worker }
179*663afb9bSAndroid Build Coastguard Worker
180*663afb9bSAndroid Build Coastguard Worker static int read_notification_count;
181*663afb9bSAndroid Build Coastguard Worker static int last_read_notification_was_et;
182*663afb9bSAndroid Build Coastguard Worker static void
read_notification_cb(evutil_socket_t fd,short event,void * arg)183*663afb9bSAndroid Build Coastguard Worker read_notification_cb(evutil_socket_t fd, short event, void *arg)
184*663afb9bSAndroid Build Coastguard Worker {
185*663afb9bSAndroid Build Coastguard Worker read_notification_count++;
186*663afb9bSAndroid Build Coastguard Worker last_read_notification_was_et = (event & EV_ET);
187*663afb9bSAndroid Build Coastguard Worker }
188*663afb9bSAndroid Build Coastguard Worker
189*663afb9bSAndroid Build Coastguard Worker static int write_notification_count;
190*663afb9bSAndroid Build Coastguard Worker static int last_write_notification_was_et;
191*663afb9bSAndroid Build Coastguard Worker static void
write_notification_cb(evutil_socket_t fd,short event,void * arg)192*663afb9bSAndroid Build Coastguard Worker write_notification_cb(evutil_socket_t fd, short event, void *arg)
193*663afb9bSAndroid Build Coastguard Worker {
194*663afb9bSAndroid Build Coastguard Worker write_notification_count++;
195*663afb9bSAndroid Build Coastguard Worker last_write_notification_was_et = (event & EV_ET);
196*663afb9bSAndroid Build Coastguard Worker }
197*663afb9bSAndroid Build Coastguard Worker
198*663afb9bSAndroid Build Coastguard Worker /* After two or more events have been registered for the same
199*663afb9bSAndroid Build Coastguard Worker * file descriptor using EV_ET, if one of the events is
200*663afb9bSAndroid Build Coastguard Worker * deleted, then the epoll_ctl() call issued by libevent drops
201*663afb9bSAndroid Build Coastguard Worker * the EPOLLET flag resulting in level triggered
202*663afb9bSAndroid Build Coastguard Worker * notifications.
203*663afb9bSAndroid Build Coastguard Worker */
204*663afb9bSAndroid Build Coastguard Worker static void
test_edge_triggered_multiple_events(void * data_)205*663afb9bSAndroid Build Coastguard Worker test_edge_triggered_multiple_events(void *data_)
206*663afb9bSAndroid Build Coastguard Worker {
207*663afb9bSAndroid Build Coastguard Worker struct basic_test_data *data = data_;
208*663afb9bSAndroid Build Coastguard Worker struct event *read_ev = NULL;
209*663afb9bSAndroid Build Coastguard Worker struct event *write_ev = NULL;
210*663afb9bSAndroid Build Coastguard Worker const char c = 'A';
211*663afb9bSAndroid Build Coastguard Worker struct event_base *base = data->base;
212*663afb9bSAndroid Build Coastguard Worker evutil_socket_t *pair = data->pair;
213*663afb9bSAndroid Build Coastguard Worker
214*663afb9bSAndroid Build Coastguard Worker if (!base_supports_et(base)) {
215*663afb9bSAndroid Build Coastguard Worker tt_skip();
216*663afb9bSAndroid Build Coastguard Worker return;
217*663afb9bSAndroid Build Coastguard Worker }
218*663afb9bSAndroid Build Coastguard Worker
219*663afb9bSAndroid Build Coastguard Worker read_notification_count = 0;
220*663afb9bSAndroid Build Coastguard Worker last_read_notification_was_et = 0;
221*663afb9bSAndroid Build Coastguard Worker write_notification_count = 0;
222*663afb9bSAndroid Build Coastguard Worker last_write_notification_was_et = 0;
223*663afb9bSAndroid Build Coastguard Worker
224*663afb9bSAndroid Build Coastguard Worker /* Make pair[1] readable */
225*663afb9bSAndroid Build Coastguard Worker tt_int_op(send(pair[0], &c, 1, 0), >, 0);
226*663afb9bSAndroid Build Coastguard Worker
227*663afb9bSAndroid Build Coastguard Worker read_ev = event_new(base, pair[1], EV_READ|EV_ET|EV_PERSIST,
228*663afb9bSAndroid Build Coastguard Worker read_notification_cb, NULL);
229*663afb9bSAndroid Build Coastguard Worker write_ev = event_new(base, pair[1], EV_WRITE|EV_ET|EV_PERSIST,
230*663afb9bSAndroid Build Coastguard Worker write_notification_cb, NULL);
231*663afb9bSAndroid Build Coastguard Worker
232*663afb9bSAndroid Build Coastguard Worker event_add(read_ev, NULL);
233*663afb9bSAndroid Build Coastguard Worker event_add(write_ev, NULL);
234*663afb9bSAndroid Build Coastguard Worker event_base_loop(base, EVLOOP_NONBLOCK|EVLOOP_ONCE);
235*663afb9bSAndroid Build Coastguard Worker event_base_loop(base, EVLOOP_NONBLOCK|EVLOOP_ONCE);
236*663afb9bSAndroid Build Coastguard Worker
237*663afb9bSAndroid Build Coastguard Worker tt_assert(last_read_notification_was_et);
238*663afb9bSAndroid Build Coastguard Worker tt_int_op(read_notification_count, ==, 1);
239*663afb9bSAndroid Build Coastguard Worker tt_assert(last_write_notification_was_et);
240*663afb9bSAndroid Build Coastguard Worker tt_int_op(write_notification_count, ==, 1);
241*663afb9bSAndroid Build Coastguard Worker
242*663afb9bSAndroid Build Coastguard Worker event_del(read_ev);
243*663afb9bSAndroid Build Coastguard Worker
244*663afb9bSAndroid Build Coastguard Worker /* trigger acitivity second time for the backend that can have multiple
245*663afb9bSAndroid Build Coastguard Worker * events for one fd (like kqueue) */
246*663afb9bSAndroid Build Coastguard Worker close(pair[0]);
247*663afb9bSAndroid Build Coastguard Worker pair[0] = -1;
248*663afb9bSAndroid Build Coastguard Worker
249*663afb9bSAndroid Build Coastguard Worker /* Verify that we are still edge-triggered for write notifications */
250*663afb9bSAndroid Build Coastguard Worker event_base_loop(base, EVLOOP_NONBLOCK|EVLOOP_ONCE);
251*663afb9bSAndroid Build Coastguard Worker event_base_loop(base, EVLOOP_NONBLOCK|EVLOOP_ONCE);
252*663afb9bSAndroid Build Coastguard Worker tt_assert(last_write_notification_was_et);
253*663afb9bSAndroid Build Coastguard Worker tt_int_op(write_notification_count, ==, 2);
254*663afb9bSAndroid Build Coastguard Worker
255*663afb9bSAndroid Build Coastguard Worker end:
256*663afb9bSAndroid Build Coastguard Worker if (read_ev)
257*663afb9bSAndroid Build Coastguard Worker event_free(read_ev);
258*663afb9bSAndroid Build Coastguard Worker if (write_ev)
259*663afb9bSAndroid Build Coastguard Worker event_free(write_ev);
260*663afb9bSAndroid Build Coastguard Worker }
261*663afb9bSAndroid Build Coastguard Worker
262*663afb9bSAndroid Build Coastguard Worker struct testcase_t edgetriggered_testcases[] = {
263*663afb9bSAndroid Build Coastguard Worker { "et", test_edgetriggered,
264*663afb9bSAndroid Build Coastguard Worker TT_FORK|TT_NEED_BASE|TT_NEED_SOCKETPAIR, &basic_setup, NULL },
265*663afb9bSAndroid Build Coastguard Worker { "et_mix_error", test_edgetriggered_mix_error,
266*663afb9bSAndroid Build Coastguard Worker TT_FORK|TT_NEED_SOCKETPAIR|TT_NO_LOGS, &basic_setup, NULL },
267*663afb9bSAndroid Build Coastguard Worker { "et_multiple_events", test_edge_triggered_multiple_events,
268*663afb9bSAndroid Build Coastguard Worker TT_FORK|TT_NEED_BASE|TT_NEED_SOCKETPAIR, &basic_setup, NULL },
269*663afb9bSAndroid Build Coastguard Worker END_OF_TESTCASES
270*663afb9bSAndroid Build Coastguard Worker };
271