Branch data Line data Source code
1 : : // SPDX-License-Identifier: BSD-3-Clause OR GPL-2.0
2 : : /******************************************************************************
3 : : *
4 : : * Module Name: dscontrol - Support for execution control opcodes -
5 : : * if/else/while/return
6 : : *
7 : : * Copyright (C) 2000 - 2020, Intel Corp.
8 : : *
9 : : *****************************************************************************/
10 : :
11 : : #include <acpi/acpi.h>
12 : : #include "accommon.h"
13 : : #include "amlcode.h"
14 : : #include "acdispat.h"
15 : : #include "acinterp.h"
16 : : #include "acdebug.h"
17 : :
18 : : #define _COMPONENT ACPI_DISPATCHER
19 : : ACPI_MODULE_NAME("dscontrol")
20 : :
21 : : /*******************************************************************************
22 : : *
23 : : * FUNCTION: acpi_ds_exec_begin_control_op
24 : : *
25 : : * PARAMETERS: walk_list - The list that owns the walk stack
26 : : * op - The control Op
27 : : *
28 : : * RETURN: Status
29 : : *
30 : : * DESCRIPTION: Handles all control ops encountered during control method
31 : : * execution.
32 : : *
33 : : ******************************************************************************/
34 : : acpi_status
35 : 2562 : acpi_ds_exec_begin_control_op(struct acpi_walk_state *walk_state,
36 : : union acpi_parse_object *op)
37 : : {
38 : 2562 : acpi_status status = AE_OK;
39 : 2562 : union acpi_generic_state *control_state;
40 : :
41 : 2562 : ACPI_FUNCTION_NAME(ds_exec_begin_control_op);
42 : :
43 : : ACPI_DEBUG_PRINT((ACPI_DB_DISPATCH, "Op=%p Opcode=%2.2X State=%p\n",
44 : 2562 : op, op->common.aml_opcode, walk_state));
45 : :
46 [ + + + + ]: 2562 : switch (op->common.aml_opcode) {
47 : 387 : case AML_WHILE_OP:
48 : : /*
49 : : * If this is an additional iteration of a while loop, continue.
50 : : * There is no need to allocate a new control state.
51 : : */
52 [ + + ]: 387 : if (walk_state->control_state) {
53 : 384 : if (walk_state->control_state->control.
54 : : aml_predicate_start ==
55 [ + - ]: 384 : (walk_state->parser_state.aml - 1)) {
56 : :
57 : : /* Reset the state to start-of-loop */
58 : :
59 : 384 : walk_state->control_state->common.state =
60 : : ACPI_CONTROL_CONDITIONAL_EXECUTING;
61 : 384 : break;
62 : : }
63 : : }
64 : :
65 : : /*lint -fallthrough */
66 : :
67 : : case AML_IF_OP:
68 : : /*
69 : : * IF/WHILE: Create a new control state to manage these
70 : : * constructs. We need to manage these as a stack, in order
71 : : * to handle nesting.
72 : : */
73 : 1779 : control_state = acpi_ut_create_control_state();
74 [ + - ]: 1779 : if (!control_state) {
75 : : status = AE_NO_MEMORY;
76 : : break;
77 : : }
78 : : /*
79 : : * Save a pointer to the predicate for multiple executions
80 : : * of a loop
81 : : */
82 : 1779 : control_state->control.aml_predicate_start =
83 : 1779 : walk_state->parser_state.aml - 1;
84 : 1779 : control_state->control.package_end =
85 : 1779 : walk_state->parser_state.pkg_end;
86 : 1779 : control_state->control.opcode = op->common.aml_opcode;
87 : 1779 : control_state->control.loop_timeout = acpi_os_get_timer() +
88 : 1779 : ((u64)acpi_gbl_max_loop_iterations * ACPI_100NSEC_PER_SEC);
89 : :
90 : : /* Push the control state on this walk's control stack */
91 : :
92 : 1779 : acpi_ut_push_generic_state(&walk_state->control_state,
93 : : control_state);
94 : 1779 : break;
95 : :
96 : 120 : case AML_ELSE_OP:
97 : :
98 : : /* Predicate is in the state object */
99 : : /* If predicate is true, the IF was executed, ignore ELSE part */
100 : :
101 [ + + ]: 120 : if (walk_state->last_predicate) {
102 : 3 : status = AE_CTRL_TRUE;
103 : : }
104 : :
105 : : break;
106 : :
107 : : case AML_RETURN_OP:
108 : :
109 : : break;
110 : :
111 : : default:
112 : :
113 : : break;
114 : : }
115 : :
116 : 2562 : return (status);
117 : : }
118 : :
119 : : /*******************************************************************************
120 : : *
121 : : * FUNCTION: acpi_ds_exec_end_control_op
122 : : *
123 : : * PARAMETERS: walk_list - The list that owns the walk stack
124 : : * op - The control Op
125 : : *
126 : : * RETURN: Status
127 : : *
128 : : * DESCRIPTION: Handles all control ops encountered during control method
129 : : * execution.
130 : : *
131 : : ******************************************************************************/
132 : :
133 : : acpi_status
134 : 2499 : acpi_ds_exec_end_control_op(struct acpi_walk_state *walk_state,
135 : : union acpi_parse_object *op)
136 : : {
137 : 2499 : acpi_status status = AE_OK;
138 : 2499 : union acpi_generic_state *control_state;
139 : :
140 : 2499 : ACPI_FUNCTION_NAME(ds_exec_end_control_op);
141 : :
142 [ + + + - : 2499 : switch (op->common.aml_opcode) {
- - + ]
143 : 1740 : case AML_IF_OP:
144 : :
145 : 1740 : ACPI_DEBUG_PRINT((ACPI_DB_DISPATCH, "[IF_OP] Op=%p\n", op));
146 : :
147 : : /*
148 : : * Save the result of the predicate in case there is an
149 : : * ELSE to come
150 : : */
151 : 1740 : walk_state->last_predicate =
152 : 1740 : (u8)walk_state->control_state->common.value;
153 : :
154 : : /*
155 : : * Pop the control state that was created at the start
156 : : * of the IF and free it
157 : : */
158 : 1740 : control_state =
159 : 1740 : acpi_ut_pop_generic_state(&walk_state->control_state);
160 : 1740 : acpi_ut_delete_generic_state(control_state);
161 : 1740 : break;
162 : :
163 : : case AML_ELSE_OP:
164 : :
165 : : break;
166 : :
167 : 387 : case AML_WHILE_OP:
168 : :
169 : 387 : ACPI_DEBUG_PRINT((ACPI_DB_DISPATCH, "[WHILE_OP] Op=%p\n", op));
170 : :
171 : 387 : control_state = walk_state->control_state;
172 [ + + ]: 387 : if (control_state->common.value) {
173 : :
174 : : /* Predicate was true, the body of the loop was just executed */
175 : :
176 : : /*
177 : : * This infinite loop detection mechanism allows the interpreter
178 : : * to escape possibly infinite loops. This can occur in poorly
179 : : * written AML when the hardware does not respond within a while
180 : : * loop and the loop does not implement a timeout.
181 : : */
182 [ + - ]: 384 : if (ACPI_TIME_AFTER(acpi_os_get_timer(),
183 : : control_state->control.
184 : : loop_timeout)) {
185 : : status = AE_AML_LOOP_TIMEOUT;
186 : : break;
187 : : }
188 : :
189 : : /*
190 : : * Go back and evaluate the predicate and maybe execute the loop
191 : : * another time
192 : : */
193 : 384 : status = AE_CTRL_PENDING;
194 : 384 : walk_state->aml_last_while =
195 : 384 : control_state->control.aml_predicate_start;
196 : 384 : break;
197 : : }
198 : :
199 : : /* Predicate was false, terminate this while loop */
200 : :
201 : : ACPI_DEBUG_PRINT((ACPI_DB_DISPATCH,
202 : 3 : "[WHILE_OP] termination! Op=%p\n", op));
203 : :
204 : : /* Pop this control state and free it */
205 : :
206 : 3 : control_state =
207 : 3 : acpi_ut_pop_generic_state(&walk_state->control_state);
208 : 3 : acpi_ut_delete_generic_state(control_state);
209 : 3 : break;
210 : :
211 : 279 : case AML_RETURN_OP:
212 : :
213 : : ACPI_DEBUG_PRINT((ACPI_DB_DISPATCH,
214 : : "[RETURN_OP] Op=%p Arg=%p\n", op,
215 : 279 : op->common.value.arg));
216 : :
217 : : /*
218 : : * One optional operand -- the return value
219 : : * It can be either an immediate operand or a result that
220 : : * has been bubbled up the tree
221 : : */
222 [ + + ]: 279 : if (op->common.value.arg) {
223 : :
224 : : /* Since we have a real Return(), delete any implicit return */
225 : :
226 : 195 : acpi_ds_clear_implicit_return(walk_state);
227 : :
228 : : /* Return statement has an immediate operand */
229 : :
230 : 195 : status =
231 : 195 : acpi_ds_create_operands(walk_state,
232 : : op->common.value.arg);
233 [ + - ]: 195 : if (ACPI_FAILURE(status)) {
234 : : return (status);
235 : : }
236 : :
237 : : /*
238 : : * If value being returned is a Reference (such as
239 : : * an arg or local), resolve it now because it may
240 : : * cease to exist at the end of the method.
241 : : */
242 : 195 : status =
243 : 195 : acpi_ex_resolve_to_value(&walk_state->operands[0],
244 : : walk_state);
245 [ + - ]: 195 : if (ACPI_FAILURE(status)) {
246 : : return (status);
247 : : }
248 : :
249 : : /*
250 : : * Get the return value and save as the last result
251 : : * value. This is the only place where walk_state->return_desc
252 : : * is set to anything other than zero!
253 : : */
254 : 195 : walk_state->return_desc = walk_state->operands[0];
255 [ + - ]: 84 : } else if (walk_state->result_count) {
256 : :
257 : : /* Since we have a real Return(), delete any implicit return */
258 : :
259 : 84 : acpi_ds_clear_implicit_return(walk_state);
260 : :
261 : : /*
262 : : * The return value has come from a previous calculation.
263 : : *
264 : : * If value being returned is a Reference (such as
265 : : * an arg or local), resolve it now because it may
266 : : * cease to exist at the end of the method.
267 : : *
268 : : * Allow references created by the Index operator to return
269 : : * unchanged.
270 : : */
271 [ + - ]: 84 : if ((ACPI_GET_DESCRIPTOR_TYPE
272 : : (walk_state->results->results.obj_desc[0]) ==
273 : : ACPI_DESC_TYPE_OPERAND)
274 : 84 : && ((walk_state->results->results.obj_desc[0])->
275 [ - + ]: 84 : common.type == ACPI_TYPE_LOCAL_REFERENCE)
276 : 0 : && ((walk_state->results->results.obj_desc[0])->
277 [ # # ]: 0 : reference.class != ACPI_REFCLASS_INDEX)) {
278 : 0 : status =
279 : 0 : acpi_ex_resolve_to_value(&walk_state->
280 : : results->results.
281 : : obj_desc[0],
282 : : walk_state);
283 [ # # ]: 0 : if (ACPI_FAILURE(status)) {
284 : : return (status);
285 : : }
286 : : }
287 : :
288 : 84 : walk_state->return_desc =
289 : 84 : walk_state->results->results.obj_desc[0];
290 : : } else {
291 : : /* No return operand */
292 : :
293 [ # # ]: 0 : if (walk_state->num_operands) {
294 : 0 : acpi_ut_remove_reference(walk_state->
295 : : operands[0]);
296 : : }
297 : :
298 : 0 : walk_state->operands[0] = NULL;
299 : 0 : walk_state->num_operands = 0;
300 : 0 : walk_state->return_desc = NULL;
301 : : }
302 : :
303 : : ACPI_DEBUG_PRINT((ACPI_DB_DISPATCH,
304 : : "Completed RETURN_OP State=%p, RetVal=%p\n",
305 : : walk_state, walk_state->return_desc));
306 : :
307 : : /* End the control method execution right now */
308 : :
309 : : status = AE_CTRL_TERMINATE;
310 : : break;
311 : :
312 : : case AML_NOOP_OP:
313 : :
314 : : /* Just do nothing! */
315 : :
316 : : break;
317 : :
318 : : case AML_BREAKPOINT_OP:
319 : :
320 : 0 : acpi_db_signal_break_point(walk_state);
321 : :
322 : : /* Call to the OSL in case OS wants a piece of the action */
323 : :
324 : 0 : status = acpi_os_signal(ACPI_SIGNAL_BREAKPOINT,
325 : : "Executed AML Breakpoint opcode");
326 : 0 : break;
327 : :
328 : : case AML_BREAK_OP:
329 : : case AML_CONTINUE_OP: /* ACPI 2.0 */
330 : :
331 : : /* Pop and delete control states until we find a while */
332 : :
333 [ # # ]: 0 : while (walk_state->control_state &&
334 [ # # ]: 0 : (walk_state->control_state->control.opcode !=
335 : : AML_WHILE_OP)) {
336 : 0 : control_state =
337 : 0 : acpi_ut_pop_generic_state(&walk_state->
338 : : control_state);
339 : 0 : acpi_ut_delete_generic_state(control_state);
340 : : }
341 : :
342 : : /* No while found? */
343 : :
344 [ # # ]: 0 : if (!walk_state->control_state) {
345 : : return (AE_AML_NO_WHILE);
346 : : }
347 : :
348 : : /* Was: walk_state->aml_last_while = walk_state->control_state->Control.aml_predicate_start; */
349 : :
350 : 0 : walk_state->aml_last_while =
351 : 0 : walk_state->control_state->control.package_end;
352 : :
353 : : /* Return status depending on opcode */
354 : :
355 [ # # ]: 0 : if (op->common.aml_opcode == AML_BREAK_OP) {
356 : : status = AE_CTRL_BREAK;
357 : : } else {
358 : 0 : status = AE_CTRL_CONTINUE;
359 : : }
360 : : break;
361 : :
362 : 0 : default:
363 : :
364 : 0 : ACPI_ERROR((AE_INFO, "Unknown control opcode=0x%X Op=%p",
365 : : op->common.aml_opcode, op));
366 : :
367 : 0 : status = AE_AML_BAD_OPCODE;
368 : 0 : break;
369 : : }
370 : :
371 : : return (status);
372 : : }
|