The FreeRADIUS server $Id: 15bac2a4c627c01d1aa2047687b3418955ac7f00 $
Loading...
Searching...
No Matches
rlm_cache_htrie.c
Go to the documentation of this file.
1/*
2 * This program is is free software; you can redistribute it and/or modify
3 * it under the terms of the GNU General Public License as published by
4 * the Free Software Foundation; either version 2 of the License, or (at
5 * your option) any later version.
6 *
7 * This program is distributed in the hope that it will be useful,
8 * but WITHOUT ANY WARRANTY; without even the implied warranty of
9 * MERCHANTABILITY or FITNESS FOR A PARTICULAR PURPOSE. See the
10 * GNU General Public License for more details.
11 *
12 * You should have received a copy of the GNU General Public License
13 * along with this program; if not, write to the Free Software
14 * Foundation, Inc., 51 Franklin St, Fifth Floor, Boston, MA 02110-1301, USA
15 */
16
17/**
18 * $Id: 70bd0f222be0b56409e0c2fc07862912ab2369d2 $
19 * @file rlm_cache_htrie.c
20 * @brief Simple htrie based cache.
21 *
22 * @copyright 2024 Arran Cudbard-Bell <a.cudbardb@freeradius.org>
23 * @copyright 2014 The FreeRADIUS server project
24 */
25#include <freeradius-devel/server/base.h>
26#include <freeradius-devel/util/heap.h>
27#include <freeradius-devel/util/debug.h>
28#include <freeradius-devel/util/value.h>
29#include <freeradius-devel/util/htrie.h>
30#include "../../rlm_cache.h"
31#include "lib/server/cf_parse.h"
32#include "lib/server/tmpl.h"
33#include "lib/util/types.h"
34
35static int cf_htrie_type_parse(TALLOC_CTX *ctx, void *out, void *parent, CONF_ITEM *ci, conf_parser_t const *rule);
36static int cf_htrie_key_parse(TALLOC_CTX *ctx, void *out, tmpl_rules_t const *t_rules, CONF_ITEM *ci,
37 call_env_ctx_t const *cec, UNUSED call_env_parser_t const *rule);
38
39typedef struct {
40 fr_htrie_t *cache; //!< Tree for looking up cache keys.
41 fr_heap_t *heap; //!< For managing entry expiry.
42
43 pthread_mutex_t mutex; //!< Protect the tree from multiple readers/writers.
45
46typedef struct {
47 fr_type_t ktype; //!< When htrie is "auto", we use this type to decide
48 ///< what type of tree to use.
49
50 fr_htrie_type_t htype; //!< The htrie type we'll be using
51 bool htrie_auto; //!< Whether the user wanted to automatically configure
52 ///< the htrie.
53
54 rlm_cache_htrie_mutable_t *mutable; //!< Mutable instance data.
56
57typedef struct {
58 rlm_cache_entry_t fields; //!< Entry data.
59 fr_heap_index_t heap_id; //!< Offset used for expiry heap.
61
63 { FR_CONF_OFFSET("type", rlm_cache_htrie_t, htype), .dflt = "auto",
67};
68
69/** Custom htrie type parsing function
70 *
71 * Sets a bool, so we known if the original type was "auto", so we can constantly re-evaluate
72 * the htrie type based on the key type.
73 */
74int cf_htrie_type_parse(TALLOC_CTX *ctx, void *out, void *parent, CONF_ITEM *ci, conf_parser_t const *rule)
75{
76 rlm_cache_htrie_t *inst = talloc_get_type_abort(parent, rlm_cache_htrie_t);
77 int ret;
78
79 ret = cf_table_parse_int(ctx, out, parent, ci, rule);
80 if (unlikely(ret < 0)) return ret;
81
82 /*
83 * Record this now, so when we overwrite this
84 * value later, we know to keep checking the
85 * htrie type value for consistency.
86 */
87 if (*(int *)out == FR_HTRIE_AUTO) inst->htrie_auto = true;
88
89 return 0;
90}
91
92/** Custom key parsing function for checking compatibility of key types
93 *
94 * This function does two things:
95 * - It selects a htrie type based on the key type.
96 * - It checks that all keys are compatible with each other.
97 */
98static int cf_htrie_key_parse(TALLOC_CTX *ctx, void *out, tmpl_rules_t const *t_rules, CONF_ITEM *ci,
99 call_env_ctx_t const *cec, UNUSED call_env_parser_t const *rule)
100{
102 rlm_cache_htrie_t *inst = talloc_get_type_abort(parent->driver_submodule->data, rlm_cache_htrie_t);
103 tmpl_t *key_tmpl;
104 fr_type_t our_ktype, old_ktype;
105 bool set_type = false;
106
107 /*
108 * Call the standard pair parsing function
109 */
110 if (unlikely(call_env_parse_pair(ctx, &key_tmpl, t_rules, ci, cec, rule) < 0)) return -1;
111 our_ktype = tmpl_expanded_type(key_tmpl);
112
113 /*
114 * We need the user to tell us what the key type is for ambiguous expansions
115 */
116 if (fr_type_is_void(our_ktype)) {
117 cf_log_err(ci, "Key type is unspecified. Add a cast to set a specific type");
118 return -1;
119 }
120
121 /*
122 * If we don't have a key type already, then just set it to the first key type we see
123 */
124 if (fr_type_is_void(inst->ktype) || fr_type_is_null(inst->ktype)) {
125 inst->ktype = our_ktype;
126 set_type = true;
127 /*
128 * Check if we can cast this key type, to the key type we've already seen
129 */
130 } else if (!fr_type_cast(our_ktype, inst->ktype)) {
131 cf_log_err(ci, "Incompatible key types '%s' and '%s', cast to a more broadly compatible "
132 "type such as 'string'", fr_type_to_str(inst->ktype), fr_type_to_str(our_ktype));
133 return -1;
134 }
135
136 /*
137 * See if we should promote inst->ktype
138 */
139 old_ktype = inst->ktype;
140 inst->ktype = fr_type_promote(inst->ktype, our_ktype);
142
143 /*
144 * If we're not automatically determining the htrie type,
145 * or the ktype hasn't changed, then don't bother figuring
146 * out the htrie type.
147 */
148 if ((!inst->htrie_auto || (old_ktype == inst->ktype)) && !set_type) goto finish;
149
150 /*
151 * We need to figure out the htrie type based on the key type
152 */
153 inst->htype = fr_htrie_hint(inst->ktype);
154 if (inst->htype == FR_HTRIE_INVALID) {
155 cf_log_err(ci, "Invalid data type '%s' for htrie key. "
156 "Cast to another type, or manually specify 'type", fr_type_to_str(inst->ktype));
157 return -1;
158 }
159
160 cf_log_info(ci, "Automatically setting htrie type to '%s' based on key type '%s'",
162
163finish:
164 *(void **)out = key_tmpl;
165 return 0;
166}
167
168/** Compare two entries by expiry time
169 *
170 * There may be multiple entries with the same expiry time.
171 */
172static int8_t cache_heap_cmp(void const *one, void const *two)
173{
174 rlm_cache_entry_t const *a = one, *b = two;
175
176 return fr_unix_time_cmp(a->expires, b->expires);
177}
178
179/** Custom allocation function for the driver
180 *
181 * Allows allocation of cache entry structures with additional fields.
182 *
183 * @copydetails cache_entry_alloc_t
184 */
186 request_t *request)
187{
189
190 c = talloc_zero(NULL, rlm_cache_htrie_entry_t);
191 if (!c) {
192 RERROR("Failed allocating cache entry");
193 return NULL;
194 }
195
196 return (rlm_cache_entry_t *)c;
197}
198
199/** Locate a cache entry
200 *
201 * @note handle not used except for sanity checks.
202 *
203 * @copydetails cache_entry_find_t
204 */
206 UNUSED rlm_cache_config_t const *config, void *instance,
207 request_t *request, UNUSED void *handle, fr_value_box_t const *key)
208{
209 rlm_cache_htrie_t *driver = talloc_get_type_abort(instance, rlm_cache_htrie_t);
210 rlm_cache_htrie_mutable_t *mutable = driver->mutable;
211 rlm_cache_entry_t find = {};
212
214
215 fr_assert(mutable->cache);
216
217 /*
218 * Clear out old entries
219 */
220 c = fr_heap_peek(mutable->heap);
221 if (c && (fr_unix_time_lt(c->expires, fr_time_to_unix_time(request->packet->timestamp)))) {
222 fr_heap_extract(&mutable->heap, c);
223 fr_htrie_delete(mutable->cache, c);
224 talloc_free(c);
225 }
226
227 fr_value_box_copy_shallow(NULL, &find.key, key);
228
229 /*
230 * Is there an entry for this key?
231 */
232 c = fr_htrie_find(mutable->cache, &find);
233 if (!c) {
234 *out = NULL;
235 return CACHE_MISS;
236 }
237 *out = c;
238
239 return CACHE_OK;
240}
241
242/** Free an entry and remove it from the data store
243 *
244 * @note handle not used except for sanity checks.
245 *
246 * @copydetails cache_entry_expire_t
247 */
249 request_t *request, UNUSED void *handle,
250 fr_value_box_t const *key)
251{
252 rlm_cache_htrie_t *driver = talloc_get_type_abort(instance, rlm_cache_htrie_t);
253 rlm_cache_htrie_mutable_t *mutable = driver->mutable;
254 rlm_cache_entry_t find = {};
256
257 if (!request) return CACHE_ERROR;
258
259 fr_value_box_copy_shallow(NULL, &find.key, key);
260
261 c = fr_htrie_find(mutable->cache, &find);
262 if (!c) return CACHE_MISS;
263
264 fr_heap_extract(&mutable->heap, c);
265 fr_htrie_delete(mutable->cache, c);
266 talloc_free(c);
267
268 return CACHE_OK;
269}
270
271/** Insert a new entry into the data store
272 *
273 * @note handle not used except for sanity checks.
274 *
275 * @copydetails cache_entry_insert_t
276 */
278 request_t *request, void *handle,
279 rlm_cache_entry_t const *c)
280{
281 cache_status_t status;
282
283 rlm_cache_htrie_t *driver = talloc_get_type_abort(instance, rlm_cache_htrie_t);
284 rlm_cache_htrie_mutable_t *mutable = driver->mutable;
285
286 fr_assert(handle == request);
287
288 if (!request) return CACHE_ERROR;
289
290 /*
291 * Allow overwriting
292 */
293 if (!fr_htrie_insert(mutable->cache, c)) {
294 status = cache_entry_expire(config, instance, request, handle, &c->key);
295 if ((status != CACHE_OK) && !fr_cond_assert(0)) return CACHE_ERROR;
296
297 if (!fr_htrie_insert(mutable->cache, c)) {
298 RERROR("Failed adding entry");
299
300 return CACHE_ERROR;
301 }
302 }
303
304 if (fr_heap_insert(&mutable->heap, UNCONST(rlm_cache_entry_t *, c)) < 0) {
305 fr_htrie_delete(mutable->cache, c);
306 RERROR("Failed adding entry to expiry heap");
307
308 return CACHE_ERROR;
309 }
310
311 return CACHE_OK;
312}
313
314/** Update the TTL of an entry
315 *
316 * @note handle not used except for sanity checks.
317 *
318 * @copydetails cache_entry_set_ttl_t
319 */
321 request_t *request, UNUSED void *handle,
323{
324 rlm_cache_htrie_t *driver = talloc_get_type_abort(instance, rlm_cache_htrie_t);
325 rlm_cache_htrie_mutable_t *mutable = driver->mutable;
326
327#ifdef NDEBUG
328 if (!request) return CACHE_ERROR;
329#endif
330
331 if (!fr_cond_assert(fr_heap_extract(&mutable->heap, c) == 0)) {
332 RERROR("Entry not in heap");
333 return CACHE_ERROR;
334 }
335
336 if (fr_heap_insert(&mutable->heap, c) < 0) {
337 fr_htrie_delete(mutable->cache, c); /* make sure we don't leak entries... */
338 RERROR("Failed updating entry TTL. Entry was forcefully expired");
339 return CACHE_ERROR;
340 }
341 return CACHE_OK;
342}
343
344/** Return the number of entries in the cache
345 *
346 * @note handle not used except for sanity checks.
347 *
348 * @copydetails cache_entry_count_t
349 */
350static uint64_t cache_entry_count(UNUSED rlm_cache_config_t const *config, void *instance,
351 request_t *request, UNUSED void *handle)
352{
353 rlm_cache_htrie_t *driver = talloc_get_type_abort(instance, rlm_cache_htrie_t);
354
355 if (!request) return CACHE_ERROR;
356
357 return fr_htrie_num_elements(driver->mutable->cache);
358}
359
360/** Lock the htrie
361 *
362 * @note handle not used except for sanity checks.
363 *
364 * @copydetails cache_acquire_t
365 */
366static int cache_acquire(void **handle, UNUSED rlm_cache_config_t const *config, void *instance,
367 request_t *request)
368{
369 rlm_cache_htrie_t *driver = talloc_get_type_abort(instance, rlm_cache_htrie_t);
370
371 pthread_mutex_lock(&driver->mutable->mutex);
372
373 *handle = request; /* handle is unused, this is just for sanity checking */
374
375 RDEBUG3("Mutex acquired");
376
377 return 0;
378}
379
380/** Release an entry unlocking any mutexes
381 *
382 * @note handle not used except for sanity checks.
383 *
384 * @copydetails cache_release_t
385 */
386static void cache_release(UNUSED rlm_cache_config_t const *config, void *instance, request_t *request,
388{
389 rlm_cache_htrie_t *driver = talloc_get_type_abort(instance, rlm_cache_htrie_t);
390
391 pthread_mutex_unlock(&driver->mutable->mutex);
392
393 RDEBUG3("Mutex released");
394}
395
396/** Cleanup a cache_htrie instance
397 *
398 */
399static int mod_detach(module_detach_ctx_t const *mctx)
400{
401 rlm_cache_htrie_t *driver = talloc_get_type_abort(mctx->mi->data, rlm_cache_htrie_t);
402 rlm_cache_htrie_mutable_t *mutable = driver->mutable;
403
404 if (!mutable) return 0;
405
406 if (mutable->cache) {
408
409 while ((c = fr_heap_peek(mutable->heap))) {
410 fr_heap_extract(&mutable->heap, c);
411 fr_htrie_delete(mutable->cache, c);
412 talloc_free(c);
413 }
414 }
415
416 pthread_mutex_destroy(&mutable->mutex);
417 talloc_free(mutable);
418
419 return 0;
420}
421
422static int8_t _value_cmp(void const *a, void const *b) {
423 fr_value_box_t const *one = a;
424 fr_value_box_t const *two = b;
425 return fr_value_box_cmp(one, two);
426}
427
428/** Create a new cache_htrie instance
429 *
430 * @param[in] mctx Data required for instantiation.
431 * @return
432 * - 0 on success.
433 * - -1 on failure.
434 */
435static int mod_instantiate(module_inst_ctx_t const *mctx)
436{
437 rlm_cache_htrie_t *driver = talloc_get_type_abort(mctx->mi->data, rlm_cache_htrie_t);
439 int ret;
440
441 /*
442 * An instance with htype of FR_HTRIE_AUTO has not been used
443 * in any policy - so instantiation can be short circuited.
444 */
445 if (driver->htype == FR_HTRIE_AUTO) return 0;
446
447 MEM(mutable = talloc_zero(NULL, rlm_cache_htrie_mutable_t));
448
449 /*
450 * The cache.
451 */
452 mutable->cache = fr_htrie_alloc(mutable, driver->htype,
456 if (!mutable->cache) {
457 PERROR("Failed to create cache");
458 error:
459 talloc_free(mutable);
460 return -1;
461 }
462
463 /*
464 * The heap of entries to expire.
465 */
466 mutable->heap = fr_heap_talloc_alloc(mutable, cache_heap_cmp, rlm_cache_htrie_entry_t, heap_id, 0);
467 if (!mutable->heap) {
468 ERROR("Failed to create heap for the cache");
469 goto error;
470 }
471
472 if ((ret = pthread_mutex_init(&mutable->mutex, NULL)) < 0) {
473 ERROR("Failed initializing mutex: %s", fr_syserror(ret));
474 goto error;
475 }
476
477 driver->mutable = mutable;
478
479 return 0;
480}
481
484 .common = {
485 .magic = MODULE_MAGIC_INIT,
486 .name = "cache_htrie",
488 .instantiate = mod_instantiate,
489 .detach = mod_detach,
490 .inst_size = sizeof(rlm_cache_htrie_t),
491 .inst_type = "rlm_cache_htrie_t",
492 },
493 .alloc = cache_entry_alloc,
494
495 .find = cache_entry_find,
496 .insert = cache_entry_insert,
497 .expire = cache_entry_expire,
498 .set_ttl = cache_entry_set_ttl,
499 .count = cache_entry_count,
500
501 .acquire = cache_acquire,
502 .release = cache_release,
503
504 .key_parse = cf_htrie_key_parse
505};
#define UNCONST(_type, _ptr)
Remove const qualification from a pointer.
Definition build.h:167
#define unlikely(_x)
Definition build.h:383
#define UNUSED
Definition build.h:317
int call_env_parse_pair(TALLOC_CTX *ctx, void *out, tmpl_rules_t const *t_rules, CONF_ITEM *ci, UNUSED call_env_ctx_t const *cec, call_env_parser_t const *rule)
Standard function we use for parsing call env pairs.
Definition call_env.c:378
module_instance_t const * mi
Module instance that the callenv is registered to.
Definition call_env.h:229
Per method call config.
Definition call_env.h:180
int cf_table_parse_int(UNUSED TALLOC_CTX *ctx, void *out, UNUSED void *parent, CONF_ITEM *ci, conf_parser_t const *rule)
Generic function for parsing conf pair values as int.
Definition cf_parse.c:1594
API to parse internal format configuration items into native C types.
#define CONF_PARSER_TERMINATOR
Definition cf_parse.h:658
cf_parse_t func
Override default parsing behaviour for the specified type with a custom parsing function.
Definition cf_parse.h:612
#define FR_CONF_OFFSET(_name, _struct, _field)
conf_parser_t which parses a single CONF_PAIR, writing the result to a field in a struct
Definition cf_parse.h:284
Defines a CONF_PAIR to C data type mapping.
Definition cf_parse.h:595
Common header for all CONF_* types.
Definition cf_priv.h:49
#define cf_log_err(_cf, _fmt,...)
Definition cf_util.h:289
#define cf_log_info(_cf, _fmt,...)
Definition cf_util.h:291
#define fr_cond_assert(_x)
Calls panic_action ifndef NDEBUG, else logs error and evaluates to value of _x.
Definition debug.h:139
#define MEM(x)
Definition debug.h:36
#define ERROR(fmt,...)
Definition dhcpclient.c:41
#define MODULE_MAGIC_INIT
Stop people using different module/library/server versions together.
Definition dl_module.h:63
uint32_t(* fr_hash_t)(void const *)
Definition hash.h:36
int fr_heap_insert(fr_heap_t **hp, void *data)
Insert a new element into the heap.
Definition heap.c:146
int fr_heap_extract(fr_heap_t **hp, void *data)
Remove a node from the heap.
Definition heap.c:239
unsigned int fr_heap_index_t
Definition heap.h:80
static void * fr_heap_peek(fr_heap_t *h)
Return the item from the top of the heap but don't pop it.
Definition heap.h:136
#define fr_heap_talloc_alloc(_ctx, _cmp, _talloc_type, _field, _init)
Creates a heap that verifies elements are of a specific talloc type.
Definition heap.h:115
The main heap structure.
Definition heap.h:66
size_t fr_htrie_type_table_len
Definition htrie.c:37
fr_table_num_sorted_t const fr_htrie_type_table[]
Definition htrie.c:31
fr_htrie_t * fr_htrie_alloc(TALLOC_CTX *ctx, fr_htrie_type_t type, fr_hash_t hash_data, fr_cmp_t cmp_data, fr_trie_key_t get_key, fr_free_t free_data)
An abstraction over our internal hashes, rb trees, and prefix tries.
Definition htrie.c:92
fr_htrie_type_t
Definition htrie.h:49
@ FR_HTRIE_AUTO
Automatically choose the best type.
Definition htrie.h:54
@ FR_HTRIE_INVALID
Definition htrie.h:50
static fr_htrie_type_t fr_htrie_hint(fr_type_t type)
Definition htrie.h:149
static bool fr_htrie_insert(fr_htrie_t *ht, void const *data)
Insert data into a htrie.
Definition htrie.h:112
static int fr_htrie_num_elements(fr_htrie_t *ht)
Return the number of elements in the htrie.
Definition htrie.h:144
static char const * fr_htrie_type_to_str(fr_htrie_type_t type)
Return a static string containing the type name.
Definition htrie.h:185
static bool fr_htrie_delete(fr_htrie_t *ht, void const *data)
Delete data from a htrie, freeing it if free_data cb was passed to fr_htrie_alloc.
Definition htrie.h:136
static void * fr_htrie_find(fr_htrie_t *ht, void const *data)
Find data in a htrie.
Definition htrie.h:104
A hash/rb/prefix trie abstraction.
Definition htrie.h:80
#define PERROR(_fmt,...)
Definition log.h:228
#define RDEBUG3(fmt,...)
Definition log.h:343
#define RERROR(fmt,...)
Definition log.h:298
talloc_free(reap)
fr_type_t
module_instance_t * mi
Module instance to detach.
Definition module_ctx.h:57
module_instance_t * mi
Instance of the module being instantiated.
Definition module_ctx.h:51
Temporary structure to hold arguments for detach calls.
Definition module_ctx.h:56
Temporary structure to hold arguments for instantiation calls.
Definition module_ctx.h:50
static const conf_parser_t config[]
Definition base.c:183
#define fr_assert(_expr)
Definition rad_assert.h:38
fr_value_box_t key
Key used to identify entry.
Definition rlm_cache.h:73
module_t common
Common fields for all loadable modules.
Definition rlm_cache.h:258
cache_status_t
Definition rlm_cache.h:39
@ CACHE_ERROR
Fatal error.
Definition rlm_cache.h:41
@ CACHE_OK
Cache entry found/updated.
Definition rlm_cache.h:42
@ CACHE_MISS
Cache entry notfound.
Definition rlm_cache.h:43
void rlm_cache_handle_t
Definition rlm_cache.h:35
fr_unix_time_t expires
When the entry expires.
Definition rlm_cache.h:76
Configuration for the rlm_cache module.
Definition rlm_cache.h:51
Definition rlm_cache.h:72
static int mod_detach(module_detach_ctx_t const *mctx)
Cleanup a cache_htrie instance.
static rlm_cache_entry_t * cache_entry_alloc(UNUSED rlm_cache_config_t const *config, UNUSED void *instance, request_t *request)
Custom allocation function for the driver.
static int cf_htrie_key_parse(TALLOC_CTX *ctx, void *out, tmpl_rules_t const *t_rules, CONF_ITEM *ci, call_env_ctx_t const *cec, UNUSED call_env_parser_t const *rule)
Custom key parsing function for checking compatibility of key types.
bool htrie_auto
Whether the user wanted to automatically configure the htrie.
fr_heap_t * heap
For managing entry expiry.
static int cf_htrie_type_parse(TALLOC_CTX *ctx, void *out, void *parent, CONF_ITEM *ci, conf_parser_t const *rule)
Custom htrie type parsing function.
rlm_cache_entry_t fields
Entry data.
static conf_parser_t driver_config[]
static uint64_t cache_entry_count(UNUSED rlm_cache_config_t const *config, void *instance, request_t *request, UNUSED void *handle)
Return the number of entries in the cache.
static int cache_acquire(void **handle, UNUSED rlm_cache_config_t const *config, void *instance, request_t *request)
Lock the htrie.
rlm_cache_htrie_mutable_t * mutable
Mutable instance data.
static int8_t _value_cmp(void const *a, void const *b)
static void cache_release(UNUSED rlm_cache_config_t const *config, void *instance, request_t *request, UNUSED rlm_cache_handle_t *handle)
Release an entry unlocking any mutexes.
rlm_cache_driver_t rlm_cache_htrie
static int8_t cache_heap_cmp(void const *one, void const *two)
Compare two entries by expiry time.
fr_htrie_t * cache
Tree for looking up cache keys.
static cache_status_t cache_entry_expire(UNUSED rlm_cache_config_t const *config, void *instance, request_t *request, UNUSED void *handle, fr_value_box_t const *key)
Free an entry and remove it from the data store.
static cache_status_t cache_entry_find(rlm_cache_entry_t **out, UNUSED rlm_cache_config_t const *config, void *instance, request_t *request, UNUSED void *handle, fr_value_box_t const *key)
Locate a cache entry.
pthread_mutex_t mutex
Protect the tree from multiple readers/writers.
fr_heap_index_t heap_id
Offset used for expiry heap.
static cache_status_t cache_entry_set_ttl(UNUSED rlm_cache_config_t const *config, void *instance, request_t *request, UNUSED void *handle, rlm_cache_entry_t *c)
Update the TTL of an entry.
static int mod_instantiate(module_inst_ctx_t const *mctx)
Create a new cache_htrie instance.
static cache_status_t cache_entry_insert(rlm_cache_config_t const *config, void *instance, request_t *request, void *handle, rlm_cache_entry_t const *c)
Insert a new entry into the data store.
fr_htrie_type_t htype
The htrie type we'll be using.
fr_type_t ktype
When htrie is "auto", we use this type to decide what type of tree to use.
void * data
Module's instance data.
Definition module.h:271
conf_parser_t const * config
How to convert a CONF_SECTION to a module instance.
Definition module.h:198
Structures and prototypes for templates.
fr_type_t tmpl_expanded_type(tmpl_t const *vpt)
Return the native data type of the expression.
Definition tmpl_eval.c:209
Optional arguments passed to vp_tmpl functions.
Definition tmpl.h:337
eap_aka_sim_process_conf_t * inst
char const * fr_syserror(int num)
Guaranteed to be thread-safe version of strerror.
Definition syserror.c:243
#define talloc_get_type_abort_const
Definition talloc.h:282
static int8_t fr_unix_time_cmp(fr_unix_time_t a, fr_unix_time_t b)
Compare two fr_unix_time_t values.
Definition time.h:944
static fr_unix_time_t fr_time_to_unix_time(fr_time_t when)
Convert an fr_time_t (internal time) to our version of unix time (wallclock time)
Definition time.h:688
#define fr_unix_time_lt(_a, _b)
Definition time.h:367
int(* fr_trie_key_t)(uint8_t **out, size_t *outlen, void const *data)
Definition trie.h:56
static fr_slen_t parent
Definition pair.h:845
bool fr_type_cast(fr_type_t dst, fr_type_t src)
Return if we're allowed to cast the types.
Definition types.c:283
fr_type_t fr_type_promote(fr_type_t a, fr_type_t b)
Return the promoted type.
Definition types.c:572
Types of values contained within an fr_value_box_t.
#define fr_type_is_void(_x)
Definition types.h:357
#define fr_type_is_null(_x)
Definition types.h:326
static char const * fr_type_to_str(fr_type_t type)
Return a static string containing the type name.
Definition types.h:433
uint32_t fr_value_box_hash(fr_value_box_t const *vb)
Hash the contents of a value box.
Definition value.c:6097
int8_t fr_value_box_cmp(fr_value_box_t const *a, fr_value_box_t const *b)
Compare two values.
Definition value.c:676
void fr_value_box_copy_shallow(TALLOC_CTX *ctx, fr_value_box_t *dst, fr_value_box_t const *src)
Perform a shallow copy of a value_box.
Definition value.c:3864
int fr_value_box_to_key(uint8_t **out, size_t *outlen, fr_value_box_t const *value)
Get a key from a value box.
Definition value.c:2084
static size_t char ** out
Definition value.h:1012