The FreeRADIUS server $Id: 15bac2a4c627c01d1aa2047687b3418955ac7f00 $
Loading...
Searching...
No Matches
module_rlm.c
Go to the documentation of this file.
1/*
2 * This program 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
5 * (at 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: e448fdac7ab12ff57480adcf5f0d83b44c68eb63 $
19 *
20 * @file src/lib/server/module_rlm.c
21 * @brief Defines functions for rlm module (re-)initialisation.
22 *
23 * @copyright 2003,2006,2016 The FreeRADIUS server project
24 * @copyright 2016,2024 Arran Cudbard-Bell (a.cudbardb@freeradius.org)
25 * @copyright 2000 Alan DeKok (aland@freeradius.org)
26 * @copyright 2000 Alan Curry (pacman@world.std.com)
27 */
28
29RCSID("$Id: e448fdac7ab12ff57480adcf5f0d83b44c68eb63 $")
30
31#include <freeradius-devel/server/cf_file.h>
32
33#include <freeradius-devel/server/global_lib.h>
34#include <freeradius-devel/server/modpriv.h>
35#include <freeradius-devel/server/module_rlm.h>
36#include <freeradius-devel/server/pair.h>
37
38#include <freeradius-devel/util/atexit.h>
39
40#include <freeradius-devel/unlang/compile.h>
41
42#include <freeradius-devel/unlang/xlat_func.h>
43#include <freeradius-devel/unlang/xlat_redundant.h>
44
45
46/** Lookup virtual module by name
47 */
49
50typedef struct {
51 fr_rb_node_t name_node; //!< Entry in the name tree.
52 char const *name; //!< module name
53 CONF_SECTION *cs; //!< CONF_SECTION where it is defined
56
57/** Compare virtual modules by name
58 */
59static int8_t module_rlm_virtual_name_cmp(void const *one, void const *two)
60{
61 module_rlm_virtual_t const *a = one;
62 module_rlm_virtual_t const *b = two;
63 int ret;
64
65 ret = strcmp(a->name, b->name);
66 return CMP(ret, 0);
67}
68
69/** Global module list for all backend modules
70 *
71 */
73
74/** Runtime instantiated list
75 *
76 */
78
79/** Print information on all loaded modules
80 *
81 */
86
87/** Initialise a module specific exfile handle
88 *
89 * @see exfile_init
90 *
91 * @param[in] ctx to bind the lifetime of the exfile handle to.
92 * @param[in] module section.
93 * @param[in] max_entries Max file descriptors to cache, and manage locks for.
94 * @param[in] max_idle Maximum time a file descriptor can be idle before it's closed.
95 * @param[in] locking Whether or not to lock the files.
96 * @param[in] triggers Should triggers be enabled.
97 * @param[in] trigger_prefix if NULL will be set automatically from the module CONF_SECTION.
98 * @param[in] trigger_args to make available in any triggers executed by the connection pool.
99 * @return
100 * - New connection pool.
101 * - NULL on error.
102 */
104 CONF_SECTION *module,
105 uint32_t max_entries,
106 fr_time_delta_t max_idle,
107 bool locking,
108 bool triggers,
109 char const *trigger_prefix,
110 fr_pair_list_t *trigger_args)
111{
112 char trigger_prefix_buff[128];
113 bool prefix_set = trigger_prefix ? true : false;
114 exfile_t *handle;
115
116 if (!trigger_prefix) {
117 snprintf(trigger_prefix_buff, sizeof(trigger_prefix_buff), "modules.%s.file", cf_section_name1(module));
118 trigger_prefix = trigger_prefix_buff;
119 }
120
121 handle = exfile_init(ctx, max_entries, max_idle, locking);
122 if (!handle) return NULL;
123
124 if (triggers) exfile_enable_triggers(handle, prefix_set ? module : cf_section_find(module, "file", NULL),
125 trigger_prefix, trigger_args);
126
127 return handle;
128}
129
130/** Resolve polymorphic item's from a module's #CONF_SECTION to a subsection in another module
131 *
132 * This allows certain module sections to reference module sections in other instances
133 * of the same module and share #CONF_DATA associated with them.
134 *
135 * @verbatim
136 example {
137 data {
138 ...
139 }
140 }
141
142 example inst {
143 data = example
144 }
145 * @endverbatim
146 *
147 * @param[out] out where to write the pointer to a module's config section. May be NULL on success,
148 * indicating the config item was not found within the module #CONF_SECTION
149 * or the chain of module references was followed and the module at the end of the chain
150 * did not a subsection.
151 * @param[in] module #CONF_SECTION.
152 * @param[in] name of the polymorphic sub-section.
153 * @return
154 * - 0 on success with referenced section.
155 * - 1 on success with local section.
156 * - -1 on failure.
157 */
159{
160 CONF_PAIR *cp;
161 CONF_SECTION *cs;
162 CONF_DATA const *cd;
163
164
166 char const *inst_name;
167
168#define FIND_SIBLING_CF_KEY "find_sibling"
169
170 *out = NULL;
171
172 /*
173 * Is a real section (not referencing sibling module).
174 */
175 cs = cf_section_find(module, name, NULL);
176 if (cs) {
177 *out = cs;
178 return 0;
179 }
180
181 /*
182 * Item omitted completely from module config.
183 */
184 cp = cf_pair_find(module, name);
185 if (!cp) return 0;
186
188 cf_log_err(cp, "Module reference loop found");
189 return -1;
190 }
191 cd = cf_data_add(module, module, FIND_SIBLING_CF_KEY, false);
192
193 /*
194 * Item found, resolve it to a module instance.
195 * This triggers module loading, so we don't have
196 * instantiation order issues.
197 */
198 inst_name = cf_pair_value(cp);
199 mi = module_instance_by_name(rlm_modules_static, NULL, inst_name);
200 if (!mi) {
201 cf_log_err(cp, "Unknown module instance \"%s\"", inst_name);
202 error:
203 cf_data_remove_by_data(module, cd);
204 return -1;
205 }
206
207 if ((mi->state != MODULE_INSTANCE_INSTANTIATED) &&
209 goto error;
210 }
211
212 /*
213 * Remove the config data we added for loop
214 * detection.
215 */
216 cf_data_remove_by_data(module, cd);
217
218 /*
219 * Check the module instances are of the same type.
220 */
221 if (strcmp(cf_section_name1(mi->conf), cf_section_name1(module)) != 0) {
222 cf_log_err(cp, "Referenced module is a rlm_%s instance, must be a rlm_%s instance",
224 return -1;
225 }
226
227 *out = cf_section_find(mi->conf, name, NULL);
228
229 return 1;
230}
231
233 char const *name, xlat_func_t func, fr_type_t return_type)
234{
235 module_instance_t *mi = mctx->mi;
236 module_rlm_instance_t *mri = talloc_get_type_abort(mi->uctx, module_rlm_instance_t);
238 xlat_t *x;
239 char inst_name[256];
240
241 fr_assert_msg(name != mctx->mi->name, "`name` must not be the same as the module "
242 "instance name. Pass a NULL `name` arg if this is required");
243
244 if (!name) {
245 name = mctx->mi->name;
246 } else {
247 if ((size_t)snprintf(inst_name, sizeof(inst_name), "%s.%s", mctx->mi->name, name) >= sizeof(inst_name)) {
248 ERROR("%s: Instance name too long", __FUNCTION__);
249 return NULL;
250 }
251 name = inst_name;
252 }
253
254 x = xlat_func_register(ctx, name, func, return_type);
255 if (unlikely(x == NULL)) return NULL;
256
257 xlat_mctx_set(x, mctx);
258
259 MEM(mrx = talloc(mi, module_rlm_xlat_t));
260 mrx->xlat = x;
261 mrx->mi = mi;
262
263 fr_dlist_insert_tail(&mri->xlats, mrx);
264
265 return x;
266}
267
268/** Initialise a module specific connection pool
269 *
270 * @see fr_pool_init
271 *
272 * @param[in] module section.
273 * @param[in] opaque data pointer to pass to callbacks.
274 * @param[in] c Callback to create new connections.
275 * @param[in] a Callback to check the status of connections.
276 * @param[in] log_prefix override, if NULL will be set automatically from the module CONF_SECTION.
277 * @param[in] trigger_prefix if NULL will be set automatically from the module CONF_SECTION.
278 * @param[in] trigger_args to make available in any triggers executed by the connection pool.
279 * @return
280 * - New connection pool.
281 * - NULL on error.
282 */
284 void *opaque,
287 char const *log_prefix,
288 char const *trigger_prefix,
289 fr_pair_list_t *trigger_args)
290{
291 CONF_SECTION *cs, *mycs;
292 char log_prefix_buff[128];
293 char trigger_prefix_buff[128];
294
295 fr_pool_t *pool;
296 char const *mod_name1, *mod_name2;
297
298 int ret;
299
300#define parent_name(_x) cf_section_name(cf_item_to_section(cf_parent(_x)))
301
302 mod_name1 = cf_section_name1(module);
303 mod_name2 = cf_section_name2(module);
304 if (!mod_name2) mod_name2 = mod_name1;
305
306 if (!trigger_prefix) {
307 snprintf(trigger_prefix_buff, sizeof(trigger_prefix_buff), "modules.%s.pool", mod_name1);
308 trigger_prefix = trigger_prefix_buff;
309 }
310
311 if (!log_prefix) {
312 snprintf(log_prefix_buff, sizeof(log_prefix_buff), "rlm_%s (%s)", mod_name1, mod_name2);
313 log_prefix = log_prefix_buff;
314 }
315
316 /*
317 * Get sibling's pool config section
318 */
319 ret = module_rlm_sibling_section_find(&cs, module, "pool");
320 switch (ret) {
321 case -1:
322 return NULL;
323
324 case 1:
325 DEBUG4("%s: Using pool section from \"%s\"", log_prefix, parent_name(cs));
326 break;
327
328 case 0:
329 DEBUG4("%s: Using local pool section", log_prefix);
330 break;
331 }
332
333 /*
334 * Get our pool config section
335 */
336 mycs = cf_section_find(module, "pool", NULL);
337 if (!mycs) {
338 DEBUG4("%s: Adding pool section to module configuration \"%s\" to store pool references", log_prefix,
339 mod_name2);
340
341 mycs = cf_section_alloc(module, module, "pool", NULL);
342 if (!mycs) return NULL;
343 }
344
345 /*
346 * Sibling didn't have a pool config section
347 * Use our own local pool.
348 */
349 if (!cs) {
350 DEBUG4("%s: \"%s.pool\" section not found, using \"%s.pool\"", log_prefix,
351 mod_name1, parent_name(mycs));
352 cs = mycs;
353 }
354
355 /*
356 * If fr_pool_init has already been called
357 * for this config section, reuse the previous instance.
358 *
359 * This allows modules to pass in the config sections
360 * they would like to use the connection pool from.
361 */
362 pool = cf_data_value(cf_data_find(cs, fr_pool_t, NULL));
363 if (!pool) {
364 DEBUG4("%s: No pool reference found for config item \"%s.pool\"", log_prefix, parent_name(cs));
365 pool = fr_pool_init(cs, cs, opaque, c, a, log_prefix);
366 if (!pool) return NULL;
367
368 fr_pool_enable_triggers(pool, trigger_prefix, trigger_args);
369
370 if (fr_pool_start(pool) < 0) {
371 ERROR("%s: Starting initial connections failed", log_prefix);
372 return NULL;
373 }
374
375 DEBUG4("%s: Adding pool reference %p to config item \"%s.pool\"", log_prefix, pool, parent_name(cs));
376 cf_data_add(cs, pool, NULL, false);
377 return pool;
378 }
379 fr_pool_ref(pool);
380
381 DEBUG4("%s: Found pool reference %p in config item \"%s.pool\"", log_prefix, pool, parent_name(cs));
382
383 /*
384 * We're reusing pool data add it to our local config
385 * section. This allows other modules to transitively
386 * reuse a pool through this module.
387 */
388 if (mycs != cs) {
389 DEBUG4("%s: Copying pool reference %p from config item \"%s.pool\" to config item \"%s.pool\"",
390 log_prefix, pool, parent_name(cs), parent_name(mycs));
391 cf_data_add(mycs, pool, NULL, false);
392 }
393
394 return pool;
395}
396
397/** Set the next section type if it's not already set
398 *
399 * @param[in] request The current request.
400 * @param[in] type_da to use. Usually attr_auth_type.
401 * @param[in] enumv Enumeration value of the specified type_da.
402 */
404{
405 fr_pair_t *vp;
406
407 switch (pair_update_control(&vp, type_da)) {
408 case 0:
409 if (unlikely(fr_value_box_copy(vp, &vp->data, enumv->value) < 0)) {
410 fr_strerror_printf("Failed to set control.%pP to %s", vp, enumv->name);
411 return false;
412 }
413 vp->data.enumv = vp->da; /* So we get the correct string alias */
414 RDEBUG2("Setting control.%pP", vp);
415 return true;
416
417 case 1:
418 RDEBUG2("control.%s already set. Not setting to %s", vp->da->name, enumv->name);
419 return false;
420
421 default:
422 return false;
423 }
424}
425
426/** Iterate over an array of named module methods, looking for matches
427 *
428 * @param[in] mmg A structure containing a terminated array of
429 * module method bindings. pre-sorted using #section_name_cmp
430 * with name2 sublists populated.
431 * @param[in] section name1 of the method being called can be one of the following:
432 * - An itenfier.
433 * - CF_IDENT_ANY if the method is a wildcard.
434 * name2 of the method being called can be one of the following:
435 * - An itenfier.
436 * - NULL to match section names with only a name1.
437 * - CF_IDENT_ANY if the method is a wildcard.
438 * @return
439 * - The module_method_name_t on success.
440 * - NULL on not found.
441 */
442static CC_HINT(nonnull)
444{
445 module_method_group_t const *mmg_p = mmg;
447
448 while (mmg_p) {
449 /*
450 * This could potentially be improved by using a binary search
451 * but given the small number of items, reduced branches and
452 * sequential access just scanning the list, it's probably not
453 * worth it.
454 */
455 for (p = mmg_p->bindings; p->section; p++) {
456 switch (section_name_match(p->section, section)) {
457 case 1: /* match */
458 return p;
459
460 case -1: /* name1 didn't match, skip to the end of the sub-list */
461 p = fr_dlist_tail(&p->same_name1);
462 break;
463
464 case 0: /* name1 did match - see if we can find a matching name2 */
465 {
466 fr_dlist_head_t const *same_name1 = &p->same_name1;
467
468 while ((p = fr_dlist_next(same_name1, p))) {
469 if (section_name2_match(p->section, section)) return p;
470 }
471 p = fr_dlist_tail(same_name1);
472 }
473 break;
474 }
475#ifdef __clang_analyzer__
476 /* Will never be NULL, worse case, p doesn't change*/
477 if (!p) break;
478#endif
479 }
480
481 /*
482 * Failed to match, search the next deepest group in the chain.
483 */
484 mmg_p = mmg_p->next;
485 }
486
487 return NULL;
488}
489
490/** Dump the available bindings for the module into the strerror stack
491 *
492 * @note Methods from _all_ linked module method groups will be pushed onto the error stack.
493 *
494 * @param[in] mmg module method group to evaluate.
495 */
497{
498 module_method_group_t const *mmg_p = mmg;
499 module_method_binding_t const *mmb_p;
500 bool first = true;
501
502 while (mmg_p) {
503 mmb_p = mmg_p->bindings;
504
505 if (!mmb_p || !mmb_p[0].section) goto next;
506
507 if (first) {
508 fr_strerror_const_push("Available methods are:");
509 first = false;
510 }
511
512 for (; mmb_p->section; mmb_p++) {
513 char const *name1 = section_name_str(mmb_p->section->name1);
514 char const *name2 = section_name_str(mmb_p->section->name2);
515
516 fr_strerror_printf_push(" %s%s%s",
517 name1, name2 ? "." : "", name2 ? name2 : "");
518 }
519 next:
520 mmg_p = mmg_p->next;
521 }
522
523 if (first) {
524 fr_strerror_const_push("No methods available");
525 }
526}
527
528/** Find an existing module instance and verify it implements the specified method
529 *
530 * Extracts the method from the module name where the format is @verbatim <module>[.<method1>[.<method2>]] @endverbatim
531 * and ensures the module implements the specified method.
532 *
533 * @param[in] ctx to allocate the dynamic module key tmpl from.
534 * @param[out] mmc_out the result from resolving the module method,
535 * plus the key tmpl for dynamic modules.
536 * This is not allocated from the ctx to save the runtime
537 * dereference.
538 * @param[in] vs Virtual server to search for alternative module names in.
539 * @param[in] section Section name containing the module call.
540 * @param[in] name The module method call i.e. module[<key>][.<method>]
541 * @param[in] t_rules for resolving the dynamic module key.
542 * @return
543 * - The module instance on success.
544 * - NULL on not found
545 *
546 * If the module exists but the method doesn't exist, then `method` is set to NULL.
547 */
549 virtual_server_t const *vs, section_name_t const *section, fr_sbuff_t *name,
550 tmpl_rules_t const *t_rules)
551{
552 fr_sbuff_term_t const *dyn_tt = &FR_SBUFF_TERMS(
553 L(""),
554 L("\t"),
555 L("\n"),
556 L(" "),
557 L("[")
558 );
559
560 fr_sbuff_term_t const *elem_tt = &FR_SBUFF_TERMS(
561 L(""),
562 L("\t"),
563 L("\n"),
564 L(" "),
565 L(".")
566 );
567
568 fr_sbuff_t *elem1;
570 module_method_call_t mmc_tmp;
571 module_method_binding_t const *mmb;
572
573 fr_sbuff_marker_t meth_start;
574 bool softfail;
575
576 fr_slen_t slen;
577 fr_sbuff_t our_name = FR_SBUFF(name);
578
579 mmc = mmc_out ? mmc_out : &mmc_tmp;
580 if (mmc_out) *mmc_out = (module_method_call_t) {};
581
582 softfail = fr_sbuff_next_if_char(&our_name, '-');
583
584 /*
585 * Advance until the start of the dynamic selector
586 * (if it exists).
587 */
588 if (fr_sbuff_adv_until(&our_name, SIZE_MAX, dyn_tt, '\0') == 0) {
589 fr_strerror_printf("Invalid module method name");
590 return fr_sbuff_error(&our_name);
591 }
592
594
595 /*
596 * If the method string contains a '['
597 *
598 * Search for a dynamic module method, e.g. `elem1[<key>]`.
599 */
600 if (fr_sbuff_is_char(&our_name, '[')) {
601 fr_sbuff_marker_t end, s_end;
602 fr_sbuff_marker(&end, &our_name);
603
604 slen = tmpl_afrom_substr(ctx, &mmc->key, &our_name, T_BARE_WORD, NULL, t_rules);
605 if (slen < 0) {
606 fr_strerror_const_push("Invalid dynamic module selector expression");
607 return slen;
608 }
609
610 if (!fr_sbuff_is_char(&our_name, ']')) {
611 fr_strerror_const_push("Missing terminating ']' for dynamic module selector");
612 error:
613 return fr_sbuff_error(&our_name);
614 }
615 fr_sbuff_marker(&s_end, &our_name);
616
617 fr_sbuff_set_to_start(&our_name);
618 slen = fr_sbuff_out_bstrncpy(elem1, &our_name, fr_sbuff_ahead(&end));
619 if (slen < 0) {
620 fr_strerror_const("Module method string too long");
621 goto error;
622 }
623 mmc->mi = module_instance_by_name(rlm_modules_dynamic, NULL, elem1->start);
624 if (!mmc->mi) {
625 fr_strerror_printf("No such dynamic module '%s'", elem1->start);
626 goto error;
627 }
629
630 fr_sbuff_set(&our_name, &s_end);
631 fr_sbuff_advance(&our_name, 1); /* Skip the ']' */
632 /*
633 * With elem1.elem2.elem3
634 *
635 * Search for a static module matching one of the following:
636 *
637 * - elem1.elem2.elem3
638 * - elem1.elem2
639 * - elem1
640 */
641 } else {
642 char *p;
643
644 fr_sbuff_set_to_start(&our_name);
645
646 slen = fr_sbuff_out_bstrncpy_until(elem1, &our_name, SIZE_MAX, dyn_tt, NULL);
647 if (slen == 0) {
648 fr_strerror_const("Invalid module name");
649 goto error;
650 }
651 if (slen < 0) {
652 fr_strerror_const("Module method string too long");
653 goto error;
654 }
655
656 /*
657 * Now we have a mutable buffer, we can start chopping
658 * it up to find the module.
659 */
660 for (;;) {
661 mmc->mi = (module_instance_t *)module_rlm_static_by_name(NULL, elem1->start);
662 if (mmc->mi) {
664 break; /* Done */
665 }
666
667 p = strrchr(elem1->start, '.');
668 if (!p) break; /* No more '.' */
669 *p = '\0'; /* Chop off the last '.' */
670 }
671
672 if (!mmc->mi) {
673 if (softfail) return fr_sbuff_set(name, &our_name);
674
675 fr_strerror_printf("No such module '%pV'", fr_box_strvalue_len(our_name.start, slen));
676 return -1;
677 }
678
679 fr_sbuff_set_to_start(&our_name);
680 fr_sbuff_advance(&our_name, strlen(elem1->start)); /* Advance past the module name */
681 if (fr_sbuff_is_char(&our_name, '.')) {
682 fr_sbuff_advance(&our_name, 1); /* Static module method, search directly */
683 } else {
684 fr_sbuff_marker(&meth_start, &our_name); /* for the errors... */
685 goto by_section; /* Get the method dynamically from the section*/
686 }
687 }
688
689 /*
690 * For both cases, the buffer should be pointing
691 * at the start of the method string.
692 */
693 fr_sbuff_marker(&meth_start, &our_name);
694
695 /*
696 * If a module method was provided, search for it in the named
697 * methods provided by the module.
698 *
699 * The method name should be either:
700 *
701 * - name1
702 * - name1.name2
703 */
704 {
705 section_name_t method;
706 fr_sbuff_t *elem2;
707
708 fr_sbuff_set_to_start(elem1); /* May have used this already for module lookups */
709
710 slen = fr_sbuff_out_bstrncpy_until(elem1, &our_name, SIZE_MAX, elem_tt, NULL);
711 if (slen < 0) {
712 fr_strerror_const("Module method string too long");
713 return fr_sbuff_error(&our_name);
714 }
715 if (slen == 0) goto by_section; /* This works for both dynamic and static modules */
716
718
719 if (fr_sbuff_is_char(&our_name, '.')) {
720 fr_sbuff_advance(&our_name, 1);
721 if (fr_sbuff_out_bstrncpy_until(elem2, &our_name, SIZE_MAX,
722 elem_tt, NULL) == MODULE_INSTANCE_LEN_MAX) {
723 fr_strerror_const("Module method string too long");
724 goto error;
725 }
726 }
727
728 method = (section_name_t) {
729 .name1 = elem1->start,
730 .name2 = fr_sbuff_used(elem2) ? elem2->start : NULL
731 };
732
733 mmb = module_binding_find(&mmc->rlm->method_group, &method);
734 if (!mmb) {
735 fr_strerror_printf("Module \"%s\" does not have method %s%s%s",
736 mmc->mi->name,
737 method.name1,
738 method.name2 ? "." : "",
739 method.name2 ? method.name2 : ""
740 );
741
743 return fr_sbuff_error(&meth_start);
744 }
745 mmc->mmb = *mmb; /* For locality of reference and fewer derefs */
746 if (mmc_out) section_name_dup(ctx, &mmc->asked, &method);
747
748 return fr_sbuff_set(name, &our_name);
749 }
750
751by_section:
752 /*
753 * First look for the section name in the module's
754 * bindings. If that fails, look for the alt
755 * section names from the virtual server section.
756 *
757 * If that fails, we're done.
758 */
759 mmb = module_binding_find(&mmc->rlm->method_group, section);
760 if (!mmb) {
761 section_name_t const **alt_p = virtual_server_section_methods(vs, section);
762 if (alt_p) {
763 for (; *alt_p; alt_p++) {
764 mmb = module_binding_find(&mmc->rlm->method_group, *alt_p);
765 if (mmb) {
766 if (mmc_out) section_name_dup(ctx, &mmc->asked, *alt_p);
767 break;
768 }
769 }
770 }
771 } else {
772 if (mmc_out) section_name_dup(ctx, &mmc->asked, section);
773 }
774 if (!mmb) {
775 fr_strerror_printf("Module \"%s\" has no method for section %s %s { ... }, i.e. %s%s%s",
776 mmc->mi->name,
777 section->name1,
778 section->name2 ? section->name2 : "",
779 section->name1,
780 section->name2 ? "." : "",
781 section->name2 ? section->name2 : ""
782 );
784
785 return fr_sbuff_error(&meth_start);
786 }
787 mmc->mmb = *mmb; /* For locality of reference and fewer derefs */
788
789 return fr_sbuff_set(name, &our_name);
790}
791
793{
795
798 .name = asked_name,
799 });
800 if (!inst) return NULL;
801
802 return inst->cs;
803}
804
809
814
815/** Create a virtual module.
816 *
817 * @param[in] cs that defines the virtual module.
818 * @return
819 * - 0 on success.
820 * - -1 on failure.
821 */
823{
824 char const *name;
825 bool all_same;
826 CONF_ITEM *sub_ci = NULL;
827 CONF_PAIR *cp;
830
832
833 /*
834 * Groups, etc. must have a name.
835 */
836 if ((strcmp(name, "group") == 0) ||
837 (strcmp(name, "redundant") == 0) ||
838 (strcmp(name, "redundant-load-balance") == 0) ||
839 (strcmp(name, "load-balance") == 0)) {
841 if (!name) {
842 cf_log_err(cs, "Keyword module must have a second name");
843 return -1;
844 }
845
846 /*
847 * name2 was already checked in modules_rlm_bootstrap()
848 */
850 } else {
851 cf_log_err(cs, "Module names cannot be unlang keywords '%s'", name);
852 return -1;
853 }
854
855 /*
856 * Ensure that the module doesn't exist.
857 */
859 if (mi) {
860 ERROR("Duplicate module \"%s\" in file %s[%d] and file %s[%d]",
861 name,
862 cf_filename(cs),
863 cf_lineno(cs),
864 cf_filename(mi->conf),
865 cf_lineno(mi->conf));
866 return -1;
867 }
868
869 /*
870 * Don't bother registering redundant xlats for a simple "group".
871 */
872 all_same = (strcmp(cf_section_name1(cs), "group") != 0);
873
874 {
875 module_t const *last = NULL;
876
877 /*
878 * Ensure that the modules we reference here exist.
879 */
880 while ((sub_ci = cf_item_next(cs, sub_ci))) {
881 if (cf_item_is_pair(sub_ci)) {
882 cp = cf_item_to_pair(sub_ci);
883 if (cf_pair_value(cp)) {
884 cf_log_err(sub_ci, "Cannot set return codes in a %s block", cf_section_name1(cs));
885 return -1;
886 }
887
889 if (!mi) {
890 cf_log_perr(sub_ci, "Failed resolving module reference '%s' in %s block",
892 return -1;
893 }
894
895 if (all_same) {
896 if (!last) {
897 last = mi->exported;
898 } else if (last != mi->exported) {
899 last = NULL;
900 all_same = false;
901 }
902 }
903 } else {
904 all_same = false;
905 }
906
907 /*
908 * Don't check subsections for now. That check
909 * happens later in the unlang compiler.
910 */
911 } /* loop over things in a virtual module section */
912 }
913
914 inst = talloc_zero(cs, module_rlm_virtual_t);
915 if (!inst) return -1;
916
917 inst->cs = cs;
918 MEM(inst->name = talloc_strdup(inst, name));
919 inst->all_same = all_same;
920
923 return -1;
924 }
925
926 return 0;
927}
928
929/** Generic conf_parser_t func for loading drivers
930 *
931 */
932int module_rlm_submodule_parse(TALLOC_CTX *ctx, void *out, void *parent,
933 CONF_ITEM *ci, conf_parser_t const *rule)
934{
935 conf_parser_t our_rule = *rule;
936
937 our_rule.uctx = &rlm_modules_static;
938
939 return module_submodule_parse(ctx, out, parent, ci, &our_rule);
940}
941
942/** Frees thread-specific data for all registered backend modules
943 *
944 */
949
950/** Allocates thread-specific data for all registered backend modules
951 *
952 * @param[in] ctx To allocate any thread-specific data in.
953 * @param[in] el to register events.
954 * @return
955 * - 0 if all modules were instantiated successfully.
956 * - -1 if a module failed instantiation.
957 */
962
963/** Runs the coord_attach method of all registered backend modules
964 *
965 * @param[in] el to register events.
966 * @return
967 * - 0 if all calls succeeded.
968 * - -1 if a call failed.
969 */
974
975/** Performs the instantiation phase for all backend modules
976 *
977 * @return
978 * - 0 if all modules were instantiated successfully.
979 * - -1 if a module failed instantiation.
980 */
985
986/** Compare the section names of two module_method_binding_t structures
987 */
988static int8_t binding_name_cmp(void const *one, void const *two)
989{
990 module_method_binding_t const *a = one;
991 module_method_binding_t const *b = two;
992
993 return section_name_cmp(a->section, b->section);
994}
995
997{
998 module_method_binding_t *p, *srt_p;
999 fr_dlist_head_t bindings;
1000 bool in_order = true;
1001
1002 /*
1003 * Not all modules export module method bindings
1004 */
1005 if (!group || !group->bindings || group->validated) return 0;
1006
1007 fr_dlist_init(&bindings, module_method_binding_t, entry);
1008
1009 for (p = group->bindings; p->section; p++) {
1011 "First section identifier can't be NULL")) return -1;
1012
1013 /*
1014 * All the bindings go in a list so we can sort them
1015 * and produce the list in the correct order.
1016 */
1017 fr_dlist_insert_tail(&bindings, p);
1018 }
1019
1020 fr_dlist_sort(&bindings, binding_name_cmp);
1021
1022 /*
1023 * Iterate over the sorted list of bindings,
1024 * and the original list, to ensure they're
1025 * in the correct order.
1026 */
1027 for (srt_p = fr_dlist_head(&bindings), p = group->bindings;
1028 srt_p;
1029 srt_p = fr_dlist_next(&bindings, srt_p), p++) {
1030 if (p != srt_p) {
1031 in_order = false;
1032 break;
1033 }
1034 }
1035
1036 /*
1037 * Rebuild the binding list in the correct order.
1038 */
1039 if (!in_order) {
1040 module_method_binding_t *ordered;
1041
1042 MEM(ordered = talloc_array(NULL, module_method_binding_t, fr_dlist_num_elements(&bindings)));
1043 for (srt_p = fr_dlist_head(&bindings), p = ordered;
1044 srt_p;
1045 srt_p = fr_dlist_next(&bindings, srt_p), p++) {
1046 *p = *srt_p;
1047 }
1048 memcpy(group->bindings, ordered, fr_dlist_num_elements(&bindings) * sizeof(*ordered));
1049 talloc_free(ordered);
1050 }
1051
1052 /*
1053 * Build the "skip" list of name1 entries
1054 */
1055 {
1056 module_method_binding_t *last_binding = NULL;
1057
1058 for (p = group->bindings; p->section; p++) {
1059 if (!last_binding ||
1060 (
1061 (last_binding->section->name1 != p->section->name1) &&
1062 (
1063 (last_binding->section->name1 == CF_IDENT_ANY) ||
1064 (p->section->name1 == CF_IDENT_ANY) ||
1065 (strcmp(last_binding->section->name1, p->section->name1) != 0)
1066 )
1067 )
1068 ) {
1070 last_binding = p;
1071 }
1072 fr_dlist_insert_tail(&last_binding->same_name1, p);
1073 }
1074 }
1075 group->validated = true;
1076
1077 return module_method_group_validate(group->next);
1078}
1079
1086
1087/** Allocate a rlm module instance
1088 *
1089 * These have extra space allocated to hold the dlist of associated xlats.
1090 *
1091 * @param[in] ml Module list to allocate from.
1092 * @param[in] parent Parent module instance.
1093 * @param[in] type Type of module instance.
1094 * @param[in] mod_name Name of the module.
1095 * @param[in] inst_name Name of the instance.
1096 * @param[in] init_state Initial state of the module instance.
1097 * @return
1098 * - The allocated module instance on success.
1099 * - NULL on failure.
1100 */
1101static inline CC_HINT(always_inline)
1104 dl_module_type_t type, char const *mod_name, char const *inst_name,
1105 module_instance_state_t init_state)
1106{
1109
1110 mi = module_instance_alloc(ml, parent, type, mod_name, inst_name, init_state);
1111 if (unlikely(mi == NULL)) return NULL;
1112
1113 MEM(mri = talloc(mi, module_rlm_instance_t));
1114 module_instance_uctx_set(mi, mri);
1115
1117
1118 return mi;
1119}
1120
1122{
1123 char const *name;
1124 char const *inst_name;
1125 module_instance_t *mi = NULL;
1126 CONF_SECTION *actions;
1127
1128 /*
1129 * name2 can't be a keyword
1130 */
1131 name = cf_section_name2(mod_conf);
1133 invalid_name:
1134 cf_log_err(mod_conf, "Module names cannot be unlang keywords '%s'", name);
1135 return -1;
1136 }
1137
1138 name = cf_section_name1(mod_conf);
1139
1140 /*
1141 * For now, ignore name1 which is a keyword.
1142 */
1144 if (!cf_section_name2(mod_conf)) {
1145 cf_log_err(mod_conf, "Missing second name at '%s'", name);
1146 return -1;
1147 }
1148 if (module_rlm_bootstrap_virtual(mod_conf) < 0) return -1;
1149 return 0;
1150 }
1151
1152 /*
1153 * Skip inline templates, and disallow "template { ... }"
1154 */
1155 if (strcmp(name, "template") == 0) {
1156 if (!cf_section_name2(mod_conf)) goto invalid_name;
1157 return 0;
1158 }
1159
1160 if (module_instance_name_from_conf(&inst_name, mod_conf) < 0) goto invalid_name;
1161
1162 mi = module_rlm_instance_alloc(ml, NULL, DL_MODULE_TYPE_MODULE, name, inst_name, 0);
1163 if (unlikely(mi == NULL)) {
1164 cf_log_perr(mod_conf, "Failed loading module");
1165 return -1;
1166 }
1167
1168 /*
1169 * First time we've loaded the dl module, so we need to
1170 * check the module methods to make sure they're ordered
1171 * correctly, and to add the "skip list" style name2
1172 * entries.
1173 */
1174 if ((mi->module->refs == 1) && (module_method_validate(mi) < 0)) {
1175 talloc_free(mi);
1176 return -1;
1177 }
1178
1179 if (module_instance_conf_parse(mi, mod_conf) < 0) {
1180 cf_log_perr(mod_conf, "Failed parsing module config");
1181 talloc_free(mi);
1182 return -1;
1183 }
1184
1185 /*
1186 * Compile the default "actions" subsection, which includes retries.
1187 */
1188 actions = cf_section_find(mod_conf, "actions", NULL);
1189 if (actions && !unlang_compile_actions(&mi->actions, actions, (mi->exported->flags & MODULE_TYPE_RETRY))) {
1190 talloc_free(mi);
1191 return -1;
1192 }
1193
1194 return 0;
1195}
1196
1197/** Bootstrap modules and virtual modules
1198 *
1199 * Parse the module config sections, and load and call each module's init() function.
1200 *
1201 * @param[in] root of the server configuration.
1202 * @return
1203 * - 0 if all modules were bootstrapped successfully.
1204 * - -1 if a module/virtual module failed to bootstrap.
1205 */
1207{
1208 CONF_SECTION *cs, *modules, *static_cs, *dynamic_cs;
1211
1212 /*
1213 * Ensure any libraries the modules depend on are instantiated
1214 */
1216
1217 /*
1218 * Remember where the modules were stored.
1219 */
1220 modules = cf_section_find(root, "modules", NULL);
1221 if (!modules) {
1222 WARN("Cannot find a \"modules\" section in the configuration file!");
1223 return 0;
1224 }
1225
1226 static_cs = cf_section_find(modules, "static", NULL);
1227 if (!static_cs) {
1228 static_cs = cf_section_alloc(modules, NULL, "static", NULL);
1229 cf_section_foreach(modules, mod_cs) {
1230 CONF_ITEM *prev;
1231 char const *name1 = cf_section_name1(mod_cs);
1232
1233 /*
1234 * Skip over the dynamic section
1235 */
1236 if ((strcmp(name1, "dynamic") == 0) && !cf_section_name2(mod_cs)) continue;
1237
1238 /*
1239 * Ignore this section if it is commented out with a magic name.
1240 */
1241 if (*name1 == '-') continue;
1242
1243 /*
1244 * Move all modules which are not in
1245 * the dynamic section into the static
1246 * section for backwards compatibility.
1247 */
1248 prev = cf_item_remove(modules, mod_cs);
1249 cf_item_add(static_cs, mod_cs);
1250
1251 /*
1252 * Find the previous item that's a section
1253 */
1254 while (prev && !cf_item_is_section(prev)) prev = cf_item_prev(modules, prev);
1255
1256 /*
1257 * Resume iterating from that item
1258 */
1259 mod_cs = cf_item_to_section(prev);
1260 }
1261 cf_item_add(modules, static_cs);
1262 }
1263 DEBUG2("#### Bootstrapping static modules ####");
1264 cf_log_debug(modules, " modules {");
1265 cf_log_debug(modules, " static {");
1266 cf_section_foreach(static_cs, mod_conf) {
1267 if (module_conf_parse(rlm_modules_static, mod_conf) < 0) return -1;
1268 }
1269 cf_log_debug(modules, " } # static");
1270
1271 /*
1272 * Now we have a module tree, run bootstrap on all the modules.
1273 * This will bootstrap modules and then submodules.
1274 */
1275 if (unlikely(modules_bootstrap(rlm_modules_static) < 0)) return -1;
1276
1277 if (fr_command_register_hook(NULL, NULL, static_cs, module_cmd_list_table) < 0) {
1278 PERROR("Failed registering radmin commands for modules");
1279 return -1;
1280 }
1281
1282 /*
1283 * Build the configuration and parse dynamic modules
1284 */
1285 dynamic_cs = cf_section_find(modules, "dynamic", NULL);
1286 if (dynamic_cs) {
1287 DEBUG2("#### Bootstrapping dynamic modules ####");
1288 /*
1289 * Parse and then instantiate any dynamic modules configure
1290 */
1291 cf_log_debug(modules, " dynamic {");
1292 cf_section_foreach(dynamic_cs, mod_conf) {
1293 if (unlikely(module_conf_parse(rlm_modules_dynamic, mod_conf) < 0)) return -1;
1294 }
1295 cf_log_debug(modules, " } # dynamic");
1296 if (unlikely(modules_bootstrap(rlm_modules_dynamic) < 0)) return -1;
1297 cf_log_debug(modules, " } # modules");
1298 }
1299
1300 /*
1301 * Check for duplicate policies. They're treated as
1302 * modules, so we might as well check them here.
1303 */
1304 cs = cf_section_find(root, "policy", NULL);
1305 if (cs) {
1306 cf_section_foreach(cs, policy_cs) {
1307 CONF_SECTION *problemcs;
1308 char const *name1 = cf_section_name1(policy_cs);
1309
1310 if (unlang_compile_is_keyword(name1)) {
1311 cf_log_err(policy_cs, "Policy name '%s' cannot be an unlang keyword", name1);
1312 return -1;
1313 }
1314
1315 if (cf_section_name2(policy_cs)) {
1316 cf_log_err(policy_cs, "Policies cannot have two names");
1317 return -1;
1318 }
1319
1320 problemcs = cf_section_find_next(cs, policy_cs, name1, CF_IDENT_ANY);
1321 if (!problemcs) continue;
1322
1323 cf_log_err(problemcs, "Duplicate policy '%s' is forbidden.",
1324 cf_section_name1(policy_cs));
1325 return -1;
1326 }
1327 }
1328
1329 /*
1330 * Now that all of the xlat things have been registered,
1331 * register our redundant xlats. But only when all of
1332 * the items in such a section are the same.
1333 */
1335 vm;
1337 if (!vm->all_same) continue;
1338
1339 if (xlat_register_redundant(vm->cs) < 0) return -1;
1340 }
1341
1342 return 0;
1343}
1344
1345/** Cleanup all global structures
1346 *
1347 * Automatically called on exit.
1348 */
1350{
1351 if (talloc_free(rlm_modules_static) < 0) return -1;
1352 rlm_modules_static = NULL;
1353
1354 if (talloc_free(rlm_modules_dynamic) < 0) return -1;
1355 rlm_modules_dynamic = NULL;
1356
1357 if (talloc_free(module_rlm_virtual_name_tree) < 0) return -1;
1359
1360 return 0;
1361}
1362
1363static int _modules_rlm_free_atexit(UNUSED void *uctx)
1364{
1365 return modules_rlm_free();
1366}
1367
1368/** Initialise the module list structure
1369 *
1370 */
1372{
1375 module_list_mask_set(rlm_modules_dynamic, MODULE_INSTANCE_INSTANTIATED); /* Ensure we never instantiate dynamic modules */
1376
1380
1381 return 0;
1382}
#define fr_atexit_global(_func, _uctx)
Add a free function to the global free list.
Definition atexit.h:58
#define RCSID(id)
Definition build.h:512
#define L(_str)
Helper for initialising arrays of string literals.
Definition build.h:228
#define CMP(_a, _b)
Same as CMP_PREFER_SMALLER use when you don't really care about ordering, you just want an ordering.
Definition build.h:113
#define unlikely(_x)
Definition build.h:407
#define UNUSED
Definition build.h:336
void const * uctx
User data accessible by the cf_parse_t func.
Definition cf_parse.h:617
Defines a CONF_PAIR to C data type mapping.
Definition cf_parse.h:594
Internal data that is associated with a configuration section.
Definition cf_priv.h:125
Common header for all CONF_* types.
Definition cf_priv.h:49
Configuration AVP similar to a fr_pair_t.
Definition cf_priv.h:72
A section grouping multiple CONF_PAIR.
Definition cf_priv.h:101
bool cf_item_is_pair(CONF_ITEM const *ci)
Determine if CONF_ITEM is a CONF_PAIR.
Definition cf_util.c:633
char const * cf_section_name2(CONF_SECTION const *cs)
Return the second identifier of a CONF_SECTION.
Definition cf_util.c:1187
void * cf_data_value(CONF_DATA const *cd)
Return the user assigned value of CONF_DATA.
Definition cf_util.c:1750
char const * cf_section_name1(CONF_SECTION const *cs)
Return the first identifier of a CONF_SECTION.
Definition cf_util.c:1173
CONF_SECTION * cf_section_find(CONF_SECTION const *cs, char const *name1, char const *name2)
Find a CONF_SECTION with name1 and optionally name2.
Definition cf_util.c:1029
CONF_SECTION * cf_item_to_section(CONF_ITEM const *ci)
Cast a CONF_ITEM to a CONF_SECTION.
Definition cf_util.c:685
CONF_PAIR * cf_pair_find(CONF_SECTION const *cs, char const *attr)
Search for a CONF_PAIR with a specific name.
Definition cf_util.c:1422
bool cf_item_is_section(CONF_ITEM const *ci)
Determine if CONF_ITEM is a CONF_SECTION.
Definition cf_util.c:619
CONF_PAIR * cf_item_to_pair(CONF_ITEM const *ci)
Cast a CONF_ITEM to a CONF_PAIR.
Definition cf_util.c:665
CONF_SECTION * cf_section_find_next(CONF_SECTION const *cs, CONF_SECTION const *prev, char const *name1, char const *name2)
Return the next matching section.
Definition cf_util.c:1050
char const * cf_pair_value(CONF_PAIR const *pair)
Return the value of a CONF_PAIR.
Definition cf_util.c:1581
char const * cf_pair_attr(CONF_PAIR const *pair)
Return the attr of a CONF_PAIR.
Definition cf_util.c:1565
#define cf_item_add(_parent, _child)
Definition cf_util.h:80
#define cf_log_err(_cf, _fmt,...)
Definition cf_util.h:285
#define cf_lineno(_cf)
Definition cf_util.h:101
#define cf_section_foreach(_parent, _iter)
Definition cf_util.h:155
#define cf_data_add(_cf, _data, _name, _free)
Definition cf_util.h:251
#define cf_data_find(_cf, _type, _name)
Definition cf_util.h:240
#define cf_data_remove_by_data(_cf, _cd)
Remove an item from a parent.
Definition cf_util.h:270
#define cf_item_prev(_parent, _curr)
Definition cf_util.h:92
#define cf_item_remove(_parent, _child)
Definition cf_util.h:86
#define cf_item_next(_parent, _curr)
Definition cf_util.h:89
#define cf_log_perr(_cf, _fmt,...)
Definition cf_util.h:292
#define cf_section_alloc(_ctx, _parent, _name1, _name2)
Definition cf_util.h:143
#define cf_filename(_cf)
Definition cf_util.h:104
#define cf_log_debug(_cf, _fmt,...)
Definition cf_util.h:288
#define CF_IDENT_ANY
Definition cf_util.h:75
fr_command_register_hook_t fr_command_register_hook
Definition command.c:41
bool unlang_compile_is_keyword(const char *name)
Check if name is an unlang keyword.
Definition compile.c:2257
bool unlang_compile_actions(unlang_mod_actions_t *actions, CONF_SECTION *cs, bool module_retry)
Definition compile.c:1091
#define fr_cond_assert(_x)
Calls panic_action ifndef NDEBUG, else logs error and evaluates to value of _x.
Definition debug.h:141
#define fr_assert_msg(_x, _msg,...)
Calls panic_action ifndef NDEBUG, else logs error and causes the server to exit immediately with code...
Definition debug.h:212
#define fr_cond_assert_msg(_x, _fmt,...)
Calls panic_action ifndef NDEBUG, else logs error and evaluates to value of _x.
Definition debug.h:158
#define MEM(x)
Definition debug.h:46
#define ERROR(fmt,...)
Definition dhcpclient.c:40
fr_value_box_t const * value
Enum value (what name maps to).
Definition dict.h:257
char const * name
Enum name.
Definition dict.h:254
Value of an enumerated attribute.
Definition dict.h:253
dl_module_type_t
Definition dl_module.h:65
@ DL_MODULE_TYPE_MODULE
Standard loadable module.
Definition dl_module.h:66
static void fr_dlist_sort(fr_dlist_head_t *list, fr_cmp_t cmp)
Sort a dlist using merge sort.
Definition dlist.h:1046
#define fr_dlist_init(_head, _type, _field)
Initialise the head structure of a doubly linked list.
Definition dlist.h:242
static void * fr_dlist_head(fr_dlist_head_t const *list_head)
Return the HEAD item of a list or NULL if the list is empty.
Definition dlist.h:468
static unsigned int fr_dlist_num_elements(fr_dlist_head_t const *head)
Return the number of elements in the dlist.
Definition dlist.h:921
static void * fr_dlist_tail(fr_dlist_head_t const *list_head)
Return the TAIL item of a list or NULL if the list is empty.
Definition dlist.h:513
static int fr_dlist_insert_tail(fr_dlist_head_t *list_head, void *ptr)
Insert an item into the tail of a list.
Definition dlist.h:360
#define fr_dlist_talloc_init(_head, _type, _field)
Initialise the head structure of a doubly linked list.
Definition dlist.h:257
static void * fr_dlist_next(fr_dlist_head_t const *list_head, void const *ptr)
Get the next item in a list.
Definition dlist.h:537
Head of a doubly linked list.
Definition dlist.h:51
exfile_t * exfile_init(TALLOC_CTX *ctx, uint32_t max_entries, fr_time_delta_t max_idle, bool locking)
Initialize a way for multiple threads to log to one or more files.
Definition exfile.c:170
void exfile_enable_triggers(exfile_t *ef, CONF_SECTION *conf, char const *trigger_prefix, fr_pair_list_t *trigger_args)
Enable triggers for an exfiles handle.
Definition exfile.c:218
int global_lib_instantiate(void)
Walk the tree of libraries and instantiate any which are pending.
Definition global_lib.c:218
talloc_free(hp)
#define PERROR(_fmt,...)
Definition log.h:228
#define DEBUG4(_fmt,...)
Definition log.h:267
Stores all information relating to an event list.
Definition event.c:377
static char const * mod_name(fr_listen_t *li)
Definition master.c:2973
fr_type_t
unsigned int uint32_t
size_t fr_sbuff_out_bstrncpy_until(fr_sbuff_t *out, fr_sbuff_t *in, size_t len, fr_sbuff_term_t const *tt, fr_sbuff_unescape_rules_t const *u_rules)
ssize_t fr_slen_t
fr_cmd_table_t module_cmd_list_table[]
Definition module.c:91
module_instance_t * mi
Instance of the module being instantiated.
Definition module_ctx.h:51
Temporary structure to hold arguments for instantiation calls.
Definition module_ctx.h:50
static module_list_t * rlm_modules_dynamic
Runtime instantiated list.
Definition module_rlm.c:77
static int _modules_rlm_free_atexit(UNUSED void *uctx)
void modules_rlm_thread_detach(void)
Frees thread-specific data for all registered backend modules.
Definition module_rlm.c:945
int modules_rlm_bootstrap(CONF_SECTION *root)
Bootstrap modules and virtual modules.
static int module_rlm_bootstrap_virtual(CONF_SECTION *cs)
Create a virtual module.
Definition module_rlm.c:822
static int8_t binding_name_cmp(void const *one, void const *two)
Compare the section names of two module_method_binding_t structures.
Definition module_rlm.c:988
static module_method_binding_t const * module_binding_find(module_method_group_t const *mmg, section_name_t const *section)
Iterate over an array of named module methods, looking for matches.
Definition module_rlm.c:443
fr_slen_t module_rlm_by_name_and_method(TALLOC_CTX *ctx, module_method_call_t *mmc_out, virtual_server_t const *vs, section_name_t const *section, fr_sbuff_t *name, tmpl_rules_t const *t_rules)
Find an existing module instance and verify it implements the specified method.
Definition module_rlm.c:548
exfile_t * module_rlm_exfile_init(TALLOC_CTX *ctx, CONF_SECTION *module, uint32_t max_entries, fr_time_delta_t max_idle, bool locking, bool triggers, char const *trigger_prefix, fr_pair_list_t *trigger_args)
Initialise a module specific exfile handle.
Definition module_rlm.c:103
xlat_t * module_rlm_xlat_register(TALLOC_CTX *ctx, module_inst_ctx_t const *mctx, char const *name, xlat_func_t func, fr_type_t return_type)
Definition module_rlm.c:232
module_instance_t * module_rlm_dynamic_by_name(module_instance_t const *parent, char const *asked_name)
Definition module_rlm.c:805
char const * name
module name
Definition module_rlm.c:52
fr_rb_node_t name_node
Entry in the name tree.
Definition module_rlm.c:51
int module_rlm_sibling_section_find(CONF_SECTION **out, CONF_SECTION *module, char const *name)
Resolve polymorphic item's from a module's CONF_SECTION to a subsection in another module.
Definition module_rlm.c:158
static int8_t module_rlm_virtual_name_cmp(void const *one, void const *two)
Compare virtual modules by name.
Definition module_rlm.c:59
void module_rlm_list_debug(void)
Print information on all loaded modules.
Definition module_rlm.c:82
static int module_method_validate(module_instance_t *mi)
static fr_rb_tree_t * module_rlm_virtual_name_tree
Lookup virtual module by name.
Definition module_rlm.c:48
int module_rlm_submodule_parse(TALLOC_CTX *ctx, void *out, void *parent, CONF_ITEM *ci, conf_parser_t const *rule)
Generic conf_parser_t func for loading drivers.
Definition module_rlm.c:932
module_instance_t * module_rlm_static_by_name(module_instance_t const *parent, char const *asked_name)
Definition module_rlm.c:810
int modules_rlm_coord_attach(fr_event_list_t *el)
Runs the coord_attach method of all registered backend modules.
Definition module_rlm.c:970
bool module_rlm_section_type_set(request_t *request, fr_dict_attr_t const *type_da, fr_dict_enum_value_t const *enumv)
Set the next section type if it's not already set.
Definition module_rlm.c:403
fr_pool_t * module_rlm_connection_pool_init(CONF_SECTION *module, void *opaque, fr_pool_connection_create_t c, fr_pool_connection_alive_t a, char const *log_prefix, char const *trigger_prefix, fr_pair_list_t *trigger_args)
Initialise a module specific connection pool.
Definition module_rlm.c:283
static int module_conf_parse(module_list_t *ml, CONF_SECTION *mod_conf)
static void module_rlm_methods_to_strerror(module_method_group_t const *mmg)
Dump the available bindings for the module into the strerror stack.
Definition module_rlm.c:496
int modules_rlm_free(void)
Cleanup all global structures.
#define FIND_SIBLING_CF_KEY
int modules_rlm_thread_instantiate(TALLOC_CTX *ctx, fr_event_list_t *el)
Allocates thread-specific data for all registered backend modules.
Definition module_rlm.c:958
static module_instance_t * module_rlm_instance_alloc(module_list_t *ml, module_instance_t const *parent, dl_module_type_t type, char const *mod_name, char const *inst_name, module_instance_state_t init_state)
Allocate a rlm module instance.
#define parent_name(_x)
CONF_SECTION * module_rlm_virtual_by_name(char const *asked_name)
Definition module_rlm.c:792
static module_list_t * rlm_modules_static
Global module list for all backend modules.
Definition module_rlm.c:72
int modules_rlm_instantiate(void)
Performs the instantiation phase for all backend modules.
Definition module_rlm.c:981
int modules_rlm_init(void)
Initialise the module list structure.
static int module_method_group_validate(module_method_group_t *group)
Definition module_rlm.c:996
CONF_SECTION * cs
CONF_SECTION where it is defined.
Definition module_rlm.c:53
module_method_group_t method_group
named methods
Definition module_rlm.h:40
module_instance_t * mi
The module instance that registered the xlat.
Definition module_rlm.h:54
fr_dlist_head_t xlats
xlats registered to this module instance.
Definition module_rlm.h:44
module_instance_t * mi
The process modules also push module calls onto the stack for execution.
Definition module_rlm.h:63
xlat_t const * xlat
The xlat function.
Definition module_rlm.h:53
tmpl_t * key
Dynamic key, only set for dynamic modules.
Definition module_rlm.h:71
section_name_t asked
The actual <name1>.
Definition module_rlm.h:67
module_method_binding_t mmb
Method we're calling.
Definition module_rlm.h:70
module_rlm_t const * rlm
Cached module_rlm_t.
Definition module_rlm.h:66
static module_rlm_t * module_rlm_from_module(module_t *module)
Definition module_rlm.h:74
The output of module_rlm_by_name_and_method.
Definition module_rlm.h:62
An xlat function registered to a module.
Definition module_rlm.h:52
int fr_pool_start(fr_pool_t *pool)
Definition pool.c:1119
void fr_pool_ref(fr_pool_t *pool)
Increment pool reference by one.
Definition pool.c:1215
fr_pool_t * fr_pool_init(TALLOC_CTX *ctx, CONF_SECTION const *cs, void *opaque, fr_pool_connection_create_t c, fr_pool_connection_alive_t a, char const *log_prefix)
Create a new connection pool.
Definition pool.c:970
void fr_pool_enable_triggers(fr_pool_t *pool, char const *trigger_prefix, fr_pair_list_t *trigger_args)
Enable triggers for a connection pool.
Definition pool.c:936
A connection pool.
Definition pool.c:85
void *(* fr_pool_connection_create_t)(TALLOC_CTX *ctx, void *opaque, fr_time_delta_t timeout)
Create a new connection handle.
Definition pool.h:111
int(* fr_pool_connection_alive_t)(void *opaque, void *connection)
Check a connection handle is still viable.
Definition pool.h:126
#define fr_assert(_expr)
Definition rad_assert.h:37
#define RDEBUG2(fmt,...)
#define DEBUG2(fmt,...)
#define WARN(fmt,...)
void * fr_rb_iter_init_inorder(fr_rb_tree_t *tree, fr_rb_iter_inorder_t *iter)
Initialise an in-order iterator.
Definition rb.c:824
void * fr_rb_find(fr_rb_tree_t const *tree, void const *data)
Find an element in the tree, returning the data, not the node.
Definition rb.c:577
bool fr_rb_insert(fr_rb_tree_t *tree, void const *data)
Insert data into a tree.
Definition rb.c:626
void * fr_rb_iter_next_inorder(UNUSED fr_rb_tree_t *tree, fr_rb_iter_inorder_t *iter)
Return the next node.
Definition rb.c:850
#define fr_rb_inline_alloc(_ctx, _type, _field, _data_cmp, _data_free)
Allocs a red black tree.
Definition rb.h:269
Iterator structure for in-order traversal of an rbtree.
Definition rb.h:319
The main red black tree structure.
Definition rb.h:71
static char const * name
size_t fr_sbuff_adv_until(fr_sbuff_t *sbuff, size_t len, fr_sbuff_term_t const *tt, char escape_chr)
Wind position until we hit a character in the terminal set.
Definition sbuff.c:1893
size_t fr_sbuff_out_bstrncpy(fr_sbuff_t *out, fr_sbuff_t *in, size_t len)
Copy as many bytes as possible from a sbuff to a sbuff.
Definition sbuff.c:736
bool fr_sbuff_next_if_char(fr_sbuff_t *sbuff, char c)
Return true if the current char matches, and if it does, advance.
Definition sbuff.c:2129
#define fr_sbuff_set(_dst, _src)
#define FR_SBUFF_TERMS(...)
Initialise a terminal structure with a list of sorted strings.
Definition sbuff.h:190
#define fr_sbuff_is_char(_sbuff_or_marker, _c)
#define fr_sbuff_error(_sbuff_or_marker)
#define FR_SBUFF(_sbuff_or_marker)
#define fr_sbuff_advance(_sbuff_or_marker, _len)
#define fr_sbuff_used(_sbuff_or_marker)
#define fr_sbuff_ahead(_sbuff_or_marker)
#define FR_SBUFF_TALLOC_THREAD_LOCAL(_out, _init, _max)
Set of terminal elements.
int8_t section_name_cmp(void const *one, void const *two)
Compare two sections.
Definition section.c:47
static int section_name2_match(section_name_t const *a, section_name_t const *b)
Definition section.h:59
static char const * section_name_str(char const *name)
Return a printable string for the section name.
Definition section.h:97
static int section_name_match(section_name_t const *a, section_name_t const *b)
Definition section.h:83
static void section_name_dup(TALLOC_CTX *ctx, section_name_t *dst, section_name_t const *src)
Definition section.h:104
char const * name2
Second section name. Usually a packet type like 'access-request', 'access-accept',...
Definition section.h:45
char const * name1
First section name. Usually a verb like 'recv', 'send', etc...
Definition section.h:44
Section name identifier.
Definition section.h:43
char const * name
Instance name e.g. user_database.
Definition module.h:357
@ MODULE_TYPE_RETRY
can handle retries
Definition module.h:51
module_flags_t flags
Flags that control how a module starts up and how a module is called.
Definition module.h:236
module_method_group_t * next
Next group in the list.
Definition module.h:168
CONF_SECTION * conf
Module's instance configuration.
Definition module.h:351
module_instance_state_t state
What's been done with this module so far.
Definition module.h:350
unlang_mod_actions_t actions
default actions and retries.
Definition module.h:325
bool validated
Set to true by module_method_group_validate.
Definition module.h:167
#define MODULE_INSTANCE_LEN_MAX
The maximum size of a module instance.
Definition module.h:148
fr_dlist_head_t same_name1
List of bindings with the same name1.
Definition module.h:187
module_instance_state_t
What state the module instance is currently in.
Definition module.h:265
@ MODULE_INSTANCE_INSTANTIATED
Module instance has been bootstrapped and instantiated.
Definition module.h:268
void * uctx
Extra data passed to module_instance_alloc.
Definition module.h:361
module_method_binding_t * bindings
named methods
Definition module.h:165
section_name_t const * section
Identifier for a section.
Definition module.h:175
module_t * exported
Public module structure.
Definition module.h:298
Module instance data.
Definition module.h:287
A list of modules.
Definition module.h:407
Named methods exported by a module.
Definition module.h:174
A group of methods exported by a module or added as an overlay.
Definition module.h:164
Struct exported by a rlm_* module.
Definition module.h:203
#define pair_update_control(_attr, _da)
Return or allocate a fr_pair_t in the control list.
Definition pair.h:140
ssize_t tmpl_afrom_substr(TALLOC_CTX *ctx, tmpl_t **out, fr_sbuff_t *in, fr_token_t quote, fr_sbuff_parse_rules_t const *p_rules, tmpl_rules_t const *t_rules))
Convert an arbitrary string into a tmpl_t.
Optional arguments passed to vp_tmpl functions.
Definition tmpl.h:336
PUBLIC int snprintf(char *string, size_t length, char *format, va_alist)
Definition snprintf.c:689
void module_list_debug(module_list_t const *ml)
Print the contents of a module list.
Definition module.c:620
module_list_type_t const module_list_type_thread_local
Callbacks for a thread local list.
Definition module.c:587
module_instance_t * module_instance_by_name(module_list_t const *ml, module_instance_t const *parent, char const *asked_name)
Find an existing module instance by its name and parent.
Definition module.c:902
void module_list_mask_set(module_list_t *ml, module_instance_state_t mask)
Set a new bootstrap/instantiate state for a list.
Definition module.c:1864
module_instance_t * module_instance_alloc(module_list_t *ml, module_instance_t const *parent, dl_module_type_t type, char const *mod_name, char const *inst_name, module_instance_state_t init_state)
Allocate a new module and add it to a module list for later bootstrap/instantiation.
Definition module.c:1680
fr_slen_t module_instance_name_from_conf(char const **name, CONF_SECTION *conf)
Avoid boilerplate when setting the module instance name.
Definition module.c:734
void modules_thread_detach(module_list_t *ml)
Remove thread-specific data for a given module list.
Definition module.c:1008
int modules_thread_instantiate(TALLOC_CTX *ctx, module_list_t const *ml, fr_event_list_t *el)
Creates per-thread instance data for modules which need it.
Definition module.c:1159
int modules_instantiate(module_list_t const *ml)
Completes instantiation of modules.
Definition module.c:1310
module_list_t * module_list_alloc(TALLOC_CTX *ctx, module_list_type_t const *type, char const *name, bool write_protect)
Allocate a new module list.
Definition module.c:1886
void module_instance_uctx_set(module_instance_t *mi, void *uctx)
Set the uctx pointer for a module instance.
Definition module.c:1648
int modules_coord_attach(module_list_t const *ml, fr_event_list_t *el)
Call the coordinator attach callback for any modules using a coordinator.
Definition module.c:1195
module_list_type_t const module_list_type_global
Callbacks for a global module list.
Definition module.c:533
int modules_bootstrap(module_list_t const *ml)
Bootstrap any modules which have not been bootstrapped already.
Definition module.c:1401
int module_instantiate(module_instance_t *instance)
Manually complete module setup by calling its instantiate function.
Definition module.c:1224
int module_submodule_parse(UNUSED TALLOC_CTX *ctx, void *out, void *parent, CONF_ITEM *ci, conf_parser_t const *rule)
Generic callback for conf_parser_t to load a submodule.
Definition module.c:846
int module_instance_conf_parse(module_instance_t *mi, CONF_SECTION *conf)
Covert a CONF_SECTION into parsed module instance data.
Definition module.c:763
eap_aka_sim_process_conf_t * inst
fr_aka_sim_id_type_t type
fr_pair_t * vp
Stores an attribute, a value and various bits of other data.
Definition pair.h:68
fr_dict_attr_t const *_CONST da
Dictionary attribute defines the attribute number, vendor and type of the pair.
Definition pair.h:69
#define talloc_strdup(_ctx, _str)
Definition talloc.h:142
A time delta, a difference in time measured in nanoseconds.
Definition time.h:80
@ T_BARE_WORD
Definition token.h:118
static fr_event_list_t * el
xlat_action_t(* xlat_func_t)(TALLOC_CTX *ctx, fr_dcursor_t *out, xlat_ctx_t const *xctx, request_t *request, fr_value_box_list_t *in)
xlat callback function
Definition xlat.h:232
static fr_slen_t parent
Definition pair.h:858
#define fr_strerror_printf(_fmt,...)
Log to thread local error buffer.
Definition strerror.h:64
#define fr_strerror_printf_push(_fmt,...)
Add a message to an existing stack of messages at the tail.
Definition strerror.h:84
#define fr_strerror_const_push(_msg)
Definition strerror.h:227
#define fr_strerror_const(_msg)
Definition strerror.h:223
int fr_value_box_copy(TALLOC_CTX *ctx, fr_value_box_t *dst, const fr_value_box_t *src)
Copy value data verbatim duplicating any buffers.
Definition value.c:4379
#define fr_box_strvalue_len(_val, _len)
Definition value.h:309
int nonnull(2, 5))
static size_t char ** out
Definition value.h:1030
section_name_t const ** virtual_server_section_methods(virtual_server_t const *vs, section_name_t const *section)
Find the component for a section.
xlat_t * xlat_func_register(TALLOC_CTX *ctx, char const *name, xlat_func_t func, fr_type_t return_type)
Register an xlat function.
Definition xlat_func.c:216
void xlat_mctx_set(xlat_t *x, module_inst_ctx_t const *mctx)
Associate a module calling ctx with the xlat.
Definition xlat_func.c:296
int xlat_register_redundant(CONF_SECTION *cs)
Registers a redundant xlat.