diff options
Diffstat (limited to 'libs/libc/list.c')
-rw-r--r-- | libs/libc/list.c | 138 |
1 files changed, 138 insertions, 0 deletions
diff --git a/libs/libc/list.c b/libs/libc/list.c new file mode 100644 index 0000000..c86b23d --- /dev/null +++ b/libs/libc/list.c @@ -0,0 +1,138 @@ +// MIT License, Copyright (c) 2020 Marvin Borner + +#include <def.h> +#include <list.h> +#include <mem.h> + +static int nonce = 0; + +struct list *list_new(void) +{ + struct list *list = malloc(sizeof(*list)); + list->head = NULL; + return list; +} + +void list_destroy(struct list *list) +{ + if (!list) + return; + struct node *iterator = list->head; + while (iterator != NULL) { + if (iterator->next == NULL) { + free(iterator); + break; + } + iterator = iterator->next; + free(iterator->prev); + } + list->head = NULL; + free(list); + list = NULL; +} + +static struct node *list_new_node(void) +{ + struct node *node = malloc(sizeof(*node)); + node->data = NULL; + node->prev = NULL; + node->next = NULL; + node->nonce = nonce++; + return node; +} + +static struct node *list_add_node(struct list *list, struct node *node) +{ + if (!list || !node) + return NULL; + + if (list->head == NULL) { + list->head = node; + return list->head; + } + + struct node *iterator = list->head; + while (iterator != NULL) { + if (iterator->next == NULL) { + iterator->next = node; + node->prev = iterator; + break; + } + iterator = iterator->next; + } + return node; +} + +struct node *list_last(struct list *list) +{ + if (!list || !list->head) + return NULL; + + struct node *iterator = list->head; + while (iterator != NULL) { + if (iterator->next == NULL) + return iterator; + iterator = iterator->next; + } + + return NULL; +} + +struct node *list_first_data(struct list *list, void *data) +{ + if (!list || !list->head || !data) + return NULL; + + struct node *iterator = list->head; + while (iterator != NULL) { + if (iterator->data == data) + return iterator; + iterator = iterator->next; + } + + return NULL; +} + +// TODO: Actually swap the nodes, not the data +struct list *list_swap(struct list *list, struct node *a, struct node *b) +{ + if (!list || !list->head || !a || !b) + return NULL; + + void *tmp = a->data; + a->data = b->data; + b->data = tmp; + + return list; +} + +struct node *list_add(struct list *list, void *data) +{ + struct node *node = list_new_node(); + node->data = data; + return list_add_node(list, node); +} + +// Maybe list_remove_node? +struct list *list_remove(struct list *list, struct node *node) +{ + if (!list || !list->head || !node) + return NULL; + + if (list->head == node) { + list->head = list->head->next; + return list; + } + + struct node *iterator = list->head->next; + while (iterator != node) { + iterator = iterator->next; + if (iterator == NULL) + return NULL; + } + + iterator->prev->next = iterator->next; + if (iterator->next) + iterator->next->prev = iterator->prev; + return list; +} |