summaryrefslogtreecommitdiffstats
path: root/src/libutil/radix.c
diff options
context:
space:
mode:
authorVsevolod Stakhov <vsevolod@highsecure.ru>2014-09-17 16:22:48 +0100
committerVsevolod Stakhov <vsevolod@highsecure.ru>2014-09-17 16:22:48 +0100
commit814bcc144f6c0d2bb91f5d2bc143b0ead5aab92f (patch)
tree691486442c0765b54ad38d031c5cac231c513f84 /src/libutil/radix.c
parent31179739be8675f1414bfb66784170f2ba3d2b19 (diff)
downloadrspamd-814bcc144f6c0d2bb91f5d2bc143b0ead5aab92f.tar.gz
rspamd-814bcc144f6c0d2bb91f5d2bc143b0ead5aab92f.zip
Check mask first.
Diffstat (limited to 'src/libutil/radix.c')
-rw-r--r--src/libutil/radix.c65
1 files changed, 40 insertions, 25 deletions
diff --git a/src/libutil/radix.c b/src/libutil/radix.c
index 1445dd0fe..f39942fc8 100644
--- a/src/libutil/radix.c
+++ b/src/libutil/radix.c
@@ -560,6 +560,37 @@ radix_move_up_compressed_leaf (struct radix_compressed_node *leaf,
}
static uintptr_t
+radix_replace_node (struct radix_compressed_node *node,
+ guint8 *key, gsize keylen,
+ uintptr_t value)
+{
+ uintptr_t oldval;
+
+ if (node->skipped) {
+ /*
+ * For leaf nodes we have to deal with the keys as well, since
+ * we might find that keys are different for the same leaf node
+ */
+ g_slice_free1 (node->d.s.keylen, node->d.s.key);
+ node->d.s.keylen = keylen;
+ node->d.s.key = g_slice_alloc (node->d.s.keylen);
+ memcpy (node->d.s.key, key, node->d.s.keylen);
+ oldval = node->value;
+ node->value = value;
+ msg_debug ("replace value for leaf node with: %p, old value: %p",
+ value, oldval);
+ }
+ else {
+ oldval = node->value;
+ node->value = value;
+ msg_debug ("replace value for node with: %p, old value: %p",
+ value, oldval);
+ }
+
+ return oldval;
+}
+
+static uintptr_t
radix_uncompress_node (struct radix_compressed_node *node,
guint8 *key, gsize keylen,
uintptr_t value,
@@ -582,15 +613,16 @@ radix_uncompress_node (struct radix_compressed_node *node,
guint8 kb = *k & bit;
guint8 nb = *nkey & bit;
- if (kb != nb) {
- msg_debug ("found available path at level %ud", cur_level);
- break;
- }
- else if (cur_level >= node->d.s.level) {
+ if (cur_level >= node->d.s.level) {
msg_debug ("found available masked path at level %ud", cur_level);
masked = TRUE;
break;
}
+ if (kb != nb) {
+ msg_debug ("found available path at level %ud", cur_level);
+ break;
+ }
+
cur_level ++;
levels_uncompress ++;
bit >>= 1;
@@ -604,10 +636,7 @@ radix_uncompress_node (struct radix_compressed_node *node,
if (kremain == 0) {
/* Nodes are equal */
- uintptr_t oldval = node->value;
- node->value = value;
- msg_debug ("replace leaf node with: %p, old value: %p", value, oldval);
- return oldval;
+ return radix_replace_node (node, key, keylen, value);
}
else {
/*
@@ -734,18 +763,7 @@ radix_insert_compressed (radix_compressed_t * tree,
* is equal to the target level
*/
if (next->d.s.level == target_level) {
- /*
- * For leaf nodes we have to deal with the keys as well, since
- * we might find that keys are different for the same leaf node
- */
- g_slice_free1 (next->d.s.keylen, next->d.s.key);
- next->d.s.keylen = keylen;
- next->d.s.key = g_slice_alloc (next->d.s.keylen);
- memcpy (next->d.s.key, key, next->d.s.keylen);
- oldval = next->value;
- next->value = value;
- msg_debug ("replace value for node with: %p, old value: %p",
- value, oldval);
+ oldval = radix_replace_node (next, key, keylen, value);
}
else if (next->d.s.level > target_level) {
/*
@@ -788,10 +806,7 @@ radix_insert_compressed (radix_compressed_t * tree,
}
}
else {
- oldval = next->value;
- next->value = value;
- msg_debug ("replace value for node with: %p, old value: %p",
- value, oldval);
+ oldval = radix_replace_node (next, key, keylen, value);
}
return oldval;
}