summaryrefslogtreecommitdiff
path: root/server/util/btreemap.c
diff options
context:
space:
mode:
Diffstat (limited to 'server/util/btreemap.c')
-rw-r--r--server/util/btreemap.c175
1 files changed, 175 insertions, 0 deletions
diff --git a/server/util/btreemap.c b/server/util/btreemap.c
new file mode 100644
index 00000000..7c20086b
--- /dev/null
+++ b/server/util/btreemap.c
@@ -0,0 +1,175 @@
+/*
+ SSSD
+
+ Service monitor
+
+ Copyright (C) Stephen Gallagher 2008
+
+ This program is free software; you can redistribute it and/or modify
+ it under the terms of the GNU General Public License as published by
+ the Free Software Foundation; either version 3 of the License, or
+ (at your option) any later version.
+
+ This program 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 General Public License for more details.
+
+ You should have received a copy of the GNU General Public License
+ along with this program. If not, see <http://www.gnu.org/licenses/>.
+ */
+
+#include "talloc.h"
+#include "util/btreemap.h"
+#include "util/util.h"
+
+struct btreemap
+{
+ /* NULL keys are not allowed */
+ void *key;
+
+ /* NULL values are permitted */
+ void *value;
+
+ struct btreemap *left;
+ struct btreemap *right;
+
+ /* comparator must return -1, 0 or 1
+ * and should only be set for the root node
+ * other nodes will be ignored.
+ */
+ btreemap_comparison_fn comparator;
+};
+
+/* btreemap_search_key
+ * Searches a btreemap for an entry with a specific key
+ * If found, it will return 0 and node will be set to the
+ * appropriate node.
+ * If not found, it will set the following:
+ * -2: The map was empty, create a new map when adding keys
+ * -1: A new node created should use node->left
+ * 1: A new node created should use node->right
+ */
+int btreemap_search_key(struct btreemap *map, void *key, struct btreemap **node)
+{
+ struct btreemap *tempnode;
+ int result;
+ int found = -2;
+
+ if (!map)
+ {
+ *node = NULL;
+ return -2;
+ }
+
+ tempnode = map;
+ while (found == -2) {
+ result = tempnode->comparator(tempnode->key, key);
+ if (result > 0)
+ {
+ if (tempnode->right)
+ tempnode=tempnode->right;
+ else
+ {
+ found = 1;
+ }
+ } else if (result < 0)
+ {
+ if (tempnode->left)
+ tempnode=tempnode->left;
+ else
+ {
+ found = -1;
+ }
+ } else
+ {
+ /* This entry matched */
+ found = 0;
+ }
+ }
+
+ *node = tempnode;
+ return found;
+}
+
+void *btreemap_get_value(struct btreemap *map, void *key)
+{
+ struct btreemap *node;
+ int found;
+
+ if (!map || !key)
+ {
+ return NULL;
+ }
+
+ /* Search for the key */
+ found = btreemap_search_key(map, key, &node);
+ if (found == 0)
+ {
+ return node->value;
+ }
+
+ /* If the key was not found, return NULL */
+ return NULL;
+}
+
+int btreemap_set_value(struct btreemap **map, void *key, void *value,
+ btreemap_comparison_fn comparator)
+{
+ struct btreemap *node;
+ struct btreemap *new_node;
+ int found;
+
+ if (!key)
+ {
+ return EINVAL;
+ }
+
+ /* Search for the key */
+ found = btreemap_search_key(*map, key, &node);
+ if (found == 0)
+ {
+ /* Update existing value */
+ node->value = value;
+ return EOK;
+ }
+
+ /* Need to add a value to the tree */
+ new_node = talloc_zero(node, struct btreemap);
+ if (!new_node)
+ {
+ return ENOMEM;
+ }
+ new_node->key = talloc_steal(*map, key);
+ new_node->value = talloc_steal(*map, value);
+ new_node->comparator = comparator;
+
+ if (found == -2)
+ {
+ *map = new_node;
+ }
+ if (found == -1)
+ {
+ node->left = new_node;
+ } else if (found == 1)
+ {
+ node->right = new_node;
+ }
+ return EOK;
+}
+
+struct btreemap *btreemap_new(void *key, void *value,
+ btreemap_comparison_fn comparator)
+{
+ struct btreemap *map;
+ int result;
+
+ map = NULL;
+ result = btreemap_set_value(&map, key, value, comparator);
+ if (result != EOK)
+ {
+ return NULL;
+ }
+
+ return map;
+}