The FreeRADIUS server $Id: 15bac2a4c627c01d1aa2047687b3418955ac7f00 $
All Data Structures Files Functions Variables Typedefs Enumerations Enumerator Macros Modules Pages
xlat_alloc.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: 7f34225cf1e1c8fdcdab788195e57df220868751 $
19 *
20 * @file xlat_alloc.c
21 * @brief Functions to allocate different types of xlat nodes
22 */
23
24RCSID("$Id: 7f34225cf1e1c8fdcdab788195e57df220868751 $")
25
26#include <freeradius-devel/server/base.h>
27#include <freeradius-devel/unlang/xlat.h>
28
29#include <freeradius-devel/util/debug.h>
30#include <freeradius-devel/util/types.h>
31
32#define _XLAT_PRIVATE
33#include <freeradius-devel/unlang/xlat_priv.h>
34
36{
38
39 MEM(head = talloc_zero(ctx, xlat_exp_head_t));
40
41 fr_dlist_init(&head->dlist, xlat_exp_t, entry);
42 head->flags.pure = head->flags.can_purify = true;
43#ifndef NDEBUG
44 head->file = file;
45 head->line = line;
46#endif
47
48 return head;
49}
50
51/** Set the type of an xlat node
52 *
53 * Also initialises any xlat_exp_head necessary
54 *
55 * @param[in] node to set type for.
56 * @param[in] type to set.
57 */
59{
60 /*
61 * Do nothing if it's the same type
62 */
63 if (node->type == type) return;
64
65 /*
66 * Free existing lists if present
67 */
68 if (node->type != 0) switch (node->type) {
69 case XLAT_GROUP:
70 TALLOC_FREE(node->group);
71 break;
72
73 case XLAT_FUNC:
75 if (type != XLAT_FUNC) {
76 TALLOC_FREE(node->call.args); /* Just switching from unresolved to resolved */
77 } else goto done;
78 break;
79
80 default:
81 break;
82 }
83
84 /*
85 * Alloc new lists to match the type
86 */
87 switch (type) {
88 case XLAT_GROUP:
90 break;
91
92 case XLAT_FUNC:
94 node->call.args = _xlat_exp_head_alloc(NDEBUG_LOCATION_VALS node);
95 break;
96
97 default:
98 break;
99 }
100
101done:
102 node->type = type;
103}
104
105static xlat_exp_t *xlat_exp_alloc_pool(NDEBUG_LOCATION_ARGS TALLOC_CTX *ctx, unsigned int extra_hdrs, size_t extra)
106{
107 xlat_exp_t *node;
108
109 MEM(node = talloc_zero_pooled_object(ctx, xlat_exp_t, extra_hdrs, extra));
110 node->flags.pure = node->flags.can_purify = true; /* everything starts pure */
111 node->quote = T_BARE_WORD;
112#ifndef NDEBUG
113 node->file = file;
114 node->line = line;
115#endif
116
117 return node;
118}
119
120/** Allocate an xlat node with no name, and no type set
121 *
122 * @param[in] ctx to allocate node in.
123 * @return A new xlat node.
124 */
129
130/** Allocate an xlat node
131 *
132 * @param[in] ctx to allocate node in.
133 * @param[in] type of the node.
134 * @param[in] in original input string.
135 * @param[in] inlen the length of the original input string.
136 * @return A new xlat node.
137 */
139{
140 xlat_exp_t *node;
141 unsigned int extra_hdrs;
142 size_t extra;
143
144 /*
145 * Figure out how much extra memory we
146 * need to allocate for this node type.
147 */
148 switch (type) {
149 case XLAT_GROUP:
150 extra_hdrs = 1;
151 extra = sizeof(xlat_exp_head_t);
152 break;
153
154 case XLAT_FUNC:
155 extra_hdrs = 1;
156 extra = sizeof(xlat_exp_head_t);
157 break;
158
159 default:
160 extra_hdrs = 0;
161 extra = 0;
162 }
163
165 ctx,
166 (in != NULL) + extra_hdrs,
167 inlen + extra);
169
170 if (!in) return node;
171
172 node->fmt = talloc_bstrndup(node, in, inlen);
173 switch (type) {
174 case XLAT_BOX:
175 fr_value_box_strdup_shallow(&node->data, NULL, node->fmt, false);
176 break;
177
178 default:
179 break;
180 }
181
182 return node;
183}
184
185/** Set the format string for an xlat node
186 *
187 * @param[in] node to set fmt for.
188 * @param[in] fmt talloced buffer to set as the fmt string.
189 * @param[in] len of fmt string.
190 */
191void xlat_exp_set_name(xlat_exp_t *node, char const *fmt, size_t len)
192{
193 fr_assert(node->fmt != fmt);
194
195 if (node->fmt) talloc_const_free(node->fmt);
196 MEM(node->fmt = talloc_bstrndup(node, fmt, len));
197}
198
199/** Set the format string for an xlat node, copying from a talloc'd buffer
200 *
201 * @param[in] node to set fmt for.
202 * @param[in] fmt talloced buffer to set as the fmt string.
203 */
205{
206 if (node->fmt) {
207 if (node->fmt == fmt) {
208 (void) talloc_steal(node, fmt);
209 } else {
210 talloc_const_free(node->fmt);
211 }
212 }
213 MEM(node->fmt = talloc_typed_strdup_buffer(node, fmt));
214}
215
216/** Set the format string for an xlat node from a pre-existing buffer
217 *
218 * @param[in] node to set fmt for.
219 * @param[in] fmt talloced buffer to set as the fmt string.
220 */
222{
223 fr_assert(node->fmt != fmt);
224
225 if (node->fmt) talloc_const_free(node->fmt);
226 node->fmt = talloc_get_type_abort_const(fmt, char);
227}
228
229/** Copy all nodes in the input list to the output list
230 *
231 * @param[in] ctx to allocate new nodes in.
232 * @param[out] out Where to write new nodes.
233 * @param[in] in Input nodes.
234 * @return
235 * - 0 on success.
236 * - -1 on failure.
237 */
239{
240 xlat_exp_head_t *head = NULL;
241
242 xlat_flags_merge(&out->flags, &in->flags);
243
244 /*
245 * Copy everything in the list of nodes
246 */
247 xlat_exp_foreach(in, p) {
248 xlat_exp_t *node;
249
250 (void)talloc_get_type_abort(p, xlat_exp_t);
251
252 /*
253 * Ensure the format string is valid... At this point
254 * they should all be talloc'd strings.
255 */
256 MEM(node = xlat_exp_alloc(ctx, p->type,
257 talloc_get_type_abort_const(p->fmt, char), talloc_array_length(p->fmt) - 1));
258
259 node->quote = p->quote;
260 node->flags = p->flags;
261
262 switch (p->type) {
263 case XLAT_INVALID:
264 fr_strerror_printf("Cannot copy xlat node of type \"invalid\"");
265 error:
267 return -1;
268
269 case XLAT_BOX:
270 if (unlikely(fr_value_box_copy(node, &node->data, &p->data) < 0)) goto error;
271 break;
272
273 case XLAT_ONE_LETTER: /* Done with format */
275 break;
276
277 case XLAT_FUNC:
278 /*
279 * Only copy the function pointer, and whether this
280 * is ephemeral.
281 *
282 * All instance data is specific to the xlat node and
283 * cannot be duplicated.
284 *
285 * The node xlat nodes will need to be registered in
286 * the xlat instantiation table later.
287 */
288 node->call.func = p->call.func;
289 node->call.dict = p->call.dict;
290 node->call.ephemeral = p->call.ephemeral;
292 node, node->call.args, p->call.args) < 0)) goto error;
293 break;
294
295 case XLAT_TMPL:
296 node->vpt = tmpl_copy(node, p->vpt);
297 break;
298
299#ifdef HAVE_REGEX
300 case XLAT_REGEX:
301 node->regex_index = p->regex_index;
302 break;
303#endif
304
305 case XLAT_GROUP:
307 node, node->group, p->group) < 0)) goto error;
308 break;
309 }
310
312 }
313
314 return 0;
315}
316
318{
319 int ret;
320
321 if (!in) return 0;
322
326
327 return ret;
328}
329
330#ifdef WITH_VERIFY_PTR
331void xlat_exp_verify(xlat_exp_t const *node)
332{
334
335 switch (node->type) {
336 case XLAT_GROUP:
337 xlat_exp_head_verify(node->group);
338 (void)talloc_get_type_abort_const(node->fmt, char);
339 return;
340
341 case XLAT_FUNC:
342 xlat_exp_head_verify(node->call.args);
343 (void)talloc_get_type_abort_const(node->fmt, char);
344 return;
345
346 case XLAT_TMPL: {
347 tmpl_t const *vpt = node->vpt;
348
349 if (node->quote != node->vpt->quote) {
350 if (node->vpt->quote == T_SOLIDUS_QUOTED_STRING) {
351 /*
352 * m'foo' versus /foo/
353 */
354 fr_assert(node->quote != T_BARE_WORD);
355 } else {
356 /*
357 * Mismatching quotes are bad.
358 */
359 fr_assert(node->quote == T_BARE_WORD);
360 }
361 }
362
363 if (tmpl_is_attr(vpt)) {
364 fr_dict_attr_t const *da;
365 da = tmpl_attr_tail_da(node->vpt);
366
367 if (tmpl_rules_cast(node->vpt) != FR_TYPE_NULL) {
368 /*
369 * Casts must be omitted, unless we're using a cast as a way to get rid
370 * of enum names.
371 */
372 if (tmpl_rules_cast(node->vpt) == da->type) {
373 fr_assert(da->flags.has_value);
374 }
375
376 } else if (node->quote != T_BARE_WORD) {
377 fr_assert(da->type != FR_TYPE_STRING);
378 }
379
380 return;
381 }
382
383 /*
384 * Casts should have been hoisted.
385 */
386 if (tmpl_is_data(node->vpt)) {
388 }
389
390#if 0
391 /*
392 * @todo - xlats SHOULD have been hoisted, unless they're quoted or cast.
393 */
394 if (tmpl_is_xlat(node->vpt)) {
395 fr_assert((node->vpt->quote != T_BARE_WORD) ||
396 (tmpl_rules_cast(node->vpt) != FR_TYPE_NULL));
397 return;
398 }
399#endif
400 return;
401 }
402
403 default:
404 break;
405 }
406}
407
408/** Performs recursive validation of node lists
409 */
410void xlat_exp_head_verify(xlat_exp_head_t const *head)
411{
413
414 xlat_exp_foreach(head, node) xlat_exp_verify(node);
415}
416#endif
int const char * file
Definition acutest.h:702
static int const char * fmt
Definition acutest.h:573
int const char int line
Definition acutest.h:702
#define RCSID(id)
Definition build.h:485
#define unlikely(_x)
Definition build.h:383
#define NDEBUG_LOCATION_VALS
Definition build.h:264
#define NDEBUG_LOCATION_ARGS
Pass caller information to the function.
Definition build.h:263
#define MEM(x)
Definition debug.h:36
static fr_slen_t in
Definition dict.h:831
#define fr_dlist_init(_head, _type, _field)
Initialise the head structure of a doubly linked list.
Definition dlist.h:260
talloc_free(reap)
@ FR_TYPE_STRING
String of printable characters.
@ FR_TYPE_NULL
Invalid (uninitialised) attribute type.
#define fr_assert(_expr)
Definition rad_assert.h:38
static bool done
Definition radclient.c:80
#define tmpl_is_xlat(vpt)
Definition tmpl.h:215
#define tmpl_is_attr(vpt)
Definition tmpl.h:213
#define tmpl_rules_cast(_tmpl)
Definition tmpl.h:947
tmpl_t * tmpl_copy(TALLOC_CTX *ctx, tmpl_t const *in)
Copy a tmpl.
#define tmpl_is_data(vpt)
Definition tmpl.h:211
static fr_slen_t vpt
Definition tmpl.h:1274
static fr_dict_attr_t const * tmpl_attr_tail_da(tmpl_t const *vpt)
Return the last attribute reference da.
Definition tmpl.h:806
fr_aka_sim_id_type_t type
char * talloc_typed_strdup_buffer(TALLOC_CTX *ctx, char const *p)
Call talloc_strndup, setting the type on the new chunk correctly.
Definition talloc.c:469
char * talloc_bstrndup(TALLOC_CTX *ctx, char const *in, size_t inlen)
Binary safe strndup function.
Definition talloc.c:564
#define talloc_get_type_abort_const
Definition talloc.h:282
#define talloc_zero_pooled_object(_ctx, _type, _num_subobjects, _total_subobjects_size)
Definition talloc.h:177
static int talloc_const_free(void const *ptr)
Free const'd memory.
Definition talloc.h:224
@ T_BARE_WORD
Definition token.h:120
@ T_SOLIDUS_QUOTED_STRING
Definition token.h:124
#define XLAT_HEAD_VERIFY(_head)
Definition xlat.h:465
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
#define fr_strerror_printf(_fmt,...)
Log to thread local error buffer.
Definition strerror.h:64
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_strdup_shallow(fr_value_box_t *dst, fr_dict_attr_t const *enumv, char const *src, bool tainted)
Assign a buffer containing a nul terminated string to a box, but don't copy it.
Definition value.c:4066
static size_t char fr_sbuff_t size_t inlen
Definition value.h:1012
int nonnull(2, 5))
static size_t char ** out
Definition value.h:1012
xlat_exp_head_t * _xlat_exp_head_alloc(NDEBUG_LOCATION_ARGS TALLOC_CTX *ctx)
Definition xlat_alloc.c:35
xlat_exp_t * _xlat_exp_alloc(NDEBUG_LOCATION_ARGS TALLOC_CTX *ctx, xlat_type_t type, char const *in, size_t inlen)
Allocate an xlat node.
Definition xlat_alloc.c:138
void xlat_exp_set_name(xlat_exp_t *node, char const *fmt, size_t len)
Set the format string for an xlat node.
Definition xlat_alloc.c:191
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
xlat_exp_t * _xlat_exp_alloc_null(NDEBUG_LOCATION_ARGS TALLOC_CTX *ctx)
Allocate an xlat node with no name, and no type set.
Definition xlat_alloc.c:125
int _xlat_copy(NDEBUG_LOCATION_ARGS TALLOC_CTX *ctx, xlat_exp_head_t *out, xlat_exp_head_t const *in)
Definition xlat_alloc.c:317
static xlat_exp_t * xlat_exp_alloc_pool(NDEBUG_LOCATION_ARGS TALLOC_CTX *ctx, unsigned int extra_hdrs, size_t extra)
Definition xlat_alloc.c:105
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
static int _xlat_copy_internal(NDEBUG_LOCATION_ARGS TALLOC_CTX *ctx, xlat_exp_head_t *out, xlat_exp_head_t const *in)
Copy all nodes in the input list to the output list.
Definition xlat_alloc.c:238
void _xlat_exp_set_type(NDEBUG_LOCATION_ARGS xlat_exp_t *node, xlat_type_t type)
Set the type of an xlat node.
Definition xlat_alloc.c:58
struct xlat_exp_head_s xlat_exp_head_t
Definition xlat_ctx.h:36
xlat_flags_t flags
Flags that control resolution and evaluation.
Definition xlat_priv.h:155
int line
Line where the xlat was allocated.
Definition xlat_priv.h:160
fr_token_t quote
Type of quoting around XLAT_GROUP types.
Definition xlat_priv.h:153
xlat_type_t
Definition xlat_priv.h:105
@ 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
char const *_CONST fmt
The original format string (a talloced buffer).
Definition xlat_priv.h:152
xlat_type_t _CONST type
type of this expansion.
Definition xlat_priv.h:156
char const *_CONST file
File where the xlat was allocated.
Definition xlat_priv.h:159
#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
An xlat expansion node.
Definition xlat_priv.h:149