summaryrefslogtreecommitdiffstats
path: root/src
diff options
context:
space:
mode:
Diffstat (limited to 'src')
-rw-r--r--src/libutil/fuzzy.c50
-rw-r--r--src/libutil/fuzzy.h6
2 files changed, 56 insertions, 0 deletions
diff --git a/src/libutil/fuzzy.c b/src/libutil/fuzzy.c
index 29a6a898c..a01377f24 100644
--- a/src/libutil/fuzzy.c
+++ b/src/libutil/fuzzy.c
@@ -30,6 +30,7 @@
#include "message.h"
#include "url.h"
#include "main.h"
+#include "xxhash.h"
#define ROLL_WINDOW_SIZE 9
#define MIN_FUZZY_BLOCK_SIZE 3
@@ -501,6 +502,55 @@ fuzzy_compare_parts (struct mime_text_part *p1, struct mime_text_part *p2)
return 0;
}
+gint
+rspamd_fuzzy_len (fuzzy_hash_t *h)
+{
+ gint len;
+ void *nullpos;
+
+ nullpos = memchr (h->hash_pipe, '\0', sizeof (h->hash_pipe));
+
+ if (nullpos == NULL) {
+ len = sizeof (h->hash_pipe);
+ }
+ else {
+ len = (char *)nullpos - h->hash_pipe;
+ }
+
+ return len;
+}
+
+guint
+rspamd_fuzzy_hash (gconstpointer key)
+{
+ fuzzy_hash_t *fh = (fuzzy_hash_t *)key;
+ void *st;
+
+ st = XXH32_init (0xdeadbeef);
+ XXH32_update (st, &fh->block_size, sizeof (fh->block_size));
+ XXH32_update (st, fh->hash_pipe, rspamd_fuzzy_len (fh));
+
+ return XXH32_digest (st);
+}
+
+gboolean
+rspamd_fuzzy_equal (gconstpointer v1, gconstpointer v2)
+{
+ fuzzy_hash_t *fh1= (fuzzy_hash_t *)v1,
+ *fh2 = (fuzzy_hash_t *)v2;
+
+ if (fh1->block_size == fh2->block_size) {
+ gint l1 = rspamd_fuzzy_len (fh1),
+ l2 = rspamd_fuzzy_len (fh2);
+
+ if (l1 == l2) {
+ return (memcmp (fh1->hash_pipe, fh2->hash_pipe, l1) == 0);
+ }
+ }
+
+ return FALSE;
+}
+
/*
* vi:ts=4
*/
diff --git a/src/libutil/fuzzy.h b/src/libutil/fuzzy.h
index 22a7cbfe6..73bddad79 100644
--- a/src/libutil/fuzzy.h
+++ b/src/libutil/fuzzy.h
@@ -67,5 +67,11 @@ gint fuzzy_compare_parts (struct mime_text_part *p1, struct mime_text_part *p2);
*/
guint32 lev_distance (gchar *s1, gint len1, gchar *s2, gint len2);
+/*
+ * Hash table utilities
+ */
+gint rspamd_fuzzy_len (fuzzy_hash_t *h);
+guint rspamd_fuzzy_hash (gconstpointer key);
+gboolean rspamd_fuzzy_equal (gconstpointer v1, gconstpointer v2);
#endif