]>
git.proxmox.com Git - systemd.git/blob - src/libsystemd/sd-event/test-event.c
1 /*-*- Mode: C; c-basic-offset: 8; indent-tabs-mode: nil -*-*/
4 This file is part of systemd.
6 Copyright 2013 Lennart Poettering
8 systemd is free software; you can redistribute it and/or modify it
9 under the terms of the GNU Lesser General Public License as published by
10 the Free Software Foundation; either version 2.1 of the License, or
11 (at your option) any later version.
13 systemd is distributed in the hope that it will be useful, but
14 WITHOUT ANY WARRANTY; without even the implied warranty of
15 MERCHANTABILITY or FITNESS FOR A PARTICULAR PURPOSE. See the GNU
16 Lesser General Public License for more details.
18 You should have received a copy of the GNU Lesser General Public License
19 along with systemd; If not, see <http://www.gnu.org/licenses/>.
26 #include "signal-util.h"
28 static int prepare_handler(sd_event_source
*s
, void *userdata
) {
29 log_info("preparing %c", PTR_TO_INT(userdata
));
33 static bool got_a
, got_b
, got_c
, got_unref
;
34 static unsigned got_d
;
36 static int unref_handler(sd_event_source
*s
, int fd
, uint32_t revents
, void *userdata
) {
37 sd_event_source_unref(s
);
42 static int io_handler(sd_event_source
*s
, int fd
, uint32_t revents
, void *userdata
) {
44 log_info("got IO on %c", PTR_TO_INT(userdata
));
46 if (userdata
== INT_TO_PTR('a')) {
47 assert_se(sd_event_source_set_enabled(s
, SD_EVENT_OFF
) >= 0);
50 } else if (userdata
== INT_TO_PTR('b')) {
53 } else if (userdata
== INT_TO_PTR('d')) {
56 assert_se(sd_event_source_set_enabled(s
, SD_EVENT_ONESHOT
) >= 0);
58 assert_se(sd_event_source_set_enabled(s
, SD_EVENT_OFF
) >= 0);
60 assert_not_reached("Yuck!");
65 static int child_handler(sd_event_source
*s
, const siginfo_t
*si
, void *userdata
) {
70 log_info("got child on %c", PTR_TO_INT(userdata
));
72 assert_se(userdata
== INT_TO_PTR('f'));
74 assert_se(sd_event_exit(sd_event_source_get_event(s
), 0) >= 0);
75 sd_event_source_unref(s
);
80 static int signal_handler(sd_event_source
*s
, const struct signalfd_siginfo
*si
, void *userdata
) {
81 sd_event_source
*p
= NULL
;
87 log_info("got signal on %c", PTR_TO_INT(userdata
));
89 assert_se(userdata
== INT_TO_PTR('e'));
91 assert_se(sigprocmask_many(SIG_BLOCK
, NULL
, SIGCHLD
, -1) >= 0);
99 assert_se(sd_event_add_child(sd_event_source_get_event(s
), &p
, pid
, WEXITED
, child_handler
, INT_TO_PTR('f')) >= 0);
100 assert_se(sd_event_source_set_enabled(p
, SD_EVENT_ONESHOT
) >= 0);
102 sd_event_source_unref(s
);
107 static int defer_handler(sd_event_source
*s
, void *userdata
) {
108 sd_event_source
*p
= NULL
;
112 log_info("got defer on %c", PTR_TO_INT(userdata
));
114 assert_se(userdata
== INT_TO_PTR('d'));
116 assert_se(sigprocmask_many(SIG_BLOCK
, NULL
, SIGUSR1
, -1) >= 0);
118 assert_se(sd_event_add_signal(sd_event_source_get_event(s
), &p
, SIGUSR1
, signal_handler
, INT_TO_PTR('e')) >= 0);
119 assert_se(sd_event_source_set_enabled(p
, SD_EVENT_ONESHOT
) >= 0);
122 sd_event_source_unref(s
);
127 static bool do_quit
= false;
129 static int time_handler(sd_event_source
*s
, uint64_t usec
, void *userdata
) {
130 log_info("got timer on %c", PTR_TO_INT(userdata
));
132 if (userdata
== INT_TO_PTR('c')) {
137 assert_se(sd_event_add_defer(sd_event_source_get_event(s
), &p
, defer_handler
, INT_TO_PTR('d')) >= 0);
138 assert_se(sd_event_source_set_enabled(p
, SD_EVENT_ONESHOT
) >= 0);
144 assert_not_reached("Huh?");
149 static bool got_exit
= false;
151 static int exit_handler(sd_event_source
*s
, void *userdata
) {
152 log_info("got quit handler on %c", PTR_TO_INT(userdata
));
159 static void test_basic(void) {
161 sd_event_source
*w
= NULL
, *x
= NULL
, *y
= NULL
, *z
= NULL
, *q
= NULL
, *t
= NULL
;
162 static const char ch
= 'x';
163 int a
[2] = { -1, -1 }, b
[2] = { -1, -1}, d
[2] = { -1, -1}, k
[2] = { -1, -1 };
165 assert_se(pipe(a
) >= 0);
166 assert_se(pipe(b
) >= 0);
167 assert_se(pipe(d
) >= 0);
168 assert_se(pipe(k
) >= 0);
170 assert_se(sd_event_default(&e
) >= 0);
172 assert_se(sd_event_set_watchdog(e
, true) >= 0);
174 /* Test whether we cleanly can destroy an io event source from its own handler */
176 assert_se(sd_event_add_io(e
, &t
, k
[0], EPOLLIN
, unref_handler
, NULL
) >= 0);
177 assert_se(write(k
[1], &ch
, 1) == 1);
178 assert_se(sd_event_run(e
, (uint64_t) -1) >= 1);
179 assert_se(got_unref
);
181 got_a
= false, got_b
= false, got_c
= false, got_d
= 0;
183 /* Add a oneshot handler, trigger it, re-enable it, and trigger
185 assert_se(sd_event_add_io(e
, &w
, d
[0], EPOLLIN
, io_handler
, INT_TO_PTR('d')) >= 0);
186 assert_se(sd_event_source_set_enabled(w
, SD_EVENT_ONESHOT
) >= 0);
187 assert_se(write(d
[1], &ch
, 1) >= 0);
188 assert_se(sd_event_run(e
, (uint64_t) -1) >= 1);
189 assert_se(got_d
== 1);
190 assert_se(write(d
[1], &ch
, 1) >= 0);
191 assert_se(sd_event_run(e
, (uint64_t) -1) >= 1);
192 assert_se(got_d
== 2);
194 assert_se(sd_event_add_io(e
, &x
, a
[0], EPOLLIN
, io_handler
, INT_TO_PTR('a')) >= 0);
195 assert_se(sd_event_add_io(e
, &y
, b
[0], EPOLLIN
, io_handler
, INT_TO_PTR('b')) >= 0);
196 assert_se(sd_event_add_time(e
, &z
, CLOCK_MONOTONIC
, 0, 0, time_handler
, INT_TO_PTR('c')) >= 0);
197 assert_se(sd_event_add_exit(e
, &q
, exit_handler
, INT_TO_PTR('g')) >= 0);
199 assert_se(sd_event_source_set_priority(x
, 99) >= 0);
200 assert_se(sd_event_source_set_enabled(y
, SD_EVENT_ONESHOT
) >= 0);
201 assert_se(sd_event_source_set_prepare(x
, prepare_handler
) >= 0);
202 assert_se(sd_event_source_set_priority(z
, 50) >= 0);
203 assert_se(sd_event_source_set_enabled(z
, SD_EVENT_ONESHOT
) >= 0);
204 assert_se(sd_event_source_set_prepare(z
, prepare_handler
) >= 0);
206 /* Test for floating event sources */
207 assert_se(sigprocmask_many(SIG_BLOCK
, NULL
, SIGRTMIN
+1, -1) >= 0);
208 assert_se(sd_event_add_signal(e
, NULL
, SIGRTMIN
+1, NULL
, NULL
) >= 0);
210 assert_se(write(a
[1], &ch
, 1) >= 0);
211 assert_se(write(b
[1], &ch
, 1) >= 0);
213 assert_se(!got_a
&& !got_b
&& !got_c
);
215 assert_se(sd_event_run(e
, (uint64_t) -1) >= 1);
217 assert_se(!got_a
&& got_b
&& !got_c
);
219 assert_se(sd_event_run(e
, (uint64_t) -1) >= 1);
221 assert_se(!got_a
&& got_b
&& got_c
);
223 assert_se(sd_event_run(e
, (uint64_t) -1) >= 1);
225 assert_se(got_a
&& got_b
&& got_c
);
227 sd_event_source_unref(x
);
228 sd_event_source_unref(y
);
231 assert_se(sd_event_source_set_time(z
, now(CLOCK_MONOTONIC
) + 200 * USEC_PER_MSEC
) >= 0);
232 assert_se(sd_event_source_set_enabled(z
, SD_EVENT_ONESHOT
) >= 0);
234 assert_se(sd_event_loop(e
) >= 0);
236 sd_event_source_unref(z
);
237 sd_event_source_unref(q
);
239 sd_event_source_unref(w
);
249 static int last_rtqueue_sigval
= 0;
250 static int n_rtqueue
= 0;
252 static int rtqueue_handler(sd_event_source
*s
, const struct signalfd_siginfo
*si
, void *userdata
) {
253 last_rtqueue_sigval
= si
->ssi_int
;
258 static void test_rtqueue(void) {
259 sd_event_source
*u
= NULL
, *v
= NULL
, *s
= NULL
;
262 assert_se(sd_event_default(&e
) >= 0);
264 assert_se(sigprocmask_many(SIG_BLOCK
, NULL
, SIGRTMIN
+2, SIGRTMIN
+3, SIGUSR2
, -1) >= 0);
265 assert_se(sd_event_add_signal(e
, &u
, SIGRTMIN
+2, rtqueue_handler
, NULL
) >= 0);
266 assert_se(sd_event_add_signal(e
, &v
, SIGRTMIN
+3, rtqueue_handler
, NULL
) >= 0);
267 assert_se(sd_event_add_signal(e
, &s
, SIGUSR2
, rtqueue_handler
, NULL
) >= 0);
269 assert_se(sd_event_source_set_priority(v
, -10) >= 0);
271 assert(sigqueue(getpid(), SIGRTMIN
+2, (union sigval
) { .sival_int
= 1 }) >= 0);
272 assert(sigqueue(getpid(), SIGRTMIN
+3, (union sigval
) { .sival_int
= 2 }) >= 0);
273 assert(sigqueue(getpid(), SIGUSR2
, (union sigval
) { .sival_int
= 3 }) >= 0);
274 assert(sigqueue(getpid(), SIGRTMIN
+3, (union sigval
) { .sival_int
= 4 }) >= 0);
275 assert(sigqueue(getpid(), SIGUSR2
, (union sigval
) { .sival_int
= 5 }) >= 0);
277 assert_se(n_rtqueue
== 0);
278 assert_se(last_rtqueue_sigval
== 0);
280 assert_se(sd_event_run(e
, (uint64_t) -1) >= 1);
281 assert_se(n_rtqueue
== 1);
282 assert_se(last_rtqueue_sigval
== 2); /* first SIGRTMIN+3 */
284 assert_se(sd_event_run(e
, (uint64_t) -1) >= 1);
285 assert_se(n_rtqueue
== 2);
286 assert_se(last_rtqueue_sigval
== 4); /* second SIGRTMIN+3 */
288 assert_se(sd_event_run(e
, (uint64_t) -1) >= 1);
289 assert_se(n_rtqueue
== 3);
290 assert_se(last_rtqueue_sigval
== 3); /* first SIGUSR2 */
292 assert_se(sd_event_run(e
, (uint64_t) -1) >= 1);
293 assert_se(n_rtqueue
== 4);
294 assert_se(last_rtqueue_sigval
== 1); /* SIGRTMIN+2 */
296 assert_se(sd_event_run(e
, 0) == 0); /* the other SIGUSR2 is dropped, because the first one was still queued */
297 assert_se(n_rtqueue
== 4);
298 assert_se(last_rtqueue_sigval
== 1);
300 sd_event_source_unref(u
);
301 sd_event_source_unref(v
);
302 sd_event_source_unref(s
);
307 int main(int argc
, char *argv
[]) {