implement cx_ll_insert()

2021-09-28

author
Mike Becker <universe@uap-core.de>
date
Tue, 28 Sep 2021 18:49:12 +0200 (2021-09-28)
changeset 448
37c5d2fdb36b
parent 447
87b2cdd668ed
child 449
68ad5750ba6b

implement cx_ll_insert()

change cx_ll_add() to use insert with index=size

src/linked_list.c file | annotate | diff | comparison | revisions
--- a/src/linked_list.c	Tue Sep 28 18:33:42 2021 +0200
+++ b/src/linked_list.c	Tue Sep 28 18:49:12 2021 +0200
@@ -109,30 +109,7 @@
     cx_linked_list_node *end;
 } cx_linked_list;
 
-static int cx_ll_add(cx_list_s *list, void *elem) {
-    cx_linked_list *ll = (cx_linked_list *) list;
-
-    cx_linked_list_node *node = cxMalloc(list->allocator,
-                                         sizeof(cx_linked_list_node) + list->itemsize);
-    if (node == NULL)
-        return 1;
-
-    memcpy(node->payload, elem, list->itemsize);
-    node->next = NULL;
-
-    if (ll->begin == NULL) {
-        ll->begin = ll->end = node;
-    } else {
-        // per contract of this linked list we know that end must be non-null
-        ll->end->next = node;
-        ll->end = node;
-    }
-    list->size++;
-
-    return 0;
-}
-
-static cx_linked_list_node *cx_ll_node_at(cx_linked_list* list, size_t index) {
+static cx_linked_list_node *cx_ll_node_at(cx_linked_list *list, size_t index) {
     if (index >= list->base.size) {
         return NULL;
     } else if (index > list->base.size / 2) {
@@ -143,9 +120,60 @@
 }
 
 static int cx_ll_insert(cx_list_s *list, size_t index, void *elem) {
+    // out-of bounds check
+    if (index > list->size) return 1;
+
+    // cast a linked list pointer
     cx_linked_list *ll = (cx_linked_list *) list;
-    // TODO: implement using low level API
-    return 1;
+
+    // create the new node
+    cx_linked_list_node *node = cxMalloc(list->allocator,
+                                         sizeof(cx_linked_list_node) + list->itemsize);
+
+    // sortir if failed
+    if (node == NULL) return 1;
+
+    // copy payload to the new node
+    memcpy(node->payload, elem, list->itemsize);
+
+    // check if this is the first node
+    if (ll->begin == NULL) {
+        node->prev = node->next = NULL;
+        ll->begin = ll->end = node;
+    } else {
+        // check if this shall be the new end node
+        if (index == list->size) {
+            ll->end->next = node;
+            node->prev = ll->end;
+            node->next = NULL;
+            ll->end = node;
+        }
+        // check if this shall be the new start node
+        else if (index == 0) {
+            ll->begin->prev = node;
+            node->next = ll->begin;
+            node->prev = NULL;
+            ll->begin = node;
+        } else {
+            // find the node at the current index
+            cx_linked_list_node *cur = cx_ll_node_at(ll, index);
+
+            // insert before that node
+            // (we know all ptr are non-null because we handled all other cases before)
+            node->next = cur;
+            node->prev = cur->prev;
+            cur->prev = node;
+            node->prev->next = node;
+        }
+    }
+
+    // increase the size and return
+    list->size++;
+    return 0;
+}
+
+static int cx_ll_add(cx_list_s *list, void *elem) {
+    return cx_ll_insert(list, list->size, elem);
 }
 
 static int cx_ll_remove(cx_list_s *list, size_t index) {

mercurial