Files
bind9/lib/isc/tests/loop_test.c
2022-03-30 09:23:14 -07:00

204 lines
3.8 KiB
C

/*
* Copyright (C) Internet Systems Consortium, Inc. ("ISC")
*
* SPDX-License-Identifier: MPL-2.0
*
* This Source Code Form is subject to the terms of the Mozilla Public
* License, v. 2.0. If a copy of the MPL was not distributed with this
* file, you can obtain one at https://mozilla.org/MPL/2.0/.
*
* See the COPYRIGHT file distributed with this work for additional
* information regarding copyright ownership.
*/
#if HAVE_CMOCKA
#include <sched.h> /* IWYU pragma: keep */
#include <setjmp.h>
#include <stdarg.h>
#include <stddef.h>
#include <stdlib.h>
#include <string.h>
#include <sys/types.h>
#include <unistd.h>
#define UNIT_TESTING
#include <cmocka.h>
#include <isc/atomic.h>
#include <isc/loop.h>
#include <isc/os.h>
#include <isc/result.h>
#include <isc/util.h>
#include "../loop.c"
#include "isctest.h"
static int
setup_mctx(void **state) {
UNUSED(state);
isc_mem_create(&test_mctx);
return (0);
}
static int
teardown_mctx(void **state) {
UNUSED(state);
isc_mem_destroy(&test_mctx);
return (0);
}
static int
setup_loopmgr(void **state) {
isc_loopmgr_t *loopmgr = NULL;
loopmgr = isc_loopmgr_new(test_mctx, isc_os_ncpus());
if (loopmgr == NULL) {
return (1);
}
*state = loopmgr;
return (0);
}
static int
teardown_loopmgr(void **state) {
isc_loopmgr_t *loopmgr = *state;
isc_loopmgr_destroy(&loopmgr);
return (0);
}
static atomic_uint scheduled = 0;
static void
count(void *arg) {
UNUSED(arg);
atomic_fetch_add(&scheduled, 1);
}
static void
shutdown_loopmgr(void *arg) {
isc_loopmgr_t *loopmgr = (isc_loopmgr_t *)arg;
while (atomic_load(&scheduled) != loopmgr->nloops) {
isc_thread_yield();
}
isc_loopmgr_shutdown(loopmgr);
}
static void
isc_loopmgr_test(void **state) {
isc_loopmgr_t *loopmgr = *state;
isc_loopmgr_schedule_ctor(loopmgr, count, loopmgr);
isc_loop_schedule_ctor(DEFAULT_LOOP(loopmgr), shutdown_loopmgr,
loopmgr);
isc_loopmgr_run(loopmgr);
assert_int_equal(atomic_load(&scheduled), loopmgr->nloops);
}
static void
pause_loopmgr(void *arg) {
isc_loopmgr_t *loopmgr = (isc_loopmgr_t *)arg;
isc_loopmgr_pause(loopmgr);
assert_true(atomic_load(&loopmgr->paused));
for (size_t i = 0; i < loopmgr->nloops; i++) {
isc_loop_t *loop = &loopmgr->loops[i];
assert_true(loop->paused);
}
atomic_init(&scheduled, loopmgr->nloops);
isc_loopmgr_resume(loopmgr);
}
static void
isc_loopmgr_pause_test(void **state) {
isc_loopmgr_t *loopmgr = *state;
isc_loop_schedule_ctor(DEFAULT_LOOP(loopmgr), pause_loopmgr, loopmgr);
isc_loop_schedule_ctor(DEFAULT_LOOP(loopmgr), shutdown_loopmgr,
loopmgr);
isc_loopmgr_run(loopmgr);
}
static void
send_sigint(void *arg) {
UNUSED(arg);
kill(getpid(), SIGINT);
}
static void
isc_loopmgr_sigint_test(void **state) {
isc_loopmgr_t *loopmgr = *state;
isc_loop_schedule_ctor(CURRENT_LOOP(loopmgr), send_sigint, loopmgr);
isc_loopmgr_run(loopmgr);
}
static void
send_sigterm(void *arg) {
UNUSED(arg);
kill(getpid(), SIGINT);
}
static void
isc_loopmgr_sigterm_test(void **state) {
isc_loopmgr_t *loopmgr = *state;
isc_loop_schedule_ctor(CURRENT_LOOP(loopmgr), send_sigterm, loopmgr);
isc_loopmgr_run(loopmgr);
}
int
main(void) {
const struct CMUnitTest tests[] = {
cmocka_unit_test_setup_teardown(isc_loopmgr_test, setup_loopmgr,
teardown_loopmgr),
cmocka_unit_test_setup_teardown(isc_loopmgr_pause_test,
setup_loopmgr,
teardown_loopmgr),
cmocka_unit_test_setup_teardown(isc_loopmgr_sigint_test,
setup_loopmgr,
teardown_loopmgr),
cmocka_unit_test_setup_teardown(isc_loopmgr_sigterm_test,
setup_loopmgr,
teardown_loopmgr),
};
return (cmocka_run_group_tests(tests, setup_mctx, teardown_mctx));
}
#else /* HAVE_CMOCKA */
#include <stdio.h>
int
main(void) {
printf("1..0 # Skipped: cmocka not available\n");
return (SKIPPED_TEST_EXIT_CODE);
}
#endif /* if HAVE_CMOCKA */