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);
115 if (base ==
NULL && item_size > 0) {
122 const size_t new_bytes = (capacity - list->
capacity) * item_size;
124 memset(
new, 0, new_bytes);
145 const size_t new_head = capacity -
prefix;
147 void *
const target =
INDEX_TO(base, new_head, item_size);
149 const void *
const src =
INDEX_TO(base, list->
head, item_size);
150 if (
prefix * item_size > 0) {
153 memmove(target, src,
prefix * item_size);
158 list->
head = new_head;
167 assert(list !=
NULL);
199 void *
const slot =
INDEX_TO(list, new_slot, item_size);
202 memcpy(slot,
item, item_size);
210 assert(index < list.
size &&
"index out of bounds");
216 for (
size_t i = 0; i < list.
size; ++i) {
218 const void *candidate =
INDEX_TO(&list, slot, item_size);
219 if (item_size == 0 || memcmp(needle, candidate, item_size) == 0) {
228 assert(list !=
NULL);
229 assert(index < list->size);
232 for (
size_t i = index + 1; i < list->
size; ++i) {
234 void *
const dst =
INDEX_TO(list, dst_slot, item_size);
236 const void *
const src =
INDEX_TO(list, src_slot, item_size);
238 memcpy(dst, src, item_size);
242 void *truncated =
INDEX_TO(list, truncated_slot, item_size);
248 assert(list !=
NULL);
250 for (
size_t i = 0; i < list->
size; ++i) {
252 void *
const to_poison =
INDEX_TO(list, slot, item_size);
268 capacity, item_size, strerror(
err));
283 for (
size_t i = 0; i < list.
size; ++i) {
285 const void *
const src =
INDEX_TO(&list, slot, item_size);
289 memcpy(dst, src, item_size);
295 void *
const to_poison =
INDEX_TO(&ret, ret.
size, item_size);
296 const size_t to_poison_len = (ret.
capacity - ret.
size) * item_size;
307 assert(list !=
NULL);
315 while (list->
head != 0) {
319 for (
size_t i = 0; i < item_size; ++i) {
321 memcpy(&lowest, list->
base,
sizeof(lowest));
322 const size_t remainder = list->
capacity * item_size -
sizeof(lowest);
323 memmove(list->
base, (
char *)list->
base +
sizeof(lowest), remainder);
324 memcpy((
char *)list->
base + remainder, &lowest,
sizeof(lowest));
339 assert(list !=
NULL);
344 if (list->
size > 0 && item_size > 0) {
349static void exchange(
void *a,
void *b,
size_t size) {
356 for (
size_t i = 0; i < size; ++i) {
362 assert(list !=
NULL);
365 for (
size_t i = 0; i < list->
size / 2; ++i) {
368 void *
const x =
INDEX_TO(list, a, item_size);
369 void *
const y =
INDEX_TO(list, b, item_size);
375 assert(list !=
NULL);
386 assert(list !=
NULL);
392 assert(list !=
NULL);
393 assert(list->
size > 0);
394 assert(into !=
NULL);
398 void *
const to_pop =
INDEX_TO(list, slot, item_size);
400 memcpy(into, to_pop, item_size);
408 assert(list !=
NULL);
409 assert(list->
size > 0);
410 assert(into !=
NULL);
414 void *
const to_pop =
INDEX_TO(list, slot, item_size);
416 memcpy(into, to_pop, item_size);
424 assert(list !=
NULL);
425 assert(datap !=
NULL);
428 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)