1*d5475af2SSiva Chandra Reddy //===-- Tests for call_once -----------------------------------------------===//
2*d5475af2SSiva Chandra Reddy //
3*d5475af2SSiva Chandra Reddy // Part of the LLVM Project, under the Apache License v2.0 with LLVM Exceptions.
4*d5475af2SSiva Chandra Reddy // See https://llvm.org/LICENSE.txt for license information.
5*d5475af2SSiva Chandra Reddy // SPDX-License-Identifier: Apache-2.0 WITH LLVM-exception
6*d5475af2SSiva Chandra Reddy //
7*d5475af2SSiva Chandra Reddy //===----------------------------------------------------------------------===//
8*d5475af2SSiva Chandra Reddy
9*d5475af2SSiva Chandra Reddy #include "src/__support/CPP/atomic.h"
10*d5475af2SSiva Chandra Reddy #include "src/threads/call_once.h"
11*d5475af2SSiva Chandra Reddy #include "src/threads/mtx_destroy.h"
12*d5475af2SSiva Chandra Reddy #include "src/threads/mtx_init.h"
13*d5475af2SSiva Chandra Reddy #include "src/threads/mtx_lock.h"
14*d5475af2SSiva Chandra Reddy #include "src/threads/mtx_unlock.h"
15*d5475af2SSiva Chandra Reddy #include "src/threads/thrd_create.h"
16*d5475af2SSiva Chandra Reddy #include "src/threads/thrd_join.h"
17*d5475af2SSiva Chandra Reddy
18*d5475af2SSiva Chandra Reddy #include "utils/IntegrationTest/test.h"
19*d5475af2SSiva Chandra Reddy
20*d5475af2SSiva Chandra Reddy #include <threads.h>
21*d5475af2SSiva Chandra Reddy
22*d5475af2SSiva Chandra Reddy static constexpr unsigned int NUM_THREADS = 5;
23*d5475af2SSiva Chandra Reddy static __llvm_libc::cpp::Atomic<unsigned int> thread_count;
24*d5475af2SSiva Chandra Reddy
25*d5475af2SSiva Chandra Reddy static unsigned int call_count;
call_once_func()26*d5475af2SSiva Chandra Reddy static void call_once_func() { ++call_count; }
27*d5475af2SSiva Chandra Reddy
func(void *)28*d5475af2SSiva Chandra Reddy static int func(void *) {
29*d5475af2SSiva Chandra Reddy static once_flag flag = ONCE_FLAG_INIT;
30*d5475af2SSiva Chandra Reddy __llvm_libc::call_once(&flag, call_once_func);
31*d5475af2SSiva Chandra Reddy
32*d5475af2SSiva Chandra Reddy thread_count.fetch_add(1);
33*d5475af2SSiva Chandra Reddy
34*d5475af2SSiva Chandra Reddy return 0;
35*d5475af2SSiva Chandra Reddy }
36*d5475af2SSiva Chandra Reddy
call_from_5_threads()37*d5475af2SSiva Chandra Reddy void call_from_5_threads() {
38*d5475af2SSiva Chandra Reddy // Ensure the call count and thread count are 0 to begin with.
39*d5475af2SSiva Chandra Reddy call_count = 0;
40*d5475af2SSiva Chandra Reddy thread_count = 0;
41*d5475af2SSiva Chandra Reddy
42*d5475af2SSiva Chandra Reddy thrd_t threads[NUM_THREADS];
43*d5475af2SSiva Chandra Reddy for (unsigned int i = 0; i < NUM_THREADS; ++i) {
44*d5475af2SSiva Chandra Reddy ASSERT_EQ(__llvm_libc::thrd_create(threads + i, func, nullptr),
45*d5475af2SSiva Chandra Reddy static_cast<int>(thrd_success));
46*d5475af2SSiva Chandra Reddy }
47*d5475af2SSiva Chandra Reddy
48*d5475af2SSiva Chandra Reddy for (unsigned int i = 0; i < NUM_THREADS; ++i) {
49*d5475af2SSiva Chandra Reddy int retval;
50*d5475af2SSiva Chandra Reddy ASSERT_EQ(__llvm_libc::thrd_join(threads + i, &retval),
51*d5475af2SSiva Chandra Reddy static_cast<int>(thrd_success));
52*d5475af2SSiva Chandra Reddy ASSERT_EQ(retval, 0);
53*d5475af2SSiva Chandra Reddy }
54*d5475af2SSiva Chandra Reddy
55*d5475af2SSiva Chandra Reddy EXPECT_EQ(thread_count.val, 5U);
56*d5475af2SSiva Chandra Reddy EXPECT_EQ(call_count, 1U);
57*d5475af2SSiva Chandra Reddy }
58*d5475af2SSiva Chandra Reddy
59*d5475af2SSiva Chandra Reddy static mtx_t once_func_blocker;
blocking_once_func()60*d5475af2SSiva Chandra Reddy static void blocking_once_func() {
61*d5475af2SSiva Chandra Reddy __llvm_libc::mtx_lock(&once_func_blocker);
62*d5475af2SSiva Chandra Reddy __llvm_libc::mtx_unlock(&once_func_blocker);
63*d5475af2SSiva Chandra Reddy }
64*d5475af2SSiva Chandra Reddy
65*d5475af2SSiva Chandra Reddy static __llvm_libc::cpp::Atomic<unsigned int> start_count;
66*d5475af2SSiva Chandra Reddy static __llvm_libc::cpp::Atomic<unsigned int> done_count;
once_func_caller(void *)67*d5475af2SSiva Chandra Reddy static int once_func_caller(void *) {
68*d5475af2SSiva Chandra Reddy static once_flag flag;
69*d5475af2SSiva Chandra Reddy start_count.fetch_add(1);
70*d5475af2SSiva Chandra Reddy __llvm_libc::call_once(&flag, blocking_once_func);
71*d5475af2SSiva Chandra Reddy done_count.fetch_add(1);
72*d5475af2SSiva Chandra Reddy return 0;
73*d5475af2SSiva Chandra Reddy }
74*d5475af2SSiva Chandra Reddy
75*d5475af2SSiva Chandra Reddy // Test the synchronization aspect of the call_once function.
76*d5475af2SSiva Chandra Reddy // This is not a fool proof test, but something which might be
77*d5475af2SSiva Chandra Reddy // useful when we add a flakiness detection scheme to UnitTest.
test_synchronization()78*d5475af2SSiva Chandra Reddy void test_synchronization() {
79*d5475af2SSiva Chandra Reddy start_count = 0;
80*d5475af2SSiva Chandra Reddy done_count = 0;
81*d5475af2SSiva Chandra Reddy
82*d5475af2SSiva Chandra Reddy ASSERT_EQ(__llvm_libc::mtx_init(&once_func_blocker, mtx_plain),
83*d5475af2SSiva Chandra Reddy static_cast<int>(thrd_success));
84*d5475af2SSiva Chandra Reddy // Lock the blocking mutex so that the once func blocks.
85*d5475af2SSiva Chandra Reddy ASSERT_EQ(__llvm_libc::mtx_lock(&once_func_blocker),
86*d5475af2SSiva Chandra Reddy static_cast<int>(thrd_success));
87*d5475af2SSiva Chandra Reddy
88*d5475af2SSiva Chandra Reddy thrd_t t1, t2;
89*d5475af2SSiva Chandra Reddy ASSERT_EQ(__llvm_libc::thrd_create(&t1, once_func_caller, nullptr),
90*d5475af2SSiva Chandra Reddy static_cast<int>(thrd_success));
91*d5475af2SSiva Chandra Reddy ASSERT_EQ(__llvm_libc::thrd_create(&t2, once_func_caller, nullptr),
92*d5475af2SSiva Chandra Reddy static_cast<int>(thrd_success));
93*d5475af2SSiva Chandra Reddy
94*d5475af2SSiva Chandra Reddy while (start_count.load() != 2)
95*d5475af2SSiva Chandra Reddy ; // Spin until both threads start.
96*d5475af2SSiva Chandra Reddy
97*d5475af2SSiva Chandra Reddy // Since the once func is blocked, the threads should not be done yet.
98*d5475af2SSiva Chandra Reddy EXPECT_EQ(done_count.val, 0U);
99*d5475af2SSiva Chandra Reddy
100*d5475af2SSiva Chandra Reddy // Unlock the blocking mutex so that the once func blocks.
101*d5475af2SSiva Chandra Reddy ASSERT_EQ(__llvm_libc::mtx_unlock(&once_func_blocker),
102*d5475af2SSiva Chandra Reddy static_cast<int>(thrd_success));
103*d5475af2SSiva Chandra Reddy
104*d5475af2SSiva Chandra Reddy int retval;
105*d5475af2SSiva Chandra Reddy ASSERT_EQ(__llvm_libc::thrd_join(&t1, &retval),
106*d5475af2SSiva Chandra Reddy static_cast<int>(thrd_success));
107*d5475af2SSiva Chandra Reddy ASSERT_EQ(retval, 0);
108*d5475af2SSiva Chandra Reddy ASSERT_EQ(__llvm_libc::thrd_join(&t2, &retval),
109*d5475af2SSiva Chandra Reddy static_cast<int>(thrd_success));
110*d5475af2SSiva Chandra Reddy ASSERT_EQ(retval, 0);
111*d5475af2SSiva Chandra Reddy
112*d5475af2SSiva Chandra Reddy ASSERT_EQ(done_count.val, 2U);
113*d5475af2SSiva Chandra Reddy
114*d5475af2SSiva Chandra Reddy __llvm_libc::mtx_destroy(&once_func_blocker);
115*d5475af2SSiva Chandra Reddy }
116*d5475af2SSiva Chandra Reddy
main()117*d5475af2SSiva Chandra Reddy int main() {
118*d5475af2SSiva Chandra Reddy call_from_5_threads();
119*d5475af2SSiva Chandra Reddy test_synchronization();
120*d5475af2SSiva Chandra Reddy return 0;
121*d5475af2SSiva Chandra Reddy }
122