| 
									
										
										
										
											1999-04-12 19:58:18 +00:00
										 |  |  | /*
 | 
					
						
							|  |  |  |  *	Filters: utility functions | 
					
						
							|  |  |  |  * | 
					
						
							|  |  |  |  *	Copyright 1998 Pavel Machek <pavel@ucw.cz> | 
					
						
							|  |  |  |  * | 
					
						
							|  |  |  |  *	Can be freely distributed and used under the terms of the GNU GPL. | 
					
						
							|  |  |  |  */ | 
					
						
							|  |  |  | 
 | 
					
						
							| 
									
										
										
										
											2010-02-17 21:53:07 +01:00
										 |  |  | #include "lib/alloca.h"
 | 
					
						
							| 
									
										
										
										
											1999-04-12 19:58:18 +00:00
										 |  |  | #include "nest/bird.h"
 | 
					
						
							|  |  |  | #include "conf/conf.h"
 | 
					
						
							|  |  |  | #include "filter/filter.h"
 | 
					
						
							| 
									
										
										
										
											2019-02-08 13:38:12 +01:00
										 |  |  | #include "filter/data.h"
 | 
					
						
							| 
									
										
										
										
											1999-04-12 19:58:18 +00:00
										 |  |  | 
 | 
					
						
							| 
									
										
										
										
											2000-06-07 13:54:06 +00:00
										 |  |  | /**
 | 
					
						
							|  |  |  |  * find_tree | 
					
						
							|  |  |  |  * @t: tree to search in | 
					
						
							|  |  |  |  * @val: value to find | 
					
						
							|  |  |  |  * | 
					
						
							|  |  |  |  * Search for given value in the tree. I relies on fact that sorted tree is populated | 
					
						
							|  |  |  |  * by &f_val structures (that can be compared by val_compare()). In each node of tree,  | 
					
						
							|  |  |  |  * either single value (then t->from==t->to) or range is present. | 
					
						
							| 
									
										
										
										
											2000-06-07 14:16:11 +00:00
										 |  |  |  * | 
					
						
							| 
									
										
										
										
											2000-06-07 14:45:55 +00:00
										 |  |  |  * Both set matching and |switch() { }| construction is implemented using this function, | 
					
						
							| 
									
										
										
										
											2000-06-07 14:16:11 +00:00
										 |  |  |  * thus both are as fast as they can be. | 
					
						
							| 
									
										
										
										
											2000-06-07 13:54:06 +00:00
										 |  |  |  */ | 
					
						
							| 
									
										
										
										
											2018-12-27 14:26:11 +01:00
										 |  |  | const struct f_tree * | 
					
						
							|  |  |  | find_tree(const struct f_tree *t, const struct f_val *val) | 
					
						
							| 
									
										
										
										
											1999-04-12 19:58:18 +00:00
										 |  |  | { | 
					
						
							|  |  |  |   if (!t) | 
					
						
							| 
									
										
										
										
											1999-10-28 21:03:36 +00:00
										 |  |  |     return NULL; | 
					
						
							| 
									
										
										
										
											2018-12-27 14:26:11 +01:00
										 |  |  |   if ((val_compare(&(t->from), val) != 1) && | 
					
						
							|  |  |  |       (val_compare(&(t->to), val) != -1)) | 
					
						
							| 
									
										
										
										
											1999-04-12 19:58:18 +00:00
										 |  |  |     return t; | 
					
						
							| 
									
										
										
										
											2018-12-27 14:26:11 +01:00
										 |  |  |   if (val_compare(&(t->from), val) == -1) | 
					
						
							| 
									
										
										
										
											1999-04-12 19:58:18 +00:00
										 |  |  |     return find_tree(t->right, val); | 
					
						
							|  |  |  |   else | 
					
						
							|  |  |  |     return find_tree(t->left, val); | 
					
						
							|  |  |  | } | 
					
						
							|  |  |  | 
 | 
					
						
							| 
									
										
										
										
											2023-01-07 20:18:44 +01:00
										 |  |  | /**
 | 
					
						
							|  |  |  |  * find_tree_linear | 
					
						
							|  |  |  |  * @t: tree to search in | 
					
						
							|  |  |  |  * @val: value to find | 
					
						
							|  |  |  |  * | 
					
						
							|  |  |  |  * Search for given value in the degenerated linear tree, which is generated by | 
					
						
							|  |  |  |  * parser before build_tree() is applied. The tree is not sorted and all nodes | 
					
						
							|  |  |  |  * are linked by left ptr. | 
					
						
							|  |  |  |  */ | 
					
						
							|  |  |  | const struct f_tree * | 
					
						
							|  |  |  | find_tree_linear(const struct f_tree *t, const struct f_val *val) | 
					
						
							|  |  |  | { | 
					
						
							|  |  |  |   for (; t; t = t->left) | 
					
						
							|  |  |  |     if ((val_compare(&(t->from), val) != 1) && | 
					
						
							|  |  |  | 	(val_compare(&(t->to), val) != -1)) | 
					
						
							|  |  |  |       return t; | 
					
						
							|  |  |  | 
 | 
					
						
							|  |  |  |   return NULL; | 
					
						
							|  |  |  | } | 
					
						
							|  |  |  | 
 | 
					
						
							| 
									
										
										
										
											2010-02-17 21:53:07 +01:00
										 |  |  | static struct f_tree * | 
					
						
							|  |  |  | build_tree_rec(struct f_tree **buf, int l, int h) | 
					
						
							|  |  |  | { | 
					
						
							|  |  |  |   struct f_tree *n; | 
					
						
							|  |  |  |   int pos; | 
					
						
							|  |  |  | 
 | 
					
						
							|  |  |  |   if (l >= h) | 
					
						
							|  |  |  |     return NULL; | 
					
						
							|  |  |  | 
 | 
					
						
							|  |  |  |   pos = (l+h)/2; | 
					
						
							|  |  |  |   n = buf[pos]; | 
					
						
							|  |  |  |   n->left = build_tree_rec(buf, l, pos); | 
					
						
							|  |  |  |   n->right = build_tree_rec(buf, pos+1, h); | 
					
						
							|  |  |  |   return n; | 
					
						
							|  |  |  | } | 
					
						
							|  |  |  | 
 | 
					
						
							| 
									
										
										
										
											2013-10-02 14:41:37 +02:00
										 |  |  | static int  | 
					
						
							|  |  |  | tree_compare(const void *p1, const void *p2) | 
					
						
							|  |  |  | { | 
					
						
							| 
									
										
										
										
											2018-12-27 14:26:11 +01:00
										 |  |  |   return val_compare(&((* (struct f_tree **) p1)->from), &((* (struct f_tree **) p2)->from)); | 
					
						
							| 
									
										
										
										
											2013-10-02 14:41:37 +02:00
										 |  |  | } | 
					
						
							| 
									
										
										
										
											2010-02-17 21:53:07 +01:00
										 |  |  | 
 | 
					
						
							| 
									
										
										
										
											2000-06-07 13:54:06 +00:00
										 |  |  | /**
 | 
					
						
							|  |  |  |  * build_tree | 
					
						
							| 
									
										
										
										
											2000-06-07 14:16:11 +00:00
										 |  |  |  * @from: degenerated tree (linked by @tree->left) to be transformed into form suitable for find_tree() | 
					
						
							| 
									
										
										
										
											2000-06-07 13:54:06 +00:00
										 |  |  |  * | 
					
						
							| 
									
										
										
										
											2016-11-08 17:46:29 +01:00
										 |  |  |  * Transforms degenerated tree into balanced tree. | 
					
						
							| 
									
										
										
										
											2000-06-07 13:54:06 +00:00
										 |  |  |  */ | 
					
						
							| 
									
										
										
										
											1999-04-12 19:58:18 +00:00
										 |  |  | struct f_tree * | 
					
						
							|  |  |  | build_tree(struct f_tree *from) | 
					
						
							|  |  |  | { | 
					
						
							| 
									
										
										
										
											2010-02-17 21:53:07 +01:00
										 |  |  |   struct f_tree *tmp, *root; | 
					
						
							|  |  |  |   struct f_tree **buf; | 
					
						
							|  |  |  |   int len, i; | 
					
						
							| 
									
										
										
										
											1999-04-12 19:58:18 +00:00
										 |  |  | 
 | 
					
						
							| 
									
										
										
										
											2010-02-17 21:53:07 +01:00
										 |  |  |   if (from == NULL) | 
					
						
							| 
									
										
										
										
											1999-04-12 19:58:18 +00:00
										 |  |  |     return NULL; | 
					
						
							|  |  |  | 
 | 
					
						
							| 
									
										
										
										
											2010-02-17 21:53:07 +01:00
										 |  |  |   len = 0; | 
					
						
							|  |  |  |   for (tmp = from; tmp != NULL; tmp = tmp->left) | 
					
						
							|  |  |  |     len++; | 
					
						
							|  |  |  | 
 | 
					
						
							|  |  |  |   if (len <= 1024) | 
					
						
							|  |  |  |     buf = alloca(len * sizeof(struct f_tree *)); | 
					
						
							|  |  |  |   else | 
					
						
							| 
									
										
										
										
											2016-09-06 17:18:15 +02:00
										 |  |  |     buf = xmalloc(len * sizeof(struct f_tree *)); | 
					
						
							| 
									
										
										
										
											2010-02-17 21:53:07 +01:00
										 |  |  | 
 | 
					
						
							|  |  |  |   /* Convert a degenerated tree into an sorted array */ | 
					
						
							|  |  |  |   i = 0; | 
					
						
							|  |  |  |   for (tmp = from; tmp != NULL; tmp = tmp->left) | 
					
						
							|  |  |  |     buf[i++] = tmp; | 
					
						
							|  |  |  | 
 | 
					
						
							|  |  |  |   qsort(buf, len, sizeof(struct f_tree *), tree_compare); | 
					
						
							|  |  |  | 
 | 
					
						
							|  |  |  |   root = build_tree_rec(buf, 0, len); | 
					
						
							|  |  |  | 
 | 
					
						
							|  |  |  |   if (len > 1024) | 
					
						
							| 
									
										
										
										
											2016-09-06 17:18:15 +02:00
										 |  |  |     xfree(buf); | 
					
						
							| 
									
										
										
										
											2010-02-17 21:53:07 +01:00
										 |  |  | 
 | 
					
						
							|  |  |  |   return root; | 
					
						
							| 
									
										
										
										
											1999-04-12 19:58:18 +00:00
										 |  |  | } | 
					
						
							|  |  |  | 
 | 
					
						
							|  |  |  | struct f_tree * | 
					
						
							|  |  |  | f_new_tree(void) | 
					
						
							|  |  |  | { | 
					
						
							| 
									
										
										
										
											2020-11-24 04:09:11 +01:00
										 |  |  |   struct f_tree *ret = cfg_allocz(sizeof(struct f_tree)); | 
					
						
							| 
									
										
										
										
											1999-04-12 19:58:18 +00:00
										 |  |  |   return ret; | 
					
						
							|  |  |  | } | 
					
						
							| 
									
										
										
										
											2000-01-31 17:44:22 +00:00
										 |  |  | 
 | 
					
						
							| 
									
										
										
										
											2000-06-07 14:16:11 +00:00
										 |  |  | /**
 | 
					
						
							|  |  |  |  * same_tree | 
					
						
							|  |  |  |  * @t1: first tree to be compared | 
					
						
							|  |  |  |  * @t2: second one | 
					
						
							|  |  |  |  * | 
					
						
							|  |  |  |  * Compares two trees and returns 1 if they are same | 
					
						
							|  |  |  |  */ | 
					
						
							| 
									
										
										
										
											2000-01-31 17:44:22 +00:00
										 |  |  | int | 
					
						
							| 
									
										
										
										
											2018-12-27 14:26:11 +01:00
										 |  |  | same_tree(const struct f_tree *t1, const struct f_tree *t2) | 
					
						
							| 
									
										
										
										
											2000-01-31 17:44:22 +00:00
										 |  |  | { | 
					
						
							|  |  |  |   if ((!!t1) != (!!t2)) | 
					
						
							|  |  |  |     return 0; | 
					
						
							|  |  |  |   if (!t1) | 
					
						
							|  |  |  |     return 1; | 
					
						
							| 
									
										
										
										
											2018-12-27 14:26:11 +01:00
										 |  |  |   if (val_compare(&(t1->from), &(t2->from))) | 
					
						
							| 
									
										
										
										
											2000-01-31 17:44:22 +00:00
										 |  |  |     return 0; | 
					
						
							| 
									
										
										
										
											2018-12-27 14:26:11 +01:00
										 |  |  |   if (val_compare(&(t1->to), &(t2->to))) | 
					
						
							| 
									
										
										
										
											2000-01-31 17:44:22 +00:00
										 |  |  |     return 0; | 
					
						
							|  |  |  |   if (!same_tree(t1->left, t2->left)) | 
					
						
							|  |  |  |     return 0; | 
					
						
							|  |  |  |   if (!same_tree(t1->right, t2->right)) | 
					
						
							|  |  |  |     return 0; | 
					
						
							| 
									
										
										
										
											2018-12-27 14:26:11 +01:00
										 |  |  |   if (!f_same(t1->data, t2->data)) | 
					
						
							| 
									
										
										
										
											2000-01-31 17:44:22 +00:00
										 |  |  |     return 0; | 
					
						
							|  |  |  |   return 1; | 
					
						
							|  |  |  | } | 
					
						
							| 
									
										
										
										
											2013-10-05 20:12:28 +02:00
										 |  |  | 
 | 
					
						
							| 
									
										
										
										
											2022-09-27 16:47:52 +02:00
										 |  |  | int | 
					
						
							|  |  |  | tree_node_count(const struct f_tree *t) | 
					
						
							|  |  |  | { | 
					
						
							|  |  |  |   if (t == NULL) | 
					
						
							|  |  |  |     return 0; | 
					
						
							|  |  |  | 
 | 
					
						
							|  |  |  |   return 1 + tree_node_count(t->left) + tree_node_count(t->right); | 
					
						
							|  |  |  | } | 
					
						
							| 
									
										
										
										
											2013-10-05 20:12:28 +02:00
										 |  |  | 
 | 
					
						
							|  |  |  | static void | 
					
						
							| 
									
										
										
										
											2018-12-27 14:26:11 +01:00
										 |  |  | tree_node_format(const struct f_tree *t, buffer *buf) | 
					
						
							| 
									
										
										
										
											2013-10-05 20:12:28 +02:00
										 |  |  | { | 
					
						
							|  |  |  |   if (t == NULL) | 
					
						
							|  |  |  |     return; | 
					
						
							|  |  |  | 
 | 
					
						
							|  |  |  |   tree_node_format(t->left, buf); | 
					
						
							|  |  |  | 
 | 
					
						
							| 
									
										
										
										
											2018-12-27 14:26:11 +01:00
										 |  |  |   val_format(&(t->from), buf); | 
					
						
							|  |  |  |   if (val_compare(&(t->from), &(t->to)) != 0) | 
					
						
							| 
									
										
										
										
											2013-10-05 20:12:28 +02:00
										 |  |  |   { | 
					
						
							|  |  |  |     buffer_puts(buf, ".."); | 
					
						
							| 
									
										
										
										
											2018-12-27 14:26:11 +01:00
										 |  |  |     val_format(&(t->to), buf); | 
					
						
							| 
									
										
										
										
											2013-10-05 20:12:28 +02:00
										 |  |  |   } | 
					
						
							|  |  |  |   buffer_puts(buf, ", "); | 
					
						
							|  |  |  | 
 | 
					
						
							|  |  |  |   tree_node_format(t->right, buf); | 
					
						
							|  |  |  | } | 
					
						
							|  |  |  | 
 | 
					
						
							|  |  |  | void | 
					
						
							| 
									
										
										
										
											2018-12-27 14:26:11 +01:00
										 |  |  | tree_format(const struct f_tree *t, buffer *buf) | 
					
						
							| 
									
										
										
										
											2013-10-05 20:12:28 +02:00
										 |  |  | { | 
					
						
							|  |  |  |   buffer_puts(buf, "["); | 
					
						
							| 
									
										
										
										
											2016-11-08 17:46:29 +01:00
										 |  |  | 
 | 
					
						
							| 
									
										
										
										
											2016-08-16 09:24:12 +02:00
										 |  |  |   tree_node_format(t, buf); | 
					
						
							| 
									
										
										
										
											2013-10-05 20:12:28 +02:00
										 |  |  | 
 | 
					
						
							| 
									
										
										
										
											2016-10-01 12:50:29 +02:00
										 |  |  |   if (buf->pos == buf->end) | 
					
						
							|  |  |  |     return; | 
					
						
							|  |  |  | 
 | 
					
						
							| 
									
										
										
										
											2016-08-16 09:24:12 +02:00
										 |  |  |   /* Undo last separator */ | 
					
						
							| 
									
										
										
										
											2013-10-05 20:12:28 +02:00
										 |  |  |   if (buf->pos[-1] != '[') | 
					
						
							|  |  |  |     buf->pos -= 2; | 
					
						
							| 
									
										
										
										
											2016-11-08 17:46:29 +01:00
										 |  |  | 
 | 
					
						
							| 
									
										
										
										
											2013-10-05 20:12:28 +02:00
										 |  |  |   buffer_puts(buf, "]"); | 
					
						
							|  |  |  | } | 
					
						
							| 
									
										
										
										
											2021-02-07 19:21:42 +01:00
										 |  |  | 
 | 
					
						
							|  |  |  | void | 
					
						
							|  |  |  | tree_walk(const struct f_tree *t, void (*hook)(const struct f_tree *, void *), void *data) | 
					
						
							|  |  |  | { | 
					
						
							|  |  |  |   if (!t) | 
					
						
							|  |  |  |     return; | 
					
						
							|  |  |  | 
 | 
					
						
							|  |  |  |   tree_walk(t->left, hook, data); | 
					
						
							|  |  |  |   hook(t, data); | 
					
						
							|  |  |  |   tree_walk(t->right, hook, data); | 
					
						
							|  |  |  | } |