22 assert(list->
base !=
NULL || index == 0 || stride == 0);
24 const char *
const base = list->
base;
25 return base + index * stride;
30 assert(list->
base !=
NULL || index == 0 || stride == 0);
32 char *
const base = list->
base;
33 return base + index * stride;
38 assert(base !=
NULL || index == 0 || stride == 0);
40 const char *
const b = base;
41 return b + index * stride;
45 assert(base !=
NULL || index == 0 || stride == 0);
48 return b + index * stride;
51#define INDEX_TO(origin, index, stride) \
53 const list_t_ *: slot_from_const_list, \
54 list_t_ *: slot_from_list, \
55 const void *: slot_from_const_base, \
56 void *: slot_from_base)((origin), (index), (stride)))
72 void *
const slot =
INDEX_TO(list, new_slot, item_size);
93 void *
const slot =
INDEX_TO(list, list->
head, item_size);
101 assert(list !=
NULL);
109 assert(capacity > 0);
110 if (
SIZE_MAX / capacity < item_size) {
114 void *
const base = realloc(list->
base, capacity * item_size);
122 const size_t new_bytes = (capacity - list->
capacity) * item_size;
123 memset(
new, 0, new_bytes);
143 const size_t new_head = capacity -
prefix;
145 void *
const target =
INDEX_TO(base, new_head, item_size);
147 const void *
const src =
INDEX_TO(base, list->
head, item_size);
148 memmove(target, src,
prefix * item_size);
152 list->
head = new_head;
161 assert(list !=
NULL);
193 void *
const slot =
INDEX_TO(list, new_slot, item_size);
195 memcpy(slot,
item, item_size);
202 assert(index < list.
size &&
"index out of bounds");
208 for (
size_t i = 0; i < list.
size; ++i) {
210 const void *candidate =
INDEX_TO(&list, slot, item_size);
211 if (memcmp(needle, candidate, item_size) == 0) {
220 assert(list !=
NULL);
221 assert(index < list->size);
224 for (
size_t i = index + 1; i < list->
size; ++i) {
226 void *
const dst =
INDEX_TO(list, dst_slot, item_size);
228 const void *
const src =
INDEX_TO(list, src_slot, item_size);
229 memcpy(dst, src, item_size);
232 void *truncated =
INDEX_TO(list, truncated_slot, item_size);
238 assert(list !=
NULL);
240 for (
size_t i = 0; i < list->
size; ++i) {
242 void *
const to_poison =
INDEX_TO(list, slot, item_size);
258 capacity, item_size, strerror(
err));
273 for (
size_t i = 0; i < list.
size; ++i) {
275 const void *
const src =
INDEX_TO(&list, slot, item_size);
278 memcpy(dst, src, item_size);
283 void *
const to_poison =
INDEX_TO(&ret, ret.
size, item_size);
284 const size_t to_poison_len = (ret.
capacity - ret.
size) * item_size;
295 assert(list !=
NULL);
303 while (list->
head != 0) {
307 for (
size_t i = 0; i < item_size; ++i) {
309 memcpy(&lowest, list->
base,
sizeof(lowest));
310 const size_t remainder = list->
capacity * item_size -
sizeof(lowest);
311 memmove(list->
base, (
char *)list->
base +
sizeof(lowest), remainder);
312 memcpy((
char *)list->
base + remainder, &lowest,
sizeof(lowest));
327 assert(list !=
NULL);
337static void exchange(
void *a,
void *b,
size_t size) {
344 for (
size_t i = 0; i < size; ++i) {
350 assert(list !=
NULL);
353 for (
size_t i = 0; i < list->
size / 2; ++i) {
356 void *
const x =
INDEX_TO(list, a, item_size);
357 void *
const y =
INDEX_TO(list, b, item_size);
363 assert(list !=
NULL);
374 assert(list !=
NULL);
380 assert(list !=
NULL);
381 assert(list->
size > 0);
382 assert(into !=
NULL);
386 void *
const to_pop =
INDEX_TO(list, slot, item_size);
387 memcpy(into, to_pop, item_size);
394 assert(list !=
NULL);
395 assert(list->
size > 0);
396 assert(into !=
NULL);
400 void *
const to_pop =
INDEX_TO(list, slot, item_size);
401 memcpy(into, to_pop, item_size);
408 assert(list !=
NULL);
409 assert(datap !=
NULL);
412 memcpy(datap, &list->
base,
sizeof(
void *));
Memory allocation wrappers that exit on failure.
static void * gv_recalloc(void *ptr, size_t old_nmemb, size_t new_nmemb, size_t size)
static void * gv_calloc(size_t nmemb, size_t size)
macros for interacting with Address Sanitizer
#define ASAN_POISON(addr, size)
#define ASAN_UNPOISON(addr, size)
static NORETURN void graphviz_exit(int status)
static int cmp(const void *key, const void *candidate)
require define api prefix
Arithmetic helper functions.
internal implementation details of list.h
static void exchange(void *a, void *b, size_t size)
list_t_ gv_list_copy_(const list_t_ list, size_t item_size)
bool gv_list_try_append_(list_t_ *list, const void *item, size_t item_size)
static const void * slot_from_const_list(const list_t_ *list, size_t index, size_t stride)
size_t gv_list_prepend_slot_(list_t_ *list, size_t item_size)
void gv_list_pop_front_(list_t_ *list, void *into, size_t item_size)
static const void * slot_from_const_base(const void *base, size_t index, size_t stride)
#define INDEX_TO(origin, index, stride)
bool gv_list_contains_(const list_t_ list, const void *needle, size_t item_size)
size_t gv_list_append_slot_(list_t_ *list, size_t item_size)
void gv_list_reserve_(list_t_ *list, size_t capacity, size_t item_size)
void gv_list_free_(list_t_ *list)
static void * slot_from_base(void *base, size_t index, size_t stride)
void gv_list_clear_(list_t_ *list, size_t item_size)
void gv_list_detach_(list_t_ *list, void *datap, size_t *sizep, size_t item_size)
bool gv_list_is_contiguous_(const list_t_ list)
void gv_list_shrink_to_fit_(list_t_ *list, size_t item_size)
void gv_list_sort_(list_t_ *list, int(*cmp)(const void *, const void *), size_t item_size)
void gv_list_reverse_(list_t_ *list, size_t item_size)
void gv_list_pop_back_(list_t_ *list, void *into, size_t item_size)
size_t gv_list_find_(const list_t_ list, const void *needle, size_t item_size)
void gv_list_sync_(list_t_ *list, size_t item_size)
static int try_reserve(list_t_ *list, size_t capacity, size_t item_size)
static void * slot_from_list(list_t_ *list, size_t index, size_t stride)
void gv_list_remove_(list_t_ *list, size_t index, size_t item_size)
size_t gv_list_get_(const list_t_ list, size_t index)
size_t size
size <= capacity
size_t capacity
available storage slots
void * base
(base == NULL && capacity == 0) || (base != NULL && capacity > 0)
size_t head
(capacity == 0 && head == 0) || (capacity > 0 && head < capacity)