The FreeRADIUS server $Id: 15bac2a4c627c01d1aa2047687b3418955ac7f00 $
Loading...
Searching...
No Matches
switch.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: f1056e18a03ef8b2d89267446b5a6b54607589f2 $
19 *
20 * @file unlang/switch.c
21 * @brief Unlang "switch" keyword evaluation.
22 *
23 * @copyright 2006-2019 The FreeRADIUS server project
24 */
25RCSID("$Id: f1056e18a03ef8b2d89267446b5a6b54607589f2 $")
26
27#include "group_priv.h"
28#include "switch_priv.h"
29
31{
32 unlang_t *found;
33
34 unlang_group_t *switch_g;
35 unlang_switch_t *switch_gext;
36
37 fr_value_box_t const *box = NULL;
38
40
41 /*
42 * Mock up an unlang_cast_t. Note that these on-stack
43 * buffers are the reason why case_cmp(), case_hash(),
44 * and case_to_key() use direct casts, and not the
45 * "generic to x" functions.
46 */
47 tmpl_t case_vpt = (tmpl_t) {
48 .type = TMPL_TYPE_DATA,
49 };
50 unlang_case_t my_case = (unlang_case_t) {
52 .self = (unlang_t) {
54 },
55 },
56 .vpt = &case_vpt,
57 };
58
59 switch_g = unlang_generic_to_group(frame->instruction);
60 switch_gext = unlang_group_to_switch(switch_g);
61
62 found = NULL;
63
64 /*
65 * The attribute doesn't exist. We can skip
66 * directly to the default 'case' statement.
67 */
68 if (tmpl_is_attr(switch_gext->vpt)) {
69 if (tmpl_find_vp(&vp, request, switch_gext->vpt) < 0) {
70 found = switch_gext->default_case;
71 goto do_null_case;
72 } else {
73 box = &vp->data;
74 }
75
76 /*
77 * Expand the template if necessary, so that it
78 * is evaluated once instead of for each 'case'
79 * statement.
80 */
81 } else if (tmpl_is_xlat(switch_gext->vpt) ||
82 tmpl_is_exec(switch_gext->vpt)) {
83 ssize_t slen;
84
86 request, switch_gext->vpt);
87 if (slen < 0) goto find_null_case;
88
89 tmpl_debug(switch_gext->vpt);
90 ERROR("GOT DATA %s - %pV", fr_type_to_str(box->type), box);
91
92 } else if (!fr_cond_assert_msg(0, "Invalid tmpl type %s", tmpl_type_to_str(switch_gext->vpt->type))) {
93 return UNLANG_ACTION_FAIL;
94 }
95
96 /*
97 * case_gext->vpt.data.literal is an in-line box, so we
98 * have to make a shallow copy of its contents.
99 *
100 * Note: We do not pass a ctx here as we don't want to
101 * create a reference.
102 */
103 fr_value_box_copy_shallow(NULL, &case_vpt.data.literal, box);
104 found = fr_htrie_find(switch_gext->ht, &my_case);
105 if (!found) {
106 find_null_case:
107 found = switch_gext->default_case;
108 }
109
110do_null_case:
111 /*
112 * Nothing found. Just continue, and ignore the "switch"
113 * statement.
114 */
115 if (!found) return UNLANG_ACTION_EXECUTE_NEXT;
116
117 if (unlang_interpret_push(request, found, frame->result, UNLANG_NEXT_STOP, UNLANG_SUB_FRAME) < 0) {
118 *p_result = RLM_MODULE_FAIL;
120 }
121
123}
124
125
127{
129
130 if (!g->children) {
131 *p_result = RLM_MODULE_NOOP;
133 }
134
135 return unlang_group(p_result, request, frame);
136}
137
139{
141 &(unlang_op_t){
142 .name = "switch",
143 .interpret = unlang_switch,
144 .debug_braces = true
145 });
146
148 &(unlang_op_t){
149 .name = "case",
150 .interpret = unlang_case,
151 .debug_braces = true
152 });
153}
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_EXECUTE_NEXT
Execute the next unlang_t.
Definition action.h:38
@ UNLANG_ACTION_STOP_PROCESSING
Break out of processing the current request (unwind).
Definition action.h:43
@ UNLANG_ACTION_FAIL
Encountered an unexpected error.
Definition action.h:36
@ UNLANG_ACTION_CALCULATE_RESULT
Calculate a new section rlm_rcode_t value.
Definition action.h:37
#define RCSID(id)
Definition build.h:485
#define fr_cond_assert_msg(_x, _fmt,...)
Calls panic_action ifndef NDEBUG, else logs error and evaluates to value of _x.
Definition debug.h:156
#define ERROR(fmt,...)
Definition dhcpclient.c:41
unlang_action_t unlang_group(rlm_rcode_t *p_result, request_t *request, unlang_stack_frame_t *frame)
Definition group.c:30
static void * fr_htrie_find(fr_htrie_t *ht, void const *data)
Find data in a htrie.
Definition htrie.h:104
TALLOC_CTX * unlang_interpret_frame_talloc_ctx(request_t *request)
Get a talloc_ctx which is valid only for this frame.
Definition interpret.c:1418
int unlang_interpret_push(request_t *request, unlang_t const *instruction, rlm_rcode_t default_rcode, bool do_next_sibling, bool top_frame)
Push a new frame onto the stack.
Definition interpret.c:161
#define UNLANG_SUB_FRAME
Definition interpret.h:36
void unlang_register(int type, unlang_op_t *op)
Register an operation with the interpreter.
Definition base.c:63
@ FR_TYPE_VALUE_BOX
A boxed value.
long int ssize_t
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
int tmpl_find_vp(fr_pair_t **out, request_t *request, tmpl_t const *vpt))
Returns the first VP matching a tmpl_t.
Definition tmpl_eval.c:880
static char const * tmpl_type_to_str(tmpl_type_t type)
Return a static string containing the type name.
Definition tmpl.h:639
#define tmpl_is_xlat(vpt)
Definition tmpl.h:215
#define tmpl_is_attr(vpt)
Definition tmpl.h:213
#define tmpl_is_exec(vpt)
Definition tmpl.h:216
@ TMPL_TYPE_DATA
Value in native boxed format.
Definition tmpl.h:142
void tmpl_debug(tmpl_t const *vpt)
#define tmpl_aexpand_type(_ctx, _out, _type, _request, _vpt)
Expand a tmpl to a C type, allocing a new buffer to hold the string.
Definition tmpl.h:1076
fr_pair_t * vp
Stores an attribute, a value and various bits of other data.
Definition pair.h:68
void unlang_switch_init(void)
Definition switch.c:138
static unlang_action_t unlang_switch(rlm_rcode_t *p_result, request_t *request, unlang_stack_frame_t *frame)
Definition switch.c:30
static unlang_action_t unlang_case(rlm_rcode_t *p_result, request_t *request, unlang_stack_frame_t *frame)
Definition switch.c:126
fr_htrie_t * ht
Definition switch_priv.h:36
unlang_group_t group
Definition switch_priv.h:56
static unlang_switch_t * unlang_group_to_switch(unlang_group_t *g)
Cast a group structure to the switch keyword extension.
Definition switch_priv.h:42
unlang_t * default_case
Definition switch_priv.h:34
#define UNLANG_NEXT_STOP
Definition unlang_priv.h:92
static unlang_group_t * unlang_generic_to_group(unlang_t const *p)
@ UNLANG_TYPE_SWITCH
Switch section.
Definition unlang_priv.h:57
@ UNLANG_TYPE_CASE
Case section (within a UNLANG_TYPE_SWITCH).
Definition unlang_priv.h:58
unlang_t const * instruction
The unlang node we're evaluating.
rlm_rcode_t result
The result from executing the instruction.
struct unlang_s unlang_t
Definition unlang_priv.h:98
unlang_type_t type
The specialisation of this node.
unlang_t * children
Children beneath this group.
Generic representation of a grouping.
An unlang operation.
A node in a graph of unlang_op_t (s) that we execute.
Our interpreter stack, as distinct from the C stack.
static char const * fr_type_to_str(fr_type_t type)
Return a static string containing the type name.
Definition types.h:433
void fr_value_box_copy_shallow(TALLOC_CTX *ctx, fr_value_box_t *dst, fr_value_box_t const *src)
Perform a shallow copy of a value_box.
Definition value.c:3864