kmod/shared/array.c
Lucas De Marchi dea2dfee9b Remove FSF mailing address
It has changed in the past, and these days, anyone can get a copy of the
LGPL via the web rather than by post.

Like 657a122 (Remove FSF mailing address) in libabc by Josh Tripplet,
but let the FSF website in which the license can be found.
2014-12-25 23:41:34 -02:00

116 lines
2.7 KiB
C

/*
* libkmod - interface to kernel module operations
*
* Copyright (C) 2011-2013 ProFUSION embedded systems
*
* This library is free software; you can redistribute it and/or
* modify it under the terms of the GNU Lesser General Public
* License as published by the Free Software Foundation; either
* version 2.1 of the License, or (at your option) any later version.
*
* This library is distributed in the hope that it will be useful,
* but WITHOUT ANY WARRANTY; without even the implied warranty of
* MERCHANTABILITY or FITNESS FOR A PARTICULAR PURPOSE. See the GNU
* Lesser General Public License for more details.
*
* You should have received a copy of the GNU Lesser General Public
* License along with this library; if not, see <http://www.gnu.org/licenses/>.
*/
#include <assert.h>
#include <errno.h>
#include <stdlib.h>
#include <string.h>
#include <shared/array.h>
#include <libkmod.h>
/* basic pointer array growing in steps */
static int array_realloc(struct array *array, size_t new_total)
{
void *tmp = realloc(array->array, sizeof(void *) * new_total);
if (tmp == NULL)
return -ENOMEM;
array->array = tmp;
array->total = new_total;
return 0;
}
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) {
int r = array_realloc(array, array->total + array->step);
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;
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) {
array->count--;
if (array->count + array->step < array->total) {
int r = array_realloc(array, array->total - array->step);
if (r < 0)
return;
}
}
void array_free_array(struct array *array) {
free(array->array);
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, unsigned int 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));
if (array->count + array->step < array->total) {
int r = array_realloc(array, array->total - array->step);
/* ignore error */
if (r < 0)
return 0;
}
return 0;
}