blob: 098cccb68df51880522f63ff520fccf57e4ba0b9 [file] [log] [blame]
Chris Lattner771cbf32006-09-28 00:31:55 +00001//===-- ManagedStatic.cpp - Static Global wrapper -------------------------===//
2//
3// The LLVM Compiler Infrastructure
4//
Chris Lattner4ee451d2007-12-29 20:36:04 +00005// This file is distributed under the University of Illinois Open Source
6// License. See LICENSE.TXT for details.
Chris Lattner771cbf32006-09-28 00:31:55 +00007//
8//===----------------------------------------------------------------------===//
9//
10// This file implements the ManagedStatic class and llvm_shutdown().
11//
12//===----------------------------------------------------------------------===//
13
14#include "llvm/Support/ManagedStatic.h"
Owen Andersonb4d97b72009-05-20 00:39:20 +000015#include "llvm/Config/config.h"
Michael J. Spencer1f6efa32010-11-29 18:16:10 +000016#include "llvm/Support/Atomic.h"
Chris Lattner771cbf32006-09-28 00:31:55 +000017#include <cassert>
18using namespace llvm;
19
20static const ManagedStaticBase *StaticList = 0;
21
Owen Andersonb4d97b72009-05-20 00:39:20 +000022void ManagedStaticBase::RegisterManagedStatic(void *(*Creator)(),
Chris Lattner771cbf32006-09-28 00:31:55 +000023 void (*Deleter)(void*)) const {
Owen Anderson4c7ac182009-06-16 17:33:51 +000024 if (llvm_is_multithreaded()) {
25 llvm_acquire_global_lock();
Owen Andersonb4d97b72009-05-20 00:39:20 +000026
27 if (Ptr == 0) {
28 void* tmp = Creator ? Creator() : 0;
29
Nick Lewycky4d0a9ff2011-11-14 20:50:16 +000030 TsanHappensBefore(this);
Owen Andersonb4d97b72009-05-20 00:39:20 +000031 sys::MemoryFence();
Nick Lewycky4d0a9ff2011-11-14 20:50:16 +000032
33 // This write is racy against the first read in the ManagedStatic
34 // accessors. The race is benign because it does a second read after a
35 // memory fence, at which point it isn't possible to get a partial value.
36 TsanIgnoreWritesBegin();
Owen Andersonb4d97b72009-05-20 00:39:20 +000037 Ptr = tmp;
Nick Lewycky4d0a9ff2011-11-14 20:50:16 +000038 TsanIgnoreWritesEnd();
Owen Andersonb4d97b72009-05-20 00:39:20 +000039 DeleterFn = Deleter;
40
41 // Add to list of managed statics.
42 Next = StaticList;
43 StaticList = this;
44 }
45
Owen Anderson4c7ac182009-06-16 17:33:51 +000046 llvm_release_global_lock();
Owen Andersonb4d97b72009-05-20 00:39:20 +000047 } else {
48 assert(Ptr == 0 && DeleterFn == 0 && Next == 0 &&
Bill Wendling51b16f42009-05-30 01:09:53 +000049 "Partially initialized ManagedStatic!?");
Owen Andersonb4d97b72009-05-20 00:39:20 +000050 Ptr = Creator ? Creator() : 0;
51 DeleterFn = Deleter;
Chris Lattner771cbf32006-09-28 00:31:55 +000052
Owen Andersonb4d97b72009-05-20 00:39:20 +000053 // Add to list of managed statics.
54 Next = StaticList;
55 StaticList = this;
56 }
Chris Lattner771cbf32006-09-28 00:31:55 +000057}
58
59void ManagedStaticBase::destroy() const {
Chris Lattnerd2835662007-02-20 06:18:57 +000060 assert(DeleterFn && "ManagedStatic not initialized correctly!");
Chris Lattner771cbf32006-09-28 00:31:55 +000061 assert(StaticList == this &&
62 "Not destroyed in reverse order of construction?");
63 // Unlink from list.
64 StaticList = Next;
65 Next = 0;
66
67 // Destroy memory.
68 DeleterFn(Ptr);
69
70 // Cleanup.
71 Ptr = 0;
72 DeleterFn = 0;
73}
74
75/// llvm_shutdown - Deallocate and destroy all ManagedStatic variables.
Chris Lattner151880b2006-09-29 18:43:14 +000076void llvm::llvm_shutdown() {
Chris Lattner771cbf32006-09-28 00:31:55 +000077 while (StaticList)
78 StaticList->destroy();
Owen Andersonb4d97b72009-05-20 00:39:20 +000079
Owen Anderson4c7ac182009-06-16 17:33:51 +000080 if (llvm_is_multithreaded()) llvm_stop_multithreaded();
Chris Lattner771cbf32006-09-28 00:31:55 +000081}