Add llvm_start_multithreaded(), which starts up the LLVM internals in thread-safe mode. Provide double-check locking

initialization of ManagedStatic's when running in thread-safe mode.


git-svn-id: https://llvm.org/svn/llvm-project/llvm/trunk@72151 91177308-0d34-0410-b5e6-96231b3b80d8
This commit is contained in:
Owen Anderson 2009-05-20 00:39:20 +00:00
parent 513fae2db5
commit b4d97b78df
2 changed files with 78 additions and 19 deletions

View File

@ -14,8 +14,16 @@
#ifndef LLVM_SUPPORT_MANAGED_STATIC_H
#define LLVM_SUPPORT_MANAGED_STATIC_H
#include "llvm/System/Atomic.h"
namespace llvm {
/// object_creator - Helper method for ManagedStatic.
template<class C>
void* object_creator() {
return new C();
}
/// object_deleter - Helper method for ManagedStatic.
///
template<class C>
@ -32,7 +40,7 @@ protected:
mutable void (*DeleterFn)(void*);
mutable const ManagedStaticBase *Next;
void RegisterManagedStatic(void *ObjPtr, void (*deleter)(void*)) const;
void RegisterManagedStatic(void *(*creator)(), void (*deleter)(void*)) const;
public:
/// isConstructed - Return true if this object has not been created yet.
bool isConstructed() const { return Ptr != 0; }
@ -51,25 +59,32 @@ public:
// Accessors.
C &operator*() {
if (!Ptr) LazyInit();
void* tmp = Ptr;
sys::MemoryFence();
if (!tmp) RegisterManagedStatic(object_creator<C>, object_deleter<C>);
return *static_cast<C*>(Ptr);
}
C *operator->() {
if (!Ptr) LazyInit();
void* tmp = Ptr;
sys::MemoryFence();
if (!tmp) RegisterManagedStatic(object_creator<C>, object_deleter<C>);
return static_cast<C*>(Ptr);
}
const C &operator*() const {
if (!Ptr) LazyInit();
void* tmp = Ptr;
sys::MemoryFence();
if (!tmp) RegisterManagedStatic(object_creator<C>, object_deleter<C>);
return *static_cast<C*>(Ptr);
}
const C *operator->() const {
if (!Ptr) LazyInit();
return static_cast<C*>(Ptr);
}
void* tmp = Ptr;
sys::MemoryFence();
if (!tmp) RegisterManagedStatic(object_creator<C>, object_deleter<C>);
public:
void LazyInit() const {
RegisterManagedStatic(new C(), object_deleter<C>);
return static_cast<C*>(Ptr);
}
};
@ -80,6 +95,10 @@ public:
};
/// llvm_start_multithreaded - Allocate and initialize structures needed to
/// make LLVM safe for multithreading.
void llvm_start_multithreaded();
/// llvm_shutdown - Deallocate and destroy all ManagedStatic variables.
void llvm_shutdown();
@ -87,7 +106,10 @@ void llvm_shutdown();
/// llvm_shutdown_obj - This is a simple helper class that calls
/// llvm_shutdown() when it is destroyed.
struct llvm_shutdown_obj {
llvm_shutdown_obj() {}
llvm_shutdown_obj() { }
explicit llvm_shutdown_obj(bool multithreaded) {
if (multithreaded) llvm_start_multithreaded();
}
~llvm_shutdown_obj() { llvm_shutdown(); }
};

View File

@ -12,21 +12,44 @@
//===----------------------------------------------------------------------===//
#include "llvm/Support/ManagedStatic.h"
#include "llvm/Config/config.h"
#include "llvm/System/Atomic.h"
#include "llvm/System/Mutex.h"
#include <cassert>
using namespace llvm;
static const ManagedStaticBase *StaticList = 0;
void ManagedStaticBase::RegisterManagedStatic(void *ObjPtr,
static sys::Mutex* ManagedStaticMutex = 0;
void ManagedStaticBase::RegisterManagedStatic(void *(*Creator)(),
void (*Deleter)(void*)) const {
assert(Ptr == 0 && DeleterFn == 0 && Next == 0 &&
"Partially init static?");
Ptr = ObjPtr;
DeleterFn = Deleter;
if (ManagedStaticMutex) {
ManagedStaticMutex->acquire();
if (Ptr == 0) {
void* tmp = Creator ? Creator() : 0;
sys::MemoryFence();
Ptr = tmp;
DeleterFn = Deleter;
// Add to list of managed statics.
Next = StaticList;
StaticList = this;
}
ManagedStaticMutex->release();
} else {
assert(Ptr == 0 && DeleterFn == 0 && Next == 0 &&
"Partially initialized ManagedStatic!?");
Ptr = Creator ? Creator() : 0;
DeleterFn = Deleter;
// Add to list of managed statics.
Next = StaticList;
StaticList = this;
// Add to list of managed statics.
Next = StaticList;
StaticList = this;
}
}
void ManagedStaticBase::destroy() const {
@ -45,9 +68,23 @@ void ManagedStaticBase::destroy() const {
DeleterFn = 0;
}
void llvm::llvm_start_multithreaded() {
#if LLVM_MULTITHREADED
assert(ManagedStaticMutex == 0 && "Multithreaded LLVM already initialized!");
ManagedStaticMutex = new sys::Mutex(true);
#else
assert(0 && "LLVM built without multithreading support!");
#endif
}
/// llvm_shutdown - Deallocate and destroy all ManagedStatic variables.
void llvm::llvm_shutdown() {
while (StaticList)
StaticList->destroy();
if (ManagedStaticMutex) {
delete ManagedStaticMutex;
ManagedStaticMutex = 0;
}
}