aboutsummaryrefslogtreecommitdiffstats
diff options
context:
space:
mode:
authorVsevolod Stakhov <vsevolod@highsecure.ru>2019-03-26 15:57:43 +0000
committerVsevolod Stakhov <vsevolod@highsecure.ru>2019-03-26 15:57:43 +0000
commita5af7ff8d409b63743889c5562c7b735e19313a8 (patch)
tree19f2c2b60a455db000df12869de4bd9712bab88d
parentab4950c5eac19bd5d5aeac2a1c4772e9fd893f37 (diff)
downloadrspamd-a5af7ff8d409b63743889c5562c7b735e19313a8.tar.gz
rspamd-a5af7ff8d409b63743889c5562c7b735e19313a8.zip
[Minor] Apply all matching special handlers
-rw-r--r--contrib/libucl/ucl_internal.h9
-rw-r--r--contrib/libucl/ucl_parser.c13
-rw-r--r--contrib/libucl/ucl_util.c21
3 files changed, 31 insertions, 12 deletions
diff --git a/contrib/libucl/ucl_internal.h b/contrib/libucl/ucl_internal.h
index ed4a7d767..edf647827 100644
--- a/contrib/libucl/ucl_internal.h
+++ b/contrib/libucl/ucl_internal.h
@@ -205,6 +205,13 @@ struct ucl_stack {
struct ucl_chunk *chunk;
};
+struct ucl_parser_special_handler_chain {
+ unsigned char *begin;
+ size_t len;
+ struct ucl_parser_special_handler *special_handler;
+ struct ucl_parser_special_handler_chain *next;
+};
+
struct ucl_chunk {
const unsigned char *begin;
const unsigned char *end;
@@ -216,7 +223,7 @@ struct ucl_chunk {
unsigned priority;
enum ucl_duplicate_strategy strategy;
enum ucl_parse_type parse_type;
- struct ucl_parser_special_handler *special_handler;
+ struct ucl_parser_special_handler_chain *special_handlers;
struct ucl_chunk *next;
};
diff --git a/contrib/libucl/ucl_parser.c b/contrib/libucl/ucl_parser.c
index 59bfd7727..7752f6661 100644
--- a/contrib/libucl/ucl_parser.c
+++ b/contrib/libucl/ucl_parser.c
@@ -2861,6 +2861,7 @@ ucl_parser_add_chunk_full (struct ucl_parser *parser, const unsigned char *data,
memset (chunk, 0, sizeof (*chunk));
+ /* Apply all matching handlers from the first to the last */
LL_FOREACH (parser->special_handlers, special_handler) {
if ((special_handler->flags & UCL_SPECIAL_HANDLER_PREPROCESS_ALL) ||
(len >= special_handler->magic_len &&
@@ -2874,11 +2875,17 @@ ucl_parser_add_chunk_full (struct ucl_parser *parser, const unsigned char *data,
return false;
}
+ struct ucl_parser_special_handler_chain *nchain;
+ nchain = UCL_ALLOC (sizeof (*nchain));
+ nchain->begin = ndata;
+ nchain->len = nlen;
+ nchain->special_handler = special_handler;
+
+ /* Free order is reversed */
+ LL_PREPEND (chunk->special_handlers, nchain);
+
data = ndata;
len = nlen;
- chunk->special_handler = special_handler;
-
- break;
}
}
diff --git a/contrib/libucl/ucl_util.c b/contrib/libucl/ucl_util.c
index 11799ab1e..6f36e5e73 100644
--- a/contrib/libucl/ucl_util.c
+++ b/contrib/libucl/ucl_util.c
@@ -520,18 +520,23 @@ void
ucl_chunk_free (struct ucl_chunk *chunk)
{
if (chunk) {
- if (chunk->special_handler) {
- if (chunk->special_handler->free_function) {
- chunk->special_handler->free_function (
- (unsigned char *) chunk->begin,
- chunk->end - chunk->begin,
- chunk->special_handler->user_data);
+ struct ucl_parser_special_handler_chain *chain, *tmp;
+
+ LL_FOREACH_SAFE (chunk->special_handlers, chain, tmp) {
+ if (chain->special_handler->free_function) {
+ chain->special_handler->free_function (
+ chain->begin,
+ chain->len,
+ chain->special_handler->user_data);
} else {
- UCL_FREE (chunk->end - chunk->begin,
- (unsigned char *) chunk->begin);
+ UCL_FREE (chain->len, chain->begin);
}
+
+ UCL_FREE (sizeof (*chain), chain);
}
+ chunk->special_handlers = NULL;
+
if (chunk->fname) {
free (chunk->fname);
}