8 #define INITIAL_CAPACITY 64
11 #define CHECK(x) (void)(x)
13 #define CHECK(x) _check(x)
21 void (*elt_dtor)(void *);
25 _check(const vector *v)
28 assert(v->capacity > 0);
29 /* test that capacity is a power of two if not maxed out
30 * https://graphics.stanford.edu/~seander/bithacks.html#DetermineIfPowerOf2
32 if (v->capacity < SIZE_MAX)
33 assert((v->capacity & (v->capacity - 1)) == 0);
34 assert(v->length <= v->capacity);
38 v_new(void (*elt_dtor)(void *))
40 vector *v = malloc(sizeof *v);
41 void **elts = malloc(INITIAL_CAPACITY * sizeof *elts);
49 .capacity = INITIAL_CAPACITY,
68 v_length(const vector *v)
70 return v ? v->length : 0;
74 v_set_length(vector *v, size_t desired)
78 if (v->elt_dtor) /* free any, if necessary */
79 for (size_t i = desired; i < v->length; i++)
80 v->elt_dtor(v->elts[i]);
81 if (v_reserve_capacity(v, desired) < desired)
83 for (size_t i = v->length; i < desired; i++)
92 v_capacity(const vector *v)
94 return v ? v->capacity : 0;
98 v_reserve_capacity(vector *v, size_t desired)
102 if (desired <= v->capacity)
104 size_t n = v->capacity;
105 /* SIZE_MAX is one less than a power of two, so if
106 * we keep looping too long we'll hit zero */
107 while (0 < n && n < desired)
111 void **enlarged = realloc(v->elts, n);
122 v_is_empty(const vector *v)
124 return v_length(v) == 0;
128 v_at(const vector *v, size_t i)
130 if (!v || i >= v->length)
136 v_first(const vector *v)
142 v_last(const vector *v)
144 /* successfully fails when length is 0 */
145 return v_at(v, v_length(v)-1);
149 v_append(vector *v, void *e)
151 return v_insert(v, v_length(v), e);
155 v_prepend(vector *v, void *e)
157 return v_insert(v, 0, e);
161 v_remove_first(vector *v)
163 return v_remove(v, 0);
167 v_remove_last(vector *v)
169 return v_remove(v, v_length(v)-1);
173 v_remove(vector *v, size_t i)
175 if (!v || i >= v->length)
177 void *elt = v->elts[i];
178 memmove(v->elts+i, v->elts+i+1, (v->length - (i+1)) * sizeof *v->elts);
186 v_insert(vector *v, size_t i, void *elt)
188 if (!v || v_reserve_capacity(v, v->length+1) < v->length+1)
190 memmove(v->elts+i+1, v->elts+i, (v->length - i) * sizeof *v->elts);
198 #define SWAP(x, y) do { void *swaptmp = (x); (x) = (y); (y) = swaptmp; } while (0)
201 v_swap(vector *v, size_t i, size_t j)
203 if (!v || i >= v->length || j >= v->length)
205 SWAP(v->elts[i], v->elts[j]);
218 v_find_index(const vector *v, const void *needle,
219 comparator *cmp, void *aux)
223 for (size_t i = 0; i < v->length; i++)
224 if (cmp(v->elts[i], needle, aux) == 0)
230 v_find_last_index(const vector *v, const void *needle,
231 comparator *cmp, void *aux)
235 for (size_t i = v->length-1; i < SIZE_MAX; i--)
236 if (cmp(v->elts[i], needle, aux) == 0)
241 /* from Bentley, https://www.youtube.com/watch?v=QvgYAQzg1z8 */
243 _quicksort(vector *v, size_t lo, size_t hi,
244 comparator *cmp, void *aux)
249 for (i = lo+1; i <= hi; i++)
250 if (cmp(v->elts[i], v->elts[lo], aux) < 0)
253 SWAP(v->elts[i], v->elts[m]);
255 SWAP(v->elts[lo], v->elts[m]);
257 if (m > 0) /* avoid wrapping size_t */
258 _quicksort(v, lo, m-1, cmp, aux);
259 _quicksort(v, m+1, hi, cmp, aux);
263 v_sort(vector *v, comparator *cmp, void *aux)
267 _quicksort(v, 0, v->length-1, cmp, aux);
276 size_t n = v_length(v);
279 for (size_t i = n-1; i >= n/2; i--)
281 void *t = v->elts[i];
282 v->elts[i] = v->elts[n-i-1];