diff options
author | Vsevolod Stakhov <vsevolod@highsecure.ru> | 2016-04-15 12:49:24 +0100 |
---|---|---|
committer | Vsevolod Stakhov <vsevolod@highsecure.ru> | 2016-04-15 12:49:24 +0100 |
commit | 023d00d0115ae931c6513c6dd30c3d12ca1f26dd (patch) | |
tree | 8d0480fa26ae4f2b9471c40ac57b09f0b8cf215e /src/libutil/multipattern.c | |
parent | 45aeb65baa5793a1177bb823bd6bdd0b959d25f8 (diff) | |
download | rspamd-023d00d0115ae931c6513c6dd30c3d12ca1f26dd.tar.gz rspamd-023d00d0115ae931c6513c6dd30c3d12ca1f26dd.zip |
[Fix] Allow multipattern scans to be nested for the case of hyperscan
Diffstat (limited to 'src/libutil/multipattern.c')
-rw-r--r-- | src/libutil/multipattern.c | 36 |
1 files changed, 30 insertions, 6 deletions
diff --git a/src/libutil/multipattern.c b/src/libutil/multipattern.c index e50574c92..092e9f889 100644 --- a/src/libutil/multipattern.c +++ b/src/libutil/multipattern.c @@ -28,16 +28,19 @@ #include "acism.h" #endif +#define MAX_SCRATCH 4 + static const char *hs_cache_dir = NULL; struct rspamd_multipattern { #ifdef WITH_HYPERSCAN hs_database_t *db; - hs_scratch_t *scratch; + hs_scratch_t *scratch[MAX_SCRATCH]; GArray *hs_pats; GArray *hs_ids; GArray *hs_flags; rspamd_cryptobox_hash_state_t hash_state; + guint scratch_used; #else ac_trie_t *t; GArray *pats; @@ -517,6 +520,7 @@ rspamd_multipattern_compile (struct rspamd_multipattern *mp, GError **err) g_assert (!mp->compiled); #ifdef WITH_HYPERSCAN + guint i; hs_platform_info_t plt; hs_compile_error_t *hs_errors; guchar hash[rspamd_cryptobox_HASHBYTES]; @@ -547,7 +551,10 @@ rspamd_multipattern_compile (struct rspamd_multipattern *mp, GError **err) } rspamd_multipattern_try_save_hs (mp, hash); - g_assert (hs_alloc_scratch (mp->db, &mp->scratch) == HS_SUCCESS); + + for (i = 0; i < MAX_SCRATCH; i ++) { + g_assert (hs_alloc_scratch (mp->db, &mp->scratch[i]) == HS_SUCCESS); + } } #else if (mp->cnt > 0) { @@ -622,9 +629,8 @@ rspamd_multipattern_lookup (struct rspamd_multipattern *mp, gint ret = 0; g_assert (mp != NULL); - g_assert (mp->compiled); - if (mp->cnt == 0) { + if (mp->cnt == 0 || !mp->compiled) { return 0; } @@ -637,9 +643,24 @@ rspamd_multipattern_lookup (struct rspamd_multipattern *mp, cbd.ret = 0; #ifdef WITH_HYPERSCAN - ret = hs_scan (mp->db, in, len, 0, mp->scratch, + hs_scratch_t *scr = NULL; + guint i; + + for (i = 0; i < MAX_SCRATCH; i ++) { + if (!(mp->scratch_used & (1 << i))) { + mp->scratch_used |= (1 << i); + scr = mp->scratch[i]; + break; + } + } + + g_assert (scr != NULL); + + ret = hs_scan (mp->db, in, len, 0, scr, rspamd_multipattern_hs_cb, &cbd); + mp->scratch_used &= ~(1 << i); + if (ret == HS_SUCCESS) { ret = 0; } @@ -671,7 +692,10 @@ rspamd_multipattern_destroy (struct rspamd_multipattern *mp) gchar *p; if (mp->compiled && mp->cnt > 0) { - hs_free_scratch (mp->scratch); + for (i = 0; i < MAX_SCRATCH; i ++) { + hs_free_scratch (mp->scratch[i]); + } + hs_free_database (mp->db); } |