diff options
author | Vsevolod Stakhov <vsevolod@highsecure.ru> | 2015-03-05 09:50:25 +0000 |
---|---|---|
committer | Vsevolod Stakhov <vsevolod@highsecure.ru> | 2015-03-05 09:50:25 +0000 |
commit | 6aeab25b1d1e1d49b94ee152ded2c011c6f7d5e1 (patch) | |
tree | 71c36fdb81260537aeb493e364374b691110a035 /src/libutil/regexp.c | |
parent | 58ff3a43b71928263415a8a874943de9de158018 (diff) | |
download | rspamd-6aeab25b1d1e1d49b94ee152ded2c011c6f7d5e1.tar.gz rspamd-6aeab25b1d1e1d49b94ee152ded2c011c6f7d5e1.zip |
Implement cache functions.
Diffstat (limited to 'src/libutil/regexp.c')
-rw-r--r-- | src/libutil/regexp.c | 98 |
1 files changed, 93 insertions, 5 deletions
diff --git a/src/libutil/regexp.c b/src/libutil/regexp.c index e4fbdef12..2dfe3e918 100644 --- a/src/libutil/regexp.c +++ b/src/libutil/regexp.c @@ -29,13 +29,15 @@ #include "ref.h" #include <pcre.h> +typedef guchar regexp_id_t[BLAKE2B_OUTBYTES]; + struct rspamd_regexp_s { - gdouble exec_time; /**< average execution time */ + gdouble exec_time; pcre *re; pcre_extra *extra; pcre *raw_re; pcre_extra *raw_extra; - guchar id[BLAKE2B_OUTBYTES / 2]; + regexp_id_t id; ref_entry_t ref; }; @@ -45,6 +47,18 @@ struct rspamd_regexp_cache { static struct rspamd_regexp_cache *global_re_cache = NULL; +static void +rspamd_regexp_generate_id (const gchar *pattern, const gchar *flags, + regexp_id_t out) +{ + blake2b_state st; + + blake2b_init (&st, sizeof (regexp_id_t)); + blake2b_update (&st, flags, strlen (flags)); + blake2b_update (&st, pattern, strlen (pattern)); + blake2b_final (&st, out, sizeof (regexp_id_t)); +} + rspamd_regexp_t* rspamd_regexp_new (const gchar *pattern, const gchar *flags, GError **err) @@ -70,10 +84,35 @@ rspamd_regexp_unref (rspamd_regexp_t *re) REF_RELEASE (re); } +static gboolean +rspamd_regexp_equal (gconstpointer a, gconstpointer b) +{ + const guchar *ia = a, *ib = b; + + return (memcmp (ia, ib, sizeof (regexp_id_t)) == 0); +} + +static guint32 +rspamd_regexp_hash (gconstpointer a) +{ + const guchar *ia = a; + guint32 res; + + memcpy (&res, ia, sizeof (res)); + + return res; +} + struct rspamd_regexp_cache* rspamd_regexp_cache_new (void) { - return NULL; + struct rspamd_regexp_cache *ncache; + + ncache = g_slice_alloc (sizeof (*ncache)); + ncache->tbl = g_hash_table_new_full (rspamd_regexp_hash, rspamd_regexp_equal, + NULL, (GDestroyNotify)rspamd_regexp_unref); + + return ncache; } @@ -82,7 +121,19 @@ rspamd_regexp_cache_query (struct rspamd_regexp_cache* cache, const gchar *pattern, const gchar *flags) { - return NULL; + rspamd_regexp_t *res = NULL; + regexp_id_t id; + + if (cache == NULL) { + cache = global_re_cache; + } + + g_assert (cache != NULL); + rspamd_regexp_generate_id (pattern, flags, id); + + res = g_hash_table_lookup (cache->tbl, id); + + return res; } @@ -91,11 +142,48 @@ rspamd_regexp_cache_create (struct rspamd_regexp_cache *cache, const gchar *pattern, const gchar *flags, GError **err) { - return NULL; + rspamd_regexp_t *res; + + if (cache == NULL) { + cache = global_re_cache; + } + + g_assert (cache != NULL); + res = rspamd_regexp_cache_query (cache, pattern, flags); + + if (res != NULL) { + return res; + } + + res = rspamd_regexp_new (pattern, flags, err); + + if (res) { + g_hash_table_insert (cache->tbl, res->id, res); + } + + return res; } void rspamd_regexp_cache_destroy (struct rspamd_regexp_cache *cache) { + if (cache != NULL) { + g_hash_table_destroy (cache->tbl); + } +} +void +rspamd_regexp_library_init (void) +{ + if (global_re_cache == NULL) { + global_re_cache = rspamd_regexp_cache_new (); + } +} + +void +rspamd_regexp_library_finalize (void) +{ + if (global_re_cache != NULL) { + rspamd_regexp_cache_destroy (global_re_cache); + } } |