# HG changeset patch # User Taylor R Campbell # Date 1741188556 0 # Wed Mar 05 15:29:16 2025 +0000 # Branch trunk # Node ID cd6b61c1b621c4bb80fbe22e714ba188467be421 # Parent 2cf5a043dce2476a538b5ba268481608cbe49e83 # EXP-Topic riastradh-pr59125-oncefork tests/lib/libpthread/t_once: Test fork and pthread_once race. PR lib/59125: pthread_once(3) races with concurrent fork diff -r 2cf5a043dce2 -r cd6b61c1b621 tests/lib/libpthread/t_once.c --- a/tests/lib/libpthread/t_once.c Wed Mar 05 00:41:17 2025 +0000 +++ b/tests/lib/libpthread/t_once.c Wed Mar 05 15:29:16 2025 +0000 @@ -32,14 +32,18 @@ __RCSID("$NetBSD: t_once.c,v 1.2 2017/08/25 22:59:47 ginsbach Exp $"); #include +#include + #include #include #include #include +#include #include #include "h_common.h" +#include "h_macros.h" static pthread_once_t once = PTHREAD_ONCE_INIT; static pthread_mutex_t mutex = PTHREAD_MUTEX_INITIALIZER; @@ -54,6 +58,12 @@ ofunc(void) x++; } +static void +ofunc_silent(void) +{ + x++; +} + ATF_TC(once1); ATF_TC_HEAD(once1, tc) { @@ -187,11 +197,72 @@ ATF_TC_BODY(once3, tc) printf("Test succeeded\n"); } +static void * +fork_and_once(void *cookie) +{ + pthread_barrier_t *bar = cookie; + pid_t pid, child; + int status; + + (void)pthread_barrier_wait(bar); + RL(pid = fork()); + if (pid == 0) { + (void)alarm(1); + (void)pthread_once(&once, &ofunc_silent); + _exit(x - 1); + } + RL(child = waitpid(pid, &status, 0)); + ATF_REQUIRE_EQ_MSG(child, pid, "child=%lld pid=%lld", + (long long)child, (long long)pid); + ATF_REQUIRE_MSG(!WIFSIGNALED(status), "child exited on signal %d (%s)", + WTERMSIG(status), strsignal(WTERMSIG(status))); + ATF_REQUIRE_MSG(WIFEXITED(status) && WEXITSTATUS(status) == 0, + "child exited 0x%x", status); + return NULL; +} + +ATF_TC(oncefork); +ATF_TC_HEAD(oncefork, tc) +{ + atf_tc_set_md_var(tc, "descr", "Test racing pthread_once with fork"); +} +ATF_TC_BODY(oncefork, tc) +{ + static pthread_once_t once0 = PTHREAD_ONCE_INIT; + pthread_barrier_t bar; + long i, n = atf_tc_get_config_var_as_long_wd(tc, + "pthread_once_forktrials", 0); + + if (n <= 0) { + atf_tc_skip("pthread_once takes thousands of fork trials" + " on a multicore system to detect a race; set" + " pthread_once_forktrials to the number of trials to" + " enable this test"); + } + + RZ(pthread_barrier_init(&bar, NULL, 2)); + + for (i = 0; i < n; i++) { + pthread_t t; + + once = once0; + x = 0; + + RZ(pthread_create(&t, NULL, &fork_and_once, &bar)); + (void)alarm(1); + (void)pthread_barrier_wait(&bar); + (void)pthread_once(&once, &ofunc_silent); + (void)alarm(0); + RZ(pthread_join(t, NULL)); + } +} + ATF_TP_ADD_TCS(tp) { ATF_TP_ADD_TC(tp, once1); ATF_TP_ADD_TC(tp, once2); ATF_TP_ADD_TC(tp, once3); + ATF_TP_ADD_TC(tp, oncefork); return atf_no_error(); }