From e8d6188499260d599b2002d1b17405220d22869a Mon Sep 17 00:00:00 2001 From: "Alon Zakai (kripken)" Date: Tue, 31 Jan 2017 14:17:39 -0800 Subject: use MixedArena in asm.js ast --- src/mixed_arena.h | 76 ++++++++++++++++++++++++++++++++++++------------------- 1 file changed, 50 insertions(+), 26 deletions(-) (limited to 'src/mixed_arena.h') diff --git a/src/mixed_arena.h b/src/mixed_arena.h index 28b2dd462..52e47fbde 100644 --- a/src/mixed_arena.h +++ b/src/mixed_arena.h @@ -144,39 +144,24 @@ struct MixedArena { // // A vector that allocates in an arena. // -// TODO: consider not saving the allocator, but requiring it be -// passed in when needed, would make this (and thus Blocks etc. -// smaller) -// // TODO: specialize on the initial size of the array -template -class ArenaVector { - MixedArena& allocator; +template +class ArenaVectorBase { +protected: T* data = nullptr; size_t usedElements = 0, allocatedElements = 0; - void allocate(size_t size) { - allocatedElements = size; - data = static_cast(allocator.allocSpace(sizeof(T) * allocatedElements)); - } - void reallocate(size_t size) { T* old = data; - allocate(size); + static_cast(this)->allocate(size); for (size_t i = 0; i < usedElements; i++) { data[i] = old[i]; } } public: - ArenaVector(MixedArena& allocator) : allocator(allocator) {} - - ArenaVector(ArenaVector&& other) : allocator(other.allocator) { - *this = other; - } - T& operator[](size_t index) const { assert(index < usedElements); return data[index]; @@ -216,11 +201,21 @@ public: usedElements++; } + void clear() { + usedElements = 0; + } + + void reserve(size_t size) { + if (size > allocatedElements) { + reallocate(size); + } + } + template void set(const ListType& list) { size_t size = list.size(); if (allocatedElements < size) { - allocate(size); + static_cast(this)->allocate(size); } for (size_t i = 0; i < size; i++) { data[i] = list[i]; @@ -228,14 +223,15 @@ public: usedElements = size; } - void operator=(ArenaVector& other) { + void operator=(SubType& other) { set(other); } - void operator=(ArenaVector&& other) { + void swap(SubType& other) { data = other.data; usedElements = other.usedElements; allocatedElements = other.allocatedElements; + other.data = nullptr; other.usedElements = other.allocatedElements = 0; } @@ -243,10 +239,10 @@ public: // iteration struct Iterator { - const ArenaVector* parent; + const SubType* parent; size_t index; - Iterator(const ArenaVector* parent, size_t index) : parent(parent), index(index) {} + Iterator(const SubType* parent, size_t index) : parent(parent), index(index) {} bool operator!=(const Iterator& other) const { return index != other.index || parent != other.parent; @@ -262,10 +258,38 @@ public: }; Iterator begin() const { - return Iterator(this, 0); + return Iterator(static_cast(this), 0); } Iterator end() const { - return Iterator(this, usedElements); + return Iterator(static_cast(this), usedElements); + } + + void allocate(size_t size) { + abort(); // must be implemented in children + } +}; + +// A vector that has an allocator for arena allocation +// +// TODO: consider not saving the allocator, but requiring it be +// passed in when needed, would make this (and thus Blocks etc. +// smaller) + +template +class ArenaVector : public ArenaVectorBase, T> { +private: + MixedArena& allocator; + +public: + ArenaVector(MixedArena& allocator) : allocator(allocator) {} + + ArenaVector(ArenaVector&& other) : allocator(other.allocator) { + *this = other; + } + + void allocate(size_t size) { + this->allocatedElements = size; + this->data = static_cast(allocator.allocSpace(sizeof(T) * this->allocatedElements)); } }; -- cgit v1.2.3