1 //===- subzero/src/IceMemory.h - Memory management declarations -*- C++ -*-===//
2 //
3 //                        The Subzero Code Generator
4 //
5 // This file is distributed under the University of Illinois Open Source
6 // License. See LICENSE.TXT for details.
7 //
8 //===----------------------------------------------------------------------===//
9 ///
10 /// \file
11 /// \brief Declares some useful data structures and routines dealing with
12 /// memory management in Subzero (mostly, allocator types.)
13 ///
14 //===----------------------------------------------------------------------===//
15 
16 #ifndef SUBZERO_SRC_ICEMEMORY_H
17 #define SUBZERO_SRC_ICEMEMORY_H
18 
19 #include "IceTLS.h"
20 
21 #include "llvm/Support/Allocator.h"
22 
23 #include <cstddef>
24 #include <mutex>
25 
26 namespace Ice {
27 
28 class Cfg;
29 class GlobalContext;
30 class Liveness;
31 
32 using ArenaAllocator =
33     llvm::BumpPtrAllocatorImpl<llvm::MallocAllocator, /*SlabSize=*/1024 * 1024>;
34 
35 class LockedArenaAllocator {
36   LockedArenaAllocator() = delete;
37   LockedArenaAllocator(const LockedArenaAllocator &) = delete;
38   LockedArenaAllocator &operator=(const LockedArenaAllocator &) = delete;
39 
40 public:
LockedArenaAllocator(ArenaAllocator * Alloc,std::mutex * Mutex)41   LockedArenaAllocator(ArenaAllocator *Alloc, std::mutex *Mutex)
42       : Alloc(Alloc), AutoLock(*Mutex) {}
43   LockedArenaAllocator(LockedArenaAllocator &&) = default;
44   LockedArenaAllocator &operator=(LockedArenaAllocator &&) = default;
45   ~LockedArenaAllocator() = default;
46 
47   ArenaAllocator *operator->() { return Alloc; }
48 
49 private:
50   ArenaAllocator *Alloc;
51   std::unique_lock<std::mutex> AutoLock;
52 };
53 
54 template <typename T, typename Traits> struct sz_allocator {
55   /// std::allocator interface implementation.
56   /// @{
57   using value_type = T;
58   using pointer = T *;
59   using const_pointer = const T *;
60   using reference = T &;
61   using const_reference = const T &;
62   using size_type = std::size_t;
63   using difference_type = std::ptrdiff_t;
64 
sz_allocatorsz_allocator65   sz_allocator() : Current() {}
66   template <class U>
sz_allocatorsz_allocator67   sz_allocator(const sz_allocator<U, Traits> &)
68       : Current() {}
69 
addresssz_allocator70   pointer address(reference x) const {
71     return reinterpret_cast<pointer>(&reinterpret_cast<char &>(x));
72   }
addresssz_allocator73   const_pointer address(const_reference x) const {
74     return reinterpret_cast<const_pointer>(&reinterpret_cast<const char &>(x));
75   }
76 
allocatesz_allocator77   pointer allocate(size_type num) {
78     assert(current() != nullptr);
79     return current()->template Allocate<T>(num);
80   }
81 
constructsz_allocator82   template <typename... A> void construct(pointer P, A &&... Args) {
83     new (static_cast<void *>(P)) T(std::forward<A>(Args)...);
84   }
85 
deallocatesz_allocator86   void deallocate(pointer, size_type) {}
87 
88   template <class U> struct rebind { typedef sz_allocator<U, Traits> other; };
89 
destroysz_allocator90   void destroy(pointer P) { P->~T(); }
91   /// @}
92 
93   /// Manages the current underlying allocator.
94   /// @{
currentsz_allocator95   typename Traits::allocator_type current() {
96     if (!Traits::cache_allocator) {
97       // TODO(jpp): allocators should always be cacheable... maybe. Investigate.
98       return Traits::current();
99     }
100     if (Current == nullptr) {
101       Current = Traits::current();
102     }
103     assert(Current == Traits::current());
104     return Current;
105   }
initsz_allocator106   static void init() { Traits::init(); }
107   /// @}
108   typename Traits::allocator_type Current;
109 };
110 
111 template <class Traits> struct sz_allocator_scope {
sz_allocator_scopesz_allocator_scope112   explicit sz_allocator_scope(typename Traits::manager_type *Manager) {
113     Traits::set_current(Manager);
114   }
115 
~sz_allocator_scopesz_allocator_scope116   ~sz_allocator_scope() { Traits::set_current(nullptr); }
117 };
118 
119 template <typename T, typename U, typename Traits>
120 inline bool operator==(const sz_allocator<T, Traits> &,
121                        const sz_allocator<U, Traits> &) {
122   return true;
123 }
124 
125 template <typename T, typename U, typename Traits>
126 inline bool operator!=(const sz_allocator<T, Traits> &,
127                        const sz_allocator<U, Traits> &) {
128   return false;
129 }
130 
131 class CfgAllocatorTraits {
132   CfgAllocatorTraits() = delete;
133   CfgAllocatorTraits(const CfgAllocatorTraits &) = delete;
134   CfgAllocatorTraits &operator=(const CfgAllocatorTraits &) = delete;
135   ~CfgAllocatorTraits() = delete;
136 
137 public:
138   using allocator_type = ArenaAllocator *;
139   using manager_type = Cfg;
140   static constexpr bool cache_allocator = false;
141 
init()142   static void init() { ICE_TLS_INIT_FIELD(CfgAllocator); };
143 
144   static allocator_type current();
145   static void set_current(const manager_type *Manager);
146   static void set_current(ArenaAllocator *Allocator);
147   static void set_current(std::nullptr_t);
148 
149 private:
150   ICE_TLS_DECLARE_FIELD(ArenaAllocator *, CfgAllocator);
151 };
152 
153 template <typename T>
154 using CfgLocalAllocator = sz_allocator<T, CfgAllocatorTraits>;
155 
156 using CfgLocalAllocatorScope = sz_allocator_scope<CfgAllocatorTraits>;
157 
158 class LivenessAllocatorTraits {
159   LivenessAllocatorTraits() = delete;
160   LivenessAllocatorTraits(const LivenessAllocatorTraits &) = delete;
161   LivenessAllocatorTraits &operator=(const LivenessAllocatorTraits &) = delete;
162   ~LivenessAllocatorTraits() = delete;
163 
164 public:
165   using allocator_type = ArenaAllocator *;
166   using manager_type = Liveness;
167   static constexpr bool cache_allocator = true;
168 
init()169   static void init() { ICE_TLS_INIT_FIELD(LivenessAllocator); };
170 
171   static allocator_type current();
172   static void set_current(const manager_type *Manager);
173 
174 private:
175   ICE_TLS_DECLARE_FIELD(ArenaAllocator *, LivenessAllocator);
176 };
177 
178 template <typename T>
179 using LivenessAllocator = sz_allocator<T, LivenessAllocatorTraits>;
180 
181 using LivenessAllocatorScope = sz_allocator_scope<LivenessAllocatorTraits>;
182 
183 } // end of namespace Ice
184 
185 #endif // SUBZERO_SRC_ICEMEMORY_H
186