The FreeRADIUS server $Id: 15bac2a4c627c01d1aa2047687b3418955ac7f00 $
Loading...
Searching...
No Matches
xlat_purify.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: d8be73fd9d2e56a55c1595ffe77fb8fc86243829 $
19 *
20 * @file xlat_purify.c
21 * @brief Purification functions for xlats
22 *
23 * @copyright 2022 The FreeRADIUS server project
24 * @copyright 2022 Network RADIUS SAS (legal@networkradius.com)
25 */
26
27RCSID("$Id: d8be73fd9d2e56a55c1595ffe77fb8fc86243829 $")
28
29#include <freeradius-devel/server/base.h>
30#include <freeradius-devel/unlang/xlat_priv.h>
31#include <freeradius-devel/util/calc.h>
32#include <freeradius-devel/util/dict.h>
33
34static void xlat_value_list_to_xlat(xlat_exp_head_t *head, fr_value_box_list_t *list)
35{
36 fr_value_box_t *box;
37 xlat_exp_t *node;
38
39 while ((box = fr_value_box_list_pop_head(list)) != NULL) {
40 MEM(node = xlat_exp_alloc(head, XLAT_BOX, NULL, 0));
41 fr_value_box_copy(node, &node->data, box);
42
43 if (node->data.type == FR_TYPE_STRING) {
45 xlat_exp_set_name_buffer(node, node->data.vb_strvalue); /* later changes can free strvalue */
46 } else {
47 char *name;
48
49 node->quote = T_BARE_WORD;
50 MEM(fr_value_box_aprint(node, &name, box, NULL) >= 0);
52 }
53 talloc_free(box);
54
56 }
57}
58
60
65
67{
68 int rcode;
69 bool success;
70 fr_value_box_list_t list;
71 xlat_flags_t our_flags;
72 xlat_exp_t *node, *next;
73
74 if (!head->flags.can_purify) return 0;
75
76 /*
77 * We can't purify things which need resolving,
78 */
79 if (head->flags.needs_resolving) return -1;
80
81 our_flags = head->flags;
82 our_flags.pure = true; /* we flip this if the children are not pure */
83
84 for (node = fr_dlist_head(&head->dlist);
85 next = fr_dlist_next(&head->dlist, node), node != NULL;
86 node = next) {
87 if (!node->flags.can_purify) continue;
88
89 switch (node->type) {
90 case XLAT_TMPL:
91 /*
92 * Optimize it by replacing the xlat -> tmpl -> xlat with just an xlat.
93 *
94 * That way we avoid a bounce through the tmpl code at run-time.
95 */
96 if (tmpl_is_xlat(node->vpt)) {
97 xlat_exp_head_t *xlat = tmpl_xlat(node->vpt);
98
99 rcode = xlat_purify_list_internal(xlat, request, node->vpt->quote);
100 if (rcode < 0) return rcode;
101
102 node->flags = xlat->flags;
103
104 /*
105 * @todo - for casts, check if the xlat is pure and/or contains just
106 * data. If so, cast the data.
107 */
108
109 /*
110 * If there's a cast, then keep it. The xlat_exp_t doesn't contain a
111 * cast type, so we have to leave it in the tmpl_t.
112 */
113 if (tmpl_rules_cast(node->vpt) != FR_TYPE_NULL) break;
114
115 /*
116 * We're in a string, or we will be in a string. Don't do any more
117 * optimizations.
118 *
119 * @todo - maybe add a cast here, but the expression evaluator will take
120 * care of that.
121 */
122 if ((quote != T_BARE_WORD) || (node->vpt->quote != T_BARE_WORD)) break;
123
124 /*
125 * If there's only one item here, we can just replace this node with the
126 * one item.
127 */
128 if (fr_dlist_num_elements(&xlat->dlist) == 1) {
129 xlat_exp_t *child, *prev;
130
131 prev = fr_dlist_remove(&head->dlist, node);
132 child = fr_dlist_head(&xlat->dlist);
133
134 (void) talloc_steal(child, node->vpt->name);
135 (void) talloc_steal(head, child);
136
137 (void) fr_dlist_remove(&xlat->dlist, child);
138 fr_dlist_insert_after(&head->dlist, prev, child);
139 talloc_free(node); /* and vpt and xlat */
140 node = child;
141 break;
142 }
143
144 /*
145 * There are multiple items in the child. We need to keep the group
146 * wrapper, which ensures that the entire sub-expression results in one
147 * output value.
148 */
150 (void) talloc_steal(node, node->vpt->name);
151 (void) talloc_steal(node, xlat);
152 talloc_free(node->vpt);
153 node->group = xlat;
154 break;
155 }
156 break;
157
158 case XLAT_BOX:
159 case XLAT_ONE_LETTER:
160 case XLAT_REGEX:
161 break;
162
163 case XLAT_INVALID:
165 fr_assert(0);
166 return -1;
167
168 case XLAT_GROUP:
169 rcode = xlat_purify_list_internal(node->group, request, quote);
170 if (rcode < 0) return rcode;
171
172 node->flags = node->group->flags;
173 break;
174
175 case XLAT_FUNC:
176 /*
177 * If the node is not pure, then maybe there's a callback to purify it, OR maybe
178 * we can purify the function arguments.
179 */
180 if (!node->flags.pure) {
181 if (node->call.func->purify) {
182 fr_dict_t const *dict = request->dict;
183
184 /*
185 * Swap in the node specific dictionary.
186 *
187 * The previous code stored the dictionary in the xlat_exp_head_t,
188 * and whilst this wasn't wrong, it was duplicative.
189 *
190 * This allows future code to create inline definitions of local
191 * attributes, and have them work correctly, as more deeply nested
192 * expressions would swap in the correct dictionary.
193 */
194 request->dict = node->call.dict;
195 if (node->call.func->purify(node, node->call.inst->data, request) < 0) return -1;
196 request->dict = dict;
197 } else {
198 if (xlat_purify_list_internal(node->call.args, request, T_BARE_WORD) < 0) return -1;
199 }
200
201 /*
202 * It may have been purified into an XLAT_BOX. But if not, ensure that
203 * the flags are all correct.
204 */
205 if (node->type == XLAT_FUNC) {
206 node->flags = node->call.func->flags;
207 xlat_exp_foreach(node->call.args, arg) {
208 xlat_flags_merge(&node->flags, &arg->flags);
209 }
210 }
211 break;
212 }
213
214 /*
215 * The node is entirely pure, we don't worry about any callbacks, we just
216 * evaluate the entire thing to purify it.
217 */
218 fr_assert(node->flags.pure);
219 fr_value_box_list_init(&list);
220 success = false;
221 if (unlang_xlat_push_node(head, &success, &list, request, node) < 0) {
222 return -1;
223 }
224
225 /*
226 * Hope to god it doesn't yield. :)
227 */
228
229 (void) unlang_interpret_synchronous(NULL, request);
230 if (!success) return -1;
231
232 /*
233 * The function call becomes a GROUP of boxes
234 */
236 xlat_exp_set_type(node, XLAT_GROUP); /* Frees the argument list */
237
238 xlat_value_list_to_xlat(node->group, &list);
239 node->flags = node->group->flags;
240 break;
241 }
242
243 node->flags.can_purify = false;
244 xlat_flags_merge(&our_flags, &node->flags);
245 }
246
247 /*
248 * Let's not call xlat_purify() repeatedly, so we clear the flag.
249 *
250 * @todo - if all of the children of "head" are "pure", then at the end of the purification
251 * process, there should only be one child, of type XLAT_BOX.
252 */
253 our_flags.can_purify = false;
254 head->flags = our_flags;
255
256 return 0;
257}
258
259/** Purify an xlat
260 *
261 * @param head the xlat to be purified
262 * @param intp the interpreter to use.
263 *
264 */
266{
267 int rcode;
268 request_t *request;
269
270 if (!head->flags.can_purify) return 0;
271
272 request = request_alloc_internal(NULL, (&(request_init_args_t){ .namespace = fr_dict_internal() }));
273 if (!request) return -1;
274
275 if (intp) unlang_interpret_set(request, intp);
276
277 rcode = xlat_purify_list(head, request);
278 talloc_free(request);
279 if (rcode < 0) return rcode;
280
281 fr_assert(!head->flags.can_purify);
282
283 return 0;
284}
285
287{
288#ifdef STATIC_ANALYZER
289 if (!node) return NULL;
290#endif
291
292 if (node->type == XLAT_BOX) {
293 return &node->data;
294
295 } else if ((node->type == XLAT_TMPL) && tmpl_is_data(node->vpt)) {
296 return tmpl_value(node->vpt);
297 }
298
299 return NULL;
300}
301
302
303static bool is_truthy(xlat_exp_t *node, bool *out)
304{
305 fr_value_box_t const *box;
306
307 box = xlat_value_box(node);
308 if (!box) {
309 *out = false;
310 return false;
311 }
312
314 return true;
315}
316
317/*
318 * Do some optimizations.
319 *
320 */
322{
323 bool value;
324
325 /*
326 * LHS isn't truthy, we can't do anything. If the LHS
327 * passes, we return the value of the LHS.
328 *
329 * FOO || ... --> FOO || ...
330 */
331 if (!is_truthy(lhs, &value)) {
332 /*
333 * FOO || 0 --> FOO much of the time
334 * FOO || 1 --> FOO much of the time
335 */
336 if (!is_truthy(rhs, &value)) return NULL;
337
338 /*
339 * BOOL || 1 --> 1
340 *
341 * Because if the LHS is 1, then we return the LHS (1)
342 * On the other hand, it the LHS is 0, then we return
343 * the RHS, which is also 1.
344 *
345 * But we can't do
346 *
347 * <type> || 1 --> 1
348 */
349 if (value && (lhs->type == XLAT_FUNC) && (lhs->call.func->return_type == FR_TYPE_BOOL)) {
350 talloc_free(lhs);
351 return rhs;
352 }
353
354 return NULL;
355 }
356
357 /*
358 * 1 || FOO --> 1
359 * 0 || FOO --> FOO
360 */
361 if (value) {
362 talloc_free(rhs);
363 return lhs;
364 }
365
366 talloc_free(lhs);
367 return rhs;
368}
369
370
371/*
372 * Do some optimizations.
373 *
374 */
376{
377 bool value;
378
379 /*
380 * LHS isn't truthy
381 *
382 * FOO && ... --> FOO && ...
383 */
384 if (!is_truthy(lhs, &value)) {
385 /*
386 * FOO && 0 --> 0
387 * FOO && 1 --> FOO
388 */
389 if (!is_truthy(rhs, &value)) return NULL;
390
391 if (!value) {
392 talloc_free(lhs);
393 return rhs;
394 }
395
396 talloc_free(rhs);
397 return lhs;
398 }
399
400 /*
401 * 0 && FOO --> 0
402 * 1 && FOO --> FOO
403 */
404 if (!value) {
405 talloc_free(rhs);
406 return lhs;
407 }
408
409 talloc_free(lhs);
410 return rhs;
411}
412
413/*
414 * Do peephole optimizations.
415 */
416static int binary_peephole_optimize(TALLOC_CTX *ctx, xlat_exp_t **out, xlat_exp_t *lhs, fr_token_t op, xlat_exp_t *rhs)
417{
418 fr_value_box_t *lhs_box, *rhs_box;
419 fr_value_box_t box;
420 xlat_exp_t *node;
421 char *name;
422
423#if 0
424 /*
425 * @todo - more peephole optimizations here. We can't enable this code as yet, because of
426 * upcasting rules (e.g. calc.c) where comparisons between IP prefixes and IP addresses (or
427 * v4/v6) are upcast, and then the values compared.
428 *
429 * We should probably expose some of the upcast functionality in calc.c so that this function can
430 * use it.
431 */
432
433 /*
434 * Attribute op value.
435 */
436 if ((lhs->type == XLAT_TMPL) && tmpl_is_attr(lhs->vpt) &&
437 (rhs->type == XLAT_TMPL) && (tmpl_is_data_unresolved(rhs->vpt) || tmpl_is_data(rhs->vpt))) {
438 fr_type_t dst_type;
439 fr_dict_attr_t const *da;
440
441 resolve:
442 dst_type = tmpl_rules_cast(rhs->vpt);
443 da = tmpl_attr_tail_da(lhs->vpt);
444
445 /*
446 * Cast to the final type. If there are two different casts, we ignore the one for the
447 * data.
448 */
449 if (fr_type_is_null(dst_type)) {
450 dst_type = tmpl_rules_cast(lhs->vpt);
451 if (fr_type_is_null(dst_type)) dst_type = da->type;
452 }
453
454 if (tmpl_cast_in_place(rhs->vpt, dst_type, da) < 0) return -1;
455
456 rhs->flags.needs_resolving = false;
457 return 0;
458 }
459
460 /*
461 * value op attribute
462 *
463 * We just swap LHS and RHS without caring about the operator, because we don't use the
464 * operator, and the caller has no idea that we swapped the pointers..
465 */
466 if ((rhs->type == XLAT_TMPL) && tmpl_is_attr(rhs->vpt) &&
467 (lhs->type == XLAT_TMPL) && (tmpl_is_data_unresolved(lhs->vpt) || tmpl_is_data(lhs->vpt))) {
468 xlat_exp_t *tmp = lhs;
469 lhs = rhs;
470 rhs = tmp;
471 goto resolve;
472 }
473#endif
474
475 /*
476 * The tmpl_tokenize code takes care of resolving the data if there's a cast.
477 */
478 lhs_box = xlat_value_box(lhs);
479 if (!lhs_box) return 0;
480
481 rhs_box = xlat_value_box(rhs);
482 if (!rhs_box) return 0;
483
484 if (fr_value_calc_binary_op(lhs, &box, FR_TYPE_NULL, lhs_box, op, rhs_box) < 0) return -1;
485
486 MEM(node = xlat_exp_alloc(ctx, XLAT_BOX, NULL, 0));
487
488 if (box.type == FR_TYPE_BOOL) box.enumv = attr_expr_bool_enum;
489
490 MEM(fr_value_box_aprint(node, &name, &box, NULL) >= 0);
492 fr_value_box_copy(node, &node->data, &box);
493
494 *out = node;
495
496 return 1;
497}
498
499int xlat_purify_op(TALLOC_CTX *ctx, xlat_exp_t **out, xlat_exp_t *lhs, fr_token_t op, xlat_exp_t *rhs)
500{
501 if (op == T_LOR) {
502 xlat_exp_t *node;
503
504 node = peephole_optimize_lor(lhs, rhs);
505 if (!node) return 0;
506
507 *out = node;
508 return 1;
509 }
510
511 if (op == T_LAND) {
512 xlat_exp_t *node;
513
514 node = peephole_optimize_land(lhs, rhs);
515 if (!node) return 0;
516
517 *out = node;
518 return 1;
519 }
520
521 return binary_peephole_optimize(ctx, out, lhs, op, rhs);
522}
#define RCSID(id)
Definition build.h:485
int fr_value_calc_binary_op(TALLOC_CTX *ctx, fr_value_box_t *dst, fr_type_t hint, fr_value_box_t const *a, fr_token_t op, fr_value_box_t const *b)
Calculate DST = A OP B.
Definition calc.c:1924
#define MEM(x)
Definition debug.h:36
fr_dict_t const * fr_dict_internal(void)
Definition dict_util.c:4617
Test enumeration values.
Definition dict_test.h:92
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:486
static void * fr_dlist_remove(fr_dlist_head_t *list_head, void *ptr)
Remove an item from the list.
Definition dlist.h:638
static unsigned int fr_dlist_num_elements(fr_dlist_head_t const *head)
Return the number of elements in the dlist.
Definition dlist.h:939
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:555
static int fr_dlist_insert_after(fr_dlist_head_t *list_head, void *pos, void *ptr)
Insert an item after an item already in the list.
Definition dlist.h:414
void unlang_interpret_set(request_t *request, unlang_interpret_t *intp)
Set a specific interpreter for a request.
Definition interpret.c:1760
rlm_rcode_t unlang_interpret_synchronous(fr_event_list_t *el, request_t *request)
Execute an unlang section synchronously.
talloc_free(reap)
fr_type_t
@ FR_TYPE_STRING
String of printable characters.
@ FR_TYPE_NULL
Invalid (uninitialised) attribute type.
@ FR_TYPE_BOOL
A truth value.
#define fr_assert(_expr)
Definition rad_assert.h:38
#define request_alloc_internal(_ctx, _args)
Allocate a new internal request.
Definition request.h:305
Optional arguments for initialising requests.
Definition request.h:254
static char const * name
#define tmpl_is_xlat(vpt)
Definition tmpl.h:215
#define tmpl_value(_tmpl)
Definition tmpl.h:942
#define tmpl_is_attr(vpt)
Definition tmpl.h:213
#define tmpl_xlat(_tmpl)
Definition tmpl.h:935
#define tmpl_rules_cast(_tmpl)
Definition tmpl.h:947
int tmpl_cast_in_place(tmpl_t *vpt, fr_type_t type, fr_dict_attr_t const *enumv))
Convert tmpl_t of type TMPL_TYPE_DATA_UNRESOLVED or TMPL_TYPE_DATA to TMPL_TYPE_DATA of type specifie...
#define tmpl_is_data(vpt)
Definition tmpl.h:211
#define tmpl_is_data_unresolved(vpt)
Definition tmpl.h:222
static fr_dict_attr_t const * tmpl_attr_tail_da(tmpl_t const *vpt)
Return the last attribute reference da.
Definition tmpl.h:806
goto success
Definition tmpl_eval.c:1444
enum fr_token fr_token_t
@ T_BARE_WORD
Definition token.h:120
@ T_LAND
Definition token.h:91
@ T_LOR
Definition token.h:92
@ T_DOUBLE_QUOTED_STRING
Definition token.h:121
int unlang_xlat_push_node(TALLOC_CTX *ctx, bool *p_success, fr_value_box_list_t *out, request_t *request, xlat_exp_t *node)
Push a pre-compiled xlat onto the stack for evaluation.
Definition xlat.c:307
static fr_slen_t head
Definition xlat.h:418
bool can_purify
if the xlat has a pure function with pure arguments.
Definition xlat.h:116
bool pure
has no external side effects, true for BOX, LITERAL, and some functions
Definition xlat.h:114
bool needs_resolving
Needs pass2 resolution.
Definition xlat.h:113
int xlat_instance_unregister_func(xlat_exp_t *node)
Remove a node from the list of xlat instance data.
Definition xlat_inst.c:550
Flags that control resolution and evaluation.
Definition xlat.h:112
#define fr_type_is_null(_x)
Definition types.h:326
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
bool fr_value_box_is_truthy(fr_value_box_t const *in)
Check truthiness of values.
Definition value.c:6370
static fr_slen_t fr_value_box_aprint(TALLOC_CTX *ctx, char **out, fr_value_box_t const *data, fr_sbuff_escape_rules_t const *e_rules) 1(fr_value_box_print
static size_t char ** out
Definition value.h:1012
void xlat_exp_set_name_buffer(xlat_exp_t *node, char const *fmt)
Set the format string for an xlat node, copying from a talloc'd buffer.
Definition xlat_alloc.c:204
void xlat_exp_set_name_shallow(xlat_exp_t *node, char const *fmt)
Set the format string for an xlat node from a pre-existing buffer.
Definition xlat_alloc.c:221
fr_dict_attr_t const * attr_expr_bool_enum
Definition xlat_eval.c:46
xlat_flags_t flags
Flags that control resolution and evaluation.
Definition xlat_priv.h:155
xlat_flags_t flags
Flags that control resolution and evaluation.
Definition xlat_priv.h:188
fr_token_t quote
Type of quoting around XLAT_GROUP types.
Definition xlat_priv.h:153
@ XLAT_ONE_LETTER
Special "one-letter" expansion.
Definition xlat_priv.h:108
@ XLAT_BOX
fr_value_box_t
Definition xlat_priv.h:107
@ XLAT_TMPL
xlat attribute
Definition xlat_priv.h:111
@ XLAT_FUNC
xlat module
Definition xlat_priv.h:109
@ XLAT_GROUP
encapsulated string of xlats
Definition xlat_priv.h:115
@ XLAT_FUNC_UNRESOLVED
func needs resolution during pass2.
Definition xlat_priv.h:110
@ XLAT_INVALID
Bad expansion.
Definition xlat_priv.h:106
static void xlat_flags_merge(xlat_flags_t *parent, xlat_flags_t const *child)
Merge flags from child to parent.
Definition xlat_priv.h:225
#define xlat_exp_set_type(_node, _type)
Definition xlat_priv.h:272
xlat_type_t _CONST type
type of this expansion.
Definition xlat_priv.h:156
#define xlat_exp_alloc(_ctx, _type, _in, _inlen)
Definition xlat_priv.h:278
#define xlat_exp_foreach(_list_head, _iter)
Iterate over the contents of a list, only one level.
Definition xlat_priv.h:218
static int xlat_exp_insert_tail(xlat_exp_head_t *head, xlat_exp_t *node)
Definition xlat_priv.h:234
fr_dlist_head_t dlist
Definition xlat_priv.h:187
An xlat expansion node.
Definition xlat_priv.h:149
static xlat_exp_t * peephole_optimize_lor(xlat_exp_t *lhs, xlat_exp_t *rhs)
static int binary_peephole_optimize(TALLOC_CTX *ctx, xlat_exp_t **out, xlat_exp_t *lhs, fr_token_t op, xlat_exp_t *rhs)
static fr_value_box_t * xlat_value_box(xlat_exp_t *node)
static bool is_truthy(xlat_exp_t *node, bool *out)
int xlat_purify_op(TALLOC_CTX *ctx, xlat_exp_t **out, xlat_exp_t *lhs, fr_token_t op, xlat_exp_t *rhs)
int xlat_purify(xlat_exp_head_t *head, unlang_interpret_t *intp)
Purify an xlat.
int xlat_purify_list(xlat_exp_head_t *head, request_t *request)
Definition xlat_purify.c:61
static int xlat_purify_list_internal(xlat_exp_head_t *head, request_t *request, fr_token_t quote)
Definition xlat_purify.c:66
static xlat_exp_t * peephole_optimize_land(xlat_exp_t *lhs, xlat_exp_t *rhs)
static void xlat_value_list_to_xlat(xlat_exp_head_t *head, fr_value_box_list_t *list)
Definition xlat_purify.c:34