blob: 7bedf45bccdc87b17ca85e5653f10c50cf2ccca5 [file] [log] [blame]
// SPDX-License-Identifier: LGPL-2.1-or-later
/*
* Copyright (C) 2011-2013 ProFUSION embedded systems
*/
#include <assert.h>
#include <errno.h>
#include <stdint.h>
#include <stdlib.h>
#include <string.h>
#include <shared/array.h>
#include <shared/util.h>
/* 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;
}