add cxListInsertAfter() and cxListInsertBefore()

2022-01-29

author
Mike Becker <universe@uap-core.de>
date
Sat, 29 Jan 2022 14:32:04 +0100 (2022-01-29)
changeset 499
3dc9075df822
parent 498
435c9965b2dd
child 500
eb9e7bd40a8e

add cxListInsertAfter() and cxListInsertBefore()

src/cx/list.h file | annotate | diff | comparison | revisions
src/linked_list.c file | annotate | diff | comparison | revisions
test/test_list.c file | annotate | diff | comparison | revisions
--- a/src/cx/list.h	Sat Jan 29 12:46:07 2022 +0100
+++ b/src/cx/list.h	Sat Jan 29 14:32:04 2022 +0100
@@ -80,6 +80,15 @@
     );
 
     /**
+     * Member function for inserting an element relative to an iterator position.
+     */
+    int (*insert_iter)(
+            CxIterator *iter,
+            void const *elem,
+            int prepend
+    );
+
+    /**
      * Member function for removing an element.
      */
     int (*remove)(
@@ -168,11 +177,6 @@
 /**
  * Adds an item to the end of the list.
  *
- * \remark It is implementation defined whether
- * the contents of the element are copied or the
- * pointer is added to the list. In the latter case
- * the \c itemsize of the list SHALL be \c sizeof(void*).
- *
  * @param list the list
  * @param elem a pointer to the element to add
  * @return zero on success, non-zero on memory allocation failure
@@ -189,16 +193,13 @@
  *
  * If \p index equals the list \c size, this is effectively cxListAdd().
  *
- * \remark It is implementation defined whether
- * the contents of the element are copied or the
- * pointer is added to the list. In the latter case
- * the \c itemsize of the list SHALL be \c sizeof(void*).
- *
  * @param list the list
  * @param index the index the element shall have
  * @param elem a pointer to the element to add
  * @return zero on success, non-zero on memory allocation failure
  * or when the index is out of bounds
+ * @see cxListInsertAfter()
+ * @see cxListInsertBefore()
  */
 static inline int cxListInsert(
         CxList list,
@@ -209,6 +210,50 @@
 }
 
 /**
+ * Inserts an element after the current location of the specified iterator.
+ *
+ * The used iterator remains operational, but all other active iterators should
+ * be considered invalidated.
+ *
+ * If \p iter is not a list iterator, the behavior is undefined.
+ * If \p iter is a past-the-end iterator, the new element gets appended to the list.
+ *
+ * @param iter an iterator
+ * @param elem the element to insert
+ * @return zero on success, non-zero on memory allocation failure
+ * @see cxListInsert()
+ * @see cxListInsertBefore()
+ */
+static inline int cxListInsertAfter(
+        CxIterator *iter,
+        void const *elem
+) {
+    return ((cx_list_s *) iter->src_handle)->cl->insert_iter(iter, elem, 0);
+}
+
+/**
+ * Inserts an element before the current location of the specified iterator.
+ *
+ * The used iterator remains operational, but all other active iterators should
+ * be considered invalidated.
+ *
+ * If \p iter is not a list iterator, the behavior is undefined.
+ * If \p iter is a past-the-end iterator, the new element gets appended to the list.
+ *
+ * @param iter an iterator
+ * @param elem the element to insert
+ * @return zero on success, non-zero on memory allocation failure
+ * @see cxListInsert()
+ * @see cxListInsertAfter()
+ */
+static inline int cxListInsertBefore(
+        CxIterator *iter,
+        void const *elem
+) {
+    return ((cx_list_s *) iter->src_handle)->cl->insert_iter(iter, elem, 1);
+}
+
+/**
  * Removes the element at the specified index.
  * @param list the list
  * @param index the index of the element
--- a/src/linked_list.c	Sat Jan 29 12:46:07 2022 +0100
+++ b/src/linked_list.c	Sat Jan 29 14:32:04 2022 +0100
@@ -481,16 +481,11 @@
     }
 }
 
-static int cx_ll_insert(
+static int cx_ll_insert_at(
         cx_list_s *list,
-        size_t index,
+        cx_linked_list_node *node,
         void const *elem
 ) {
-    // out-of bounds check
-    if (index > list->size) return 1;
-
-    // cast a linked list pointer
-    cx_linked_list *ll = (cx_linked_list *) list;
 
     // create the new new_node
     cx_linked_list_node *new_node = cxMalloc(list->allocator,
@@ -503,10 +498,8 @@
     new_node->prev = new_node->next = NULL;
     memcpy(new_node->payload, elem, list->itemsize);
 
-    // find position efficiently
-    cx_linked_list_node *node = index == 0 ? NULL : cx_ll_node_at(ll, index - 1);
-
     // insert
+    cx_linked_list *ll = (cx_linked_list *) list;
     cx_linked_list_insert_chain(
             (void **) &ll->begin, (void **) &ll->end,
             CX_LL_LOC_PREV, CX_LL_LOC_NEXT,
@@ -518,6 +511,21 @@
     return 0;
 }
 
+static int cx_ll_insert(
+        cx_list_s *list,
+        size_t index,
+        void const *elem
+) {
+    // out-of bounds check
+    if (index > list->size) return 1;
+
+    // find position efficiently
+    cx_linked_list_node *node = index == 0 ? NULL : cx_ll_node_at((cx_linked_list *) list, index - 1);
+
+    // perform insert
+    return cx_ll_insert_at(list, node, elem);
+}
+
 static int cx_ll_add(
         cx_list_s *list,
         void const *elem
@@ -695,21 +703,51 @@
     return iter;
 }
 
+static int cx_ll_insert_iter(
+        CxIterator *iter,
+        void const *elem,
+        int prepend
+) {
+    cx_list_s *list = iter->src_handle;
+    cx_linked_list_node *node = iter->elem_handle;
+    if (node != NULL) {
+        assert(prepend >= 0 && prepend <= 1);
+        cx_linked_list_node *choice[2] = {node, node->prev};
+        int result = cx_ll_insert_at(list, choice[prepend], elem);
+        iter->index += prepend * (0 == result);
+        return result;
+    } else {
+        int result = cx_ll_insert(list, list->size, elem);
+        iter->index = list->size;
+        return result;
+    }
+}
+
+static int cx_pll_insert_iter(
+        CxIterator *iter,
+        void const *elem,
+        int prepend
+) {
+    return cx_ll_insert_iter(iter, &elem, prepend);
+}
+
 static cx_list_class cx_linked_list_class = {
         cx_ll_add,
         cx_ll_insert,
+        cx_ll_insert_iter,
         cx_ll_remove,
         cx_ll_at,
         cx_ll_find,
         cx_ll_sort,
         cx_ll_compare,
         cx_ll_reverse,
-        cx_ll_iterator,
+        cx_ll_iterator
 };
 
 static cx_list_class cx_pointer_linked_list_class = {
         cx_pll_add,
         cx_pll_insert,
+        cx_pll_insert_iter,
         cx_ll_remove,
         cx_pll_at,
         cx_pll_find,
--- a/test/test_list.c	Sat Jan 29 12:46:07 2022 +0100
+++ b/test/test_list.c	Sat Jan 29 14:32:04 2022 +0100
@@ -1001,6 +1001,92 @@
     test_hl_linked_list_iterator_impl(list);
 }
 
+void test_hl_linked_list_insert_via_iterator(void) {
+    cxTestingAllocatorReset();
+    CxList list = cxLinkedListCreate(cxTestingAllocator, cmp_int, sizeof(int));
+    for (int i = 0; i < 5; i++) {
+        cxListAdd(list, &i);
+    }
+    CxIterator iter = cxListIterator(list, 2);
+    CU_ASSERT_EQUAL(iter.index, 2)
+    CU_ASSERT_EQUAL(*(int *) cxIteratorCurrent(&iter), 2)
+
+    int data = 10;
+    cxListInsertAfter(&iter, &data);
+    CU_ASSERT_EQUAL(iter.index, 2)
+    CU_ASSERT_EQUAL(*(int *) cxIteratorCurrent(&iter), 2)
+    data = 20;
+    cxListInsertBefore(&iter, &data);
+    CU_ASSERT_EQUAL(iter.index, 3)
+    CU_ASSERT_EQUAL(*(int *) cxIteratorCurrent(&iter), 2)
+
+    data = 30;
+    iter = cxListBegin(list);
+    cxListInsertBefore(&iter, &data);
+    CU_ASSERT_EQUAL(iter.index, 1)
+    CU_ASSERT_EQUAL(*(int *) cxIteratorCurrent(&iter), 0)
+    data = 40;
+    iter = cxListIterator(list, list->size);
+    cxListInsertBefore(&iter, &data);
+    CU_ASSERT_EQUAL(iter.index, 9)
+    CU_ASSERT_FALSE(cxIteratorValid(&iter))
+    data = 50;
+    iter = cxListIterator(list, list->size);
+    cxListInsertAfter(&iter, &data);
+    CU_ASSERT_EQUAL(iter.index, 10)
+    CU_ASSERT_FALSE(cxIteratorValid(&iter))
+
+    int expdata[] = {30, 0, 1, 20, 2, 10, 3, 4, 40, 50};
+    CxList expected = cxLinkedListFromArray(cxTestingAllocator,
+                                            cmp_int, sizeof(int), 10, expdata);
+
+    CU_ASSERT_EQUAL(0, cxListCompare(list, expected))
+    cxLinkedListDestroy(list);
+    cxLinkedListDestroy(expected);
+    CU_ASSERT_TRUE(cxTestingAllocatorVerify())
+}
+
+void test_hl_ptr_linked_list_insert_via_iterator(void) {
+    int testdata[] = {0, 1, 2, 3, 4, 10, 20, 30, 40, 50};
+    cxTestingAllocatorReset();
+    CxList list = cxPointerLinkedListCreate(cxTestingAllocator, cmp_int);
+    int i;
+    for (i = 0; i < 5; i++) {
+        cxListAdd(list, &testdata[i]);
+    }
+    CxIterator iter = cxListIterator(list, 2);
+    CU_ASSERT_EQUAL(iter.index, 2)
+    CU_ASSERT_EQUAL(*(int *) cxIteratorCurrent(&iter), 2)
+
+    cxListInsertAfter(&iter, &testdata[i++]);
+    CU_ASSERT_EQUAL(iter.index, 2)
+    CU_ASSERT_EQUAL(*(int *) cxIteratorCurrent(&iter), 2)
+    cxListInsertBefore(&iter, &testdata[i++]);
+    CU_ASSERT_EQUAL(iter.index, 3)
+    CU_ASSERT_EQUAL(*(int *) cxIteratorCurrent(&iter), 2)
+
+    iter = cxListBegin(list);
+    cxListInsertBefore(&iter, &testdata[i++]);
+    CU_ASSERT_EQUAL(iter.index, 1)
+    CU_ASSERT_EQUAL(*(int *) cxIteratorCurrent(&iter), 0)
+    iter = cxListIterator(list, list->size);
+    cxListInsertBefore(&iter, &testdata[i++]);
+    CU_ASSERT_EQUAL(iter.index, 9)
+    CU_ASSERT_FALSE(cxIteratorValid(&iter))
+    iter = cxListIterator(list, list->size);
+    cxListInsertAfter(&iter, &testdata[i++]);
+    CU_ASSERT_EQUAL(iter.index, 10)
+    CU_ASSERT_FALSE(cxIteratorValid(&iter))
+
+    int expdata[] = {30, 0, 1, 20, 2, 10, 3, 4, 40, 50};
+    for (i = 0; i < 10; i++) {
+        CU_ASSERT_EQUAL(*(int *) cxListAt(list, i), expdata[i])
+    }
+
+    cxLinkedListDestroy(list);
+    CU_ASSERT_TRUE(cxTestingAllocatorVerify())
+}
+
 int main() {
     CU_pSuite suite = NULL;
 
@@ -1037,6 +1123,7 @@
     cu_add_test(suite, test_hl_linked_list_find);
     cu_add_test(suite, test_hl_linked_list_sort);
     cu_add_test(suite, test_hl_linked_list_iterator);
+    cu_add_test(suite, test_hl_linked_list_insert_via_iterator);
 
     suite = CU_add_suite("high level pointer linked list", NULL, NULL);
 
@@ -1048,6 +1135,7 @@
     cu_add_test(suite, test_hl_ptr_linked_list_find);
     cu_add_test(suite, test_hl_ptr_linked_list_sort);
     cu_add_test(suite, test_hl_ptr_linked_list_iterator);
+    cu_add_test(suite, test_hl_ptr_linked_list_insert_via_iterator);
 
     CU_basic_set_mode(UCX_CU_BRM);
 

mercurial