[clang] Add test for CWG190 "Layout-compatible POD-struct types" (#121668)
[llvm-project.git] / llvm / lib / Support / ManagedStatic.cpp
bloba6ae67066ea0423334e8ee52106f220cd456e25e
1 //===-- ManagedStatic.cpp - Static Global wrapper -------------------------===//
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 // This file implements the ManagedStatic class and llvm_shutdown().
11 //===----------------------------------------------------------------------===//
13 #include "llvm/Support/ManagedStatic.h"
14 #include "llvm/Config/config.h"
15 #include "llvm/Support/Threading.h"
16 #include <cassert>
17 #include <mutex>
18 using namespace llvm;
20 static const ManagedStaticBase *StaticList = nullptr;
22 static std::recursive_mutex *getManagedStaticMutex() {
23 static std::recursive_mutex m;
24 return &m;
27 void ManagedStaticBase::RegisterManagedStatic(void *(*Creator)(),
28 void (*Deleter)(void*)) const {
29 assert(Creator);
30 if (llvm_is_multithreaded()) {
31 std::lock_guard<std::recursive_mutex> Lock(*getManagedStaticMutex());
33 if (!Ptr.load(std::memory_order_relaxed)) {
34 void *Tmp = Creator();
36 Ptr.store(Tmp, std::memory_order_release);
37 DeleterFn = Deleter;
39 // Add to list of managed statics.
40 Next = StaticList;
41 StaticList = this;
43 } else {
44 assert(!Ptr && !DeleterFn && !Next &&
45 "Partially initialized ManagedStatic!?");
46 Ptr = Creator();
47 DeleterFn = Deleter;
49 // Add to list of managed statics.
50 Next = StaticList;
51 StaticList = this;
55 void ManagedStaticBase::destroy() const {
56 assert(DeleterFn && "ManagedStatic not initialized correctly!");
57 assert(StaticList == this &&
58 "Not destroyed in reverse order of construction?");
59 // Unlink from list.
60 StaticList = Next;
61 Next = nullptr;
63 // Destroy memory.
64 DeleterFn(Ptr);
66 // Cleanup.
67 Ptr = nullptr;
68 DeleterFn = nullptr;
71 /// llvm_shutdown - Deallocate and destroy all ManagedStatic variables.
72 /// IMPORTANT: it's only safe to call llvm_shutdown() in single thread,
73 /// without any other threads executing LLVM APIs.
74 /// llvm_shutdown() should be the last use of LLVM APIs.
75 void llvm::llvm_shutdown() {
76 while (StaticList)
77 StaticList->destroy();