summaryrefslogtreecommitdiff
path: root/filter/tree.c
diff options
context:
space:
mode:
authorPavel Machek <pavel@ucw.cz>1999-04-12 19:58:18 +0000
committerPavel Machek <pavel@ucw.cz>1999-04-12 19:58:18 +0000
commit38506f71b0bea5580987e999a7b1a69f58aec7ec (patch)
treef9779191375233fb91582d32eed2489c0e2032ce /filter/tree.c
parent01bd7759b260b379089acf28cc47bd49572ebd22 (diff)
Sets of integers now actually work. Sets of IP will work as soon as
compare function is ready.
Diffstat (limited to 'filter/tree.c')
-rw-r--r--filter/tree.c125
1 files changed, 125 insertions, 0 deletions
diff --git a/filter/tree.c b/filter/tree.c
new file mode 100644
index 00000000..ea1f18f2
--- /dev/null
+++ b/filter/tree.c
@@ -0,0 +1,125 @@
+/*
+ * Filters: utility functions
+ *
+ * Copyright 1998 Pavel Machek <pavel@ucw.cz>
+ *
+ * Can be freely distributed and used under the terms of the GNU GPL.
+ */
+
+#include <stdio.h>
+#include <fcntl.h>
+#include <unistd.h>
+#include <sys/signal.h>
+#include <setjmp.h>
+
+#include "nest/bird.h"
+#include "lib/lists.h"
+#include "lib/resource.h"
+#include "lib/socket.h"
+#include "nest/route.h"
+#include "nest/protocol.h"
+#include "nest/iface.h"
+#include "conf/conf.h"
+#include "filter/filter.h"
+
+/* Finds n-th item in list linked by right. Trashes pointers in right. */
+static struct f_tree *
+find_nth(struct f_tree *from, int nth)
+{
+ struct f_tree *pivot;
+ int lcount = 0, rcount = 0;
+ struct f_tree *left, *right, *next;
+
+ pivot = from;
+
+ left = right = NULL;
+ next = from->right;
+ while (from = next) {
+ next = from->right;
+ if (val_compare(pivot->from, from->from)==1) {
+ from->right = left;
+ left = from;
+ lcount++;
+ } else {
+ from->right = right;
+ right = from;
+ rcount++;
+ }
+ }
+ if (lcount == nth)
+ return pivot;
+ if (lcount < nth)
+ return find_nth(right, nth-lcount-1);
+ return find_nth(left, nth);
+}
+
+/* Gets list linked by left, finds its median, trashes pointers in right */
+static struct f_tree *
+find_median(struct f_tree *from)
+{
+ struct f_tree *t = from;
+ int cnt = 0;
+
+ if (!from)
+ return NULL;
+ do {
+ t->right = t->left;
+ cnt++;
+ } while (t = t->left);
+ return find_nth(from, cnt/2);
+}
+
+struct f_tree *
+find_tree(struct f_tree *t, struct f_val val)
+{
+ if (!t)
+ return 0;
+ if ((val_compare(t->from, val) != 1) &&
+ (val_compare(t->to, val) != -1))
+ return t;
+ if (val_compare(t->from, val) == -1)
+ return find_tree(t->right, val);
+ else
+ return find_tree(t->left, val);
+}
+
+/* Gets list linked by left */
+struct f_tree *
+build_tree(struct f_tree *from)
+{
+ struct f_tree *median, *t = from, *next, *left = NULL, *right = NULL;
+
+ median = find_median(from);
+ if (!median)
+ return NULL;
+
+ do {
+ next = t->left;
+ if (t == median)
+ continue;
+
+ if (val_compare(median->from, t->from)==1) {
+ t->left = left;
+ left = t;
+ } else {
+ t->left = right;
+ right = t;
+ }
+ } while(t = next);
+
+ median->left = build_tree(left);
+ median->right = build_tree(right);
+ return median;
+}
+
+struct f_tree *
+f_new_tree(void)
+{
+ struct f_tree * ret;
+ ret = cfg_alloc(sizeof(struct f_tree));
+ ret->left = ret->right = NULL;
+ ret->from.type = ret->to.type = T_VOID;
+ ret->from.val.i = ret->to.val.i = 0;
+ ret->data = NULL;
+ return ret;
+}