1
2
3
4
5
6
7
8
9
10
11
12
13
14
15
16
17
18
19
20
21
22
23
24
25
26
27
28
29
30
31
32
33
34
35
36
37
38
39
40
41
42
43
44
45
46
47
48
49
50
51
52
53
54
55
56
57
58
59
60
61
62
63
64
65
66
67
68
69
70
71
72
73
74
75
76
77
78
79
80
81
82
83
84
85
86
87
88
89
90
91
92
93
94
95
96
97
98
99
100
101
102
103
104
105
106
107
108
109
110
111
112
113
114
115
116
117
118
119
120
121
122
123
124
125
126
127
128
129
130
131
132
133
134
135
136
137
138
139
140
141
142
143
144
145
146
147
148
149
150
151
152
153
154
155
156
157
158
159
160
161
162
163
164
165
166
167
168
169
170
171
172
173
174
175
176
177
178
179
180
181
182
183
184
185
186
187
188
189
190
191
192
193
194
195
196
197
198
199
200
201
202
203
204
205
206
207
208
209
|
/*
* Copyright 2023 Vsevolod Stakhov
*
* Licensed under the Apache License, Version 2.0 (the "License");
* you may not use this file except in compliance with the License.
* You may obtain a copy of the License at
*
* http://www.apache.org/licenses/LICENSE-2.0
*
* Unless required by applicable law or agreed to in writing, software
* distributed under the License is distributed on an "AS IS" BASIS,
* WITHOUT WARRANTIES OR CONDITIONS OF ANY KIND, either express or implied.
* See the License for the specific language governing permissions and
* limitations under the License.
*/
/**
* Symcache runtime is produced for each task and it consists of symbols
* being executed, being dynamically disabled/enabled and it also captures
* the current order of the symbols (produced by resort periodic)
*/
#ifndef RSPAMD_SYMCACHE_RUNTIME_HXX
#define RSPAMD_SYMCACHE_RUNTIME_HXX
#pragma once
#include "symcache_internal.hxx"
struct rspamd_scan_result;
namespace rspamd::symcache {
/**
* These items are saved within task structure and are used to track
* symbols execution.
* Each symcache item occupies a single dynamic item, that currently has 8 bytes
* length
*/
struct cache_dynamic_item {
std::uint16_t start_msec; /* Relative to task time */
bool started;
bool finished;
std::uint32_t async_events;
};
static_assert(sizeof(cache_dynamic_item) == sizeof(std::uint64_t));
static_assert(std::is_trivial_v<cache_dynamic_item>);
class symcache_runtime {
unsigned items_inflight;
bool profile;
bool has_slow;
double profile_start;
double lim;
struct cache_dynamic_item *cur_item;
order_generation_ptr order;
/* Dynamically expanded as needed */
mutable struct cache_dynamic_item dynamic_items[];
/* We allocate this structure merely in memory pool, so destructor is absent */
~symcache_runtime() = delete;
auto process_symbol(struct rspamd_task *task, symcache &cache, cache_item *item,
cache_dynamic_item *dyn_item) -> bool;
/* Specific stages of the processing */
auto process_pre_postfilters(struct rspamd_task *task, symcache &cache, int start_events, unsigned int stage) -> bool;
auto process_filters(struct rspamd_task *task, symcache &cache, int start_events) -> bool;
auto check_metric_limit(struct rspamd_task *task) -> bool;
auto check_item_deps(struct rspamd_task *task, symcache &cache, cache_item *item,
cache_dynamic_item *dyn_item, bool check_only) -> bool;
public:
/* Dropper for a shared ownership */
auto savepoint_dtor() -> void
{
/* Drop shared ownership */
order.reset();
}
/**
* Creates a cache runtime using task mempool
* @param task
* @param cache
* @return
*/
static auto create(struct rspamd_task *task, symcache &cache) -> symcache_runtime *;
/**
* Process task settings
* @param task
* @return
*/
auto process_settings(struct rspamd_task *task, const symcache &cache) -> bool;
/**
* Disable all symbols but not touching ones that are in the specific mask
* @param skip_mask
*/
auto disable_all_symbols(int skip_mask) -> void;
/**
* Disable a symbol (or it's parent)
* @param name
* @return
*/
auto disable_symbol(struct rspamd_task *task, const symcache &cache, std::string_view name) -> bool;
/**
* Enable a symbol (or it's parent)
* @param name
* @return
*/
auto enable_symbol(struct rspamd_task *task, const symcache &cache, std::string_view name) -> bool;
/**
* Checks if an item has been checked/disabled
* @param cache
* @param name
* @return
*/
auto is_symbol_checked(const symcache &cache, std::string_view name) -> bool;
/**
* Checks if a symbol is enabled for execution, checking all pending conditions
* @param task
* @param cache
* @param name
* @return
*/
auto is_symbol_enabled(struct rspamd_task *task, const symcache &cache, std::string_view name) -> bool;
/**
* Get the current processed item
* @return
*/
auto get_cur_item() const -> auto
{
return cur_item;
}
/**
* Set the current processed item
* @param item
* @return
*/
auto set_cur_item(cache_dynamic_item *item) -> auto
{
std::swap(item, cur_item);
return item;
}
/**
* Set profile mode for the runtime
* @param enable
* @return
*/
auto set_profile_mode(bool enable) -> auto
{
std::swap(profile, enable);
return enable;
}
/**
* Returns the dynamic item by static item id
* @param id
* @return
*/
auto get_dynamic_item(int id) const -> cache_dynamic_item *;
/**
* Returns static cache item by dynamic cache item
* @return
*/
auto get_item_by_dynamic_item(cache_dynamic_item *) const -> cache_item *;
/**
* Process symbols in the cache
* @param task
* @param cache
* @param stage
* @return
*/
auto process_symbols(struct rspamd_task *task, symcache &cache, unsigned int stage) -> bool;
/**
* Finalize execution of some item in the cache
* @param task
* @param item
*/
auto finalize_item(struct rspamd_task *task, cache_dynamic_item *item) -> void;
/**
* Process unblocked reverse dependencies of the specific item
* @param task
* @param item
*/
auto process_item_rdeps(struct rspamd_task *task, cache_item *item) -> void;
/* XXX: a helper to allow hiding internal implementation of the slow timer structure */
auto unset_slow() -> void
{
has_slow = false;
}
};
}// namespace rspamd::symcache
#endif//RSPAMD_SYMCACHE_RUNTIME_HXX
|