| //===-- llvm/Support/ManagedStatic.h - Static Global wrapper ----*- C++ -*-===// |
| // |
| // The LLVM Compiler Infrastructure |
| // |
| // This file is distributed under the University of Illinois Open Source |
| // License. See LICENSE.TXT for details. |
| // |
| //===----------------------------------------------------------------------===// |
| // |
| // This file defines the ManagedStatic class and the llvm_shutdown() function. |
| // |
| //===----------------------------------------------------------------------===// |
| |
| #ifndef LLVM_SUPPORT_MANAGEDSTATIC_H |
| #define LLVM_SUPPORT_MANAGEDSTATIC_H |
| |
| #include "llvm/Support/Compiler.h" |
| #include <atomic> |
| #include <cstddef> |
| |
| namespace llvm { |
| |
| /// object_creator - Helper method for ManagedStatic. |
| template<class C> |
| LLVM_LIBRARY_VISIBILITY void* object_creator() { |
| return new C(); |
| } |
| |
| /// object_deleter - Helper method for ManagedStatic. |
| /// |
| template <typename T> struct LLVM_LIBRARY_VISIBILITY object_deleter { |
| static void call(void *Ptr) { delete (T *)Ptr; } |
| }; |
| template <typename T, size_t N> |
| struct LLVM_LIBRARY_VISIBILITY object_deleter<T[N]> { |
| static void call(void *Ptr) { delete[](T *)Ptr; } |
| }; |
| |
| /// ManagedStaticBase - Common base class for ManagedStatic instances. |
| class ManagedStaticBase { |
| protected: |
| // This should only be used as a static variable, which guarantees that this |
| // will be zero initialized. |
| mutable std::atomic<void *> Ptr; |
| mutable void (*DeleterFn)(void*); |
| mutable const ManagedStaticBase *Next; |
| |
| 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 != nullptr; } |
| |
| void destroy() const; |
| }; |
| |
| /// ManagedStatic - This transparently changes the behavior of global statics to |
| /// be lazily constructed on demand (good for reducing startup times of dynamic |
| /// libraries that link in LLVM components) and for making destruction be |
| /// explicit through the llvm_shutdown() function call. |
| /// |
| template<class C> |
| class ManagedStatic : public ManagedStaticBase { |
| public: |
| // Accessors. |
| C &operator*() { |
| void *Tmp = Ptr.load(std::memory_order_acquire); |
| if (!Tmp) |
| RegisterManagedStatic(object_creator<C>, object_deleter<C>::call); |
| |
| return *static_cast<C *>(Ptr.load(std::memory_order_relaxed)); |
| } |
| |
| C *operator->() { return &**this; } |
| |
| const C &operator*() const { |
| void *Tmp = Ptr.load(std::memory_order_acquire); |
| if (!Tmp) |
| RegisterManagedStatic(object_creator<C>, object_deleter<C>::call); |
| |
| return *static_cast<C *>(Ptr.load(std::memory_order_relaxed)); |
| } |
| |
| const C *operator->() const { return &**this; } |
| }; |
| |
| /// llvm_shutdown - Deallocate and destroy all ManagedStatic variables. |
| 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() = default; |
| ~llvm_shutdown_obj() { llvm_shutdown(); } |
| }; |
| |
| } // end namespace llvm |
| |
| #endif // LLVM_SUPPORT_MANAGEDSTATIC_H |