The FreeRADIUS server $Id: 15bac2a4c627c01d1aa2047687b3418955ac7f00 $
Loading...
Searching...
No Matches
finally.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: 244103f7d079dcd3173e8a68d95bd778b68a3900 $
19 *
20 * @file unlang/finally.c
21 * @brief Unlang "finally" keyword evaluation. Used for running policy after a virtual server.
22 *
23 * @copyright 2025 Network RAIDUS SAS (legal@networkradius.com)
24 */
25
26RCSID("$Id: 244103f7d079dcd3173e8a68d95bd778b68a3900 $")
27
28#include <freeradius-devel/server/state.h>
29#include <freeradius-devel/server/pair.h>
30#include <freeradius-devel/unlang/unlang_priv.h>
31#include <freeradius-devel/unlang/finally.h>
32#include <freeradius-devel/util/timer.h>
33
34typedef struct {
35 fr_time_delta_t min_time; //!< minimum time to run the finally instruction.
37 unlang_result_t result; //!< Result of the finally instruction. We discard this.
38 rlm_rcode_t original_rcode; //!< The original request rcode when we entered.
39 unlang_t *instruction; //!< to run on timeout
41
43{
44 unlang_frame_state_finally_t *state = talloc_get_type_abort(ctx, unlang_frame_state_finally_t);
45 request_t *request = talloc_get_type_abort(state->request, request_t);
46
47 RDEBUG("Timeout reached, exiting finally section");
48
49 /*
50 * Cancels all frames (other than other finally sections)
51 * and marks the request runnable again.
52 */
54}
55
57{
58 unlang_frame_state_finally_t *state = talloc_get_type_abort(frame->state, unlang_frame_state_finally_t);
59
60 /*
61 * Reset the request->rcode, so that any other
62 * finally sections have access to the original
63 * rcode like 'timeout'.
64 */
65 request->rcode = state->original_rcode;
66
68}
69
71{
72 unlang_frame_state_finally_t *state = talloc_get_type_abort(frame->state, unlang_frame_state_finally_t);
73
74 state->original_rcode = request->rcode;
75
76 /*
77 * Ensure the request has at least min_time to continue
78 * executing before we cancel it.
79 */
80 if (request->timeout && fr_time_delta_lt(state->min_time, fr_timer_remaining(request->timeout))) {
82 unlang_interpret_event_list(request)->tl, &request->timeout,
83 state->min_time, false, unlang_timeout_handler, state) < 0)) {
84 fail:
85 unlang_interpret_signal(request, FR_SIGNAL_CANCEL); /* also stops the request and does cleanups */
86 return UNLANG_ACTION_FAIL;
87 }
88 }
89
90 /*
91 * Finally should be transparent to allow the rcode from
92 * process module to propagate back up, if there are no
93 * modules called.
94 */
97 goto fail;
98 }
99
101
102 /*
103 * Set a timer to cancel the request. If we don't do this
104 * then the request may never finish.
105 */
107}
108
109/** Push a finally instruction on the stack, to be evaluated as the stack is unwound
110 *
111 * @param[in] request to push timeout onto
112 * @param[in] instruction to run as we unwind
113 * @param[in] min_time max time to wait for the finally instruction to finish.
114 * This only applies if the request timeout timer has already
115 * fired, or has less than max_time to execute. i.e. this is
116 * a guarantee of a minimum amount of time for the finally
117 * instruction to run.
118 * @param[in] top_frame Set to UNLANG_TOP_FRAME if the interpreter should return.
119 * Set to UNLANG_SUB_FRAME if the interpreter should continue.
120 *
121 * @return
122 * - 0 on success.
123 * - -1 on failure.
124 */
125int unlang_finally_push_instruction(request_t *request, void *instruction, fr_time_delta_t min_time, bool top_frame)
126{
127 /** Static instruction for performing xlat evaluations
128 *
129 */
130 static unlang_t finally_instruction = {
132 .name = "finally",
133 .debug_name = "finally",
135 };
136
138 unlang_stack_t *stack = request->stack;
140
141 /*
142 * Push a new finally frame onto the stack
143 *
144 * This frame is uncancellable, and will always
145 * execute before the request completes.
146 *
147 * Its children are very much cancellable though
148 * and will be cancelled if min_time or the request
149 * timer expires.
150 */
151 if (unlang_interpret_push(NULL, request, &finally_instruction,
152 FRAME_CONF(RLM_MODULE_NOT_SET, top_frame), UNLANG_NEXT_STOP) < 0) return -1;
153 frame = &stack->frame[stack->depth];
154
155 /*
156 * Allocate its state
157 */
158 MEM(frame->state = state = talloc_zero(stack, unlang_frame_state_finally_t));
159 state->instruction = instruction;
160 state->request = request;
161 state->min_time = min_time;
162
163 frame_repeat(frame, unlang_finally); /* execute immediately... or when unwinding */
164
165 return 0;
166
167}
168
170{
172 .name = "finally",
173 .type = UNLANG_TYPE_FINALLY,
174
175 .interpret = unlang_finally,
176
177 /*
178 * No debug braces, the thing
179 * that's pushed in unlang
180 * finally should have braces
181 */
183
184 .frame_state_size = sizeof(unlang_frame_state_finally_t),
185 .frame_state_type = "unlang_frame_state_finally_t",
186 });
187}
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_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:506
#define unlikely(_x)
Definition build.h:402
#define UNUSED
Definition build.h:336
#define MEM(x)
Definition debug.h:46
void unlang_finally_init(void)
Definition finally.c:169
unlang_result_t result
Result of the finally instruction. We discard this.
Definition finally.c:37
static void unlang_timeout_handler(UNUSED fr_timer_list_t *tl, UNUSED fr_time_t now, void *ctx)
Definition finally.c:42
rlm_rcode_t original_rcode
The original request rcode when we entered.
Definition finally.c:38
static unlang_action_t unlang_finally_resume(UNUSED unlang_result_t *p_result, request_t *request, unlang_stack_frame_t *frame)
Definition finally.c:56
fr_time_delta_t min_time
minimum time to run the finally instruction.
Definition finally.c:35
unlang_t * instruction
to run on timeout
Definition finally.c:39
static unlang_action_t unlang_finally(UNUSED unlang_result_t *p_result, request_t *request, unlang_stack_frame_t *frame)
Definition finally.c:70
int unlang_finally_push_instruction(request_t *request, void *instruction, fr_time_delta_t min_time, bool top_frame)
Push a finally instruction on the stack, to be evaluated as the stack is unwound.
Definition finally.c:125
TALLOC_CTX * unlang_interpret_frame_talloc_ctx(request_t *request)
Get a talloc_ctx which is valid only for this frame.
Definition interpret.c:1681
int unlang_interpret_push_instruction(unlang_result_t *p_result, request_t *request, void *instruction, unlang_frame_conf_t const *conf)
Push an instruction onto the request stack for later interpretation.
Definition interpret.c:1192
void unlang_interpret_signal(request_t *request, fr_signal_t action)
Send a signal (usually stop) to a request.
Definition interpret.c:1417
int unlang_interpret_push(unlang_result_t *p_result, request_t *request, unlang_t const *instruction, unlang_frame_conf_t const *conf, bool do_next_sibling)
Push a new frame onto the stack.
Definition interpret.c:278
fr_event_list_t * unlang_interpret_event_list(request_t *request)
Get the event list for the current interpreter.
Definition interpret.c:2053
#define FRAME_CONF(_default_rcode, _top_frame)
Definition interpret.h:152
#define UNLANG_SUB_FRAME
Definition interpret.h:37
void unlang_register(unlang_op_t *op)
Register an operation with the interpreter.
Definition base.c:56
static char * stack[MAX_STACK]
Definition radmin.c:158
#define MOD_ACTIONS_FAIL_TIMEOUT_RETURN
Definition mod_action.h:74
#define RDEBUG(fmt,...)
rlm_rcode_t
Return codes indicating the result of the module call.
Definition rcode.h:44
@ RLM_MODULE_NOT_SET
Error resolving rcode (should not be returned by modules).
Definition rcode.h:45
@ RLM_MODULE_NOOP
Module succeeded without doing anything.
Definition rcode.h:54
@ FR_SIGNAL_CANCEL
Request has been cancelled.
Definition signal.h:40
#define fr_time_delta_lt(_a, _b)
Definition time.h:285
A time delta, a difference in time measured in nanoseconds.
Definition time.h:80
"server local" time.
Definition time.h:69
fr_time_delta_t fr_timer_remaining(fr_timer_t *ev)
Return time delta between now and when the timer should fire.
Definition timer.c:729
An event timer list.
Definition timer.c:49
#define fr_timer_in(...)
Definition timer.h:87
void * state
Stack frame specialisations.
#define UNLANG_NEXT_STOP
Definition unlang_priv.h:98
@ UNLANG_TYPE_FINALLY
run at the end of a virtual server.
Definition unlang_priv.h:78
static void frame_repeat(unlang_stack_frame_t *frame, unlang_process_t process)
Mark the current stack frame up for repeat, and set a new process function.
@ UNLANG_OP_FLAG_RETURN_POINT
Return point.
@ UNLANG_OP_FLAG_NO_FORCE_UNWIND
Must not be cancelled.
@ UNLANG_OP_FLAG_INTERNAL
it's not a real keyword
unlang_type_t type
The specialisation of this node.
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.
An unlang stack associated with a request.