diff options
| author | 2015-11-03 21:15:50 +0000 | |
|---|---|---|
| committer | 2015-11-03 21:15:50 +0000 | |
| commit | f33ea7911dadab2e49c4634cb1763f0526047e89 (patch) | |
| tree | d80b0a7ac8d15a674f1ee2c2c842fce3b9205754 /runtime/lambda | |
| parent | 530dcc8b74a0f4e5f54e2b2c326580d58594a2f7 (diff) | |
| parent | 30c475a2046951a81769c2db0b2dad66cd71e189 (diff) | |
Merge "lambda: Minor capture-variable/liberate-variable clean-up after post-merge reviews."
Diffstat (limited to 'runtime/lambda')
| -rw-r--r-- | runtime/lambda/leaking_allocator.cc | 7 | ||||
| -rw-r--r-- | runtime/lambda/leaking_allocator.h | 23 |
2 files changed, 25 insertions, 5 deletions
diff --git a/runtime/lambda/leaking_allocator.cc b/runtime/lambda/leaking_allocator.cc index 4910732a6c..22bb294d03 100644 --- a/runtime/lambda/leaking_allocator.cc +++ b/runtime/lambda/leaking_allocator.cc @@ -14,6 +14,7 @@ * limitations under the License. */ +#include "base/bit_utils.h" #include "lambda/leaking_allocator.h" #include "linear_alloc.h" #include "runtime.h" @@ -21,9 +22,11 @@ namespace art { namespace lambda { -void* LeakingAllocator::AllocateMemory(Thread* self, size_t byte_size) { +void* LeakingAllocator::AllocateMemoryImpl(Thread* self, size_t byte_size, size_t align_size) { // TODO: use GetAllocatorForClassLoader to allocate lambda ArtMethod data. - return Runtime::Current()->GetLinearAlloc()->Alloc(self, byte_size); + void* mem = Runtime::Current()->GetLinearAlloc()->Alloc(self, byte_size); + DCHECK_ALIGNED_PARAM(reinterpret_cast<uintptr_t>(mem), align_size); + return mem; } } // namespace lambda diff --git a/runtime/lambda/leaking_allocator.h b/runtime/lambda/leaking_allocator.h index c3222d0485..cb5a1bf4c3 100644 --- a/runtime/lambda/leaking_allocator.h +++ b/runtime/lambda/leaking_allocator.h @@ -17,6 +17,7 @@ #define ART_RUNTIME_LAMBDA_LEAKING_ALLOCATOR_H_ #include <utility> // std::forward +#include <type_traits> // std::aligned_storage namespace art { class Thread; // forward declaration @@ -33,20 +34,36 @@ namespace lambda { // TODO: do all of the above a/b for each callsite, and delete this class. class LeakingAllocator { public: + // An opaque type which is guaranteed for: + // * a) be large enough to hold T (e.g. for in-place new) + // * b) be well-aligned (so that reads/writes are well-defined) to T + // * c) strict-aliasing compatible with T* + // + // Nominally used to allocate memory for yet unconstructed instances of T. + template <typename T> + using AlignedMemoryStorage = typename std::aligned_storage<sizeof(T), alignof(T)>::type; + // Allocate byte_size bytes worth of memory. Never freed. - static void* AllocateMemory(Thread* self, size_t byte_size); + template <typename T> + static AlignedMemoryStorage<T>* AllocateMemory(Thread* self, size_t byte_size = sizeof(T)) { + return reinterpret_cast<AlignedMemoryStorage<T>*>( + AllocateMemoryImpl(self, byte_size, alignof(T))); + } // Make a new instance of T, flexibly sized, in-place at newly allocated memory. Never freed. template <typename T, typename... Args> static T* MakeFlexibleInstance(Thread* self, size_t byte_size, Args&&... args) { - return new (AllocateMemory(self, byte_size)) T(std::forward<Args>(args)...); + return new (AllocateMemory<T>(self, byte_size)) T(std::forward<Args>(args)...); } // Make a new instance of T in-place at newly allocated memory. Never freed. template <typename T, typename... Args> static T* MakeInstance(Thread* self, Args&&... args) { - return new (AllocateMemory(self, sizeof(T))) T(std::forward<Args>(args)...); + return new (AllocateMemory<T>(self, sizeof(T))) T(std::forward<Args>(args)...); } + + private: + static void* AllocateMemoryImpl(Thread* self, size_t byte_size, size_t align_size); }; } // namespace lambda |