diff options
author | Tucker Evans <tucker@tuckerevans.com> | 2020-05-29 17:15:53 -0400 |
---|---|---|
committer | Tucker Evans <tucker@tuckerevans.com> | 2020-05-29 17:15:53 -0400 |
commit | 4e4704b0251bb2b03d0fa573437b77b15567441c (patch) | |
tree | e543b7def10af6a6dd93d0bf7a0cd9d00ec7db03 | |
parent | dbb5ba792719088f99b58a954a53b1cdab213f0c (diff) |
Fix change beg, end to indices.
These are now indices of the buffer rather than pointers into buffer,
as this is simpler to maintain and extend.
end is also changed to point to next location for storage rather than
the last element.
-rw-r--r-- | collections/double_ended_queue.c | 154 |
1 files changed, 78 insertions, 76 deletions
diff --git a/collections/double_ended_queue.c b/collections/double_ended_queue.c index b555e3a..956e362 100644 --- a/collections/double_ended_queue.c +++ b/collections/double_ended_queue.c @@ -17,14 +17,14 @@ /* Double ended queue as a circular buffer * base is pointer to buffer. - * beg is where the first element is stored. - * end is where the last element is stored. - * limit is the maximum number of elements. + * beg: postiton of the first element is stored. + * end: position for the next element to be stored. + * limit: the maximum number of elements. */ struct double_ended_queue { - void **base, **end, **beg; - int limit; + void **base; + int beg, end, limit; }; deq* deq_new() @@ -35,7 +35,8 @@ deq* deq_new() assert(root); root->limit = START_SIZE; - root->base = root->end = root->beg = malloc(root->limit * sizeof(void*)); + root->beg = root->end = 0; + root->base = malloc(root->limit * sizeof(void*)); assert(root->base); return root; @@ -50,7 +51,8 @@ int n; assert(root); root->limit = n; - root->base = root->end = root->beg = malloc(root->limit * sizeof(void*)); + root->beg = root->end = 0; + root->base = malloc(root->limit * sizeof(void*)); assert(root->base); return root; @@ -59,15 +61,13 @@ int n; int deq_size(root) deq *root; { - if (!root) { + if (!root) return -1; - } - if (root->beg <= root->end) { + if (root->beg <= root->end) return (root->end - root->beg); - } - return (root->base + root->limit - root->beg) + (root->end - root->base); + return (root->limit - root->beg) + (root->end); } void deq_resize(root) @@ -79,26 +79,25 @@ deq *root; tmp = malloc(root->limit * 2 * sizeof(void*)); assert(root->base); - if (root->beg > root->end) { + if (root->beg <= root->end) { + tmp = memcpy(tmp, root->base + root->beg, + deq_size(root) * sizeof(void*)); + } else { /*copy beg<->limit*/ - size = (root->base + (root->limit*sizeof(void*)) - root->beg); - tmp = memcpy(tmp, root->beg, size * sizeof(void*); + size = root->limit - root->beg; + tmp = memcpy(tmp, root->base + root->beg, size * sizeof(void*)); assert(tmp); /*copy base<->end*/ - tmp = memcpy(tmp + size, root->beg, - (root->end - root->base)* sizeof(void*)); + tmp = memcpy(tmp + size, root->base, + root->end * sizeof(void*)); assert(tmp); - } else { - size = deq_size(root); - tmp = memcpy(tmp, root->beg, size*sizeof(void*)); } root->limit *= 2; free(root->base); - root->base = root->beg = tmp; - root->end = root->base + size; + root->base = tmp; } void* deq_index(root, index) @@ -107,16 +106,15 @@ int index; { void **tmp; - if (!root) { + if (!root || index > root->limit + || (index >= root->end && index < root->beg)) return NULL; - } - tmp = root->base + ((root->beg + index - root->base) % root->limit); - if (tmp > root->end) { + index = (root->beg + index) % root->limit; + if (index >= root->end) return NULL; - } - return *tmp; + return root->base[index]; } void deq_push_back(root, item) @@ -125,17 +123,17 @@ void *item; { void **tmp; - if (!root) { + if (!root) return; - } - tmp = (root->base + (root->end - root->base) % root->limit); - if (tmp == root->beg) { + if (root->end >= root->limit) deq_resize(root); - tmp = (root->base + (root->end - root->base) % root->limit); - } - *(root->end = tmp) = item; + root->base[root->end++] = item; + root->end %= root->limit; + + if (root->end == root->beg) + root->end = root->limit; } void deq_push_front(root, item) @@ -144,29 +142,32 @@ void *item; { void **tmp; - if (!root) { + if (!root) return; - } - tmp = (root->base + ((root->beg - 1) - root->base) % root->limit); - if (tmp == root->beg) { + if (root->end >= root->limit) deq_resize(root); - tmp = (root->base + ((root->beg - 1) - root->base) % root->limit); - } - *(root->beg = tmp) = item; + --root->beg; + root->beg %= root->limit; + + root->base[root->beg] = item; + + if (root->end == root->beg) + root->end = root->limit; } void* deq_pop_front(root) deq *root; { void* tmp; - if (!root || root->beg == root->end) { + if (!root || root->beg == root->end) return NULL; - } - tmp = *(root->beg++); - root->beg = (root->base + (root->beg - root->base) % root->limit); + tmp = root->base[root->beg]; + + ++root->beg; + root->beg %= root->limit; return tmp; } @@ -174,15 +175,13 @@ deq *root; void* deq_pop_back(root) deq *root; { - void* tmp; - if (!root || root->end == root->beg) { + if (!root || root->end == root->beg) return NULL; - } - tmp = *(root->end--); - root->end = (root->base + (root->end - root->base) % root->limit); + --root->end; + root->end %= root->limit; - return tmp; + return root->base[root->end]; } void deq_free(root) @@ -190,8 +189,6 @@ deq *root; { free(root->base); root->base = NULL; - root->end = NULL; - root->beg = NULL; free(root); } @@ -216,9 +213,9 @@ deq *root; { void **tmp; - fprintf(stderr, "VEC[b: %p, p: %p, n:%p]:\n\t ", + fprintf(stderr, "VEC[b: %p, beg: %d, end:%d]:\n\t ", root->base, root->beg, root->end); - for (tmp = root->beg; tmp < root->end; tmp++){ + for (tmp = root->base; tmp < root->base + root->limit; tmp++){ fprintf(stderr, "[%p]", *tmp); } fprintf(stderr, "\n"); @@ -230,13 +227,16 @@ deq *root; deq *copy; copy = deq_with_capacity(root->limit); + assert(copy->base); - copy->base = memcpy(copy->base, root->beg, - deq_size(root) * sizeof(void*)); + copy->base = memcpy(copy->base, root->base, + root->limit * sizeof(void*)); assert(copy->base); - copy->beg = copy->base; - copy->end = copy->base + deq_size(root); + copy->beg = root->beg; + copy->end = root->end; + + return copy; } void deq_swap(root, i, j) @@ -246,48 +246,50 @@ int i, j; int len; void *tmp; - if (!root) { + if (!root) return; - } len = deq_size(root); - if (i >= len || j >= len) { + if (i >= len || j >= len) return; - } - tmp = root->end[i]; - root->end[j] = root->end[i]; - root->end[i] = tmp; + i += root->beg; + i %= root->limit; + + j += root->beg; + j %= root->limit; + + tmp = root->base[i]; + root->base[i] = root->base[j]; + root->base[j] = tmp; } +/*Note: Does not currently reduce memory footprint*/ void deq_truncate(root, size) deq *root; int size; { - if ((!root) || size > deq_size(root)) { + if (!root || size > deq_size(root)) return; - } - root->end = root->beg + size; + root->end = (root->beg + size) % root->limit; } void* deq_front(root) deq *root; { - if (!root) { + if (!root) return NULL; - } - return *root->beg; + return root->base[root->beg]; } void* deq_back(root) deq *root; { - if (!root) { + if (!root) return NULL; - } - return *root->end; + return root->base[(root->end - 1) % root->limit]; } |