The FreeRADIUS server $Id: 15bac2a4c627c01d1aa2047687b3418955ac7f00 $
Loading...
Searching...
No Matches
foreach.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: 71b073199db1c522927674481f4539b75a92063c $
19 *
20 * @file unlang/foreach.c
21 * @brief Unlang "foreach" keyword evaluation.
22 *
23 * @copyright 2006-2019 The FreeRADIUS server project
24 */
25RCSID("$Id: 71b073199db1c522927674481f4539b75a92063c $")
26
27#include <freeradius-devel/server/tmpl_dcursor.h>
28#include <freeradius-devel/unlang/xlat_func.h>
29
30#include "foreach_priv.h"
31#include "return_priv.h"
32#include "xlat_priv.h"
33
34#define BUFFER_SIZE (256)
35
36/** State of a foreach loop
37 *
38 */
39typedef struct {
40 request_t *request; //!< The current request.
41 fr_dcursor_t cursor; //!< Used to track our place in the list
42 fr_pair_t *key; //!< local variable which contains the key
43 fr_pair_t *value; //!< local variable which contains the value
44 tmpl_t const *vpt; //!< pointer to the vpt
45
46 uint32_t index; //!< for xlat results
47 char *buffer; //!< for key values
48
49 bool success; //!< for xlat expansion
50 fr_value_box_list_t list; //!< value box list for looping over xlats
51
52 tmpl_dcursor_ctx_t cc; //!< tmpl cursor state
53
54#ifndef NDEBUG
55 int indent; //!< for catching indentation issues
56#endif
58
59/*
60 * Brute-force things instead of doing it the "right" way.
61 *
62 * We would ideally like to have the local variable be a ref to the current vp from the cursor. However,
63 * that isn't (yet) supported. In order to support that, we would likely have to add a new data type
64 * FR_TYPE_DCURSOR, and put the cursor into in vp->vp_ptr. We would then have to update a lot of things:
65 *
66 * - the foreach code has to put the dcursor into state->value->vp_ptr.
67 * - the pair code (all of it, perhaps) has to check for "is this thing a cursor), and if so
68 * return the next pair from the cursor instead of the given pair. This is a huge change.
69 * - update all of the pair / value-box APIs to handle the new data type
70 * - check performance, etc, and that nothing else breaks.
71 * - also need to ensure that the pair with the cursor _cannot_ be copied, as that would add two
72 * refs to the cursor.
73 * - if we're lucky, we could perhaps _instead_ update only the tmpl code, but the cursor
74 * still has to be in the pair.
75 * - we can update tmpl_eval_pair(), because that's what's used in the xlat code. That gets us all
76 * references to the _source_ VP.
77 * - we also have to update the edit.c code, which calls tmpl_dcursor_init() to get pairs from
78 * a tmpl_t of type ATTR.
79 * - for LHS assignment, the edit code has to be updated: apply_edits_to_leaf() and apply_edits_to_list()
80 * which calls fr_edit_list_apply_pair_assignment() to do the actual work. But we could likely just
81 * check current->lhs.vp, and dereference that to get the underlying thing.
82 *
83 * What we ACTUALLY do instead is in the compiler when we call define_local_variable(), we clone the "da"
84 * hierarchy via fr_dict_attr_acopy_local(). That function which should go away when we add refs.
85 *
86 * Then this horrific function copies the pairs by number, which re-parents them to the correct
87 * destination da. It's brute-force and expensive, but it's easy. And for now, it's less work than
88 * re-doing substantial parts of the server core and utility libraries.
89 */
90static int unlang_foreach_pair_copy(fr_pair_t *to, fr_pair_t *from, fr_dict_attr_t const *from_parent)
91{
92 fr_assert(fr_type_is_structural(to->vp_type));
93 fr_assert(fr_type_is_structural(from->vp_type));
94
95 fr_pair_list_foreach(&from->vp_group, vp) {
96 fr_pair_t *child;
97
98 /*
99 * We only copy children of the parent TLV, but we can copy internal attributes, as they
100 * can exist anywhere.
101 */
102 if (vp->da->parent != from_parent) {
103 if (vp->da->flags.internal) {
104 child = fr_pair_copy(to, vp);
105 if (child) fr_pair_append(&to->vp_group, child);
106 }
107 continue;
108 }
109
110 child = fr_pair_afrom_child_num(to, to->da, vp->da->attr);
111 if (!child) continue;
112
113 fr_pair_append(&to->vp_group, child);
114
115 if (fr_type_is_leaf(child->vp_type)) {
116 if (fr_value_box_copy(child, &child->data, &vp->data) < 0) return -1;
117 continue;
118 }
119
120 if (unlang_foreach_pair_copy(child, vp, vp->da) < 0) return -1;
121 }
122
123 return 0;
124}
125
126/** Ensure request data is pulled out of the request if the frame is popped
127 *
128 */
130{
131 request_t *request = state->request;
132 fr_pair_t *vp;
133
134 fr_assert(state->value);
135
136 if (tmpl_is_xlat(state->vpt)) return 0;
137
138 tmpl_dcursor_clear(&state->cc);
139
140 /*
141 * Now that we're done, the leaf entries can be changed again.
142 */
143 vp = tmpl_dcursor_init(NULL, NULL, &state->cc, &state->cursor, request, state->vpt);
144 if (!vp) {
145 tmpl_dcursor_clear(&state->cc);
146 return 0;
147 }
148 do {
149 vp->vp_edit = false;
150 } while ((vp = fr_dcursor_next(&state->cursor)) != NULL);
151 tmpl_dcursor_clear(&state->cc);
152
153 return 0;
154}
155
157{
158 fr_value_box_t box;
159
160 if (!state->key) return 0;
161
162 fr_value_box_clear_value(&state->key->data);
163
164 fr_value_box(&box, state->index, false);
165
166 if (fr_value_box_cast(state->key, &state->key->data, state->key->vp_type, state->key->da, &box) < 0) {
167 RDEBUG("Failed casting 'foreach' key variable '%s' from %u", state->key->da->name, state->index);
168 return -1;
169 }
170
171 return 0;
172}
173
174
176{
177 unlang_frame_state_foreach_t *state = talloc_get_type_abort(frame->state, unlang_frame_state_foreach_t);
178 fr_value_box_t *box;
179
180next:
181 state->index++;
182
183 box = fr_dcursor_next(&state->cursor);
184 if (!box) {
185 *p_result = frame->result;
187 }
188
189 if (unlang_foreach_xlat_key_update(request, state) < 0) goto next;
190
191 fr_value_box_clear_value(&state->value->data);
192 if (fr_value_box_cast(state->value, &state->value->data, state->value->vp_type, state->value->da, box) < 0) {
193 RDEBUG("Failed casting 'foreach' iteration variable '%s' from %pV", state->value->da->name, box);
194 goto next;
195 }
196
197 repeatable_set(frame);
198
199 /*
200 * Push the child, and yield for a later return.
201 */
202 return unlang_interpret_push_children(p_result, request, frame->result, UNLANG_NEXT_SIBLING);
203}
204
205
207{
208 unlang_frame_state_foreach_t *state = talloc_get_type_abort(frame->state, unlang_frame_state_foreach_t);
209 fr_value_box_t *box;
210
211 if (!state->success) {
212 RDEBUG("Failed expanding 'foreach' list");
213 *p_result = RLM_MODULE_FAIL;
215 }
216
217 box = fr_dcursor_init(&state->cursor, fr_value_box_list_dlist_head(&state->list));
218 if (!box) {
219 done:
220 *p_result = RLM_MODULE_NOOP;
222 }
223
224 fr_value_box_clear_value(&state->value->data);
225
226next:
227 if (fr_value_box_cast(state->value, &state->value->data, state->value->vp_type, state->value->da, box) < 0) {
228 RDEBUG("Failed casting 'foreach' iteration variable '%s' from %pV", state->value->da->name, box);
229 box = fr_dcursor_next(&state->cursor);
230 if (!box) goto done;
231
232 goto next;
233 }
234
236 repeatable_set(frame);
237
238 /*
239 * Push the child, and yield for a later return.
240 */
241 return unlang_interpret_push_children(p_result, request, frame->result, UNLANG_NEXT_SIBLING);
242}
243
244
245/*
246 * Loop over an xlat expansion
247 */
250{
251 fr_value_box_list_init(&state->list);
252
253 if (unlang_xlat_push(state, &state->success, &state->list, request, tmpl_xlat(state->vpt), false) < 0) {
254 REDEBUG("Failed starting expansion of %s", state->vpt->name);
255 *p_result = RLM_MODULE_FAIL;
257 }
258
259 if (unlang_foreach_xlat_key_update(request, state) < 0) {
260 *p_result = RLM_MODULE_FAIL;
262 }
263
265 repeatable_set(frame);
266
268}
269
271{
272 if (!state->key) return;
273
274 switch (state->key->vp_type) {
275 case FR_TYPE_UINT32:
276 state->key->vp_uint32++;
277 break;
278
279 case FR_TYPE_STRING:
280 fr_value_box_clear_value(&state->key->data);
281 if (tmpl_dcursor_print(&FR_SBUFF_IN(state->buffer, BUFFER_SIZE), &state->cc) > 0) {
282 fr_value_box_strdup(state->key, &state->key->data, NULL, state->buffer, false);
283 }
284 break;
285
286 default:
287 fr_assert(0);
288 break;
289
290 }
291}
292
294{
295 unlang_frame_state_foreach_t *state = talloc_get_type_abort(frame->state, unlang_frame_state_foreach_t);
296 fr_pair_t *vp;
297
299
300 vp = fr_dcursor_current(&state->cursor);
301 fr_assert(vp != NULL);
302
303 /*
304 * If we modified the value, copy it back to the original pair. Note that the copy does NOT
305 * check the "immutable" flag. That flag is for the people using unlang, not for the
306 * interpreter.
307 */
308 if (fr_type_is_leaf(vp->vp_type)) {
309 if (vp->vp_type == state->value->vp_type) {
311 (void) fr_value_box_copy(vp, &vp->data, &state->value->data);
312 } else {
313 /*
314 * @todo - this shouldn't happen?
315 */
316 }
317 } else {
318 /*
319 * @todo - copy the pairs back?
320 */
321 }
322
323next:
324 vp = fr_dcursor_next(&state->cursor);
325 if (!vp) {
326 *p_result = frame->result;
327#ifndef NDEBUG
328 fr_assert(state->indent == request->log.indent.unlang);
329#endif
331 }
332
333 unlang_foreach_attr_key_update(request, state);
334
335 /*
336 * Copy the data.
337 */
338 if (vp->vp_type == FR_TYPE_GROUP) {
339 fr_assert(state->value->vp_type == FR_TYPE_GROUP);
340
341 fr_pair_list_free(&state->value->vp_group);
342
343 if (fr_pair_list_copy(state->value, &state->value->vp_group, &vp->vp_group) < 0) {
344 REDEBUG("Failed copying members of %s", state->value->da->name);
345 *p_result = RLM_MODULE_FAIL;
347 }
348
349 } else if (fr_type_is_structural(vp->vp_type)) {
350 if (state->value->vp_type != vp->vp_type) goto next;
351
352 fr_pair_list_free(&state->value->vp_group);
353
354 if (unlang_foreach_pair_copy(state->value, vp, vp->da) < 0) {
355 REDEBUG("Failed copying children of %s", state->value->da->name);
356 *p_result = RLM_MODULE_FAIL;
358 }
359
360 } else {
361 fr_value_box_clear_value(&state->value->data);
362 if (fr_value_box_cast(state->value, &state->value->data, state->value->vp_type, state->value->da, &vp->data) < 0) {
363 RDEBUG("Failed casting 'foreach' iteration variable '%s' from %pP", state->value->da->name, vp);
364 goto next;
365 }
366
367#ifndef NDEBUG
368 RDEBUG2("# looping with: %s = %pV", state->value->da->name, &vp->data);
369#endif
370 }
371
372 repeatable_set(frame);
373
374 /*
375 * Push the child, and yield for a later return.
376 */
377 return unlang_interpret_push_children(p_result, request, frame->result, UNLANG_NEXT_SIBLING);
378}
379
380/*
381 * Loop over an attribute
382 */
385{
386 fr_pair_t *vp;
387
388 /*
389 * No matching attributes, we can't do anything.
390 */
391 vp = tmpl_dcursor_init(NULL, NULL, &state->cc, &state->cursor, request, state->vpt);
392 if (!vp) {
393 tmpl_dcursor_clear(&state->cc);
394 *p_result = RLM_MODULE_NOOP;
396 }
397
398 /*
399 * Before we loop over the variables, ensure that the user can't pull the rug out from
400 * under us.
401 */
402 do {
403 if (vp->vp_edit) {
404 REDEBUG("Cannot do nested 'foreach' loops over the same attribute %pP", vp);
405 fail:
406 tmpl_dcursor_clear(&state->cc);
407 *p_result = RLM_MODULE_FAIL;
409 }
410
411 vp->vp_edit = true;
412 } while ((vp = fr_dcursor_next(&state->cursor)) != NULL);
413 tmpl_dcursor_clear(&state->cc);
414
415 vp = tmpl_dcursor_init(NULL, NULL, &state->cc, &state->cursor, request, state->vpt);
416 fr_assert(vp != NULL);
417
418next:
419 /*
420 * Update the key with the current path. Attribute indexes start at zero.
421 */
422 if (state->key && (state->key->vp_type == FR_TYPE_STRING)) unlang_foreach_attr_key_update(request, state);
423
424 if (vp->vp_type == FR_TYPE_GROUP) {
425 fr_assert(state->value->vp_type == FR_TYPE_GROUP);
426
427 if (fr_pair_list_copy(state->value, &state->value->vp_group, &vp->vp_group) < 0) {
428 REDEBUG("Failed copying members of %s", state->value->da->name);
429 goto fail;
430 }
431
432 } else if (fr_type_is_structural(vp->vp_type)) {
433 if (state->value->vp_type != vp->vp_type) {
434 vp = fr_dcursor_next(&state->cursor);
435 if (vp) goto next;
436
437 *p_result = frame->result;
438 fr_assert(state->indent == request->log.indent.unlang);
440 }
441
442 if (unlang_foreach_pair_copy(state->value, vp, vp->da) < 0) {
443 REDEBUG("Failed copying children of %s", state->value->da->name);
444 goto fail;
445 }
446
447 } else {
448 fr_value_box_clear_value(&state->value->data);
449 while (vp && (fr_value_box_cast(state->value, &state->value->data, state->value->vp_type, state->value->da, &vp->data) < 0)) {
450 RDEBUG("Failed casting 'foreach' iteration variable '%s' from %pP", state->value->da->name, vp);
451 vp = fr_dcursor_next(&state->cursor);
452 }
453
454 /*
455 * Couldn't cast anything, the loop can't be run.
456 */
457 if (!vp) {
458 tmpl_dcursor_clear(&state->cc);
459 *p_result = RLM_MODULE_NOOP;
461 }
462 }
463
465
466 repeatable_set(frame);
467
468 /*
469 * Push the child, and go process it.
470 */
471 return unlang_interpret_push_children(p_result, request, frame->result, UNLANG_NEXT_SIBLING);
472}
473
474
476{
480
481 /*
482 * Ensure any breaks terminate here...
483 */
484 break_point_set(frame);
485
486 MEM(frame->state = state = talloc_zero(request->stack, unlang_frame_state_foreach_t));
487 talloc_set_destructor(state, _free_unlang_frame_state_foreach);
488
489 state->request = request;
490#ifndef NDEBUG
491 state->indent = request->log.indent.unlang;
492#endif
493
494 /*
495 * Get the value.
496 */
497 fr_assert(gext->value);
498
499 state->vpt = gext->vpt;
500
501 fr_assert(fr_pair_find_by_da(&request->local_pairs, NULL, gext->value) == NULL);
502
503 /*
504 * Create the local variable and populate its value.
505 */
506 if (fr_pair_append_by_da(request->local_ctx, &state->value, &request->local_pairs, gext->value) < 0) {
507 REDEBUG("Failed creating %s", gext->value->name);
508 *p_result = RLM_MODULE_FAIL;
510 }
511 fr_assert(state->value != NULL);
512
513 if (gext->key) {
514 fr_assert(fr_pair_find_by_da(&request->local_pairs, NULL, gext->key) == NULL);
515
516 if (fr_pair_append_by_da(request->local_ctx, &state->key, &request->local_pairs, gext->key) < 0) {
517 REDEBUG("Failed creating %s", gext->key->name);
518 *p_result = RLM_MODULE_FAIL;
520 }
521 fr_assert(state->key != NULL);
522 }
523
524 if (tmpl_is_attr(gext->vpt)) {
525 MEM(state->buffer = talloc_array(state, char, BUFFER_SIZE));
526 return unlang_foreach_attr_init(p_result, request, frame, state);
527 }
528
529 fr_assert(tmpl_is_xlat(gext->vpt));
530
531 return unlang_foreach_xlat_init(p_result, request, frame, state);
532}
533
535{
536 RDEBUG2("%s", unlang_ops[frame->instruction->type].name);
537
538 *p_result = frame->result;
539
540 /*
541 * Stop at the next break point, or if we hit
542 * the a top frame.
543 */
544 return unwind_to_break(request->stack);
545}
546
548{
550 &(unlang_op_t){
551 .name = "foreach",
552 .interpret = unlang_foreach,
553 .debug_braces = true
554 });
555
557 &(unlang_op_t){
558 .name = "break",
559 .interpret = unlang_break,
560 });
561}
unlang_action_t
Returned by unlang_op_t calls, determine the next action of the interpreter.
Definition action.h:35
@ UNLANG_ACTION_PUSHED_CHILD
unlang_t pushed a new child onto the stack, execute it instead of continuing.
Definition action.h:39
@ UNLANG_ACTION_CALCULATE_RESULT
Calculate a new section rlm_rcode_t value.
Definition action.h:37
#define RCSID(id)
Definition build.h:485
#define UNUSED
Definition build.h:317
static void * fr_dcursor_next(fr_dcursor_t *cursor)
Advanced the cursor to the next item.
Definition dcursor.h:288
#define fr_dcursor_init(_cursor, _head)
Initialise a cursor.
Definition dcursor.h:732
static void * fr_dcursor_current(fr_dcursor_t *cursor)
Return the item the cursor current points to.
Definition dcursor.h:337
#define MEM(x)
Definition debug.h:36
static int unlang_foreach_pair_copy(fr_pair_t *to, fr_pair_t *from, fr_dict_attr_t const *from_parent)
Definition foreach.c:90
uint32_t index
for xlat results
Definition foreach.c:46
static void unlang_foreach_attr_key_update(UNUSED request_t *request, unlang_frame_state_foreach_t *state)
Definition foreach.c:270
static unlang_action_t unlang_foreach_attr_next(rlm_rcode_t *p_result, request_t *request, unlang_stack_frame_t *frame)
Definition foreach.c:293
char * buffer
for key values
Definition foreach.c:47
fr_dcursor_t cursor
Used to track our place in the list.
Definition foreach.c:41
tmpl_t const * vpt
pointer to the vpt
Definition foreach.c:44
fr_pair_t * key
local variable which contains the key
Definition foreach.c:42
static unlang_action_t unlang_foreach_attr_init(rlm_rcode_t *p_result, request_t *request, unlang_stack_frame_t *frame, unlang_frame_state_foreach_t *state)
Definition foreach.c:383
static unlang_action_t unlang_foreach_xlat_next(rlm_rcode_t *p_result, request_t *request, unlang_stack_frame_t *frame)
Definition foreach.c:175
bool success
for xlat expansion
Definition foreach.c:49
fr_value_box_list_t list
value box list for looping over xlats
Definition foreach.c:50
#define BUFFER_SIZE
Definition foreach.c:34
fr_pair_t * value
local variable which contains the value
Definition foreach.c:43
static unlang_action_t unlang_foreach(rlm_rcode_t *p_result, request_t *request, unlang_stack_frame_t *frame)
Definition foreach.c:475
static unlang_action_t unlang_break(rlm_rcode_t *p_result, request_t *request, unlang_stack_frame_t *frame)
Definition foreach.c:534
tmpl_dcursor_ctx_t cc
tmpl cursor state
Definition foreach.c:52
int indent
for catching indentation issues
Definition foreach.c:55
request_t * request
The current request.
Definition foreach.c:40
static unlang_action_t unlang_foreach_xlat_expanded(rlm_rcode_t *p_result, request_t *request, unlang_stack_frame_t *frame)
Definition foreach.c:206
static unlang_action_t unlang_foreach_xlat_init(rlm_rcode_t *p_result, request_t *request, unlang_stack_frame_t *frame, unlang_frame_state_foreach_t *state)
Definition foreach.c:248
void unlang_foreach_init(void)
Definition foreach.c:547
static int unlang_foreach_xlat_key_update(request_t *request, unlang_frame_state_foreach_t *state)
Definition foreach.c:156
static int _free_unlang_frame_state_foreach(unlang_frame_state_foreach_t *state)
Ensure request data is pulled out of the request if the frame is popped.
Definition foreach.c:129
State of a foreach loop.
Definition foreach.c:39
static unlang_foreach_t * unlang_group_to_foreach(unlang_group_t *g)
Cast a group structure to the foreach keyword extension.
fr_dict_attr_t const * value
value variable in the foreach loop
fr_dict_attr_t const * key
key variable for the foreach loop
unlang_action_t unlang_interpret_push_children(rlm_rcode_t *p_result, request_t *request, rlm_rcode_t default_rcode, bool do_next_sibling)
Push the children of the current frame onto a new frame onto the stack.
Definition interpret.c:254
void unlang_register(int type, unlang_op_t *op)
Register an operation with the interpreter.
Definition base.c:63
unlang_op_t unlang_ops[UNLANG_TYPE_MAX]
Different operations the interpreter can execute.
Definition base.c:31
@ FR_TYPE_STRING
String of printable characters.
@ FR_TYPE_UINT32
32 Bit unsigned integer.
@ FR_TYPE_GROUP
A grouping of other attributes.
unsigned int uint32_t
int fr_pair_append_by_da(TALLOC_CTX *ctx, fr_pair_t **out, fr_pair_list_t *list, fr_dict_attr_t const *da)
Alloc a new fr_pair_t (and append)
Definition pair.c:1468
int fr_pair_list_copy(TALLOC_CTX *ctx, fr_pair_list_t *to, fr_pair_list_t const *from)
Duplicate a list of pairs.
Definition pair.c:2321
fr_pair_t * fr_pair_find_by_da(fr_pair_list_t const *list, fr_pair_t const *prev, fr_dict_attr_t const *da)
Find the first pair with a matching da.
Definition pair.c:695
int fr_pair_append(fr_pair_list_t *list, fr_pair_t *to_add)
Add a VP to the end of the list.
Definition pair.c:1347
fr_pair_t * fr_pair_copy(TALLOC_CTX *ctx, fr_pair_t const *vp)
Copy a single valuepair.
Definition pair.c:491
fr_pair_t * fr_pair_afrom_child_num(TALLOC_CTX *ctx, fr_dict_attr_t const *parent, unsigned int attr)
Create a new valuepair.
Definition pair.c:373
#define fr_assert(_expr)
Definition rad_assert.h:38
static bool done
Definition radclient.c:80
#define REDEBUG(fmt,...)
Definition radclient.h:52
#define RDEBUG2(fmt,...)
Definition radclient.h:54
#define RDEBUG(fmt,...)
Definition radclient.h:53
rlm_rcode_t
Return codes indicating the result of the module call.
Definition rcode.h:40
@ RLM_MODULE_FAIL
Module failed, don't reply.
Definition rcode.h:42
@ RLM_MODULE_NOOP
Module succeeded without doing anything.
Definition rcode.h:48
Declarations for the "return" keyword, used to implement other keywords.
#define FR_SBUFF_IN(_start, _len_or_end)
#define tmpl_is_xlat(vpt)
Definition tmpl.h:215
#define tmpl_is_attr(vpt)
Definition tmpl.h:213
#define tmpl_xlat(_tmpl)
Definition tmpl.h:935
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
ssize_t tmpl_dcursor_print(fr_sbuff_t *out, tmpl_dcursor_ctx_t const *cc)
void tmpl_dcursor_clear(tmpl_dcursor_ctx_t *cc)
Clear any temporary state allocations.
#define tmpl_dcursor_init(_err, _ctx, _cc, _cursor, _request, _vpt)
Maintains state between cursor calls.
int unlang_xlat_push(TALLOC_CTX *ctx, bool *p_success, fr_value_box_list_t *out, request_t *request, xlat_exp_head_t const *xlat, bool top_frame)
Push a pre-compiled xlat onto the stack for evaluation.
Definition xlat.c:287
#define UNLANG_NEXT_SIBLING
Definition unlang_priv.h:93
void * state
Stack frame specialisations.
static unlang_group_t * unlang_generic_to_group(unlang_t const *p)
@ UNLANG_TYPE_BREAK
Break statement (within a UNLANG_TYPE_FOREACH).
Definition unlang_priv.h:60
@ UNLANG_TYPE_FOREACH
Foreach section.
Definition unlang_priv.h:59
unlang_t const * instruction
The unlang node we're evaluating.
static void break_point_set(unlang_stack_frame_t *frame)
rlm_rcode_t result
The result from executing the instruction.
char const * name
Name of the operation.
static bool is_stack_unwinding_to_break(unlang_stack_t *stack)
static void repeatable_set(unlang_stack_frame_t *frame)
static unlang_action_t unwind_to_break(unlang_stack_t *stack)
unlang_process_t process
function to call for interpreting this stack frame
unlang_type_t type
The specialisation of this node.
Generic representation of a grouping.
An unlang operation.
Our interpreter stack, as distinct from the C stack.
#define fr_pair_list_foreach(_list_head, _iter)
Iterate over the contents of a fr_pair_list_t.
Definition pair.h:261
void fr_pair_list_free(fr_pair_list_t *list)
Free memory used by a valuepair list.
#define fr_type_is_structural(_x)
Definition types.h:371
#define fr_type_is_leaf(_x)
Definition types.h:372
int fr_value_box_cast(TALLOC_CTX *ctx, fr_value_box_t *dst, fr_type_t dst_type, fr_dict_attr_t const *dst_enumv, fr_value_box_t const *src)
Convert one type of fr_value_box_t to another.
Definition value.c:3370
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:3759
void fr_value_box_clear_value(fr_value_box_t *data)
Clear/free any existing value.
Definition value.c:3700
int fr_value_box_strdup(TALLOC_CTX *ctx, fr_value_box_t *dst, fr_dict_attr_t const *enumv, char const *src, bool tainted)
Copy a nul terminated string to a fr_value_box_t.
Definition value.c:3957
#define fr_value_box(_box, _var, _tainted)
Automagically fill in a box, determining the value type from the type of the C variable.
Definition value.h:886
String expansion ("translation").