// SPDX-License-Identifier: LGPL-2.1-or-later /* * Copyright (C) 2011-2013 ProFUSION embedded systems */ #include #include #include #include #include #include #include /* basic pointer array growing in steps */ static int array_realloc(struct array *array, size_t new_total) { size_t total_size; void *tmp; if (umulsz_overflow(sizeof(void *), new_total, &total_size)) return -ENOMEM; tmp = realloc(array->array, total_size); if (tmp == NULL) return -ENOMEM; array->array = tmp; array->total = new_total; return 0; } static void array_trim(struct array *array) { if (array->count + array->step < array->total) { /* ignore error */ array_realloc(array, array->total - array->step); } } void array_init(struct array *array, size_t step) { assert(step > 0); array->array = NULL; array->count = 0; array->total = 0; array->step = step; } int array_append(struct array *array, const void *element) { size_t idx; if (array->count + 1 >= array->total) { size_t new_size; int r; if (uaddsz_overflow(array->total, array->step, &new_size)) return -ENOMEM; r = array_realloc(array, new_size); if (r < 0) return r; } idx = array->count; array->array[idx] = (void *)element; array->count++; return idx; } int array_append_unique(struct array *array, const void *element) { void **itr = array->array; if (array->count != 0) { void **itr_end = itr + array->count; for (; itr < itr_end; itr++) if (*itr == element) return -EEXIST; } return array_append(array, element); } void array_pop(struct array *array) { if (array->count == 0) return; array->count--; array_trim(array); } void array_free_array(struct array *array) { free(array->array); array->array = NULL; array->count = 0; array->total = 0; } void array_sort(struct array *array, int (*cmp)(const void *a, const void *b)) { qsort(array->array, array->count, sizeof(void *), cmp); } int array_remove_at(struct array *array, size_t pos) { if (array->count <= pos) return -ENOENT; array->count--; if (pos < array->count) memmove(array->array + pos, array->array + pos + 1, sizeof(void *) * (array->count - pos)); array_trim(array); return 0; }