summaryrefslogtreecommitdiffstats
path: root/ksysguard/CContLib
diff options
context:
space:
mode:
Diffstat (limited to 'ksysguard/CContLib')
-rw-r--r--ksysguard/CContLib/Makefile.am6
-rw-r--r--ksysguard/CContLib/ccont.c364
-rw-r--r--ksysguard/CContLib/ccont.h161
3 files changed, 531 insertions, 0 deletions
diff --git a/ksysguard/CContLib/Makefile.am b/ksysguard/CContLib/Makefile.am
new file mode 100644
index 000000000..a3f5d786d
--- /dev/null
+++ b/ksysguard/CContLib/Makefile.am
@@ -0,0 +1,6 @@
+AUTOMAKE_OPTIONS = foreign
+
+noinst_LIBRARIES = libccont.a
+noinst_HEADERS = ccont.h
+
+libccont_a_SOURCES = ccont.c
diff --git a/ksysguard/CContLib/ccont.c b/ksysguard/CContLib/ccont.c
new file mode 100644
index 000000000..501429337
--- /dev/null
+++ b/ksysguard/CContLib/ccont.c
@@ -0,0 +1,364 @@
+/*
+ Lightweight C Container Library
+
+ Copyright (c) 2002 Tobias Koenig <[email protected]>
+
+ Original code was written by Chris Schlaeger <[email protected]>
+
+ This library is free software; you can redistribute it and/or
+ modify it under the terms of the GNU Library General Public
+ License as published by the Free Software Foundation; either
+ version 2 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
+ Library General Public License for more details.
+
+ You should have received a copy of the GNU Library General Public License
+ along with this library; see the file COPYING.LIB. If not, write to
+ the Free Software Foundation, Inc., 51 Franklin Street, Fifth Floor,
+ Boston, MA 02110-1301, USA.
+
+*/
+
+#include "ccont.h"
+
+#include <stdio.h>
+#include <stdlib.h>
+
+struct container_info {
+ INDEX count;
+ CONTAINER currentNode;
+};
+
+typedef struct container_info T_CONTAINER_INFO;
+typedef struct container_info* CONTAINER_INFO;
+
+#define rpterr(x) fprintf(stderr, "%s\n", x)
+
+CONTAINER new_ctnr(void)
+{
+ CONTAINER_INFO info;
+ CONTAINER rootNode;
+
+ rootNode = (CONTAINER)malloc(sizeof(T_CONTAINER));
+
+ info = (CONTAINER_INFO)malloc(sizeof(T_CONTAINER_INFO));
+ info->count = 0;
+ info->currentNode = rootNode;
+
+ rootNode->next = rootNode;
+ rootNode->prev = rootNode;
+ rootNode->data = info;
+
+ return rootNode;
+}
+
+void zero_destr_ctnr(CONTAINER rootNode, DESTR_FUNC destr_func)
+{
+ INDEX counter;
+
+ if (rootNode == NIL || destr_func == NIL) {
+ rpterr("destr_ctnr: NIL argument");
+ return;
+ }
+
+ for (counter = level_ctnr(rootNode); counter > -1; --counter)
+ destr_func(pop_ctnr(rootNode));
+
+ if (rootNode->data)
+ free(rootNode->data);
+
+ free(rootNode);
+ rootNode = 0;
+
+ return;
+}
+
+INDEX level_ctnr(CONTAINER rootNode)
+{
+ if (rootNode == NIL) {
+ rpterr("level_ctnr: NIL argument");
+ return -1;
+ }
+
+ return ((CONTAINER_INFO)rootNode->data)->count;
+}
+
+void insert_ctnr(CONTAINER rootNode, void* object, INDEX pos)
+{
+ CONTAINER it;
+ INDEX counter = 0;
+
+ if (rootNode == NIL || object == NIL) {
+ rpterr("insert_ctnr: NIL argument");
+ return;
+ }
+
+ for (it = rootNode->next; it != rootNode; it = it->next) {
+ if (counter == pos) {
+ CONTAINER newNode = (CONTAINER)malloc(sizeof(T_CONTAINER));
+
+ newNode->prev = it;
+ newNode->next = it->next;
+ it->next->prev = newNode;
+ it->next = newNode;
+
+ newNode->data = object;
+ ((CONTAINER_INFO)rootNode->data)->count++;
+ return;
+ }
+
+ counter++;
+ }
+}
+
+void push_ctnr(CONTAINER rootNode, void* object)
+{
+ CONTAINER newNode;
+
+ if (rootNode == NIL || object == NIL) {
+ rpterr("push_ctnr: NIL argument");
+ return;
+ }
+
+ newNode = (CONTAINER)malloc(sizeof(T_CONTAINER));
+ newNode->next = rootNode;
+ newNode->prev = rootNode->prev;
+ rootNode->prev->next = newNode;
+ rootNode->prev = newNode;
+
+ newNode->data = object;
+ ((CONTAINER_INFO)rootNode->data)->count++;
+}
+
+void* remove_at_ctnr(CONTAINER rootNode, INDEX pos)
+{
+ CONTAINER it;
+ INDEX counter = 0;
+ void* retval;
+
+ if (rootNode == NIL) {
+ rpterr("remove_ctnr: NIL argument");
+ return NIL;
+ }
+
+ for (it = rootNode->next; it != rootNode; it = it->next) {
+ if (counter == pos) {
+ retval = it->data;
+
+ it->prev->next = it->next;
+ it->next->prev = it->prev;
+
+ free(it);
+
+ ((CONTAINER_INFO)rootNode->data)->count--;
+ return retval;
+ }
+
+ counter++;
+ }
+
+ return NIL;
+}
+
+void* pop_ctnr(CONTAINER rootNode)
+{
+ CONTAINER ptr;
+ void* retval;
+
+ if (rootNode == NIL) {
+ rpterr("pop_ctnr: NIL argument");
+ return NIL;
+ }
+
+ if (rootNode->next == rootNode)
+ return NIL;
+
+ ptr = rootNode->next;
+ retval = ptr->data;
+
+ rootNode->next = ptr->next;
+ ptr->next->prev = rootNode;
+
+ ((CONTAINER_INFO)rootNode->data)->count--;
+
+ free(ptr);
+
+ return retval;
+}
+
+void* get_ctnr(CONTAINER rootNode, INDEX pos)
+{
+ CONTAINER it;
+ INDEX counter = 0;
+
+ if (rootNode == NIL) {
+ rpterr("get_ctnr: NIL argument");
+ return NIL;
+ }
+
+ for (it = rootNode->next; it != rootNode; it = it->next) {
+ if (counter == pos)
+ return it->data;
+
+ counter++;
+ }
+
+ return NIL;
+}
+
+INDEX search_ctnr(CONTAINER rootNode, COMPARE_FUNC compare_func, void* pattern)
+{
+ CONTAINER it;
+ INDEX counter = 0;
+
+ if (rootNode == NIL || compare_func == NIL || pattern == NIL) {
+ rpterr("search_ctnr: NIL argument");
+ return -1;
+ }
+
+ for (it = rootNode->next; it != rootNode; it = it->next) {
+ if (compare_func(pattern, it->data) == 0)
+ return counter;
+
+ counter++;
+ }
+
+ return -1;
+}
+
+void swop_ctnr(CONTAINER rootNode, INDEX pos1, INDEX pos2)
+{
+ CONTAINER it, node1, node2;
+ INDEX counter = 0;
+ int found = 0;
+ void* tmpData;
+
+ if (rootNode == NIL) {
+ rpterr("swop_ctnr: NIL argument");
+ return;
+ }
+
+ if (pos1 == pos2)
+ return;
+
+ /**
+ * it is a bit hackish because of the 'goto' but it's fast
+ * since we have to run through the list only once
+ */
+ for (it = rootNode->next; it != rootNode; it = it->next) {
+ if (counter == pos1) {
+ node1 = it;
+ if (found)
+ goto swapIt;
+ else
+ found = 1;
+ }
+ if (counter == pos2) {
+ node2 = it;
+ if (found)
+ goto swapIt;
+ else
+ found = 1;
+ }
+
+ counter++;
+ }
+
+ return;
+
+swapIt:
+ tmpData = node1->data;
+ node1->data = node2->data;
+ node2->data = tmpData;
+ return;
+}
+
+void bsort_ctnr(CONTAINER rootNode, COMPARE_FUNC compare_func)
+{
+ INDEX right, i, level, last;
+
+ if (rootNode == NIL || compare_func == NIL) {
+ rpterr("destr_ctnr: NIL argument");
+ return;
+ }
+
+ last = level = level_ctnr(rootNode);
+ do
+ {
+ right = last;
+ last = 0;
+ for (i = 1; i < right; i++)
+ if (compare_func(get_ctnr(rootNode, i - 1), get_ctnr(rootNode, i)) > 0)
+ swop_ctnr(rootNode, i - 1, last = i);
+ } while (last > 0);
+}
+
+void* first_ctnr(CONTAINER rootNode)
+{
+ if (rootNode == NIL) {
+ rpterr("first_ctnr: NIL argument");
+ return NIL;
+ }
+
+ if (rootNode->next == rootNode)
+ return NIL;
+
+ ((CONTAINER_INFO)rootNode->data)->currentNode = rootNode->next;
+
+ return rootNode->next->data;
+}
+
+void* next_ctnr(CONTAINER rootNode)
+{
+ CONTAINER_INFO info;
+
+ if (rootNode == NIL) {
+ rpterr("next_ctnr: NIL argument");
+ return NIL;
+ }
+
+ info = (CONTAINER_INFO)rootNode->data;
+
+ if (info->currentNode->next == rootNode)
+ return NIL;
+
+ info->currentNode = info->currentNode->next;
+
+ return info->currentNode->data;
+}
+
+void* remove_ctnr(CONTAINER rootNode)
+{
+ CONTAINER currentNode, tmp;
+ CONTAINER_INFO info;
+ void* retval;
+
+ if (rootNode == NIL) {
+ rpterr("remove_curr_ctnr: NIL argument");
+ return NIL;
+ }
+
+ info = (CONTAINER_INFO)rootNode->data;
+ currentNode = info->currentNode;
+
+ if (currentNode == rootNode) { /* should never happen */
+ rpterr("remove_curr_ctnr: delete root node");
+ return NIL;
+ }
+
+ retval = currentNode->data;
+ tmp = currentNode->prev;
+
+ currentNode->prev->next = currentNode->next;
+ currentNode->next->prev = currentNode->prev;
+
+ free(currentNode);
+
+ info->count--;
+ info->currentNode = tmp;
+
+ return retval;
+}
diff --git a/ksysguard/CContLib/ccont.h b/ksysguard/CContLib/ccont.h
new file mode 100644
index 000000000..66b22eba4
--- /dev/null
+++ b/ksysguard/CContLib/ccont.h
@@ -0,0 +1,161 @@
+/*
+ Lightweight C Container Library
+
+ Copyright (c) 2002 Tobias Koenig <[email protected]>
+
+ Original code was written by Chris Schlaeger <[email protected]>
+
+ This library is free software; you can redistribute it and/or
+ modify it under the terms of the GNU Library General Public
+ License as published by the Free Software Foundation; either
+ version 2 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
+ Library General Public License for more details.
+
+ You should have received a copy of the GNU Library General Public License
+ along with this library; see the file COPYING.LIB. If not, write to
+ the Free Software Foundation, Inc., 51 Franklin Street, Fifth Floor,
+ Boston, MA 02110-1301, USA.
+
+*/
+
+#ifndef _CCONT_H
+#define _CCONT_H
+
+#ifndef NIL
+#define NIL ((void*) 0)
+#endif
+
+#define destr_ctnr(x, y) zero_destr_ctnr(x, y); x=0
+
+struct container {
+ struct container* next;
+ struct container* prev;
+ void* data;
+};
+
+typedef struct container T_CONTAINER;
+typedef struct container* CONTAINER;
+
+typedef long INDEX;
+
+typedef void (*DESTR_FUNC)(void*);
+typedef int (*COMPARE_FUNC)(void*, void*);
+
+/**
+ * Initialize the container @p ctnr.
+ */
+CONTAINER new_ctnr(void);
+
+/**
+ * Remove all entries from @p ctnr and reset its
+ * internal structure. Use @ref new_ctnr() @em before @em
+ * access the container next time.
+ *
+ * Note: use the 'destr_ctnr' macro to get zeroed pointer
+ * automatically.
+ *
+ * @param destr_func The function that is called to
+ * free the single entries.
+ */
+void zero_destr_ctnr(CONTAINER ctnr, DESTR_FUNC destr_func);
+
+/**
+ * @return the number of entries in @p ctnr.
+ */
+INDEX level_ctnr(CONTAINER ctnr);
+
+/**
+ * Insert a new entry in container.
+ *
+ * @param object A pointer to the object.
+ * @param pos The position where the object should be insert.
+ */
+void insert_ctnr(CONTAINER ctnr, void* object, INDEX pos);
+
+/**
+ * Add a new entry at the end of container.
+ *
+ * @param object The object that should be added.
+ */
+void push_ctnr(CONTAINER ctnr, void* object);
+
+/**
+ * Remove an entry from container.
+ *
+ * @param pos The position of the object that should be removed.
+ *
+ * @return A pointer to the removed object or @p 0L if it doesn't exist.
+ */
+void* remove_at_ctnr(CONTAINER ctnr, INDEX pos);
+
+/**
+ * Remove the first entry of container.
+ *
+ * @return A pointer to the removed object or @p 0L if there is now entry.
+ */
+void* pop_ctnr(CONTAINER ctnr);
+
+/**
+ * @return A pointer to the object at position @a pos
+ * or @p 0L if it doesn't exist.
+ */
+void* get_ctnr(CONTAINER ctnr, INDEX pos);
+
+/**
+ * @return The position of a matching entry.
+ *
+ * @param compare_func A Pointer to the function that is
+ * called to compare all entries in the
+ * container with the given pattern.
+ * @param pattern The pattern for coparison.
+ */
+INDEX search_ctnr(CONTAINER ctnr, COMPARE_FUNC compare_func, void* pattern);
+
+/**
+ * Swap two objects in container.
+ *
+ * @param pos1 Position of the first object.
+ * @param pos2 Position of the second object.
+ */
+void swop_ctnr(CONTAINER ctnr, INDEX pos1, INDEX pos2);
+
+/**
+ * Sort all entries of container.
+ *
+ * @param compare_func A Pointer to the function that is
+ * called to compare to entries of the
+ * container.
+ */
+void bsort_ctnr(CONTAINER ctnr, COMPARE_FUNC compare_func);
+
+/**
+ * Use this function to iterate over the container.
+ *
+ * for (ptr = first_ctnr(ctnr); ptr; ptr = next_ctnr(ctnr)) {
+ * do_anything(ptr);
+ * }
+ *
+ * @return A Pointer to the first object in container.
+ */
+void* first_ctnr(CONTAINER ctnr);
+
+/**
+ * Use this function to iterate over the container.
+ *
+ * @return A Pointer to the next object in container.
+ */
+void* next_ctnr(CONTAINER ctnr);
+
+/**
+ * Use this function to remove the current entry while
+ * iterating over the container.
+ *
+ * @return A Pointer to the removed object or @p 0L if it doesn't exist.
+ */
+void* remove_ctnr(CONTAINER ctnr);
+
+#endif