//===-- Tests for pthread_once --------------------------------------------===// // // Part of the LLVM Project, under the Apache License v2.0 with LLVM Exceptions. // See https://llvm.org/LICENSE.txt for license information. // SPDX-License-Identifier: Apache-2.0 WITH LLVM-exception // //===----------------------------------------------------------------------===// #include "src/__support/CPP/atomic.h" #include "src/pthread/pthread_create.h" #include "src/pthread/pthread_join.h" #include "src/pthread/pthread_mutex_destroy.h" #include "src/pthread/pthread_mutex_init.h" #include "src/pthread/pthread_mutex_lock.h" #include "src/pthread/pthread_mutex_unlock.h" #include "src/pthread/pthread_once.h" #include "test/IntegrationTest/test.h" #include #include // uintptr_t static constexpr unsigned int NUM_THREADS = 5; static LIBC_NAMESPACE::cpp::Atomic thread_count; static unsigned int call_count; static void pthread_once_func() { ++call_count; } static void *func(void *) { static pthread_once_t flag = PTHREAD_ONCE_INIT; ASSERT_EQ(LIBC_NAMESPACE::pthread_once(&flag, pthread_once_func), 0); thread_count.fetch_add(1); return nullptr; } void call_from_5_threads() { // Ensure the call count and thread count are 0 to begin with. call_count = 0; thread_count = 0; pthread_t threads[NUM_THREADS]; for (unsigned int i = 0; i < NUM_THREADS; ++i) { ASSERT_EQ( LIBC_NAMESPACE::pthread_create(threads + i, nullptr, func, nullptr), 0); } for (unsigned int i = 0; i < NUM_THREADS; ++i) { void *retval; ASSERT_EQ(LIBC_NAMESPACE::pthread_join(threads[i], &retval), 0); ASSERT_EQ(uintptr_t(retval), uintptr_t(0)); } EXPECT_EQ(thread_count.val, 5U); EXPECT_EQ(call_count, 1U); } static pthread_mutex_t once_func_blocker; static void blocking_once_func() { LIBC_NAMESPACE::pthread_mutex_lock(&once_func_blocker); LIBC_NAMESPACE::pthread_mutex_unlock(&once_func_blocker); } static LIBC_NAMESPACE::cpp::Atomic start_count; static LIBC_NAMESPACE::cpp::Atomic done_count; static void *once_func_caller(void *) { static pthread_once_t flag; start_count.fetch_add(1); LIBC_NAMESPACE::pthread_once(&flag, blocking_once_func); done_count.fetch_add(1); return nullptr; } // Test the synchronization aspect of the pthread_once function. // This is not a fool proof test, but something which might be // useful when we add a flakiness detection scheme to UnitTest. void test_synchronization() { start_count = 0; done_count = 0; ASSERT_EQ(LIBC_NAMESPACE::pthread_mutex_init(&once_func_blocker, nullptr), 0); // Lock the blocking mutex so that the once func blocks. ASSERT_EQ(LIBC_NAMESPACE::pthread_mutex_lock(&once_func_blocker), 0); pthread_t t1, t2; ASSERT_EQ( LIBC_NAMESPACE::pthread_create(&t1, nullptr, once_func_caller, nullptr), 0); ASSERT_EQ( LIBC_NAMESPACE::pthread_create(&t2, nullptr, once_func_caller, nullptr), 0); while (start_count.load() != 2) ; // Spin until both threads start. // Since the once func is blocked, the threads should not be done yet. EXPECT_EQ(done_count.val, 0U); // Unlock the blocking mutex so that the once func blocks. ASSERT_EQ(LIBC_NAMESPACE::pthread_mutex_unlock(&once_func_blocker), 0); void *retval; ASSERT_EQ(LIBC_NAMESPACE::pthread_join(t1, &retval), uintptr_t(0)); ASSERT_EQ(uintptr_t(retval), 0); ASSERT_EQ(LIBC_NAMESPACE::pthread_join(t2, &retval), uintptr_t(0)); ASSERT_EQ(uintptr_t(retval), 0); ASSERT_EQ(done_count.val, 2U); LIBC_NAMESPACE::pthread_mutex_destroy(&once_func_blocker); } TEST_MAIN() { call_from_5_threads(); test_synchronization(); return 0; }