2015-08-04 21:30:40 -04:00
|
|
|
#include "list.h"
|
2017-05-10 12:53:48 +12:00
|
|
|
#include "log.h"
|
2017-05-09 21:16:14 +12:00
|
|
|
|
|
|
|
|
#include <stdbool.h>
|
2017-05-10 13:21:12 +12:00
|
|
|
#include <stdint.h>
|
|
|
|
|
#include <stdlib.h>
|
2015-08-04 21:30:40 -04:00
|
|
|
#include <string.h>
|
|
|
|
|
|
2017-05-09 21:16:14 +12:00
|
|
|
list_t *list_new(size_t memb_size, size_t capacity) {
|
2017-05-10 13:18:20 +12:00
|
|
|
if (capacity == 0) {
|
2017-05-09 21:16:14 +12:00
|
|
|
capacity = 8;
|
2017-05-10 13:18:20 +12:00
|
|
|
}
|
2017-05-09 21:16:14 +12:00
|
|
|
|
2017-05-10 13:18:20 +12:00
|
|
|
list_t *list = malloc(sizeof(*list));
|
|
|
|
|
if (!list) {
|
2016-12-15 17:52:53 -05:00
|
|
|
return NULL;
|
|
|
|
|
}
|
2017-05-09 21:16:14 +12:00
|
|
|
|
2017-05-10 13:18:20 +12:00
|
|
|
list->capacity = capacity;
|
|
|
|
|
list->length = 0;
|
|
|
|
|
list->memb_size = memb_size;
|
2017-05-09 21:16:14 +12:00
|
|
|
|
2017-05-10 13:18:20 +12:00
|
|
|
list->items = malloc(memb_size * capacity);
|
|
|
|
|
if (!list->items) {
|
|
|
|
|
free(list);
|
2017-05-10 01:50:54 +12:00
|
|
|
return NULL;
|
|
|
|
|
}
|
|
|
|
|
|
2017-05-10 13:18:20 +12:00
|
|
|
return list;
|
2015-08-04 21:30:40 -04:00
|
|
|
}
|
|
|
|
|
|
2017-05-10 01:50:54 +12:00
|
|
|
void list_free(list_t *list) {
|
|
|
|
|
if (!list) {
|
|
|
|
|
return;
|
|
|
|
|
}
|
2017-05-09 21:16:14 +12:00
|
|
|
|
2017-05-10 13:18:20 +12:00
|
|
|
free(list->items);
|
2017-05-10 01:50:54 +12:00
|
|
|
free(list);
|
|
|
|
|
}
|
|
|
|
|
|
|
|
|
|
static bool resize(list_t *list) {
|
|
|
|
|
if (list->length < list->capacity) {
|
2017-05-09 21:16:14 +12:00
|
|
|
return true;
|
2015-08-11 10:44:29 -07:00
|
|
|
}
|
2017-05-09 21:16:14 +12:00
|
|
|
|
2017-05-10 01:50:54 +12:00
|
|
|
size_t new_cap = list->capacity * 2;
|
2017-05-10 13:18:20 +12:00
|
|
|
void *items = realloc(list->items, list->memb_size * new_cap);
|
|
|
|
|
if (!items) {
|
2017-05-09 21:16:14 +12:00
|
|
|
return false;
|
|
|
|
|
}
|
|
|
|
|
|
2017-05-10 13:18:20 +12:00
|
|
|
list->items = items;
|
2017-05-10 01:50:54 +12:00
|
|
|
list->capacity = new_cap;
|
2017-05-09 21:16:14 +12:00
|
|
|
return true;
|
2015-08-11 10:44:29 -07:00
|
|
|
}
|
|
|
|
|
|
2017-05-10 01:50:54 +12:00
|
|
|
void list_insert(list_t *list, size_t index, const void *data) {
|
2017-05-10 12:53:48 +12:00
|
|
|
if (!sway_assert(list && data && index <= list->length, "Invalid argument") ||
|
|
|
|
|
!resize(list)) {
|
|
|
|
|
|
2015-12-22 15:35:37 +01:00
|
|
|
return;
|
|
|
|
|
}
|
2017-05-09 21:16:14 +12:00
|
|
|
|
2017-05-10 01:50:54 +12:00
|
|
|
size_t size = list->memb_size;
|
2017-05-10 13:21:12 +12:00
|
|
|
uint8_t (*array)[size] = list->items;
|
2017-05-10 01:50:54 +12:00
|
|
|
memmove(&array[index + 1], &array[index], size * (list->length - index));
|
2017-05-09 21:16:14 +12:00
|
|
|
memcpy(&array[index], data, size);
|
2017-05-10 01:50:54 +12:00
|
|
|
++list->length;
|
2017-05-09 21:16:14 +12:00
|
|
|
}
|
|
|
|
|
|
2017-05-13 14:10:11 +12:00
|
|
|
void list_add(list_t *list, const void *data) {
|
|
|
|
|
if (!sway_assert(list && data, "Invalid argument")) {
|
|
|
|
|
return;
|
|
|
|
|
}
|
|
|
|
|
|
|
|
|
|
list_insert(list, list->length, data);
|
|
|
|
|
}
|
|
|
|
|
|
2017-05-13 00:22:49 +12:00
|
|
|
static void shrink(list_t *list) {
|
2017-05-10 13:11:49 +12:00
|
|
|
/* We shrink very sparse lists, but only down to a certain size.
|
|
|
|
|
* The choice of >= 8 is somewhat arbitrary, but leaves a minimum
|
|
|
|
|
* size of 4 elements.
|
|
|
|
|
*/
|
|
|
|
|
if (list->length <= list->capacity / 4 && list->capacity >= 8) {
|
|
|
|
|
size_t new_cap = list->capacity / 2;
|
2017-05-10 13:18:20 +12:00
|
|
|
void *items = realloc(list->items, list->memb_size * new_cap);
|
|
|
|
|
if (!items) {
|
2017-05-10 13:11:49 +12:00
|
|
|
return;
|
|
|
|
|
}
|
|
|
|
|
|
2017-05-10 13:18:20 +12:00
|
|
|
list->items = items;
|
2017-05-10 13:11:49 +12:00
|
|
|
list->capacity = new_cap;
|
|
|
|
|
}
|
2017-05-09 21:16:14 +12:00
|
|
|
}
|
|
|
|
|
|
2017-05-13 00:22:49 +12:00
|
|
|
void list_delete(list_t *list, size_t index) {
|
|
|
|
|
if (!sway_assert(list && index < list->length, "Invalid argument")) {
|
|
|
|
|
return;
|
|
|
|
|
}
|
|
|
|
|
|
|
|
|
|
size_t size = list->memb_size;
|
|
|
|
|
uint8_t (*array)[size] = list->items;
|
|
|
|
|
|
|
|
|
|
memmove(&array[index], &array[index + 1], size * (list->length - index));
|
|
|
|
|
--list->length;
|
|
|
|
|
shrink(list);
|
|
|
|
|
}
|
|
|
|
|
|
2017-05-13 14:10:11 +12:00
|
|
|
void list_remove(list_t *list) {
|
|
|
|
|
if (!sway_assert(list, "Invalid argument") || list->length == 0) {
|
|
|
|
|
return;
|
|
|
|
|
}
|
|
|
|
|
|
|
|
|
|
list_delete(list, list->length - 1);
|
|
|
|
|
}
|
|
|
|
|
|
2017-05-09 21:16:14 +12:00
|
|
|
void list_swap(list_t *list, size_t i1, size_t i2) {
|
2017-05-10 12:53:48 +12:00
|
|
|
if (!sway_assert(list && i1 < list->length && i2 < list->length, "Invalid argument")) {
|
2017-05-09 21:16:14 +12:00
|
|
|
return;
|
|
|
|
|
}
|
|
|
|
|
|
|
|
|
|
size_t size = list->memb_size;
|
2017-05-10 13:21:12 +12:00
|
|
|
uint8_t (*array)[size] = list->items;
|
2017-05-09 21:16:14 +12:00
|
|
|
|
2017-05-10 13:21:12 +12:00
|
|
|
uint8_t tmp[size];
|
2017-05-09 21:16:14 +12:00
|
|
|
memcpy(tmp, &array[i1], size);
|
|
|
|
|
memcpy(&array[i1], &array[i2], size);
|
|
|
|
|
memcpy(&array[i2], tmp, size);
|
|
|
|
|
}
|
|
|
|
|
|
|
|
|
|
void *list_get(list_t *list, size_t index) {
|
2017-05-10 12:53:48 +12:00
|
|
|
if (!sway_assert(list && index < list->length, "Invalid argument")) {
|
2017-05-09 21:16:14 +12:00
|
|
|
return NULL;
|
|
|
|
|
}
|
|
|
|
|
|
2017-05-10 01:50:54 +12:00
|
|
|
size_t size = list->memb_size;
|
2017-05-10 13:21:12 +12:00
|
|
|
uint8_t (*array)[size] = list->items;
|
2017-05-10 01:50:54 +12:00
|
|
|
|
|
|
|
|
return &array[index];
|
2017-05-09 21:16:14 +12:00
|
|
|
}
|
|
|
|
|
|
2017-05-13 13:37:17 +12:00
|
|
|
void *list_getp(list_t *list, size_t index) {
|
|
|
|
|
void **elem = list_get(list, index);
|
|
|
|
|
return elem ? *elem : NULL;
|
|
|
|
|
}
|
|
|
|
|
|
2017-05-09 21:16:14 +12:00
|
|
|
void list_qsort(list_t *list, int compare(const void *, const void *)) {
|
2017-05-10 12:53:48 +12:00
|
|
|
if (!sway_assert(list && compare, "Invalid argument")) {
|
2017-05-09 21:16:14 +12:00
|
|
|
return;
|
|
|
|
|
}
|
|
|
|
|
|
2017-05-10 13:18:20 +12:00
|
|
|
qsort(list->items, list->length, list->memb_size, compare);
|
2017-05-09 21:16:14 +12:00
|
|
|
}
|
|
|
|
|
|
|
|
|
|
void list_isort(list_t *list, int compare(const void *, const void *)) {
|
2017-05-10 12:53:48 +12:00
|
|
|
if (!sway_assert(list && compare, "Invalid argument")) {
|
2017-05-09 21:16:14 +12:00
|
|
|
return;
|
|
|
|
|
}
|
|
|
|
|
|
|
|
|
|
size_t size = list->memb_size;
|
2017-05-10 13:21:12 +12:00
|
|
|
uint8_t (*array)[size] = list->items;
|
2017-05-09 21:16:14 +12:00
|
|
|
|
|
|
|
|
for (size_t i = 1; i < list->length; ++i) {
|
2017-05-10 13:21:12 +12:00
|
|
|
uint8_t tmp[size];
|
2017-05-09 21:16:14 +12:00
|
|
|
memcpy(tmp, &array[i], size);
|
|
|
|
|
|
|
|
|
|
ssize_t j = i - 1;
|
|
|
|
|
while (j >= 0 && compare(&array[j], tmp) > 0) {
|
|
|
|
|
memcpy(&array[j + 1], &array[j], size);
|
|
|
|
|
--j;
|
|
|
|
|
}
|
|
|
|
|
|
|
|
|
|
memcpy(array[j + 1], tmp, size);
|
2015-12-22 15:35:37 +01:00
|
|
|
}
|
|
|
|
|
}
|
|
|
|
|
|
2017-05-10 13:38:49 +12:00
|
|
|
ssize_t list_bsearch(const list_t *list, int compare(const void *key, const void *item),
|
2017-05-10 12:53:48 +12:00
|
|
|
const void *key, void *ret) {
|
|
|
|
|
|
|
|
|
|
if (!sway_assert(list && compare && key, "Invalid argument")) {
|
2017-05-09 21:16:14 +12:00
|
|
|
return -1;
|
|
|
|
|
}
|
2015-08-04 21:30:40 -04:00
|
|
|
|
2017-05-10 13:21:12 +12:00
|
|
|
const uint8_t *ptr = bsearch(key, list->items, list->length, list->memb_size, compare);
|
2017-05-09 21:16:14 +12:00
|
|
|
if (!ptr) {
|
|
|
|
|
return -1;
|
|
|
|
|
} else {
|
|
|
|
|
if (ret) {
|
|
|
|
|
memcpy(ret, ptr, list->memb_size);
|
|
|
|
|
}
|
2017-05-10 13:21:12 +12:00
|
|
|
return (ptr - (uint8_t *)list->items) / list->memb_size;
|
2015-08-04 21:30:40 -04:00
|
|
|
}
|
|
|
|
|
}
|
2015-09-08 08:54:57 -07:00
|
|
|
|
2017-05-10 13:38:49 +12:00
|
|
|
ssize_t list_lsearch(const list_t *list, int compare(const void *key, const void *item),
|
2017-05-10 12:53:48 +12:00
|
|
|
const void *key, void *ret) {
|
|
|
|
|
|
|
|
|
|
if (!sway_assert(list && compare && key, "Invalid argument")) {
|
2017-05-09 21:16:14 +12:00
|
|
|
return -1;
|
|
|
|
|
}
|
2015-11-18 21:12:20 +01:00
|
|
|
|
2017-05-09 21:16:14 +12:00
|
|
|
size_t size = list->memb_size;
|
2017-05-10 13:21:12 +12:00
|
|
|
uint8_t (*array)[size] = list->items;
|
2017-05-09 21:16:14 +12:00
|
|
|
|
|
|
|
|
for (size_t i = 0; i < list->length; ++i) {
|
2017-05-13 13:37:17 +12:00
|
|
|
if (compare(key, &array[i]) == 0) {
|
2017-05-09 21:16:14 +12:00
|
|
|
if (ret) {
|
|
|
|
|
memcpy(ret, &array[i], size);
|
|
|
|
|
}
|
2015-11-18 21:12:20 +01:00
|
|
|
return i;
|
|
|
|
|
}
|
|
|
|
|
}
|
2017-05-09 21:16:14 +12:00
|
|
|
|
2015-11-18 21:12:20 +01:00
|
|
|
return -1;
|
|
|
|
|
}
|
2016-06-02 15:48:14 -05:00
|
|
|
|
2017-05-10 13:38:49 +12:00
|
|
|
void list_foreach(list_t *list, void callback(void *item)) {
|
2017-05-10 12:53:48 +12:00
|
|
|
if (!sway_assert(list && callback, "Invalid argument")) {
|
2016-06-02 15:48:14 -05:00
|
|
|
return;
|
|
|
|
|
}
|
|
|
|
|
|
2017-05-09 21:16:14 +12:00
|
|
|
size_t size = list->memb_size;
|
2017-05-10 13:21:12 +12:00
|
|
|
uint8_t (*array)[size] = list->items;
|
2016-06-02 15:48:14 -05:00
|
|
|
|
2017-05-09 21:16:14 +12:00
|
|
|
for (size_t i = 0; i < list->length; ++i) {
|
|
|
|
|
callback(&array[i]);
|
2016-06-02 15:48:14 -05:00
|
|
|
}
|
|
|
|
|
}
|
2017-05-10 01:50:54 +12:00
|
|
|
|
2017-05-13 14:10:11 +12:00
|
|
|
void list_foreachp(list_t *list, void callback(void *item)) {
|
|
|
|
|
if (!sway_assert(list && callback, "Invalid argument")) {
|
|
|
|
|
return;
|
|
|
|
|
}
|
|
|
|
|
|
|
|
|
|
size_t size = list->memb_size;
|
|
|
|
|
uint8_t (*array)[size] = list->items;
|
|
|
|
|
|
|
|
|
|
for (size_t i = 0; i < list->length; ++i) {
|
|
|
|
|
callback(*(void **)&array[i]);
|
|
|
|
|
}
|
|
|
|
|
}
|
|
|
|
|
|
|
|
|
|
void list_free_with(list_t *list, freefn_t callback) {
|
2017-05-13 00:22:49 +12:00
|
|
|
if (!sway_assert(callback, "Invalid argument") || !list) {
|
|
|
|
|
return;
|
|
|
|
|
}
|
|
|
|
|
|
|
|
|
|
list_foreach(list, callback);
|
|
|
|
|
list_free(list);
|
|
|
|
|
}
|
|
|
|
|
|
2017-05-13 14:10:11 +12:00
|
|
|
void list_free_withp(list_t *list, freefn_t callback) {
|
|
|
|
|
if (!sway_assert(callback, "Invalid argument") || !list) {
|
|
|
|
|
return;
|
|
|
|
|
}
|
|
|
|
|
|
|
|
|
|
list_foreachp(list, callback);
|
|
|
|
|
list_free(list);
|
|
|
|
|
}
|
|
|
|
|
|
2017-05-10 01:50:54 +12:00
|
|
|
void *list_end(list_t *list) {
|
2017-05-10 12:53:48 +12:00
|
|
|
if (!sway_assert(list, "Invalid argument")) {
|
2017-05-10 01:50:54 +12:00
|
|
|
return NULL;
|
|
|
|
|
}
|
|
|
|
|
|
2017-05-10 13:21:12 +12:00
|
|
|
return (uint8_t *)list->items + list->memb_size * list->length;
|
2017-05-10 01:50:54 +12:00
|
|
|
}
|