2008-02-19 18:06:52 +00:00
|
|
|
/* Copyright (c) 2008, Avian Contributors
|
|
|
|
|
|
|
|
Permission to use, copy, modify, and/or distribute this software
|
|
|
|
for any purpose with or without fee is hereby granted, provided
|
|
|
|
that the above copyright notice and this permission notice appear
|
|
|
|
in all copies.
|
|
|
|
|
|
|
|
There is NO WARRANTY for this software. See license.txt for
|
|
|
|
details. */
|
|
|
|
|
2007-12-09 20:03:21 +00:00
|
|
|
#ifndef VECTOR_H
|
|
|
|
#define VECTOR_H
|
|
|
|
|
|
|
|
#include "system.h"
|
|
|
|
|
|
|
|
namespace vm {
|
|
|
|
|
|
|
|
class Vector {
|
|
|
|
public:
|
2008-01-14 23:37:24 +00:00
|
|
|
Vector(System* s, Allocator* allocator, unsigned minimumCapacity):
|
2007-12-09 20:03:21 +00:00
|
|
|
s(s),
|
2008-01-13 22:05:08 +00:00
|
|
|
allocator(allocator),
|
2007-12-09 20:03:21 +00:00
|
|
|
data(0),
|
|
|
|
position(0),
|
|
|
|
capacity(0),
|
|
|
|
minimumCapacity(minimumCapacity)
|
|
|
|
{ }
|
|
|
|
|
|
|
|
~Vector() {
|
|
|
|
dispose();
|
|
|
|
}
|
|
|
|
|
|
|
|
void dispose() {
|
2007-12-14 00:27:09 +00:00
|
|
|
if (data and minimumCapacity >= 0) {
|
2008-04-13 18:15:04 +00:00
|
|
|
allocator->free(data, capacity);
|
2007-12-09 20:03:21 +00:00
|
|
|
}
|
|
|
|
}
|
|
|
|
|
2007-12-14 00:27:09 +00:00
|
|
|
void wrap(uint8_t* data, unsigned capacity) {
|
|
|
|
dispose();
|
|
|
|
|
|
|
|
this->data = data;
|
|
|
|
this->position = 0;
|
|
|
|
this->capacity = capacity;
|
|
|
|
this->minimumCapacity = -1;
|
|
|
|
}
|
|
|
|
|
2007-12-09 20:03:21 +00:00
|
|
|
void ensure(unsigned space) {
|
|
|
|
if (position + space > capacity) {
|
2007-12-14 00:27:09 +00:00
|
|
|
assert(s, minimumCapacity >= 0);
|
|
|
|
|
2007-12-09 20:03:21 +00:00
|
|
|
unsigned newCapacity = max
|
|
|
|
(position + space, max(minimumCapacity, capacity * 2));
|
2008-01-13 22:05:08 +00:00
|
|
|
uint8_t* newData = static_cast<uint8_t*>
|
2008-04-13 18:15:04 +00:00
|
|
|
(allocator->allocate(newCapacity));
|
2007-12-09 20:03:21 +00:00
|
|
|
if (data) {
|
|
|
|
memcpy(newData, data, position);
|
2008-04-13 18:15:04 +00:00
|
|
|
allocator->free(data, capacity);
|
2007-12-09 20:03:21 +00:00
|
|
|
}
|
|
|
|
data = newData;
|
2007-12-11 00:48:09 +00:00
|
|
|
capacity = newCapacity;
|
2007-12-09 20:03:21 +00:00
|
|
|
}
|
|
|
|
}
|
|
|
|
|
|
|
|
void get(unsigned offset, void* dst, unsigned size) {
|
|
|
|
assert(s, offset + size <= position);
|
2007-12-09 22:45:43 +00:00
|
|
|
memcpy(dst, data + offset, size);
|
2007-12-09 20:03:21 +00:00
|
|
|
}
|
|
|
|
|
|
|
|
void set(unsigned offset, const void* src, unsigned size) {
|
|
|
|
assert(s, offset + size <= position);
|
2007-12-09 22:45:43 +00:00
|
|
|
memcpy(data + offset, src, size);
|
2007-12-09 20:03:21 +00:00
|
|
|
}
|
|
|
|
|
|
|
|
void pop(void* dst, unsigned size) {
|
|
|
|
get(position - size, dst, size);
|
|
|
|
position -= size;
|
|
|
|
}
|
|
|
|
|
2007-12-09 22:45:43 +00:00
|
|
|
void* allocate(unsigned size) {
|
2007-12-09 20:03:21 +00:00
|
|
|
ensure(size);
|
|
|
|
void* r = data + position;
|
|
|
|
position += size;
|
|
|
|
return r;
|
|
|
|
}
|
|
|
|
|
2007-12-09 22:45:43 +00:00
|
|
|
void* append(const void* p, unsigned size) {
|
|
|
|
void* r = allocate(size);
|
|
|
|
memcpy(r, p, size);
|
|
|
|
return r;
|
|
|
|
}
|
|
|
|
|
2007-12-09 20:03:21 +00:00
|
|
|
void append(uint8_t v) {
|
|
|
|
append(&v, 1);
|
|
|
|
}
|
|
|
|
|
|
|
|
void append2(uint16_t v) {
|
|
|
|
append(&v, 2);
|
|
|
|
}
|
|
|
|
|
|
|
|
void append4(uint32_t v) {
|
|
|
|
append(&v, 4);
|
|
|
|
}
|
|
|
|
|
|
|
|
void appendAddress(uintptr_t v) {
|
|
|
|
append(&v, BytesPerWord);
|
|
|
|
}
|
|
|
|
|
2007-12-11 00:48:09 +00:00
|
|
|
void appendAddress(void* v) {
|
|
|
|
append(&v, BytesPerWord);
|
|
|
|
}
|
|
|
|
|
2008-01-07 16:01:35 +00:00
|
|
|
unsigned get(unsigned offset) {
|
|
|
|
uint8_t v; get(offset, &v, 1);
|
|
|
|
return v;
|
|
|
|
}
|
|
|
|
|
|
|
|
unsigned get2(unsigned offset) {
|
|
|
|
uint16_t v; get(offset, &v, 2);
|
|
|
|
return v;
|
|
|
|
}
|
|
|
|
|
|
|
|
unsigned get4(unsigned offset) {
|
|
|
|
uint32_t v; get(offset, &v, 4);
|
|
|
|
return v;
|
|
|
|
}
|
|
|
|
|
|
|
|
uintptr_t getAddress(unsigned offset) {
|
|
|
|
uintptr_t v; get(offset, &v, BytesPerWord);
|
|
|
|
return v;
|
|
|
|
}
|
|
|
|
|
2007-12-09 22:45:43 +00:00
|
|
|
unsigned length() {
|
|
|
|
return position;
|
2007-12-09 20:03:21 +00:00
|
|
|
}
|
|
|
|
|
2007-12-09 22:45:43 +00:00
|
|
|
template <class T>
|
|
|
|
T* peek(unsigned offset) {
|
|
|
|
assert(s, offset + sizeof(T) <= position);
|
|
|
|
return reinterpret_cast<T*>(data + offset);
|
|
|
|
}
|
2007-12-09 20:03:21 +00:00
|
|
|
|
|
|
|
System* s;
|
2008-01-13 22:05:08 +00:00
|
|
|
Allocator* allocator;
|
2007-12-09 20:03:21 +00:00
|
|
|
uint8_t* data;
|
|
|
|
unsigned position;
|
|
|
|
unsigned capacity;
|
2007-12-14 00:27:09 +00:00
|
|
|
int minimumCapacity;
|
2007-12-09 20:03:21 +00:00
|
|
|
};
|
|
|
|
|
|
|
|
} // namespace vm
|
|
|
|
|
|
|
|
#endif//VECTOR_H
|