diff options
-rw-r--r-- | src/libserver/re_cache.c | 73 |
1 files changed, 59 insertions, 14 deletions
diff --git a/src/libserver/re_cache.c b/src/libserver/re_cache.c index f82f1d782..c438b6c61 100644 --- a/src/libserver/re_cache.c +++ b/src/libserver/re_cache.c @@ -121,9 +121,19 @@ struct rspamd_re_cache { #endif }; +struct rspamd_re_selector_result { + guchar **scvec; + guint *lenvec; + guint cnt; +}; + +KHASH_INIT (selectors_results_hash, int, struct rspamd_re_selector_result, 1, + kh_int_hash_func, kh_int_hash_equal); + struct rspamd_re_runtime { guchar *checked; guchar *results; + khash_t (selectors_results_hash) *sel_cache; struct rspamd_re_cache *cache; struct rspamd_re_cache_stat stat; gboolean has_hs; @@ -482,11 +492,11 @@ rspamd_re_cache_runtime_new (struct rspamd_re_cache *cache) struct rspamd_re_runtime *rt; g_assert (cache != NULL); - rt = g_malloc0 (sizeof (*rt)); + rt = g_malloc0 (sizeof (*rt) + NBYTES (cache->nre) + cache->nre); rt->cache = cache; REF_RETAIN (cache); - rt->checked = g_malloc0 (NBYTES (cache->nre)); - rt->results = g_malloc0 (cache->nre); + rt->checked = ((guchar *)rt) + sizeof (*rt); + rt->results = rt->checked + NBYTES (cache->nre); rt->stat.regexp_total = cache->nre; #ifdef WITH_HYPERSCAN rt->has_hs = cache->hyperscan_loaded; @@ -773,7 +783,7 @@ rspamd_re_cache_finish_class (struct rspamd_re_runtime *rt, static gboolean rspamd_re_cache_process_selector (struct rspamd_task *task, - struct rspamd_re_cache *cache, + struct rspamd_re_runtime *rt, const gchar *name, guchar ***svec, guint **lenvec, @@ -786,6 +796,8 @@ rspamd_re_cache_process_selector (struct rspamd_task *task, GString *tb; struct rspamd_task **ptask; gboolean result = FALSE; + struct rspamd_re_cache *cache = rt->cache; + struct rspamd_re_selector_result *sr; L = cache->L; k = kh_get (lua_selectors_hash, cache->selectors, (gchar *)name); @@ -798,6 +810,24 @@ rspamd_re_cache_process_selector (struct rspamd_task *task, ref = kh_value (cache->selectors, k); + /* First, search for the cached result */ + if (rt->sel_cache) { + k = kh_get (selectors_results_hash, rt->sel_cache, ref); + + if (k != kh_end (rt->sel_cache)) { + sr = &kh_value (rt->sel_cache, k); + + *svec = sr->scvec; + *lenvec = sr->lenvec; + *n = sr->cnt; + + return TRUE; + } + } + else { + rt->sel_cache = kh_init (selectors_results_hash); + } + lua_pushcfunction (L, &rspamd_lua_traceback); err_idx = lua_gettop (L); @@ -848,6 +878,15 @@ rspamd_re_cache_process_selector (struct rspamd_task *task, lua_settop (L, err_idx - 1); + if (result) { + k = kh_put (selectors_results_hash, rt->sel_cache, ref, &ret); + sr = &kh_value (rt->sel_cache, k); + + sr->cnt = *n; + sr->scvec = *svec; + sr->lenvec = *lenvec; + } + return result; } @@ -1161,7 +1200,7 @@ rspamd_re_cache_exec_re (struct rspamd_task *task, } break; case RSPAMD_RE_SELECTOR: - if (rspamd_re_cache_process_selector (task, rt->cache, + if (rspamd_re_cache_process_selector (task, rt, re_class->type_data, (guchar ***)&scvec, &lenvec, &cnt)) { @@ -1172,13 +1211,7 @@ rspamd_re_cache_exec_re (struct rspamd_task *task, re_class->type_data, rspamd_regexp_get_pattern (re), ret); - /* TODO: add caching logic for this data */ - for (i = 0; i < cnt; i ++) { - g_free ((gpointer)scvec[i]); - } - - g_free (scvec); - g_free (lenvec); + /* Do not free vectors as they are managed by rt->sel_cache */ } break; case RSPAMD_RE_MAX: @@ -1271,8 +1304,20 @@ rspamd_re_cache_runtime_destroy (struct rspamd_re_runtime *rt) { g_assert (rt != NULL); - g_free (rt->checked); - g_free (rt->results); + if (rt->sel_cache) { + struct rspamd_re_selector_result sr; + + kh_foreach_value (rt->sel_cache, sr, { + for (guint i = 0; i < sr.cnt; i ++) { + g_free ((gpointer)sr.scvec[i]); + } + + g_free (sr.scvec); + g_free (sr.lenvec); + }); + kh_destroy (selectors_results_hash, rt->sel_cache); + } + REF_RELEASE (rt->cache); g_free (rt); } |