1 /*
2  * Copyright (C) 2014 The Android Open Source Project
3  *
4  * Licensed under the Apache License, Version 2.0 (the "License");
5  * you may not use this file except in compliance with the License.
6  * You may obtain a copy of the License at
7  *
8  *      http://www.apache.org/licenses/LICENSE-2.0
9  *
10  * Unless required by applicable law or agreed to in writing, software
11  * distributed under the License is distributed on an "AS IS" BASIS,
12  * WITHOUT WARRANTIES OR CONDITIONS OF ANY KIND, either express or implied.
13  * See the License for the specific language governing permissions and
14  * limitations under the License.
15  */
16 
17 #ifndef ART_LIBARTBASE_BASE_SCOPED_ARENA_ALLOCATOR_H_
18 #define ART_LIBARTBASE_BASE_SCOPED_ARENA_ALLOCATOR_H_
19 
20 #include <android-base/logging.h>
21 
22 #include "arena_allocator.h"
23 #include "debug_stack.h"
24 #include "globals.h"
25 #include "macros.h"
26 
27 namespace art {
28 
29 class ArenaStack;
30 class ScopedArenaAllocator;
31 
32 template <typename T>
33 class ScopedArenaAllocatorAdapter;
34 
35 // Tag associated with each allocation to help prevent double free.
36 enum class ArenaFreeTag : uint8_t {
37   // Allocation is used and has not yet been destroyed.
38   kUsed,
39   // Allocation has been destroyed.
40   kFree,
41 };
42 
43 // Holds a list of Arenas for use by ScopedArenaAllocator stack.
44 // The memory is returned to the ArenaPool when the ArenaStack is destroyed.
45 class ArenaStack : private DebugStackRefCounter, private ArenaAllocatorMemoryTool {
46  public:
47   explicit ArenaStack(ArenaPool* arena_pool);
48   ~ArenaStack();
49 
50   using ArenaAllocatorMemoryTool::IsRunningOnMemoryTool;
51   using ArenaAllocatorMemoryTool::MakeDefined;
52   using ArenaAllocatorMemoryTool::MakeUndefined;
53   using ArenaAllocatorMemoryTool::MakeInaccessible;
54 
55   void Reset();
56 
PeakBytesAllocated()57   size_t PeakBytesAllocated() {
58     DebugStackRefCounter::CheckNoRefs();
59     return PeakStats()->BytesAllocated();
60   }
61 
62   size_t ApproximatePeakBytes();
63 
64   MemStats GetPeakStats() const;
65 
66   // Return the arena tag associated with a pointer.
ArenaTagForAllocation(void * ptr)67   static ArenaFreeTag& ArenaTagForAllocation(void* ptr) {
68     DCHECK(kIsDebugBuild) << "Only debug builds have tags";
69     return *(reinterpret_cast<ArenaFreeTag*>(ptr) - 1);
70   }
71 
72   // The alignment guaranteed for individual allocations.
73   static constexpr size_t kAlignment = 8u;
74 
75  private:
76   struct Peak;
77   struct Current;
78   template <typename Tag> struct TaggedStats : ArenaAllocatorStats { };
79   struct StatsAndPool : TaggedStats<Peak>, TaggedStats<Current> {
StatsAndPoolStatsAndPool80     explicit StatsAndPool(ArenaPool* arena_pool) : pool(arena_pool) { }
81     ArenaPool* const pool;
82   };
83 
PeakStats()84   ArenaAllocatorStats* PeakStats() {
85     return static_cast<TaggedStats<Peak>*>(&stats_and_pool_);
86   }
87 
PeakStats()88   const ArenaAllocatorStats* PeakStats() const {
89     return static_cast<const TaggedStats<Peak>*>(&stats_and_pool_);
90   }
91 
CurrentStats()92   ArenaAllocatorStats* CurrentStats() {
93     return static_cast<TaggedStats<Current>*>(&stats_and_pool_);
94   }
95 
96   // Private - access via ScopedArenaAllocator or ScopedArenaAllocatorAdapter.
Alloc(size_t bytes,ArenaAllocKind kind)97   void* Alloc(size_t bytes, ArenaAllocKind kind) ALWAYS_INLINE {
98     if (UNLIKELY(IsRunningOnMemoryTool())) {
99       return AllocWithMemoryTool(bytes, kind);
100     }
101     // Add kAlignment for the free or used tag. Required to preserve alignment.
102     size_t rounded_bytes = RoundUp(bytes + (kIsDebugBuild ? kAlignment : 0u), kAlignment);
103     uint8_t* ptr = top_ptr_;
104     if (UNLIKELY(static_cast<size_t>(top_end_ - ptr) < rounded_bytes)) {
105       ptr = AllocateFromNextArena(rounded_bytes);
106     }
107     CurrentStats()->RecordAlloc(bytes, kind);
108     top_ptr_ = ptr + rounded_bytes;
109     if (kIsDebugBuild) {
110       ptr += kAlignment;
111       ArenaTagForAllocation(ptr) = ArenaFreeTag::kUsed;
112     }
113     return ptr;
114   }
115 
116   uint8_t* AllocateFromNextArena(size_t rounded_bytes);
117   void UpdatePeakStatsAndRestore(const ArenaAllocatorStats& restore_stats);
118   void UpdateBytesAllocated();
119   void* AllocWithMemoryTool(size_t bytes, ArenaAllocKind kind);
120 
121   StatsAndPool stats_and_pool_;
122   Arena* bottom_arena_;
123   Arena* top_arena_;
124   uint8_t* top_ptr_;
125   uint8_t* top_end_;
126 
127   friend class ScopedArenaAllocator;
128   template <typename T>
129   friend class ScopedArenaAllocatorAdapter;
130 
131   DISALLOW_COPY_AND_ASSIGN(ArenaStack);
132 };
133 
134 // Fast single-threaded allocator. Allocated chunks are _not_ guaranteed to be zero-initialized.
135 //
136 // Unlike the ArenaAllocator, ScopedArenaAllocator is intended for relatively short-lived
137 // objects and allows nesting multiple allocators. Only the top allocator can be used but
138 // once it's destroyed, its memory can be reused by the next ScopedArenaAllocator on the
139 // stack. This is facilitated by returning the memory to the ArenaStack.
140 class ScopedArenaAllocator
141     : private DebugStackReference, private DebugStackRefCounter, private ArenaAllocatorStats {
142  public:
143   ScopedArenaAllocator(ScopedArenaAllocator&& other) noexcept;
144   explicit ScopedArenaAllocator(ArenaStack* arena_stack);
145   ~ScopedArenaAllocator();
146 
GetArenaStack()147   ArenaStack* GetArenaStack() const {
148     return arena_stack_;
149   }
150 
151   void Reset();
152 
153   void* Alloc(size_t bytes, ArenaAllocKind kind = kArenaAllocMisc) ALWAYS_INLINE {
154     DebugStackReference::CheckTop();
155     return arena_stack_->Alloc(bytes, kind);
156   }
157 
158   template <typename T>
159   T* Alloc(ArenaAllocKind kind = kArenaAllocMisc) {
160     return AllocArray<T>(1, kind);
161   }
162 
163   template <typename T>
164   T* AllocArray(size_t length, ArenaAllocKind kind = kArenaAllocMisc) {
165     return static_cast<T*>(Alloc(length * sizeof(T), kind));
166   }
167 
168   // Get adapter for use in STL containers. See scoped_arena_containers.h .
169   ScopedArenaAllocatorAdapter<void> Adapter(ArenaAllocKind kind = kArenaAllocSTL);
170 
171   size_t ApproximatePeakBytes();
172 
173   // Allow a delete-expression to destroy but not deallocate allocators created by Create().
delete(void * ptr)174   static void operator delete([[maybe_unused]] void* ptr) {}
175 
176  private:
177   ArenaStack* arena_stack_;
178   Arena* mark_arena_;
179   uint8_t* mark_ptr_;
180   uint8_t* mark_end_;
181 
182   void DoReset();
183 
184   template <typename T>
185   friend class ScopedArenaAllocatorAdapter;
186 
187   DISALLOW_COPY_AND_ASSIGN(ScopedArenaAllocator);
188 };
189 
190 }  // namespace art
191 
192 #endif  // ART_LIBARTBASE_BASE_SCOPED_ARENA_ALLOCATOR_H_
193