aboutsummaryrefslogtreecommitdiffstats
path: root/src/libutil/expression.c
diff options
context:
space:
mode:
Diffstat (limited to 'src/libutil/expression.c')
-rw-r--r--src/libutil/expression.c38
1 files changed, 38 insertions, 0 deletions
diff --git a/src/libutil/expression.c b/src/libutil/expression.c
index b2655b518..83ee76a07 100644
--- a/src/libutil/expression.c
+++ b/src/libutil/expression.c
@@ -26,11 +26,16 @@
#include "expression.h"
#include "printf.h"
#include "regexp.h"
+#include "util.h"
#include "utlist.h"
+#include "ottery.h"
#define RSPAMD_EXPR_FLAG_NEGATE (1 << 0)
#define RSPAMD_EXPR_FLAG_PROCESSED (1 << 1)
+#define MIN_RESORT_EVALS 50
+#define MAX_RESORT_EVALS 150
+
enum rspamd_expression_op {
OP_INVALID = 0,
OP_PLUS, /* || or + */
@@ -71,6 +76,8 @@ struct rspamd_expression {
GArray *expressions;
GPtrArray *expression_stack;
GNode *ast;
+ guint next_resort;
+ guint evals;
};
static GQuark
@@ -477,6 +484,8 @@ rspamd_parse_expression (const gchar *line, gsize len,
e->ast = NULL;
e->expression_stack = g_ptr_array_sized_new (32);
e->subr = subr;
+ e->evals = 0;
+ e->next_resort = ottery_rand_range (MAX_RESORT_EVALS) + MIN_RESORT_EVALS;
/* Shunting-yard algorithm */
while (p < end) {
@@ -812,13 +821,34 @@ rspamd_ast_process_node (struct rspamd_expression *expr, gint flags, GNode *node
struct rspamd_expression_elt *elt, *celt, *parelt;
GNode *cld;
gint acc = G_MININT, lim = G_MININT, val;
+ gdouble t1, t2;
+ gboolean calc_ticks = FALSE;
elt = node->data;
switch (elt->type) {
case ELT_ATOM:
if (!(elt->flags & RSPAMD_EXPR_FLAG_PROCESSED)) {
+
+ /*
+ * Sometimes get ticks for this expression. 'Sometimes' here means
+ * that we get lowest 5 bits of the counter `evals` and 5 bits
+ * of some shifted address to provide some sort of jittering for
+ * ticks evaluation
+ */
+ if ((expr->evals & 0x1F) == (GPOINTER_TO_UINT (node) >> 4 & 0x1F)) {
+ calc_ticks = TRUE;
+ t1 = rspamd_get_ticks ();
+ }
+
elt->value = expr->subr->process (data, elt->p.atom);
+
+ if (calc_ticks) {
+ t2 = rspamd_get_ticks ();
+ elt->p.atom->avg_ticks += ((t2 - t1) - elt->p.atom->avg_ticks) /
+ (expr->evals);
+ }
+
elt->flags |= RSPAMD_EXPR_FLAG_PROCESSED;
}
@@ -893,6 +923,14 @@ rspamd_process_expression (struct rspamd_expression *expr, gint flags,
g_node_traverse (expr->ast, G_IN_ORDER, G_TRAVERSE_ALL, -1,
rspamd_ast_cleanup_traverse, NULL);
+ expr->evals ++;
+
+ /* Check if we need to resort */
+ if (expr->evals == expr->next_resort) {
+ expr->next_resort = ottery_rand_range (MAX_RESORT_EVALS) +
+ MIN_RESORT_EVALS;
+ }
+
return ret;
}