aboutsummaryrefslogtreecommitdiffstats
path: root/src/libutil/regexp.c
diff options
context:
space:
mode:
Diffstat (limited to 'src/libutil/regexp.c')
-rw-r--r--src/libutil/regexp.c75
1 files changed, 74 insertions, 1 deletions
diff --git a/src/libutil/regexp.c b/src/libutil/regexp.c
index f9eef0385..57515703e 100644
--- a/src/libutil/regexp.c
+++ b/src/libutil/regexp.c
@@ -255,11 +255,84 @@ fin:
gboolean
rspamd_regexp_search (rspamd_regexp_t *re, const gchar *text, gsize len,
- gboolean raw)
+ const gchar **start, const gchar **end, gboolean raw)
{
+ pcre *r;
+ pcre_extra *ext;
+#ifdef HAVE_PCRE_JIT
+ pcre_jit_stack *st;
+#endif
+ const gchar *mt;
+ gsize remain;
+ gint rc, match_flags = 0, ovec[10];
+
g_assert (re != NULL);
g_assert (text != NULL);
+ if (end != NULL && *end != NULL) {
+ /* Incremental search */
+ mt = (*end) + 1;
+ remain = len - (mt - text);
+ }
+ else {
+ mt = text;
+ remain = len;
+ }
+
+ match_flags = PCRE_NEWLINE_ANYCRLF;
+ if ((re->flags & RSPAMD_REGEXP_FLAG_RAW) || raw) {
+ r = re->raw_re;
+ ext = re->raw_extra;
+#ifdef HAVE_PCRE_JIT
+ st = re->raw_jstack;
+#endif
+ }
+ else {
+ match_flags |= PCRE_NO_UTF8_CHECK;
+ r = re->re;
+ ext = re->extra;
+#ifdef HAVE_PCRE_JIT
+ st = re->jstack;
+#endif
+ }
+
+ g_assert (r != NULL);
+
+#ifdef HAVE_PCRE_JIT
+ if (re->mtx) {
+ rspamd_mutex_lock (re->mtx);
+ }
+#endif
+
+ if (!(re->flags & RSPAMD_REGEXP_FLAG_NOOPT)) {
+#ifdef HAVE_PCRE_JIT
+ rc = pcre_jit_exec (r, ext, mt, remain, 0, match_flags, ovec,
+ G_N_ELEMENTS (ovec), st);
+#else
+ rc = pcre_exec (r, ext, mt, remain, 0, match_flags, ovec,
+ G_N_ELEMENTS (ovec));
+#endif
+ }
+ else {
+ rc = pcre_exec (r, ext, mt, remain, 0, match_flags, ovec,
+ G_N_ELEMENTS (ovec));
+ }
+#ifdef HAVE_PCRE_JIT
+ if (re->mtx) {
+ rspamd_mutex_unlock (re->mtx);
+ }
+#endif
+ if (rc > 0) {
+ if (start) {
+ *start = mt + ovec[0];
+ }
+ if (end) {
+ *end = mt + ovec[1];
+ }
+
+ return TRUE;
+ }
+
return FALSE;
}