blob: 18ed710157634161ec579c673f4d0dad40f1859e [file] [log] [blame]
//===-- Unittests for call_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 "include/threads.h"
#include "src/threads/call_once.h"
#include "src/threads/mtx_destroy.h"
#include "src/threads/mtx_init.h"
#include "src/threads/mtx_lock.h"
#include "src/threads/mtx_unlock.h"
#include "src/threads/thrd_create.h"
#include "src/threads/thrd_join.h"
#include "utils/UnitTest/Test.h"
#include <stdatomic.h>
static constexpr unsigned int num_threads = 5;
static atomic_uint thread_count;
static unsigned int call_count;
static void call_once_func() { ++call_count; }
static int func(void *) {
static once_flag flag = ONCE_FLAG_INIT;
__llvm_libc::call_once(&flag, call_once_func);
++thread_count; // This is a an atomic update.
return 0;
}
TEST(LlvmLibcCallOnceTest, CallFrom5Threads) {
// Ensure the call count and thread count are 0 to begin with.
call_count = 0;
thread_count = 0;
thrd_t threads[num_threads];
for (unsigned int i = 0; i < num_threads; ++i) {
ASSERT_EQ(__llvm_libc::thrd_create(threads + i, func, nullptr),
static_cast<int>(thrd_success));
}
for (unsigned int i = 0; i < num_threads; ++i) {
int retval;
ASSERT_EQ(__llvm_libc::thrd_join(threads + i, &retval),
static_cast<int>(thrd_success));
ASSERT_EQ(retval, 0);
}
EXPECT_EQ(static_cast<unsigned int>(thread_count), 5U);
EXPECT_EQ(call_count, 1U);
}
static mtx_t once_func_blocker;
static void blocking_once_func() {
__llvm_libc::mtx_lock(&once_func_blocker);
__llvm_libc::mtx_unlock(&once_func_blocker);
}
static atomic_uint start_count;
static atomic_uint done_count;
static int once_func_caller(void *) {
static once_flag flag;
++start_count;
__llvm_libc::call_once(&flag, blocking_once_func);
++done_count;
return 0;
}
// Test the synchronization aspect of the call_once function.
// This is not a fool proof test, but something which might be
// useful when we add a flakiness detection scheme to UnitTest.
TEST(LlvmLibcCallOnceTest, TestSynchronization) {
start_count = 0;
done_count = 0;
ASSERT_EQ(__llvm_libc::mtx_init(&once_func_blocker, mtx_plain),
static_cast<int>(thrd_success));
// Lock the blocking mutex so that the once func blocks.
ASSERT_EQ(__llvm_libc::mtx_lock(&once_func_blocker),
static_cast<int>(thrd_success));
thrd_t t1, t2;
ASSERT_EQ(__llvm_libc::thrd_create(&t1, once_func_caller, nullptr),
static_cast<int>(thrd_success));
ASSERT_EQ(__llvm_libc::thrd_create(&t2, once_func_caller, nullptr),
static_cast<int>(thrd_success));
while (start_count != 2)
; // Spin until both threads start.
// Since the once func is blocked, the threads should not be done yet.
EXPECT_EQ(static_cast<unsigned int>(done_count), 0U);
// Unlock the blocking mutex so that the once func blocks.
ASSERT_EQ(__llvm_libc::mtx_unlock(&once_func_blocker),
static_cast<int>(thrd_success));
int retval;
ASSERT_EQ(__llvm_libc::thrd_join(&t1, &retval),
static_cast<int>(thrd_success));
ASSERT_EQ(retval, 0);
ASSERT_EQ(__llvm_libc::thrd_join(&t2, &retval),
static_cast<int>(thrd_success));
ASSERT_EQ(retval, 0);
ASSERT_EQ(static_cast<unsigned int>(done_count), 2U);
__llvm_libc::mtx_destroy(&once_func_blocker);
}