]>
git.ipfire.org Git - thirdparty/systemd.git/blob - src/libsystemd/sd-event/test-event.c
1 /* SPDX-License-Identifier: LGPL-2.1+ */
3 This file is part of systemd.
5 Copyright 2013 Lennart Poettering
7 systemd is free software; you can redistribute it and/or modify it
8 under the terms of the GNU Lesser General Public License as published by
9 the Free Software Foundation; either version 2.1 of the License, or
10 (at your option) any later version.
12 systemd is distributed in the hope that it will be useful, but
13 WITHOUT ANY WARRANTY; without even the implied warranty of
14 MERCHANTABILITY or FITNESS FOR A PARTICULAR PURPOSE. See the GNU
15 Lesser General Public License for more details.
17 You should have received a copy of the GNU Lesser General Public License
18 along with systemd; If not, see <http://www.gnu.org/licenses/>.
28 #include "signal-util.h"
31 static int prepare_handler(sd_event_source
*s
, void *userdata
) {
32 log_info("preparing %c", PTR_TO_INT(userdata
));
36 static bool got_a
, got_b
, got_c
, got_unref
;
37 static unsigned got_d
;
39 static int unref_handler(sd_event_source
*s
, int fd
, uint32_t revents
, void *userdata
) {
40 sd_event_source_unref(s
);
45 static int io_handler(sd_event_source
*s
, int fd
, uint32_t revents
, void *userdata
) {
47 log_info("got IO on %c", PTR_TO_INT(userdata
));
49 if (userdata
== INT_TO_PTR('a')) {
50 assert_se(sd_event_source_set_enabled(s
, SD_EVENT_OFF
) >= 0);
53 } else if (userdata
== INT_TO_PTR('b')) {
56 } else if (userdata
== INT_TO_PTR('d')) {
59 assert_se(sd_event_source_set_enabled(s
, SD_EVENT_ONESHOT
) >= 0);
61 assert_se(sd_event_source_set_enabled(s
, SD_EVENT_OFF
) >= 0);
63 assert_not_reached("Yuck!");
68 static int child_handler(sd_event_source
*s
, const siginfo_t
*si
, void *userdata
) {
73 log_info("got child on %c", PTR_TO_INT(userdata
));
75 assert_se(userdata
== INT_TO_PTR('f'));
77 assert_se(sd_event_exit(sd_event_source_get_event(s
), 0) >= 0);
78 sd_event_source_unref(s
);
83 static int signal_handler(sd_event_source
*s
, const struct signalfd_siginfo
*si
, void *userdata
) {
84 sd_event_source
*p
= NULL
;
90 log_info("got signal on %c", PTR_TO_INT(userdata
));
92 assert_se(userdata
== INT_TO_PTR('e'));
94 assert_se(sigprocmask_many(SIG_BLOCK
, NULL
, SIGCHLD
, -1) >= 0);
102 assert_se(sd_event_add_child(sd_event_source_get_event(s
), &p
, pid
, WEXITED
, child_handler
, INT_TO_PTR('f')) >= 0);
103 assert_se(sd_event_source_set_enabled(p
, SD_EVENT_ONESHOT
) >= 0);
105 sd_event_source_unref(s
);
110 static int defer_handler(sd_event_source
*s
, void *userdata
) {
111 sd_event_source
*p
= NULL
;
115 log_info("got defer on %c", PTR_TO_INT(userdata
));
117 assert_se(userdata
== INT_TO_PTR('d'));
119 assert_se(sigprocmask_many(SIG_BLOCK
, NULL
, SIGUSR1
, -1) >= 0);
121 assert_se(sd_event_add_signal(sd_event_source_get_event(s
), &p
, SIGUSR1
, signal_handler
, INT_TO_PTR('e')) >= 0);
122 assert_se(sd_event_source_set_enabled(p
, SD_EVENT_ONESHOT
) >= 0);
125 sd_event_source_unref(s
);
130 static bool do_quit
= false;
132 static int time_handler(sd_event_source
*s
, uint64_t usec
, void *userdata
) {
133 log_info("got timer on %c", PTR_TO_INT(userdata
));
135 if (userdata
== INT_TO_PTR('c')) {
140 assert_se(sd_event_add_defer(sd_event_source_get_event(s
), &p
, defer_handler
, INT_TO_PTR('d')) >= 0);
141 assert_se(sd_event_source_set_enabled(p
, SD_EVENT_ONESHOT
) >= 0);
147 assert_not_reached("Huh?");
152 static bool got_exit
= false;
154 static int exit_handler(sd_event_source
*s
, void *userdata
) {
155 log_info("got quit handler on %c", PTR_TO_INT(userdata
));
162 static bool got_post
= false;
164 static int post_handler(sd_event_source
*s
, void *userdata
) {
165 log_info("got post handler");
172 static void test_basic(void) {
174 sd_event_source
*w
= NULL
, *x
= NULL
, *y
= NULL
, *z
= NULL
, *q
= NULL
, *t
= NULL
;
175 static const char ch
= 'x';
176 int a
[2] = { -1, -1 }, b
[2] = { -1, -1}, d
[2] = { -1, -1}, k
[2] = { -1, -1 };
180 assert_se(pipe(a
) >= 0);
181 assert_se(pipe(b
) >= 0);
182 assert_se(pipe(d
) >= 0);
183 assert_se(pipe(k
) >= 0);
185 assert_se(sd_event_default(&e
) >= 0);
186 assert_se(sd_event_now(e
, CLOCK_MONOTONIC
, &event_now
) > 0);
188 assert_se(sd_event_set_watchdog(e
, true) >= 0);
190 /* Test whether we cleanly can destroy an io event source from its own handler */
192 assert_se(sd_event_add_io(e
, &t
, k
[0], EPOLLIN
, unref_handler
, NULL
) >= 0);
193 assert_se(write(k
[1], &ch
, 1) == 1);
194 assert_se(sd_event_run(e
, (uint64_t) -1) >= 1);
195 assert_se(got_unref
);
197 got_a
= false, got_b
= false, got_c
= false, got_d
= 0;
199 /* Add a oneshot handler, trigger it, re-enable it, and trigger
201 assert_se(sd_event_add_io(e
, &w
, d
[0], EPOLLIN
, io_handler
, INT_TO_PTR('d')) >= 0);
202 assert_se(sd_event_source_set_enabled(w
, SD_EVENT_ONESHOT
) >= 0);
203 assert_se(write(d
[1], &ch
, 1) >= 0);
204 assert_se(sd_event_run(e
, (uint64_t) -1) >= 1);
205 assert_se(got_d
== 1);
206 assert_se(write(d
[1], &ch
, 1) >= 0);
207 assert_se(sd_event_run(e
, (uint64_t) -1) >= 1);
208 assert_se(got_d
== 2);
210 assert_se(sd_event_add_io(e
, &x
, a
[0], EPOLLIN
, io_handler
, INT_TO_PTR('a')) >= 0);
211 assert_se(sd_event_add_io(e
, &y
, b
[0], EPOLLIN
, io_handler
, INT_TO_PTR('b')) >= 0);
212 assert_se(sd_event_add_time(e
, &z
, CLOCK_MONOTONIC
, 0, 0, time_handler
, INT_TO_PTR('c')) >= 0);
213 assert_se(sd_event_add_exit(e
, &q
, exit_handler
, INT_TO_PTR('g')) >= 0);
215 assert_se(sd_event_source_set_priority(x
, 99) >= 0);
216 assert_se(sd_event_source_get_priority(x
, &priority
) >= 0);
217 assert_se(priority
== 99);
218 assert_se(sd_event_source_set_enabled(y
, SD_EVENT_ONESHOT
) >= 0);
219 assert_se(sd_event_source_set_prepare(x
, prepare_handler
) >= 0);
220 assert_se(sd_event_source_set_priority(z
, 50) >= 0);
221 assert_se(sd_event_source_set_enabled(z
, SD_EVENT_ONESHOT
) >= 0);
222 assert_se(sd_event_source_set_prepare(z
, prepare_handler
) >= 0);
224 /* Test for floating event sources */
225 assert_se(sigprocmask_many(SIG_BLOCK
, NULL
, SIGRTMIN
+1, -1) >= 0);
226 assert_se(sd_event_add_signal(e
, NULL
, SIGRTMIN
+1, NULL
, NULL
) >= 0);
228 assert_se(write(a
[1], &ch
, 1) >= 0);
229 assert_se(write(b
[1], &ch
, 1) >= 0);
231 assert_se(!got_a
&& !got_b
&& !got_c
);
233 assert_se(sd_event_run(e
, (uint64_t) -1) >= 1);
235 assert_se(!got_a
&& got_b
&& !got_c
);
237 assert_se(sd_event_run(e
, (uint64_t) -1) >= 1);
239 assert_se(!got_a
&& got_b
&& got_c
);
241 assert_se(sd_event_run(e
, (uint64_t) -1) >= 1);
243 assert_se(got_a
&& got_b
&& got_c
);
245 sd_event_source_unref(x
);
246 sd_event_source_unref(y
);
249 assert_se(sd_event_add_post(e
, NULL
, post_handler
, NULL
) >= 0);
250 assert_se(sd_event_now(e
, CLOCK_MONOTONIC
, &event_now
) == 0);
251 assert_se(sd_event_source_set_time(z
, event_now
+ 200 * USEC_PER_MSEC
) >= 0);
252 assert_se(sd_event_source_set_enabled(z
, SD_EVENT_ONESHOT
) >= 0);
254 assert_se(sd_event_loop(e
) >= 0);
258 sd_event_source_unref(z
);
259 sd_event_source_unref(q
);
261 sd_event_source_unref(w
);
271 static void test_sd_event_now(void) {
272 _cleanup_(sd_event_unrefp
) sd_event
*e
= NULL
;
275 assert_se(sd_event_new(&e
) >= 0);
276 assert_se(sd_event_now(e
, CLOCK_MONOTONIC
, &event_now
) > 0);
277 assert_se(sd_event_now(e
, CLOCK_REALTIME
, &event_now
) > 0);
278 assert_se(sd_event_now(e
, CLOCK_REALTIME_ALARM
, &event_now
) > 0);
279 if (clock_boottime_supported()) {
280 assert_se(sd_event_now(e
, CLOCK_BOOTTIME
, &event_now
) > 0);
281 assert_se(sd_event_now(e
, CLOCK_BOOTTIME_ALARM
, &event_now
) > 0);
283 assert_se(sd_event_now(e
, -1, &event_now
) == -EOPNOTSUPP
);
284 assert_se(sd_event_now(e
, 900 /* arbitrary big number */, &event_now
) == -EOPNOTSUPP
);
286 assert_se(sd_event_run(e
, 0) == 0);
288 assert_se(sd_event_now(e
, CLOCK_MONOTONIC
, &event_now
) == 0);
289 assert_se(sd_event_now(e
, CLOCK_REALTIME
, &event_now
) == 0);
290 assert_se(sd_event_now(e
, CLOCK_REALTIME_ALARM
, &event_now
) == 0);
291 if (clock_boottime_supported()) {
292 assert_se(sd_event_now(e
, CLOCK_BOOTTIME
, &event_now
) == 0);
293 assert_se(sd_event_now(e
, CLOCK_BOOTTIME_ALARM
, &event_now
) == 0);
295 assert_se(sd_event_now(e
, -1, &event_now
) == -EOPNOTSUPP
);
296 assert_se(sd_event_now(e
, 900 /* arbitrary big number */, &event_now
) == -EOPNOTSUPP
);
299 static int last_rtqueue_sigval
= 0;
300 static int n_rtqueue
= 0;
302 static int rtqueue_handler(sd_event_source
*s
, const struct signalfd_siginfo
*si
, void *userdata
) {
303 last_rtqueue_sigval
= si
->ssi_int
;
308 static void test_rtqueue(void) {
309 sd_event_source
*u
= NULL
, *v
= NULL
, *s
= NULL
;
312 assert_se(sd_event_default(&e
) >= 0);
314 assert_se(sigprocmask_many(SIG_BLOCK
, NULL
, SIGRTMIN
+2, SIGRTMIN
+3, SIGUSR2
, -1) >= 0);
315 assert_se(sd_event_add_signal(e
, &u
, SIGRTMIN
+2, rtqueue_handler
, NULL
) >= 0);
316 assert_se(sd_event_add_signal(e
, &v
, SIGRTMIN
+3, rtqueue_handler
, NULL
) >= 0);
317 assert_se(sd_event_add_signal(e
, &s
, SIGUSR2
, rtqueue_handler
, NULL
) >= 0);
319 assert_se(sd_event_source_set_priority(v
, -10) >= 0);
321 assert_se(sigqueue(getpid_cached(), SIGRTMIN
+2, (union sigval
) { .sival_int
= 1 }) >= 0);
322 assert_se(sigqueue(getpid_cached(), SIGRTMIN
+3, (union sigval
) { .sival_int
= 2 }) >= 0);
323 assert_se(sigqueue(getpid_cached(), SIGUSR2
, (union sigval
) { .sival_int
= 3 }) >= 0);
324 assert_se(sigqueue(getpid_cached(), SIGRTMIN
+3, (union sigval
) { .sival_int
= 4 }) >= 0);
325 assert_se(sigqueue(getpid_cached(), SIGUSR2
, (union sigval
) { .sival_int
= 5 }) >= 0);
327 assert_se(n_rtqueue
== 0);
328 assert_se(last_rtqueue_sigval
== 0);
330 assert_se(sd_event_run(e
, (uint64_t) -1) >= 1);
331 assert_se(n_rtqueue
== 1);
332 assert_se(last_rtqueue_sigval
== 2); /* first SIGRTMIN+3 */
334 assert_se(sd_event_run(e
, (uint64_t) -1) >= 1);
335 assert_se(n_rtqueue
== 2);
336 assert_se(last_rtqueue_sigval
== 4); /* second SIGRTMIN+3 */
338 assert_se(sd_event_run(e
, (uint64_t) -1) >= 1);
339 assert_se(n_rtqueue
== 3);
340 assert_se(last_rtqueue_sigval
== 3); /* first SIGUSR2 */
342 assert_se(sd_event_run(e
, (uint64_t) -1) >= 1);
343 assert_se(n_rtqueue
== 4);
344 assert_se(last_rtqueue_sigval
== 1); /* SIGRTMIN+2 */
346 assert_se(sd_event_run(e
, 0) == 0); /* the other SIGUSR2 is dropped, because the first one was still queued */
347 assert_se(n_rtqueue
== 4);
348 assert_se(last_rtqueue_sigval
== 1);
350 sd_event_source_unref(u
);
351 sd_event_source_unref(v
);
352 sd_event_source_unref(s
);
357 int main(int argc
, char *argv
[]) {
359 log_set_max_level(LOG_DEBUG
);
360 log_parse_environment();