xref: /OK3568_Linux_fs/kernel/drivers/acpi/acpica/dswstate.c (revision 4882a59341e53eb6f0b4789bf948001014eff981)
1*4882a593Smuzhiyun // SPDX-License-Identifier: BSD-3-Clause OR GPL-2.0
2*4882a593Smuzhiyun /******************************************************************************
3*4882a593Smuzhiyun  *
4*4882a593Smuzhiyun  * Module Name: dswstate - Dispatcher parse tree walk management routines
5*4882a593Smuzhiyun  *
6*4882a593Smuzhiyun  * Copyright (C) 2000 - 2020, Intel Corp.
7*4882a593Smuzhiyun  *
8*4882a593Smuzhiyun  *****************************************************************************/
9*4882a593Smuzhiyun 
10*4882a593Smuzhiyun #include <acpi/acpi.h>
11*4882a593Smuzhiyun #include "accommon.h"
12*4882a593Smuzhiyun #include "acparser.h"
13*4882a593Smuzhiyun #include "acdispat.h"
14*4882a593Smuzhiyun #include "acnamesp.h"
15*4882a593Smuzhiyun 
16*4882a593Smuzhiyun #define _COMPONENT          ACPI_DISPATCHER
17*4882a593Smuzhiyun ACPI_MODULE_NAME("dswstate")
18*4882a593Smuzhiyun 
19*4882a593Smuzhiyun   /* Local prototypes */
20*4882a593Smuzhiyun static acpi_status
21*4882a593Smuzhiyun acpi_ds_result_stack_push(struct acpi_walk_state *walk_state);
22*4882a593Smuzhiyun static acpi_status acpi_ds_result_stack_pop(struct acpi_walk_state *walk_state);
23*4882a593Smuzhiyun 
24*4882a593Smuzhiyun /*******************************************************************************
25*4882a593Smuzhiyun  *
26*4882a593Smuzhiyun  * FUNCTION:    acpi_ds_result_pop
27*4882a593Smuzhiyun  *
28*4882a593Smuzhiyun  * PARAMETERS:  object              - Where to return the popped object
29*4882a593Smuzhiyun  *              walk_state          - Current Walk state
30*4882a593Smuzhiyun  *
31*4882a593Smuzhiyun  * RETURN:      Status
32*4882a593Smuzhiyun  *
33*4882a593Smuzhiyun  * DESCRIPTION: Pop an object off the top of this walk's result stack
34*4882a593Smuzhiyun  *
35*4882a593Smuzhiyun  ******************************************************************************/
36*4882a593Smuzhiyun 
37*4882a593Smuzhiyun acpi_status
acpi_ds_result_pop(union acpi_operand_object ** object,struct acpi_walk_state * walk_state)38*4882a593Smuzhiyun acpi_ds_result_pop(union acpi_operand_object **object,
39*4882a593Smuzhiyun 		   struct acpi_walk_state *walk_state)
40*4882a593Smuzhiyun {
41*4882a593Smuzhiyun 	u32 index;
42*4882a593Smuzhiyun 	union acpi_generic_state *state;
43*4882a593Smuzhiyun 	acpi_status status;
44*4882a593Smuzhiyun 
45*4882a593Smuzhiyun 	ACPI_FUNCTION_NAME(ds_result_pop);
46*4882a593Smuzhiyun 
47*4882a593Smuzhiyun 	state = walk_state->results;
48*4882a593Smuzhiyun 
49*4882a593Smuzhiyun 	/* Incorrect state of result stack */
50*4882a593Smuzhiyun 
51*4882a593Smuzhiyun 	if (state && !walk_state->result_count) {
52*4882a593Smuzhiyun 		ACPI_ERROR((AE_INFO, "No results on result stack"));
53*4882a593Smuzhiyun 		return (AE_AML_INTERNAL);
54*4882a593Smuzhiyun 	}
55*4882a593Smuzhiyun 
56*4882a593Smuzhiyun 	if (!state && walk_state->result_count) {
57*4882a593Smuzhiyun 		ACPI_ERROR((AE_INFO, "No result state for result stack"));
58*4882a593Smuzhiyun 		return (AE_AML_INTERNAL);
59*4882a593Smuzhiyun 	}
60*4882a593Smuzhiyun 
61*4882a593Smuzhiyun 	/* Empty result stack */
62*4882a593Smuzhiyun 
63*4882a593Smuzhiyun 	if (!state) {
64*4882a593Smuzhiyun 		ACPI_ERROR((AE_INFO, "Result stack is empty! State=%p",
65*4882a593Smuzhiyun 			    walk_state));
66*4882a593Smuzhiyun 		return (AE_AML_NO_RETURN_VALUE);
67*4882a593Smuzhiyun 	}
68*4882a593Smuzhiyun 
69*4882a593Smuzhiyun 	/* Return object of the top element and clean that top element result stack */
70*4882a593Smuzhiyun 
71*4882a593Smuzhiyun 	walk_state->result_count--;
72*4882a593Smuzhiyun 	index = (u32)walk_state->result_count % ACPI_RESULTS_FRAME_OBJ_NUM;
73*4882a593Smuzhiyun 
74*4882a593Smuzhiyun 	*object = state->results.obj_desc[index];
75*4882a593Smuzhiyun 	if (!*object) {
76*4882a593Smuzhiyun 		ACPI_ERROR((AE_INFO,
77*4882a593Smuzhiyun 			    "No result objects on result stack, State=%p",
78*4882a593Smuzhiyun 			    walk_state));
79*4882a593Smuzhiyun 		return (AE_AML_NO_RETURN_VALUE);
80*4882a593Smuzhiyun 	}
81*4882a593Smuzhiyun 
82*4882a593Smuzhiyun 	state->results.obj_desc[index] = NULL;
83*4882a593Smuzhiyun 	if (index == 0) {
84*4882a593Smuzhiyun 		status = acpi_ds_result_stack_pop(walk_state);
85*4882a593Smuzhiyun 		if (ACPI_FAILURE(status)) {
86*4882a593Smuzhiyun 			return (status);
87*4882a593Smuzhiyun 		}
88*4882a593Smuzhiyun 	}
89*4882a593Smuzhiyun 
90*4882a593Smuzhiyun 	ACPI_DEBUG_PRINT((ACPI_DB_EXEC,
91*4882a593Smuzhiyun 			  "Obj=%p [%s] Index=%X State=%p Num=%X\n", *object,
92*4882a593Smuzhiyun 			  acpi_ut_get_object_type_name(*object),
93*4882a593Smuzhiyun 			  index, walk_state, walk_state->result_count));
94*4882a593Smuzhiyun 
95*4882a593Smuzhiyun 	return (AE_OK);
96*4882a593Smuzhiyun }
97*4882a593Smuzhiyun 
98*4882a593Smuzhiyun /*******************************************************************************
99*4882a593Smuzhiyun  *
100*4882a593Smuzhiyun  * FUNCTION:    acpi_ds_result_push
101*4882a593Smuzhiyun  *
102*4882a593Smuzhiyun  * PARAMETERS:  object              - Where to return the popped object
103*4882a593Smuzhiyun  *              walk_state          - Current Walk state
104*4882a593Smuzhiyun  *
105*4882a593Smuzhiyun  * RETURN:      Status
106*4882a593Smuzhiyun  *
107*4882a593Smuzhiyun  * DESCRIPTION: Push an object onto the current result stack
108*4882a593Smuzhiyun  *
109*4882a593Smuzhiyun  ******************************************************************************/
110*4882a593Smuzhiyun 
111*4882a593Smuzhiyun acpi_status
acpi_ds_result_push(union acpi_operand_object * object,struct acpi_walk_state * walk_state)112*4882a593Smuzhiyun acpi_ds_result_push(union acpi_operand_object *object,
113*4882a593Smuzhiyun 		    struct acpi_walk_state *walk_state)
114*4882a593Smuzhiyun {
115*4882a593Smuzhiyun 	union acpi_generic_state *state;
116*4882a593Smuzhiyun 	acpi_status status;
117*4882a593Smuzhiyun 	u32 index;
118*4882a593Smuzhiyun 
119*4882a593Smuzhiyun 	ACPI_FUNCTION_NAME(ds_result_push);
120*4882a593Smuzhiyun 
121*4882a593Smuzhiyun 	if (walk_state->result_count > walk_state->result_size) {
122*4882a593Smuzhiyun 		ACPI_ERROR((AE_INFO, "Result stack is full"));
123*4882a593Smuzhiyun 		return (AE_AML_INTERNAL);
124*4882a593Smuzhiyun 	} else if (walk_state->result_count == walk_state->result_size) {
125*4882a593Smuzhiyun 
126*4882a593Smuzhiyun 		/* Extend the result stack */
127*4882a593Smuzhiyun 
128*4882a593Smuzhiyun 		status = acpi_ds_result_stack_push(walk_state);
129*4882a593Smuzhiyun 		if (ACPI_FAILURE(status)) {
130*4882a593Smuzhiyun 			ACPI_ERROR((AE_INFO,
131*4882a593Smuzhiyun 				    "Failed to extend the result stack"));
132*4882a593Smuzhiyun 			return (status);
133*4882a593Smuzhiyun 		}
134*4882a593Smuzhiyun 	}
135*4882a593Smuzhiyun 
136*4882a593Smuzhiyun 	if (!(walk_state->result_count < walk_state->result_size)) {
137*4882a593Smuzhiyun 		ACPI_ERROR((AE_INFO, "No free elements in result stack"));
138*4882a593Smuzhiyun 		return (AE_AML_INTERNAL);
139*4882a593Smuzhiyun 	}
140*4882a593Smuzhiyun 
141*4882a593Smuzhiyun 	state = walk_state->results;
142*4882a593Smuzhiyun 	if (!state) {
143*4882a593Smuzhiyun 		ACPI_ERROR((AE_INFO, "No result stack frame during push"));
144*4882a593Smuzhiyun 		return (AE_AML_INTERNAL);
145*4882a593Smuzhiyun 	}
146*4882a593Smuzhiyun 
147*4882a593Smuzhiyun 	if (!object) {
148*4882a593Smuzhiyun 		ACPI_ERROR((AE_INFO,
149*4882a593Smuzhiyun 			    "Null Object! Obj=%p State=%p Num=%u",
150*4882a593Smuzhiyun 			    object, walk_state, walk_state->result_count));
151*4882a593Smuzhiyun 		return (AE_BAD_PARAMETER);
152*4882a593Smuzhiyun 	}
153*4882a593Smuzhiyun 
154*4882a593Smuzhiyun 	/* Assign the address of object to the top free element of result stack */
155*4882a593Smuzhiyun 
156*4882a593Smuzhiyun 	index = (u32)walk_state->result_count % ACPI_RESULTS_FRAME_OBJ_NUM;
157*4882a593Smuzhiyun 	state->results.obj_desc[index] = object;
158*4882a593Smuzhiyun 	walk_state->result_count++;
159*4882a593Smuzhiyun 
160*4882a593Smuzhiyun 	ACPI_DEBUG_PRINT((ACPI_DB_EXEC, "Obj=%p [%s] State=%p Num=%X Cur=%X\n",
161*4882a593Smuzhiyun 			  object,
162*4882a593Smuzhiyun 			  acpi_ut_get_object_type_name((union
163*4882a593Smuzhiyun 							acpi_operand_object *)
164*4882a593Smuzhiyun 						       object), walk_state,
165*4882a593Smuzhiyun 			  walk_state->result_count,
166*4882a593Smuzhiyun 			  walk_state->current_result));
167*4882a593Smuzhiyun 
168*4882a593Smuzhiyun 	return (AE_OK);
169*4882a593Smuzhiyun }
170*4882a593Smuzhiyun 
171*4882a593Smuzhiyun /*******************************************************************************
172*4882a593Smuzhiyun  *
173*4882a593Smuzhiyun  * FUNCTION:    acpi_ds_result_stack_push
174*4882a593Smuzhiyun  *
175*4882a593Smuzhiyun  * PARAMETERS:  walk_state          - Current Walk state
176*4882a593Smuzhiyun  *
177*4882a593Smuzhiyun  * RETURN:      Status
178*4882a593Smuzhiyun  *
179*4882a593Smuzhiyun  * DESCRIPTION: Push an object onto the walk_state result stack
180*4882a593Smuzhiyun  *
181*4882a593Smuzhiyun  ******************************************************************************/
182*4882a593Smuzhiyun 
acpi_ds_result_stack_push(struct acpi_walk_state * walk_state)183*4882a593Smuzhiyun static acpi_status acpi_ds_result_stack_push(struct acpi_walk_state *walk_state)
184*4882a593Smuzhiyun {
185*4882a593Smuzhiyun 	union acpi_generic_state *state;
186*4882a593Smuzhiyun 
187*4882a593Smuzhiyun 	ACPI_FUNCTION_NAME(ds_result_stack_push);
188*4882a593Smuzhiyun 
189*4882a593Smuzhiyun 	/* Check for stack overflow */
190*4882a593Smuzhiyun 
191*4882a593Smuzhiyun 	if (((u32) walk_state->result_size + ACPI_RESULTS_FRAME_OBJ_NUM) >
192*4882a593Smuzhiyun 	    ACPI_RESULTS_OBJ_NUM_MAX) {
193*4882a593Smuzhiyun 		ACPI_ERROR((AE_INFO, "Result stack overflow: State=%p Num=%u",
194*4882a593Smuzhiyun 			    walk_state, walk_state->result_size));
195*4882a593Smuzhiyun 		return (AE_STACK_OVERFLOW);
196*4882a593Smuzhiyun 	}
197*4882a593Smuzhiyun 
198*4882a593Smuzhiyun 	state = acpi_ut_create_generic_state();
199*4882a593Smuzhiyun 	if (!state) {
200*4882a593Smuzhiyun 		return (AE_NO_MEMORY);
201*4882a593Smuzhiyun 	}
202*4882a593Smuzhiyun 
203*4882a593Smuzhiyun 	state->common.descriptor_type = ACPI_DESC_TYPE_STATE_RESULT;
204*4882a593Smuzhiyun 	acpi_ut_push_generic_state(&walk_state->results, state);
205*4882a593Smuzhiyun 
206*4882a593Smuzhiyun 	/* Increase the length of the result stack by the length of frame */
207*4882a593Smuzhiyun 
208*4882a593Smuzhiyun 	walk_state->result_size += ACPI_RESULTS_FRAME_OBJ_NUM;
209*4882a593Smuzhiyun 
210*4882a593Smuzhiyun 	ACPI_DEBUG_PRINT((ACPI_DB_EXEC, "Results=%p State=%p\n",
211*4882a593Smuzhiyun 			  state, walk_state));
212*4882a593Smuzhiyun 
213*4882a593Smuzhiyun 	return (AE_OK);
214*4882a593Smuzhiyun }
215*4882a593Smuzhiyun 
216*4882a593Smuzhiyun /*******************************************************************************
217*4882a593Smuzhiyun  *
218*4882a593Smuzhiyun  * FUNCTION:    acpi_ds_result_stack_pop
219*4882a593Smuzhiyun  *
220*4882a593Smuzhiyun  * PARAMETERS:  walk_state          - Current Walk state
221*4882a593Smuzhiyun  *
222*4882a593Smuzhiyun  * RETURN:      Status
223*4882a593Smuzhiyun  *
224*4882a593Smuzhiyun  * DESCRIPTION: Pop an object off of the walk_state result stack
225*4882a593Smuzhiyun  *
226*4882a593Smuzhiyun  ******************************************************************************/
227*4882a593Smuzhiyun 
acpi_ds_result_stack_pop(struct acpi_walk_state * walk_state)228*4882a593Smuzhiyun static acpi_status acpi_ds_result_stack_pop(struct acpi_walk_state *walk_state)
229*4882a593Smuzhiyun {
230*4882a593Smuzhiyun 	union acpi_generic_state *state;
231*4882a593Smuzhiyun 
232*4882a593Smuzhiyun 	ACPI_FUNCTION_NAME(ds_result_stack_pop);
233*4882a593Smuzhiyun 
234*4882a593Smuzhiyun 	/* Check for stack underflow */
235*4882a593Smuzhiyun 
236*4882a593Smuzhiyun 	if (walk_state->results == NULL) {
237*4882a593Smuzhiyun 		ACPI_DEBUG_PRINT((ACPI_DB_EXEC,
238*4882a593Smuzhiyun 				  "Result stack underflow - State=%p\n",
239*4882a593Smuzhiyun 				  walk_state));
240*4882a593Smuzhiyun 		return (AE_AML_NO_OPERAND);
241*4882a593Smuzhiyun 	}
242*4882a593Smuzhiyun 
243*4882a593Smuzhiyun 	if (walk_state->result_size < ACPI_RESULTS_FRAME_OBJ_NUM) {
244*4882a593Smuzhiyun 		ACPI_ERROR((AE_INFO, "Insufficient result stack size"));
245*4882a593Smuzhiyun 		return (AE_AML_INTERNAL);
246*4882a593Smuzhiyun 	}
247*4882a593Smuzhiyun 
248*4882a593Smuzhiyun 	state = acpi_ut_pop_generic_state(&walk_state->results);
249*4882a593Smuzhiyun 	acpi_ut_delete_generic_state(state);
250*4882a593Smuzhiyun 
251*4882a593Smuzhiyun 	/* Decrease the length of result stack by the length of frame */
252*4882a593Smuzhiyun 
253*4882a593Smuzhiyun 	walk_state->result_size -= ACPI_RESULTS_FRAME_OBJ_NUM;
254*4882a593Smuzhiyun 
255*4882a593Smuzhiyun 	ACPI_DEBUG_PRINT((ACPI_DB_EXEC,
256*4882a593Smuzhiyun 			  "Result=%p RemainingResults=%X State=%p\n",
257*4882a593Smuzhiyun 			  state, walk_state->result_count, walk_state));
258*4882a593Smuzhiyun 
259*4882a593Smuzhiyun 	return (AE_OK);
260*4882a593Smuzhiyun }
261*4882a593Smuzhiyun 
262*4882a593Smuzhiyun /*******************************************************************************
263*4882a593Smuzhiyun  *
264*4882a593Smuzhiyun  * FUNCTION:    acpi_ds_obj_stack_push
265*4882a593Smuzhiyun  *
266*4882a593Smuzhiyun  * PARAMETERS:  object              - Object to push
267*4882a593Smuzhiyun  *              walk_state          - Current Walk state
268*4882a593Smuzhiyun  *
269*4882a593Smuzhiyun  * RETURN:      Status
270*4882a593Smuzhiyun  *
271*4882a593Smuzhiyun  * DESCRIPTION: Push an object onto this walk's object/operand stack
272*4882a593Smuzhiyun  *
273*4882a593Smuzhiyun  ******************************************************************************/
274*4882a593Smuzhiyun 
275*4882a593Smuzhiyun acpi_status
acpi_ds_obj_stack_push(void * object,struct acpi_walk_state * walk_state)276*4882a593Smuzhiyun acpi_ds_obj_stack_push(void *object, struct acpi_walk_state *walk_state)
277*4882a593Smuzhiyun {
278*4882a593Smuzhiyun 	ACPI_FUNCTION_NAME(ds_obj_stack_push);
279*4882a593Smuzhiyun 
280*4882a593Smuzhiyun 	/* Check for stack overflow */
281*4882a593Smuzhiyun 
282*4882a593Smuzhiyun 	if (walk_state->num_operands >= ACPI_OBJ_NUM_OPERANDS) {
283*4882a593Smuzhiyun 		ACPI_ERROR((AE_INFO,
284*4882a593Smuzhiyun 			    "Object stack overflow! Obj=%p State=%p #Ops=%u",
285*4882a593Smuzhiyun 			    object, walk_state, walk_state->num_operands));
286*4882a593Smuzhiyun 		return (AE_STACK_OVERFLOW);
287*4882a593Smuzhiyun 	}
288*4882a593Smuzhiyun 
289*4882a593Smuzhiyun 	/* Put the object onto the stack */
290*4882a593Smuzhiyun 
291*4882a593Smuzhiyun 	walk_state->operands[walk_state->operand_index] = object;
292*4882a593Smuzhiyun 	walk_state->num_operands++;
293*4882a593Smuzhiyun 
294*4882a593Smuzhiyun 	/* For the usual order of filling the operand stack */
295*4882a593Smuzhiyun 
296*4882a593Smuzhiyun 	walk_state->operand_index++;
297*4882a593Smuzhiyun 
298*4882a593Smuzhiyun 	ACPI_DEBUG_PRINT((ACPI_DB_EXEC, "Obj=%p [%s] State=%p #Ops=%X\n",
299*4882a593Smuzhiyun 			  object,
300*4882a593Smuzhiyun 			  acpi_ut_get_object_type_name((union
301*4882a593Smuzhiyun 							acpi_operand_object *)
302*4882a593Smuzhiyun 						       object), walk_state,
303*4882a593Smuzhiyun 			  walk_state->num_operands));
304*4882a593Smuzhiyun 
305*4882a593Smuzhiyun 	return (AE_OK);
306*4882a593Smuzhiyun }
307*4882a593Smuzhiyun 
308*4882a593Smuzhiyun /*******************************************************************************
309*4882a593Smuzhiyun  *
310*4882a593Smuzhiyun  * FUNCTION:    acpi_ds_obj_stack_pop
311*4882a593Smuzhiyun  *
312*4882a593Smuzhiyun  * PARAMETERS:  pop_count           - Number of objects/entries to pop
313*4882a593Smuzhiyun  *              walk_state          - Current Walk state
314*4882a593Smuzhiyun  *
315*4882a593Smuzhiyun  * RETURN:      Status
316*4882a593Smuzhiyun  *
317*4882a593Smuzhiyun  * DESCRIPTION: Pop this walk's object stack. Objects on the stack are NOT
318*4882a593Smuzhiyun  *              deleted by this routine.
319*4882a593Smuzhiyun  *
320*4882a593Smuzhiyun  ******************************************************************************/
321*4882a593Smuzhiyun 
322*4882a593Smuzhiyun acpi_status
acpi_ds_obj_stack_pop(u32 pop_count,struct acpi_walk_state * walk_state)323*4882a593Smuzhiyun acpi_ds_obj_stack_pop(u32 pop_count, struct acpi_walk_state *walk_state)
324*4882a593Smuzhiyun {
325*4882a593Smuzhiyun 	u32 i;
326*4882a593Smuzhiyun 
327*4882a593Smuzhiyun 	ACPI_FUNCTION_NAME(ds_obj_stack_pop);
328*4882a593Smuzhiyun 
329*4882a593Smuzhiyun 	for (i = 0; i < pop_count; i++) {
330*4882a593Smuzhiyun 
331*4882a593Smuzhiyun 		/* Check for stack underflow */
332*4882a593Smuzhiyun 
333*4882a593Smuzhiyun 		if (walk_state->num_operands == 0) {
334*4882a593Smuzhiyun 			ACPI_ERROR((AE_INFO,
335*4882a593Smuzhiyun 				    "Object stack underflow! Count=%X State=%p #Ops=%u",
336*4882a593Smuzhiyun 				    pop_count, walk_state,
337*4882a593Smuzhiyun 				    walk_state->num_operands));
338*4882a593Smuzhiyun 			return (AE_STACK_UNDERFLOW);
339*4882a593Smuzhiyun 		}
340*4882a593Smuzhiyun 
341*4882a593Smuzhiyun 		/* Just set the stack entry to null */
342*4882a593Smuzhiyun 
343*4882a593Smuzhiyun 		walk_state->num_operands--;
344*4882a593Smuzhiyun 		walk_state->operands[walk_state->num_operands] = NULL;
345*4882a593Smuzhiyun 	}
346*4882a593Smuzhiyun 
347*4882a593Smuzhiyun 	ACPI_DEBUG_PRINT((ACPI_DB_EXEC, "Count=%X State=%p #Ops=%u\n",
348*4882a593Smuzhiyun 			  pop_count, walk_state, walk_state->num_operands));
349*4882a593Smuzhiyun 
350*4882a593Smuzhiyun 	return (AE_OK);
351*4882a593Smuzhiyun }
352*4882a593Smuzhiyun 
353*4882a593Smuzhiyun /*******************************************************************************
354*4882a593Smuzhiyun  *
355*4882a593Smuzhiyun  * FUNCTION:    acpi_ds_obj_stack_pop_and_delete
356*4882a593Smuzhiyun  *
357*4882a593Smuzhiyun  * PARAMETERS:  pop_count           - Number of objects/entries to pop
358*4882a593Smuzhiyun  *              walk_state          - Current Walk state
359*4882a593Smuzhiyun  *
360*4882a593Smuzhiyun  * RETURN:      Status
361*4882a593Smuzhiyun  *
362*4882a593Smuzhiyun  * DESCRIPTION: Pop this walk's object stack and delete each object that is
363*4882a593Smuzhiyun  *              popped off.
364*4882a593Smuzhiyun  *
365*4882a593Smuzhiyun  ******************************************************************************/
366*4882a593Smuzhiyun 
367*4882a593Smuzhiyun void
acpi_ds_obj_stack_pop_and_delete(u32 pop_count,struct acpi_walk_state * walk_state)368*4882a593Smuzhiyun acpi_ds_obj_stack_pop_and_delete(u32 pop_count,
369*4882a593Smuzhiyun 				 struct acpi_walk_state *walk_state)
370*4882a593Smuzhiyun {
371*4882a593Smuzhiyun 	s32 i;
372*4882a593Smuzhiyun 	union acpi_operand_object *obj_desc;
373*4882a593Smuzhiyun 
374*4882a593Smuzhiyun 	ACPI_FUNCTION_NAME(ds_obj_stack_pop_and_delete);
375*4882a593Smuzhiyun 
376*4882a593Smuzhiyun 	if (pop_count == 0) {
377*4882a593Smuzhiyun 		return;
378*4882a593Smuzhiyun 	}
379*4882a593Smuzhiyun 
380*4882a593Smuzhiyun 	for (i = (s32)pop_count - 1; i >= 0; i--) {
381*4882a593Smuzhiyun 		if (walk_state->num_operands == 0) {
382*4882a593Smuzhiyun 			return;
383*4882a593Smuzhiyun 		}
384*4882a593Smuzhiyun 
385*4882a593Smuzhiyun 		/* Pop the stack and delete an object if present in this stack entry */
386*4882a593Smuzhiyun 
387*4882a593Smuzhiyun 		walk_state->num_operands--;
388*4882a593Smuzhiyun 		obj_desc = walk_state->operands[i];
389*4882a593Smuzhiyun 		if (obj_desc) {
390*4882a593Smuzhiyun 			acpi_ut_remove_reference(walk_state->operands[i]);
391*4882a593Smuzhiyun 			walk_state->operands[i] = NULL;
392*4882a593Smuzhiyun 		}
393*4882a593Smuzhiyun 	}
394*4882a593Smuzhiyun 
395*4882a593Smuzhiyun 	ACPI_DEBUG_PRINT((ACPI_DB_EXEC, "Count=%X State=%p #Ops=%X\n",
396*4882a593Smuzhiyun 			  pop_count, walk_state, walk_state->num_operands));
397*4882a593Smuzhiyun }
398*4882a593Smuzhiyun 
399*4882a593Smuzhiyun /*******************************************************************************
400*4882a593Smuzhiyun  *
401*4882a593Smuzhiyun  * FUNCTION:    acpi_ds_get_current_walk_state
402*4882a593Smuzhiyun  *
403*4882a593Smuzhiyun  * PARAMETERS:  thread          - Get current active state for this Thread
404*4882a593Smuzhiyun  *
405*4882a593Smuzhiyun  * RETURN:      Pointer to the current walk state
406*4882a593Smuzhiyun  *
407*4882a593Smuzhiyun  * DESCRIPTION: Get the walk state that is at the head of the list (the "current"
408*4882a593Smuzhiyun  *              walk state.)
409*4882a593Smuzhiyun  *
410*4882a593Smuzhiyun  ******************************************************************************/
411*4882a593Smuzhiyun 
acpi_ds_get_current_walk_state(struct acpi_thread_state * thread)412*4882a593Smuzhiyun struct acpi_walk_state *acpi_ds_get_current_walk_state(struct acpi_thread_state
413*4882a593Smuzhiyun 						       *thread)
414*4882a593Smuzhiyun {
415*4882a593Smuzhiyun 	ACPI_FUNCTION_NAME(ds_get_current_walk_state);
416*4882a593Smuzhiyun 
417*4882a593Smuzhiyun 	if (!thread) {
418*4882a593Smuzhiyun 		return (NULL);
419*4882a593Smuzhiyun 	}
420*4882a593Smuzhiyun 
421*4882a593Smuzhiyun 	ACPI_DEBUG_PRINT((ACPI_DB_PARSE, "Current WalkState %p\n",
422*4882a593Smuzhiyun 			  thread->walk_state_list));
423*4882a593Smuzhiyun 
424*4882a593Smuzhiyun 	return (thread->walk_state_list);
425*4882a593Smuzhiyun }
426*4882a593Smuzhiyun 
427*4882a593Smuzhiyun /*******************************************************************************
428*4882a593Smuzhiyun  *
429*4882a593Smuzhiyun  * FUNCTION:    acpi_ds_push_walk_state
430*4882a593Smuzhiyun  *
431*4882a593Smuzhiyun  * PARAMETERS:  walk_state      - State to push
432*4882a593Smuzhiyun  *              thread          - Thread state object
433*4882a593Smuzhiyun  *
434*4882a593Smuzhiyun  * RETURN:      None
435*4882a593Smuzhiyun  *
436*4882a593Smuzhiyun  * DESCRIPTION: Place the Thread state at the head of the state list
437*4882a593Smuzhiyun  *
438*4882a593Smuzhiyun  ******************************************************************************/
439*4882a593Smuzhiyun 
440*4882a593Smuzhiyun void
acpi_ds_push_walk_state(struct acpi_walk_state * walk_state,struct acpi_thread_state * thread)441*4882a593Smuzhiyun acpi_ds_push_walk_state(struct acpi_walk_state *walk_state,
442*4882a593Smuzhiyun 			struct acpi_thread_state *thread)
443*4882a593Smuzhiyun {
444*4882a593Smuzhiyun 	ACPI_FUNCTION_TRACE(ds_push_walk_state);
445*4882a593Smuzhiyun 
446*4882a593Smuzhiyun 	walk_state->next = thread->walk_state_list;
447*4882a593Smuzhiyun 	thread->walk_state_list = walk_state;
448*4882a593Smuzhiyun 
449*4882a593Smuzhiyun 	return_VOID;
450*4882a593Smuzhiyun }
451*4882a593Smuzhiyun 
452*4882a593Smuzhiyun /*******************************************************************************
453*4882a593Smuzhiyun  *
454*4882a593Smuzhiyun  * FUNCTION:    acpi_ds_pop_walk_state
455*4882a593Smuzhiyun  *
456*4882a593Smuzhiyun  * PARAMETERS:  thread      - Current thread state
457*4882a593Smuzhiyun  *
458*4882a593Smuzhiyun  * RETURN:      A walk_state object popped from the thread's stack
459*4882a593Smuzhiyun  *
460*4882a593Smuzhiyun  * DESCRIPTION: Remove and return the walkstate object that is at the head of
461*4882a593Smuzhiyun  *              the walk stack for the given walk list. NULL indicates that
462*4882a593Smuzhiyun  *              the list is empty.
463*4882a593Smuzhiyun  *
464*4882a593Smuzhiyun  ******************************************************************************/
465*4882a593Smuzhiyun 
acpi_ds_pop_walk_state(struct acpi_thread_state * thread)466*4882a593Smuzhiyun struct acpi_walk_state *acpi_ds_pop_walk_state(struct acpi_thread_state *thread)
467*4882a593Smuzhiyun {
468*4882a593Smuzhiyun 	struct acpi_walk_state *walk_state;
469*4882a593Smuzhiyun 
470*4882a593Smuzhiyun 	ACPI_FUNCTION_TRACE(ds_pop_walk_state);
471*4882a593Smuzhiyun 
472*4882a593Smuzhiyun 	walk_state = thread->walk_state_list;
473*4882a593Smuzhiyun 
474*4882a593Smuzhiyun 	if (walk_state) {
475*4882a593Smuzhiyun 
476*4882a593Smuzhiyun 		/* Next walk state becomes the current walk state */
477*4882a593Smuzhiyun 
478*4882a593Smuzhiyun 		thread->walk_state_list = walk_state->next;
479*4882a593Smuzhiyun 
480*4882a593Smuzhiyun 		/*
481*4882a593Smuzhiyun 		 * Don't clear the NEXT field, this serves as an indicator
482*4882a593Smuzhiyun 		 * that there is a parent WALK STATE
483*4882a593Smuzhiyun 		 * Do Not: walk_state->Next = NULL;
484*4882a593Smuzhiyun 		 */
485*4882a593Smuzhiyun 	}
486*4882a593Smuzhiyun 
487*4882a593Smuzhiyun 	return_PTR(walk_state);
488*4882a593Smuzhiyun }
489*4882a593Smuzhiyun 
490*4882a593Smuzhiyun /*******************************************************************************
491*4882a593Smuzhiyun  *
492*4882a593Smuzhiyun  * FUNCTION:    acpi_ds_create_walk_state
493*4882a593Smuzhiyun  *
494*4882a593Smuzhiyun  * PARAMETERS:  owner_id        - ID for object creation
495*4882a593Smuzhiyun  *              origin          - Starting point for this walk
496*4882a593Smuzhiyun  *              method_desc     - Method object
497*4882a593Smuzhiyun  *              thread          - Current thread state
498*4882a593Smuzhiyun  *
499*4882a593Smuzhiyun  * RETURN:      Pointer to the new walk state.
500*4882a593Smuzhiyun  *
501*4882a593Smuzhiyun  * DESCRIPTION: Allocate and initialize a new walk state. The current walk
502*4882a593Smuzhiyun  *              state is set to this new state.
503*4882a593Smuzhiyun  *
504*4882a593Smuzhiyun  ******************************************************************************/
505*4882a593Smuzhiyun 
acpi_ds_create_walk_state(acpi_owner_id owner_id,union acpi_parse_object * origin,union acpi_operand_object * method_desc,struct acpi_thread_state * thread)506*4882a593Smuzhiyun struct acpi_walk_state *acpi_ds_create_walk_state(acpi_owner_id owner_id,
507*4882a593Smuzhiyun 						  union acpi_parse_object
508*4882a593Smuzhiyun 						  *origin,
509*4882a593Smuzhiyun 						  union acpi_operand_object
510*4882a593Smuzhiyun 						  *method_desc,
511*4882a593Smuzhiyun 						  struct acpi_thread_state
512*4882a593Smuzhiyun 						  *thread)
513*4882a593Smuzhiyun {
514*4882a593Smuzhiyun 	struct acpi_walk_state *walk_state;
515*4882a593Smuzhiyun 
516*4882a593Smuzhiyun 	ACPI_FUNCTION_TRACE(ds_create_walk_state);
517*4882a593Smuzhiyun 
518*4882a593Smuzhiyun 	walk_state = ACPI_ALLOCATE_ZEROED(sizeof(struct acpi_walk_state));
519*4882a593Smuzhiyun 	if (!walk_state) {
520*4882a593Smuzhiyun 		return_PTR(NULL);
521*4882a593Smuzhiyun 	}
522*4882a593Smuzhiyun 
523*4882a593Smuzhiyun 	walk_state->descriptor_type = ACPI_DESC_TYPE_WALK;
524*4882a593Smuzhiyun 	walk_state->method_desc = method_desc;
525*4882a593Smuzhiyun 	walk_state->owner_id = owner_id;
526*4882a593Smuzhiyun 	walk_state->origin = origin;
527*4882a593Smuzhiyun 	walk_state->thread = thread;
528*4882a593Smuzhiyun 
529*4882a593Smuzhiyun 	walk_state->parser_state.start_op = origin;
530*4882a593Smuzhiyun 
531*4882a593Smuzhiyun 	/* Init the method args/local */
532*4882a593Smuzhiyun 
533*4882a593Smuzhiyun #ifndef ACPI_CONSTANT_EVAL_ONLY
534*4882a593Smuzhiyun 	acpi_ds_method_data_init(walk_state);
535*4882a593Smuzhiyun #endif
536*4882a593Smuzhiyun 
537*4882a593Smuzhiyun 	/* Put the new state at the head of the walk list */
538*4882a593Smuzhiyun 
539*4882a593Smuzhiyun 	if (thread) {
540*4882a593Smuzhiyun 		acpi_ds_push_walk_state(walk_state, thread);
541*4882a593Smuzhiyun 	}
542*4882a593Smuzhiyun 
543*4882a593Smuzhiyun 	return_PTR(walk_state);
544*4882a593Smuzhiyun }
545*4882a593Smuzhiyun 
546*4882a593Smuzhiyun /*******************************************************************************
547*4882a593Smuzhiyun  *
548*4882a593Smuzhiyun  * FUNCTION:    acpi_ds_init_aml_walk
549*4882a593Smuzhiyun  *
550*4882a593Smuzhiyun  * PARAMETERS:  walk_state      - New state to be initialized
551*4882a593Smuzhiyun  *              op              - Current parse op
552*4882a593Smuzhiyun  *              method_node     - Control method NS node, if any
553*4882a593Smuzhiyun  *              aml_start       - Start of AML
554*4882a593Smuzhiyun  *              aml_length      - Length of AML
555*4882a593Smuzhiyun  *              info            - Method info block (params, etc.)
556*4882a593Smuzhiyun  *              pass_number     - 1, 2, or 3
557*4882a593Smuzhiyun  *
558*4882a593Smuzhiyun  * RETURN:      Status
559*4882a593Smuzhiyun  *
560*4882a593Smuzhiyun  * DESCRIPTION: Initialize a walk state for a pass 1 or 2 parse tree walk
561*4882a593Smuzhiyun  *
562*4882a593Smuzhiyun  ******************************************************************************/
563*4882a593Smuzhiyun 
564*4882a593Smuzhiyun acpi_status
acpi_ds_init_aml_walk(struct acpi_walk_state * walk_state,union acpi_parse_object * op,struct acpi_namespace_node * method_node,u8 * aml_start,u32 aml_length,struct acpi_evaluate_info * info,u8 pass_number)565*4882a593Smuzhiyun acpi_ds_init_aml_walk(struct acpi_walk_state *walk_state,
566*4882a593Smuzhiyun 		      union acpi_parse_object *op,
567*4882a593Smuzhiyun 		      struct acpi_namespace_node *method_node,
568*4882a593Smuzhiyun 		      u8 * aml_start,
569*4882a593Smuzhiyun 		      u32 aml_length,
570*4882a593Smuzhiyun 		      struct acpi_evaluate_info *info, u8 pass_number)
571*4882a593Smuzhiyun {
572*4882a593Smuzhiyun 	acpi_status status;
573*4882a593Smuzhiyun 	struct acpi_parse_state *parser_state = &walk_state->parser_state;
574*4882a593Smuzhiyun 	union acpi_parse_object *extra_op;
575*4882a593Smuzhiyun 
576*4882a593Smuzhiyun 	ACPI_FUNCTION_TRACE(ds_init_aml_walk);
577*4882a593Smuzhiyun 
578*4882a593Smuzhiyun 	walk_state->parser_state.aml =
579*4882a593Smuzhiyun 	    walk_state->parser_state.aml_start = aml_start;
580*4882a593Smuzhiyun 	walk_state->parser_state.aml_end =
581*4882a593Smuzhiyun 	    walk_state->parser_state.pkg_end = aml_start + aml_length;
582*4882a593Smuzhiyun 
583*4882a593Smuzhiyun 	/* The next_op of the next_walk will be the beginning of the method */
584*4882a593Smuzhiyun 
585*4882a593Smuzhiyun 	walk_state->next_op = NULL;
586*4882a593Smuzhiyun 	walk_state->pass_number = pass_number;
587*4882a593Smuzhiyun 
588*4882a593Smuzhiyun 	if (info) {
589*4882a593Smuzhiyun 		walk_state->params = info->parameters;
590*4882a593Smuzhiyun 		walk_state->caller_return_desc = &info->return_object;
591*4882a593Smuzhiyun 	}
592*4882a593Smuzhiyun 
593*4882a593Smuzhiyun 	status = acpi_ps_init_scope(&walk_state->parser_state, op);
594*4882a593Smuzhiyun 	if (ACPI_FAILURE(status)) {
595*4882a593Smuzhiyun 		return_ACPI_STATUS(status);
596*4882a593Smuzhiyun 	}
597*4882a593Smuzhiyun 
598*4882a593Smuzhiyun 	if (method_node) {
599*4882a593Smuzhiyun 		walk_state->parser_state.start_node = method_node;
600*4882a593Smuzhiyun 		walk_state->walk_type = ACPI_WALK_METHOD;
601*4882a593Smuzhiyun 		walk_state->method_node = method_node;
602*4882a593Smuzhiyun 		walk_state->method_desc =
603*4882a593Smuzhiyun 		    acpi_ns_get_attached_object(method_node);
604*4882a593Smuzhiyun 
605*4882a593Smuzhiyun 		/* Push start scope on scope stack and make it current  */
606*4882a593Smuzhiyun 
607*4882a593Smuzhiyun 		status =
608*4882a593Smuzhiyun 		    acpi_ds_scope_stack_push(method_node, ACPI_TYPE_METHOD,
609*4882a593Smuzhiyun 					     walk_state);
610*4882a593Smuzhiyun 		if (ACPI_FAILURE(status)) {
611*4882a593Smuzhiyun 			return_ACPI_STATUS(status);
612*4882a593Smuzhiyun 		}
613*4882a593Smuzhiyun 
614*4882a593Smuzhiyun 		/* Init the method arguments */
615*4882a593Smuzhiyun 
616*4882a593Smuzhiyun 		status = acpi_ds_method_data_init_args(walk_state->params,
617*4882a593Smuzhiyun 						       ACPI_METHOD_NUM_ARGS,
618*4882a593Smuzhiyun 						       walk_state);
619*4882a593Smuzhiyun 		if (ACPI_FAILURE(status)) {
620*4882a593Smuzhiyun 			return_ACPI_STATUS(status);
621*4882a593Smuzhiyun 		}
622*4882a593Smuzhiyun 	} else {
623*4882a593Smuzhiyun 		/*
624*4882a593Smuzhiyun 		 * Setup the current scope.
625*4882a593Smuzhiyun 		 * Find a Named Op that has a namespace node associated with it.
626*4882a593Smuzhiyun 		 * search upwards from this Op. Current scope is the first
627*4882a593Smuzhiyun 		 * Op with a namespace node.
628*4882a593Smuzhiyun 		 */
629*4882a593Smuzhiyun 		extra_op = parser_state->start_op;
630*4882a593Smuzhiyun 		while (extra_op && !extra_op->common.node) {
631*4882a593Smuzhiyun 			extra_op = extra_op->common.parent;
632*4882a593Smuzhiyun 		}
633*4882a593Smuzhiyun 
634*4882a593Smuzhiyun 		if (!extra_op) {
635*4882a593Smuzhiyun 			parser_state->start_node = NULL;
636*4882a593Smuzhiyun 		} else {
637*4882a593Smuzhiyun 			parser_state->start_node = extra_op->common.node;
638*4882a593Smuzhiyun 		}
639*4882a593Smuzhiyun 
640*4882a593Smuzhiyun 		if (parser_state->start_node) {
641*4882a593Smuzhiyun 
642*4882a593Smuzhiyun 			/* Push start scope on scope stack and make it current  */
643*4882a593Smuzhiyun 
644*4882a593Smuzhiyun 			status =
645*4882a593Smuzhiyun 			    acpi_ds_scope_stack_push(parser_state->start_node,
646*4882a593Smuzhiyun 						     parser_state->start_node->
647*4882a593Smuzhiyun 						     type, walk_state);
648*4882a593Smuzhiyun 			if (ACPI_FAILURE(status)) {
649*4882a593Smuzhiyun 				return_ACPI_STATUS(status);
650*4882a593Smuzhiyun 			}
651*4882a593Smuzhiyun 		}
652*4882a593Smuzhiyun 	}
653*4882a593Smuzhiyun 
654*4882a593Smuzhiyun 	status = acpi_ds_init_callbacks(walk_state, pass_number);
655*4882a593Smuzhiyun 	return_ACPI_STATUS(status);
656*4882a593Smuzhiyun }
657*4882a593Smuzhiyun 
658*4882a593Smuzhiyun /*******************************************************************************
659*4882a593Smuzhiyun  *
660*4882a593Smuzhiyun  * FUNCTION:    acpi_ds_delete_walk_state
661*4882a593Smuzhiyun  *
662*4882a593Smuzhiyun  * PARAMETERS:  walk_state      - State to delete
663*4882a593Smuzhiyun  *
664*4882a593Smuzhiyun  * RETURN:      Status
665*4882a593Smuzhiyun  *
666*4882a593Smuzhiyun  * DESCRIPTION: Delete a walk state including all internal data structures
667*4882a593Smuzhiyun  *
668*4882a593Smuzhiyun  ******************************************************************************/
669*4882a593Smuzhiyun 
acpi_ds_delete_walk_state(struct acpi_walk_state * walk_state)670*4882a593Smuzhiyun void acpi_ds_delete_walk_state(struct acpi_walk_state *walk_state)
671*4882a593Smuzhiyun {
672*4882a593Smuzhiyun 	union acpi_generic_state *state;
673*4882a593Smuzhiyun 
674*4882a593Smuzhiyun 	ACPI_FUNCTION_TRACE_PTR(ds_delete_walk_state, walk_state);
675*4882a593Smuzhiyun 
676*4882a593Smuzhiyun 	if (!walk_state) {
677*4882a593Smuzhiyun 		return_VOID;
678*4882a593Smuzhiyun 	}
679*4882a593Smuzhiyun 
680*4882a593Smuzhiyun 	if (walk_state->descriptor_type != ACPI_DESC_TYPE_WALK) {
681*4882a593Smuzhiyun 		ACPI_ERROR((AE_INFO, "%p is not a valid walk state",
682*4882a593Smuzhiyun 			    walk_state));
683*4882a593Smuzhiyun 		return_VOID;
684*4882a593Smuzhiyun 	}
685*4882a593Smuzhiyun 
686*4882a593Smuzhiyun 	/* There should not be any open scopes */
687*4882a593Smuzhiyun 
688*4882a593Smuzhiyun 	if (walk_state->parser_state.scope) {
689*4882a593Smuzhiyun 		ACPI_ERROR((AE_INFO, "%p walk still has a scope list",
690*4882a593Smuzhiyun 			    walk_state));
691*4882a593Smuzhiyun 		acpi_ps_cleanup_scope(&walk_state->parser_state);
692*4882a593Smuzhiyun 	}
693*4882a593Smuzhiyun 
694*4882a593Smuzhiyun 	/* Always must free any linked control states */
695*4882a593Smuzhiyun 
696*4882a593Smuzhiyun 	while (walk_state->control_state) {
697*4882a593Smuzhiyun 		state = walk_state->control_state;
698*4882a593Smuzhiyun 		walk_state->control_state = state->common.next;
699*4882a593Smuzhiyun 
700*4882a593Smuzhiyun 		acpi_ut_delete_generic_state(state);
701*4882a593Smuzhiyun 	}
702*4882a593Smuzhiyun 
703*4882a593Smuzhiyun 	/* Always must free any linked parse states */
704*4882a593Smuzhiyun 
705*4882a593Smuzhiyun 	while (walk_state->scope_info) {
706*4882a593Smuzhiyun 		state = walk_state->scope_info;
707*4882a593Smuzhiyun 		walk_state->scope_info = state->common.next;
708*4882a593Smuzhiyun 
709*4882a593Smuzhiyun 		acpi_ut_delete_generic_state(state);
710*4882a593Smuzhiyun 	}
711*4882a593Smuzhiyun 
712*4882a593Smuzhiyun 	/* Always must free any stacked result states */
713*4882a593Smuzhiyun 
714*4882a593Smuzhiyun 	while (walk_state->results) {
715*4882a593Smuzhiyun 		state = walk_state->results;
716*4882a593Smuzhiyun 		walk_state->results = state->common.next;
717*4882a593Smuzhiyun 
718*4882a593Smuzhiyun 		acpi_ut_delete_generic_state(state);
719*4882a593Smuzhiyun 	}
720*4882a593Smuzhiyun 
721*4882a593Smuzhiyun 	ACPI_FREE(walk_state);
722*4882a593Smuzhiyun 	return_VOID;
723*4882a593Smuzhiyun }
724