1  // SPDX-License-Identifier: BSD-3-Clause OR GPL-2.0
2  /*******************************************************************************
3   *
4   * Module Name: utstate - state object support procedures
5   *
6   ******************************************************************************/
7  
8  #include <acpi/acpi.h>
9  #include "accommon.h"
10  
11  #define _COMPONENT          ACPI_UTILITIES
12  ACPI_MODULE_NAME("utstate")
13  
14  /*******************************************************************************
15   *
16   * FUNCTION:    acpi_ut_push_generic_state
17   *
18   * PARAMETERS:  list_head           - Head of the state stack
19   *              state               - State object to push
20   *
21   * RETURN:      None
22   *
23   * DESCRIPTION: Push a state object onto a state stack
24   *
25   ******************************************************************************/
26  void
acpi_ut_push_generic_state(union acpi_generic_state ** list_head,union acpi_generic_state * state)27  acpi_ut_push_generic_state(union acpi_generic_state **list_head,
28  			   union acpi_generic_state *state)
29  {
30  	ACPI_FUNCTION_ENTRY();
31  
32  	/* Push the state object onto the front of the list (stack) */
33  
34  	state->common.next = *list_head;
35  	*list_head = state;
36  	return;
37  }
38  
39  /*******************************************************************************
40   *
41   * FUNCTION:    acpi_ut_pop_generic_state
42   *
43   * PARAMETERS:  list_head           - Head of the state stack
44   *
45   * RETURN:      The popped state object
46   *
47   * DESCRIPTION: Pop a state object from a state stack
48   *
49   ******************************************************************************/
50  
acpi_ut_pop_generic_state(union acpi_generic_state ** list_head)51  union acpi_generic_state *acpi_ut_pop_generic_state(union acpi_generic_state
52  						    **list_head)
53  {
54  	union acpi_generic_state *state;
55  
56  	ACPI_FUNCTION_ENTRY();
57  
58  	/* Remove the state object at the head of the list (stack) */
59  
60  	state = *list_head;
61  	if (state) {
62  
63  		/* Update the list head */
64  
65  		*list_head = state->common.next;
66  	}
67  
68  	return (state);
69  }
70  
71  /*******************************************************************************
72   *
73   * FUNCTION:    acpi_ut_create_generic_state
74   *
75   * PARAMETERS:  None
76   *
77   * RETURN:      The new state object. NULL on failure.
78   *
79   * DESCRIPTION: Create a generic state object. Attempt to obtain one from
80   *              the global state cache;  If none available, create a new one.
81   *
82   ******************************************************************************/
83  
acpi_ut_create_generic_state(void)84  union acpi_generic_state *acpi_ut_create_generic_state(void)
85  {
86  	union acpi_generic_state *state;
87  
88  	ACPI_FUNCTION_ENTRY();
89  
90  	state = acpi_os_acquire_object(acpi_gbl_state_cache);
91  	if (state) {
92  
93  		/* Initialize */
94  		state->common.descriptor_type = ACPI_DESC_TYPE_STATE;
95  	}
96  
97  	return (state);
98  }
99  
100  /*******************************************************************************
101   *
102   * FUNCTION:    acpi_ut_create_thread_state
103   *
104   * PARAMETERS:  None
105   *
106   * RETURN:      New Thread State. NULL on failure
107   *
108   * DESCRIPTION: Create a "Thread State" - a flavor of the generic state used
109   *              to track per-thread info during method execution
110   *
111   ******************************************************************************/
112  
acpi_ut_create_thread_state(void)113  struct acpi_thread_state *acpi_ut_create_thread_state(void)
114  {
115  	union acpi_generic_state *state;
116  
117  	ACPI_FUNCTION_ENTRY();
118  
119  	/* Create the generic state object */
120  
121  	state = acpi_ut_create_generic_state();
122  	if (!state) {
123  		return (NULL);
124  	}
125  
126  	/* Init fields specific to the update struct */
127  
128  	state->common.descriptor_type = ACPI_DESC_TYPE_STATE_THREAD;
129  	state->thread.thread_id = acpi_os_get_thread_id();
130  
131  	/* Check for invalid thread ID - zero is very bad, it will break things */
132  
133  	if (!state->thread.thread_id) {
134  		ACPI_ERROR((AE_INFO, "Invalid zero ID from AcpiOsGetThreadId"));
135  		state->thread.thread_id = (acpi_thread_id) 1;
136  	}
137  
138  	return ((struct acpi_thread_state *)state);
139  }
140  
141  /*******************************************************************************
142   *
143   * FUNCTION:    acpi_ut_create_update_state
144   *
145   * PARAMETERS:  object          - Initial Object to be installed in the state
146   *              action          - Update action to be performed
147   *
148   * RETURN:      New state object, null on failure
149   *
150   * DESCRIPTION: Create an "Update State" - a flavor of the generic state used
151   *              to update reference counts and delete complex objects such
152   *              as packages.
153   *
154   ******************************************************************************/
155  
acpi_ut_create_update_state(union acpi_operand_object * object,u16 action)156  union acpi_generic_state *acpi_ut_create_update_state(union acpi_operand_object
157  						      *object, u16 action)
158  {
159  	union acpi_generic_state *state;
160  
161  	ACPI_FUNCTION_ENTRY();
162  
163  	/* Create the generic state object */
164  
165  	state = acpi_ut_create_generic_state();
166  	if (!state) {
167  		return (NULL);
168  	}
169  
170  	/* Init fields specific to the update struct */
171  
172  	state->common.descriptor_type = ACPI_DESC_TYPE_STATE_UPDATE;
173  	state->update.object = object;
174  	state->update.value = action;
175  	return (state);
176  }
177  
178  /*******************************************************************************
179   *
180   * FUNCTION:    acpi_ut_create_pkg_state
181   *
182   * PARAMETERS:  object          - Initial Object to be installed in the state
183   *              action          - Update action to be performed
184   *
185   * RETURN:      New state object, null on failure
186   *
187   * DESCRIPTION: Create a "Package State"
188   *
189   ******************************************************************************/
190  
acpi_ut_create_pkg_state(void * internal_object,void * external_object,u32 index)191  union acpi_generic_state *acpi_ut_create_pkg_state(void *internal_object,
192  						   void *external_object,
193  						   u32 index)
194  {
195  	union acpi_generic_state *state;
196  
197  	ACPI_FUNCTION_ENTRY();
198  
199  	/* Create the generic state object */
200  
201  	state = acpi_ut_create_generic_state();
202  	if (!state) {
203  		return (NULL);
204  	}
205  
206  	/* Init fields specific to the update struct */
207  
208  	state->common.descriptor_type = ACPI_DESC_TYPE_STATE_PACKAGE;
209  	state->pkg.source_object = (union acpi_operand_object *)internal_object;
210  	state->pkg.dest_object = external_object;
211  	state->pkg.index = index;
212  	state->pkg.num_packages = 1;
213  
214  	return (state);
215  }
216  
217  /*******************************************************************************
218   *
219   * FUNCTION:    acpi_ut_create_control_state
220   *
221   * PARAMETERS:  None
222   *
223   * RETURN:      New state object, null on failure
224   *
225   * DESCRIPTION: Create a "Control State" - a flavor of the generic state used
226   *              to support nested IF/WHILE constructs in the AML.
227   *
228   ******************************************************************************/
229  
acpi_ut_create_control_state(void)230  union acpi_generic_state *acpi_ut_create_control_state(void)
231  {
232  	union acpi_generic_state *state;
233  
234  	ACPI_FUNCTION_ENTRY();
235  
236  	/* Create the generic state object */
237  
238  	state = acpi_ut_create_generic_state();
239  	if (!state) {
240  		return (NULL);
241  	}
242  
243  	/* Init fields specific to the control struct */
244  
245  	state->common.descriptor_type = ACPI_DESC_TYPE_STATE_CONTROL;
246  	state->common.state = ACPI_CONTROL_CONDITIONAL_EXECUTING;
247  
248  	return (state);
249  }
250  
251  /*******************************************************************************
252   *
253   * FUNCTION:    acpi_ut_delete_generic_state
254   *
255   * PARAMETERS:  state               - The state object to be deleted
256   *
257   * RETURN:      None
258   *
259   * DESCRIPTION: Release a state object to the state cache. NULL state objects
260   *              are ignored.
261   *
262   ******************************************************************************/
263  
acpi_ut_delete_generic_state(union acpi_generic_state * state)264  void acpi_ut_delete_generic_state(union acpi_generic_state *state)
265  {
266  	ACPI_FUNCTION_ENTRY();
267  
268  	/* Ignore null state */
269  
270  	if (state) {
271  		(void)acpi_os_release_object(acpi_gbl_state_cache, state);
272  	}
273  
274  	return;
275  }
276