summaryrefslogtreecommitdiff
path: root/src/mixed_arena.h
diff options
context:
space:
mode:
authorAlon Zakai <alonzakai@gmail.com>2016-04-22 22:01:43 -0700
committerAlon Zakai <alonzakai@gmail.com>2016-04-26 19:50:14 -0700
commitd9096704dfa84aea94bb6ac2ffa02d88fa546bfa (patch)
tree817992a1e811af71087a228b20f3c35df9485b96 /src/mixed_arena.h
parent7fd8aac5084223c48901ca20f9d233e9764d536a (diff)
downloadbinaryen-d9096704dfa84aea94bb6ac2ffa02d88fa546bfa.tar.gz
binaryen-d9096704dfa84aea94bb6ac2ffa02d88fa546bfa.tar.bz2
binaryen-d9096704dfa84aea94bb6ac2ffa02d88fa546bfa.zip
add an ArenaVector for internal array allocations in expression nodes
Diffstat (limited to 'src/mixed_arena.h')
-rw-r--r--src/mixed_arena.h131
1 files changed, 130 insertions, 1 deletions
diff --git a/src/mixed_arena.h b/src/mixed_arena.h
index d8b487e4d..930bf3c0e 100644
--- a/src/mixed_arena.h
+++ b/src/mixed_arena.h
@@ -107,7 +107,7 @@ struct MixedArena {
template<class T>
T* alloc() {
auto* ret = static_cast<T*>(allocSpace(sizeof(T)));
- new (ret) T();
+ new (ret) T(*this); // allocated objects receive the allocator, so they can allocate more later if necessary
return ret;
}
@@ -124,4 +124,133 @@ 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 <typename T>
+class ArenaVector {
+ MixedArena& allocator;
+ T* data = nullptr;
+ size_t usedElements = 0,
+ allocatedElements = 0;
+
+ void allocate(size_t size) {
+ allocatedElements = size;
+ data = static_cast<T*>(allocator.allocSpace(sizeof(T) * allocatedElements));
+ }
+
+ void reallocate(size_t size) {
+ T* old = data;
+ allocate(size);
+ for (size_t i = 0; i < usedElements; i++) {
+ data[i] = old[i];
+ }
+ }
+
+public:
+ ArenaVector(MixedArena& allocator) : allocator(allocator) {}
+
+ ArenaVector(ArenaVector<T>&& other) : allocator(other.allocator) {
+ *this = other;
+ }
+
+ T& operator[](size_t index) const {
+ assert(index < usedElements);
+ return data[index];
+ }
+
+ size_t size() const {
+ return usedElements;
+ }
+
+ void resize(size_t size) {
+ if (size > allocatedElements) {
+ reallocate(size);
+ }
+ // construct new elements
+ for (size_t i = usedElements; i < size; i++) {
+ new (data + i) T();
+ }
+ usedElements = size;
+ }
+
+ T& back() const {
+ assert(usedElements > 0);
+ return data[usedElements - 1];
+ }
+
+ T& pop_back() {
+ assert(usedElements > 0);
+ usedElements--;
+ return data[usedElements];
+ }
+
+ void push_back(T item) {
+ if (usedElements == allocatedElements) {
+ reallocate((allocatedElements + 1) * 2); // TODO: optimize
+ }
+ data[usedElements] = item;
+ usedElements++;
+ }
+
+ template<typename ListType>
+ void set(ListType& list) {
+ size_t size = list.size();
+ if (allocatedElements < size) {
+ allocate(size);
+ }
+ for (size_t i = 0; i < size; i++) {
+ data[i] = list[i];
+ }
+ usedElements = size;
+ }
+
+ void operator=(ArenaVector<T>& other) {
+ set(other);
+ }
+
+ void operator=(ArenaVector<T>&& other) {
+ data = other.data;
+ usedElements = other.usedElements;
+ allocatedElements = other.allocatedElements;
+ other.data = nullptr;
+ other.usedElements = other.allocatedElements = 0;
+ }
+
+ // iteration
+
+ struct Iterator {
+ const ArenaVector<T>* parent;
+ size_t index;
+
+ Iterator(const ArenaVector<T>* parent, size_t index) : parent(parent), index(index) {}
+
+ bool operator!=(const Iterator& other) const {
+ return index != other.index || parent != other.parent;
+ }
+
+ void operator++() {
+ index++;
+ }
+
+ T& operator*() {
+ return (*parent)[index];
+ }
+ };
+
+ Iterator begin() const {
+ return Iterator(this, 0);
+ }
+ Iterator end() const {
+ return Iterator(this, usedElements);
+ }
+};
+
#endif // wasm_mixed_arena_h