mirror of
https://github.com/odin-lang/Odin.git
synced 2025-12-29 17:34:34 +00:00
Currently, the Queue will never reuse it's full capacity if you call `pop_front`, even if you empty it before pushing more items. With this change, if you empty the Queue with `pop_front`, then the offset will be set back to the start of the underlying array when you pop the last item. Future pushes will then reuse the already-allocated--but now empty--space.
176 lines
3.6 KiB
Odin
176 lines
3.6 KiB
Odin
package container
|
|
|
|
Queue :: struct(T: typeid) {
|
|
data: Array(T),
|
|
len: int,
|
|
offset: int,
|
|
}
|
|
|
|
/*
|
|
queue_init :: proc{
|
|
queue_init_none,
|
|
queue_init_len,
|
|
queue_init_len_cap,
|
|
}
|
|
queue_delete
|
|
queue_clear
|
|
queue_len
|
|
queue_cap
|
|
queue_space
|
|
queue_get
|
|
queue_set
|
|
queue_reserve
|
|
queue_resize
|
|
queue_push :: proc{
|
|
queue_push_back,
|
|
queue_push_elems,
|
|
};
|
|
queue_push_front
|
|
queue_pop_front
|
|
queue_pop_back
|
|
queue_consume
|
|
*/
|
|
|
|
queue_init_none :: proc(q: ^$Q/Queue($T), allocator := context.allocator) {
|
|
queue_init_len(q, 0, allocator);
|
|
}
|
|
queue_init_len :: proc(q: ^$Q/Queue($T), len: int, allocator := context.allocator) {
|
|
queue_init_len_cap(q, 0, 16, allocator);
|
|
}
|
|
queue_init_len_cap :: proc(q: ^$Q/Queue($T), len: int, cap: int, allocator := context.allocator) {
|
|
array_init(&q.data, len, cap, allocator);
|
|
q.len = len;
|
|
q.offset = 0;
|
|
}
|
|
|
|
queue_init :: proc{queue_init_none, queue_init_len, queue_init_len_cap};
|
|
|
|
queue_delete :: proc(q: $Q/Queue($T)) {
|
|
array_delete(q.data);
|
|
}
|
|
|
|
queue_clear :: proc(q: ^$Q/Queue($T)) {
|
|
q.len = 0;
|
|
}
|
|
|
|
queue_len :: proc(q: $Q/Queue($T)) -> int {
|
|
return q.len;
|
|
}
|
|
|
|
queue_cap :: proc(q: $Q/Queue($T)) -> int {
|
|
return array_cap(q.data);
|
|
}
|
|
|
|
queue_space :: proc(q: $Q/Queue($T)) -> int {
|
|
return array_len(q.data) - q.len;
|
|
}
|
|
|
|
queue_get :: proc(q: $Q/Queue($T), index: int) -> T {
|
|
i := (index + q.offset) % array_len(q.data);
|
|
data := array_slice(q.data);
|
|
return data[i];
|
|
}
|
|
|
|
queue_set :: proc(q: ^$Q/Queue($T), index: int, item: T) {
|
|
i := (index + q.offset) % array_len(q.data);
|
|
data := array_slice(q.data);
|
|
data[i] = item;
|
|
}
|
|
|
|
|
|
queue_reserve :: proc(q: ^$Q/Queue($T), capacity: int) {
|
|
if capacity > q.len {
|
|
_queue_increase_capacity(q, capacity);
|
|
}
|
|
}
|
|
|
|
queue_resize :: proc(q: ^$Q/Queue($T), length: int) {
|
|
if length > q.len {
|
|
_queue_increase_capacity(q, length);
|
|
}
|
|
q.len = length;
|
|
}
|
|
|
|
queue_push_back :: proc(q: ^$Q/Queue($T), item: T) {
|
|
if queue_space(q^) == 0 {
|
|
_queue_grow(q);
|
|
}
|
|
|
|
queue_set(q, q.len, item);
|
|
q.len += 1;
|
|
}
|
|
|
|
queue_push_front :: proc(q: ^$Q/Queue($T), item: T) {
|
|
if queue_space(q^) == 0 {
|
|
_queue_grow(q);
|
|
}
|
|
|
|
q.offset = (q.offset - 1 + array_len(q.data)) % array_len(q.data);
|
|
q.len += 1;
|
|
queue_set(q, 0, item);
|
|
}
|
|
|
|
queue_pop_front :: proc(q: ^$Q/Queue($T)) -> T {
|
|
assert(q.len > 0);
|
|
item := queue_get(q^, 0);
|
|
q.offset = (q.offset + 1) % array_len(q.data);
|
|
q.len -= 1;
|
|
if q.len == 0 {
|
|
q.offset = 0;
|
|
}
|
|
return item;
|
|
}
|
|
|
|
queue_pop_back :: proc(q: ^$Q/Queue($T)) -> T {
|
|
assert(q.len > 0);
|
|
item := queue_get(q^, q.len-1);
|
|
q.len -= 1;
|
|
return item;
|
|
}
|
|
|
|
queue_consume :: proc(q: ^$Q/Queue($T), count: int) {
|
|
q.offset = (q.offset + count) & array_len(q.data);
|
|
q.len -= count;
|
|
}
|
|
|
|
|
|
queue_push_elems :: proc(q: ^$Q/Queue($T), items: ..T) {
|
|
if queue_space(q^) < len(items) {
|
|
_queue_grow(q, q.len + len(items));
|
|
}
|
|
size := array_len(q.data);
|
|
insert := (q.offset + q.len) % size;
|
|
|
|
to_insert := len(items);
|
|
if insert + to_insert > size {
|
|
to_insert = size - insert;
|
|
}
|
|
|
|
the_items := items[:];
|
|
|
|
data := array_slice(q.data);
|
|
|
|
q.len += copy(data[insert:][:to_insert], the_items);
|
|
the_items = the_items[to_insert:];
|
|
q.len += copy(data[:], the_items);
|
|
}
|
|
|
|
queue_push :: proc{queue_push_back, queue_push_elems};
|
|
|
|
|
|
|
|
_queue_increase_capacity :: proc(q: ^$Q/Queue($T), new_capacity: int) {
|
|
end := array_len(q.data);
|
|
array_resize(&q.data, new_capacity);
|
|
if q.offset + q.len > end {
|
|
end_items := q.len + end;
|
|
data := array_slice(q.data);
|
|
copy(data[new_capacity-end_items:][:end_items], data[q.offset:][:end_items]);
|
|
q.offset += new_capacity - end;
|
|
}
|
|
}
|
|
_queue_grow :: proc(q: ^$Q/Queue($T), min_capacity: int = 0) {
|
|
new_capacity := max(array_len(q.data)*2 + 8, min_capacity);
|
|
_queue_increase_capacity(q, new_capacity);
|
|
}
|