1 //===- llvm/unittest/Support/ManagedStatic.cpp - ManagedStatic tests ------===//
2 //
3 // Part of the LLVM Project, under the Apache License v2.0 with LLVM Exceptions.
4 // See https://llvm.org/LICENSE.txt for license information.
5 // SPDX-License-Identifier: Apache-2.0 WITH LLVM-exception
6 //
7 //===----------------------------------------------------------------------===//
8 
9 #include "llvm/Support/Allocator.h"
10 #include "llvm/Support/ManagedStatic.h"
11 #include "llvm/Config/config.h"
12 #ifdef HAVE_PTHREAD_H
13 #include <pthread.h>
14 #endif
15 
16 #include "gtest/gtest.h"
17 
18 using namespace llvm;
19 
20 namespace {
21 
22 #if LLVM_ENABLE_THREADS != 0 && defined(HAVE_PTHREAD_H) && \
23   !__has_feature(memory_sanitizer)
24 namespace test1 {
25   llvm::ManagedStatic<int> ms;
helper(void *)26   void *helper(void*) {
27     *ms;
28     return nullptr;
29   }
30 
31   // Valgrind's leak checker complains glibc's stack allocation.
32   // To appease valgrind, we provide our own stack for each thread.
allocate_stack(pthread_attr_t & a,size_t n=65536)33   void *allocate_stack(pthread_attr_t &a, size_t n = 65536) {
34     void *stack = safe_malloc(n);
35     pthread_attr_init(&a);
36 #if defined(__linux__)
37     pthread_attr_setstack(&a, stack, n);
38 #endif
39     return stack;
40   }
41 }
42 
TEST(Initialize,MultipleThreads)43 TEST(Initialize, MultipleThreads) {
44   // Run this test under tsan: http://code.google.com/p/data-race-test/
45 
46   pthread_attr_t a1, a2;
47   void *p1 = test1::allocate_stack(a1);
48   void *p2 = test1::allocate_stack(a2);
49 
50   pthread_t t1, t2;
51   pthread_create(&t1, &a1, test1::helper, nullptr);
52   pthread_create(&t2, &a2, test1::helper, nullptr);
53   pthread_join(t1, nullptr);
54   pthread_join(t2, nullptr);
55   free(p1);
56   free(p2);
57 }
58 #endif
59 
60 namespace NestedStatics {
61 static ManagedStatic<int> Ms1;
62 struct Nest {
Nest__anond28357bd0111::NestedStatics::Nest63   Nest() {
64     ++(*Ms1);
65   }
66 
~Nest__anond28357bd0111::NestedStatics::Nest67   ~Nest() {
68     assert(Ms1.isConstructed());
69     ++(*Ms1);
70   }
71 };
72 static ManagedStatic<Nest> Ms2;
73 
TEST(ManagedStaticTest,NestedStatics)74 TEST(ManagedStaticTest, NestedStatics) {
75   EXPECT_FALSE(Ms1.isConstructed());
76   EXPECT_FALSE(Ms2.isConstructed());
77 
78   *Ms2;
79   EXPECT_TRUE(Ms1.isConstructed());
80   EXPECT_TRUE(Ms2.isConstructed());
81 }
82 } // namespace NestedStatics
83 
84 namespace CustomCreatorDeletor {
85 struct CustomCreate {
call__anond28357bd0111::CustomCreatorDeletor::CustomCreate86   static void *call() {
87     void *Mem = safe_malloc(sizeof(int));
88     *((int *)Mem) = 42;
89     return Mem;
90   }
91 };
92 struct CustomDelete {
call__anond28357bd0111::CustomCreatorDeletor::CustomDelete93   static void call(void *P) { std::free(P); }
94 };
95 static ManagedStatic<int, CustomCreate, CustomDelete> Custom;
TEST(ManagedStaticTest,CustomCreatorDeletor)96 TEST(ManagedStaticTest, CustomCreatorDeletor) {
97   EXPECT_EQ(42, *Custom);
98 }
99 } // namespace CustomCreatorDeletor
100 
101 } // anonymous namespace
102