1*9880d681SAndroid Build Coastguard Worker //===-- ManagedStatic.cpp - Static Global wrapper -------------------------===// 2*9880d681SAndroid Build Coastguard Worker // 3*9880d681SAndroid Build Coastguard Worker // The LLVM Compiler Infrastructure 4*9880d681SAndroid Build Coastguard Worker // 5*9880d681SAndroid Build Coastguard Worker // This file is distributed under the University of Illinois Open Source 6*9880d681SAndroid Build Coastguard Worker // License. See LICENSE.TXT for details. 7*9880d681SAndroid Build Coastguard Worker // 8*9880d681SAndroid Build Coastguard Worker //===----------------------------------------------------------------------===// 9*9880d681SAndroid Build Coastguard Worker // 10*9880d681SAndroid Build Coastguard Worker // This file implements the ManagedStatic class and llvm_shutdown(). 11*9880d681SAndroid Build Coastguard Worker // 12*9880d681SAndroid Build Coastguard Worker //===----------------------------------------------------------------------===// 13*9880d681SAndroid Build Coastguard Worker 14*9880d681SAndroid Build Coastguard Worker #include "llvm/Support/ManagedStatic.h" 15*9880d681SAndroid Build Coastguard Worker #include "llvm/Config/config.h" 16*9880d681SAndroid Build Coastguard Worker #include "llvm/Support/Mutex.h" 17*9880d681SAndroid Build Coastguard Worker #include "llvm/Support/MutexGuard.h" 18*9880d681SAndroid Build Coastguard Worker #include "llvm/Support/Threading.h" 19*9880d681SAndroid Build Coastguard Worker #include <cassert> 20*9880d681SAndroid Build Coastguard Worker using namespace llvm; 21*9880d681SAndroid Build Coastguard Worker 22*9880d681SAndroid Build Coastguard Worker static const ManagedStaticBase *StaticList = nullptr; 23*9880d681SAndroid Build Coastguard Worker static sys::Mutex *ManagedStaticMutex = nullptr; 24*9880d681SAndroid Build Coastguard Worker LLVM_DEFINE_ONCE_FLAG(mutex_init_flag); 25*9880d681SAndroid Build Coastguard Worker initializeMutex()26*9880d681SAndroid Build Coastguard Workerstatic void initializeMutex() { 27*9880d681SAndroid Build Coastguard Worker ManagedStaticMutex = new sys::Mutex(); 28*9880d681SAndroid Build Coastguard Worker } 29*9880d681SAndroid Build Coastguard Worker getManagedStaticMutex()30*9880d681SAndroid Build Coastguard Workerstatic sys::Mutex* getManagedStaticMutex() { 31*9880d681SAndroid Build Coastguard Worker // We need to use a function local static here, since this can get called 32*9880d681SAndroid Build Coastguard Worker // during a static constructor and we need to guarantee that it's initialized 33*9880d681SAndroid Build Coastguard Worker // correctly. 34*9880d681SAndroid Build Coastguard Worker llvm::call_once(mutex_init_flag, initializeMutex); 35*9880d681SAndroid Build Coastguard Worker return ManagedStaticMutex; 36*9880d681SAndroid Build Coastguard Worker } 37*9880d681SAndroid Build Coastguard Worker RegisterManagedStatic(void * (* Creator)(),void (* Deleter)(void *)) const38*9880d681SAndroid Build Coastguard Workervoid ManagedStaticBase::RegisterManagedStatic(void *(*Creator)(), 39*9880d681SAndroid Build Coastguard Worker void (*Deleter)(void*)) const { 40*9880d681SAndroid Build Coastguard Worker assert(Creator); 41*9880d681SAndroid Build Coastguard Worker if (llvm_is_multithreaded()) { 42*9880d681SAndroid Build Coastguard Worker MutexGuard Lock(*getManagedStaticMutex()); 43*9880d681SAndroid Build Coastguard Worker 44*9880d681SAndroid Build Coastguard Worker if (!Ptr.load(std::memory_order_relaxed)) { 45*9880d681SAndroid Build Coastguard Worker void *Tmp = Creator(); 46*9880d681SAndroid Build Coastguard Worker 47*9880d681SAndroid Build Coastguard Worker Ptr.store(Tmp, std::memory_order_release); 48*9880d681SAndroid Build Coastguard Worker DeleterFn = Deleter; 49*9880d681SAndroid Build Coastguard Worker 50*9880d681SAndroid Build Coastguard Worker // Add to list of managed statics. 51*9880d681SAndroid Build Coastguard Worker Next = StaticList; 52*9880d681SAndroid Build Coastguard Worker StaticList = this; 53*9880d681SAndroid Build Coastguard Worker } 54*9880d681SAndroid Build Coastguard Worker } else { 55*9880d681SAndroid Build Coastguard Worker assert(!Ptr && !DeleterFn && !Next && 56*9880d681SAndroid Build Coastguard Worker "Partially initialized ManagedStatic!?"); 57*9880d681SAndroid Build Coastguard Worker Ptr = Creator(); 58*9880d681SAndroid Build Coastguard Worker DeleterFn = Deleter; 59*9880d681SAndroid Build Coastguard Worker 60*9880d681SAndroid Build Coastguard Worker // Add to list of managed statics. 61*9880d681SAndroid Build Coastguard Worker Next = StaticList; 62*9880d681SAndroid Build Coastguard Worker StaticList = this; 63*9880d681SAndroid Build Coastguard Worker } 64*9880d681SAndroid Build Coastguard Worker } 65*9880d681SAndroid Build Coastguard Worker destroy() const66*9880d681SAndroid Build Coastguard Workervoid ManagedStaticBase::destroy() const { 67*9880d681SAndroid Build Coastguard Worker assert(DeleterFn && "ManagedStatic not initialized correctly!"); 68*9880d681SAndroid Build Coastguard Worker assert(StaticList == this && 69*9880d681SAndroid Build Coastguard Worker "Not destroyed in reverse order of construction?"); 70*9880d681SAndroid Build Coastguard Worker // Unlink from list. 71*9880d681SAndroid Build Coastguard Worker StaticList = Next; 72*9880d681SAndroid Build Coastguard Worker Next = nullptr; 73*9880d681SAndroid Build Coastguard Worker 74*9880d681SAndroid Build Coastguard Worker // Destroy memory. 75*9880d681SAndroid Build Coastguard Worker DeleterFn(Ptr); 76*9880d681SAndroid Build Coastguard Worker 77*9880d681SAndroid Build Coastguard Worker // Cleanup. 78*9880d681SAndroid Build Coastguard Worker Ptr = nullptr; 79*9880d681SAndroid Build Coastguard Worker DeleterFn = nullptr; 80*9880d681SAndroid Build Coastguard Worker } 81*9880d681SAndroid Build Coastguard Worker 82*9880d681SAndroid Build Coastguard Worker /// llvm_shutdown - Deallocate and destroy all ManagedStatic variables. llvm_shutdown()83*9880d681SAndroid Build Coastguard Workervoid llvm::llvm_shutdown() { 84*9880d681SAndroid Build Coastguard Worker MutexGuard Lock(*getManagedStaticMutex()); 85*9880d681SAndroid Build Coastguard Worker 86*9880d681SAndroid Build Coastguard Worker while (StaticList) 87*9880d681SAndroid Build Coastguard Worker StaticList->destroy(); 88*9880d681SAndroid Build Coastguard Worker } 89