xref: /OK3568_Linux_fs/kernel/drivers/acpi/acpica/nsparse.c (revision 4882a59341e53eb6f0b4789bf948001014eff981)
1*4882a593Smuzhiyun // SPDX-License-Identifier: BSD-3-Clause OR GPL-2.0
2*4882a593Smuzhiyun /******************************************************************************
3*4882a593Smuzhiyun  *
4*4882a593Smuzhiyun  * Module Name: nsparse - namespace interface to AML parser
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 "acnamesp.h"
13*4882a593Smuzhiyun #include "acparser.h"
14*4882a593Smuzhiyun #include "acdispat.h"
15*4882a593Smuzhiyun #include "actables.h"
16*4882a593Smuzhiyun #include "acinterp.h"
17*4882a593Smuzhiyun 
18*4882a593Smuzhiyun #define _COMPONENT          ACPI_NAMESPACE
19*4882a593Smuzhiyun ACPI_MODULE_NAME("nsparse")
20*4882a593Smuzhiyun 
21*4882a593Smuzhiyun /*******************************************************************************
22*4882a593Smuzhiyun  *
23*4882a593Smuzhiyun  * FUNCTION:    ns_execute_table
24*4882a593Smuzhiyun  *
25*4882a593Smuzhiyun  * PARAMETERS:  table_desc      - An ACPI table descriptor for table to parse
26*4882a593Smuzhiyun  *              start_node      - Where to enter the table into the namespace
27*4882a593Smuzhiyun  *
28*4882a593Smuzhiyun  * RETURN:      Status
29*4882a593Smuzhiyun  *
30*4882a593Smuzhiyun  * DESCRIPTION: Load ACPI/AML table by executing the entire table as a single
31*4882a593Smuzhiyun  *              large control method.
32*4882a593Smuzhiyun  *
33*4882a593Smuzhiyun  * NOTE: The point of this is to execute any module-level code in-place
34*4882a593Smuzhiyun  * as the table is parsed. Some AML code depends on this behavior.
35*4882a593Smuzhiyun  *
36*4882a593Smuzhiyun  * It is a run-time option at this time, but will eventually become
37*4882a593Smuzhiyun  * the default.
38*4882a593Smuzhiyun  *
39*4882a593Smuzhiyun  * Note: This causes the table to only have a single-pass parse.
40*4882a593Smuzhiyun  * However, this is compatible with other ACPI implementations.
41*4882a593Smuzhiyun  *
42*4882a593Smuzhiyun  ******************************************************************************/
43*4882a593Smuzhiyun acpi_status
acpi_ns_execute_table(u32 table_index,struct acpi_namespace_node * start_node)44*4882a593Smuzhiyun acpi_ns_execute_table(u32 table_index, struct acpi_namespace_node *start_node)
45*4882a593Smuzhiyun {
46*4882a593Smuzhiyun 	acpi_status status;
47*4882a593Smuzhiyun 	struct acpi_table_header *table;
48*4882a593Smuzhiyun 	acpi_owner_id owner_id;
49*4882a593Smuzhiyun 	struct acpi_evaluate_info *info = NULL;
50*4882a593Smuzhiyun 	u32 aml_length;
51*4882a593Smuzhiyun 	u8 *aml_start;
52*4882a593Smuzhiyun 	union acpi_operand_object *method_obj = NULL;
53*4882a593Smuzhiyun 
54*4882a593Smuzhiyun 	ACPI_FUNCTION_TRACE(ns_execute_table);
55*4882a593Smuzhiyun 
56*4882a593Smuzhiyun 	status = acpi_get_table_by_index(table_index, &table);
57*4882a593Smuzhiyun 	if (ACPI_FAILURE(status)) {
58*4882a593Smuzhiyun 		return_ACPI_STATUS(status);
59*4882a593Smuzhiyun 	}
60*4882a593Smuzhiyun 
61*4882a593Smuzhiyun 	/* Table must consist of at least a complete header */
62*4882a593Smuzhiyun 
63*4882a593Smuzhiyun 	if (table->length < sizeof(struct acpi_table_header)) {
64*4882a593Smuzhiyun 		return_ACPI_STATUS(AE_BAD_HEADER);
65*4882a593Smuzhiyun 	}
66*4882a593Smuzhiyun 
67*4882a593Smuzhiyun 	aml_start = (u8 *)table + sizeof(struct acpi_table_header);
68*4882a593Smuzhiyun 	aml_length = table->length - sizeof(struct acpi_table_header);
69*4882a593Smuzhiyun 
70*4882a593Smuzhiyun 	status = acpi_tb_get_owner_id(table_index, &owner_id);
71*4882a593Smuzhiyun 	if (ACPI_FAILURE(status)) {
72*4882a593Smuzhiyun 		return_ACPI_STATUS(status);
73*4882a593Smuzhiyun 	}
74*4882a593Smuzhiyun 
75*4882a593Smuzhiyun 	/* Create, initialize, and link a new temporary method object */
76*4882a593Smuzhiyun 
77*4882a593Smuzhiyun 	method_obj = acpi_ut_create_internal_object(ACPI_TYPE_METHOD);
78*4882a593Smuzhiyun 	if (!method_obj) {
79*4882a593Smuzhiyun 		return_ACPI_STATUS(AE_NO_MEMORY);
80*4882a593Smuzhiyun 	}
81*4882a593Smuzhiyun 
82*4882a593Smuzhiyun 	/* Allocate the evaluation information block */
83*4882a593Smuzhiyun 
84*4882a593Smuzhiyun 	info = ACPI_ALLOCATE_ZEROED(sizeof(struct acpi_evaluate_info));
85*4882a593Smuzhiyun 	if (!info) {
86*4882a593Smuzhiyun 		status = AE_NO_MEMORY;
87*4882a593Smuzhiyun 		goto cleanup;
88*4882a593Smuzhiyun 	}
89*4882a593Smuzhiyun 
90*4882a593Smuzhiyun 	ACPI_DEBUG_PRINT_RAW((ACPI_DB_PARSE,
91*4882a593Smuzhiyun 			      "%s: Create table pseudo-method for [%4.4s] @%p, method %p\n",
92*4882a593Smuzhiyun 			      ACPI_GET_FUNCTION_NAME, table->signature, table,
93*4882a593Smuzhiyun 			      method_obj));
94*4882a593Smuzhiyun 
95*4882a593Smuzhiyun 	method_obj->method.aml_start = aml_start;
96*4882a593Smuzhiyun 	method_obj->method.aml_length = aml_length;
97*4882a593Smuzhiyun 	method_obj->method.owner_id = owner_id;
98*4882a593Smuzhiyun 	method_obj->method.info_flags |= ACPI_METHOD_MODULE_LEVEL;
99*4882a593Smuzhiyun 
100*4882a593Smuzhiyun 	info->pass_number = ACPI_IMODE_EXECUTE;
101*4882a593Smuzhiyun 	info->node = start_node;
102*4882a593Smuzhiyun 	info->obj_desc = method_obj;
103*4882a593Smuzhiyun 	info->node_flags = info->node->flags;
104*4882a593Smuzhiyun 	info->full_pathname = acpi_ns_get_normalized_pathname(info->node, TRUE);
105*4882a593Smuzhiyun 	if (!info->full_pathname) {
106*4882a593Smuzhiyun 		status = AE_NO_MEMORY;
107*4882a593Smuzhiyun 		goto cleanup;
108*4882a593Smuzhiyun 	}
109*4882a593Smuzhiyun 
110*4882a593Smuzhiyun 	/* Optional object evaluation log */
111*4882a593Smuzhiyun 
112*4882a593Smuzhiyun 	ACPI_DEBUG_PRINT_RAW((ACPI_DB_EVALUATION,
113*4882a593Smuzhiyun 			      "%-26s:  (Definition Block level)\n",
114*4882a593Smuzhiyun 			      "Module-level evaluation"));
115*4882a593Smuzhiyun 
116*4882a593Smuzhiyun 	status = acpi_ps_execute_table(info);
117*4882a593Smuzhiyun 
118*4882a593Smuzhiyun 	/* Optional object evaluation log */
119*4882a593Smuzhiyun 
120*4882a593Smuzhiyun 	ACPI_DEBUG_PRINT_RAW((ACPI_DB_EVALUATION,
121*4882a593Smuzhiyun 			      "%-26s:  (Definition Block level)\n",
122*4882a593Smuzhiyun 			      "Module-level complete"));
123*4882a593Smuzhiyun 
124*4882a593Smuzhiyun cleanup:
125*4882a593Smuzhiyun 	if (info) {
126*4882a593Smuzhiyun 		ACPI_FREE(info->full_pathname);
127*4882a593Smuzhiyun 		info->full_pathname = NULL;
128*4882a593Smuzhiyun 	}
129*4882a593Smuzhiyun 	ACPI_FREE(info);
130*4882a593Smuzhiyun 	acpi_ut_remove_reference(method_obj);
131*4882a593Smuzhiyun 	return_ACPI_STATUS(status);
132*4882a593Smuzhiyun }
133*4882a593Smuzhiyun 
134*4882a593Smuzhiyun /*******************************************************************************
135*4882a593Smuzhiyun  *
136*4882a593Smuzhiyun  * FUNCTION:    ns_one_complete_parse
137*4882a593Smuzhiyun  *
138*4882a593Smuzhiyun  * PARAMETERS:  pass_number             - 1 or 2
139*4882a593Smuzhiyun  *              table_desc              - The table to be parsed.
140*4882a593Smuzhiyun  *
141*4882a593Smuzhiyun  * RETURN:      Status
142*4882a593Smuzhiyun  *
143*4882a593Smuzhiyun  * DESCRIPTION: Perform one complete parse of an ACPI/AML table.
144*4882a593Smuzhiyun  *
145*4882a593Smuzhiyun  ******************************************************************************/
146*4882a593Smuzhiyun 
147*4882a593Smuzhiyun acpi_status
acpi_ns_one_complete_parse(u32 pass_number,u32 table_index,struct acpi_namespace_node * start_node)148*4882a593Smuzhiyun acpi_ns_one_complete_parse(u32 pass_number,
149*4882a593Smuzhiyun 			   u32 table_index,
150*4882a593Smuzhiyun 			   struct acpi_namespace_node *start_node)
151*4882a593Smuzhiyun {
152*4882a593Smuzhiyun 	union acpi_parse_object *parse_root;
153*4882a593Smuzhiyun 	acpi_status status;
154*4882a593Smuzhiyun 	u32 aml_length;
155*4882a593Smuzhiyun 	u8 *aml_start;
156*4882a593Smuzhiyun 	struct acpi_walk_state *walk_state;
157*4882a593Smuzhiyun 	struct acpi_table_header *table;
158*4882a593Smuzhiyun 	acpi_owner_id owner_id;
159*4882a593Smuzhiyun 
160*4882a593Smuzhiyun 	ACPI_FUNCTION_TRACE(ns_one_complete_parse);
161*4882a593Smuzhiyun 
162*4882a593Smuzhiyun 	status = acpi_get_table_by_index(table_index, &table);
163*4882a593Smuzhiyun 	if (ACPI_FAILURE(status)) {
164*4882a593Smuzhiyun 		return_ACPI_STATUS(status);
165*4882a593Smuzhiyun 	}
166*4882a593Smuzhiyun 
167*4882a593Smuzhiyun 	/* Table must consist of at least a complete header */
168*4882a593Smuzhiyun 
169*4882a593Smuzhiyun 	if (table->length < sizeof(struct acpi_table_header)) {
170*4882a593Smuzhiyun 		return_ACPI_STATUS(AE_BAD_HEADER);
171*4882a593Smuzhiyun 	}
172*4882a593Smuzhiyun 
173*4882a593Smuzhiyun 	aml_start = (u8 *)table + sizeof(struct acpi_table_header);
174*4882a593Smuzhiyun 	aml_length = table->length - sizeof(struct acpi_table_header);
175*4882a593Smuzhiyun 
176*4882a593Smuzhiyun 	status = acpi_tb_get_owner_id(table_index, &owner_id);
177*4882a593Smuzhiyun 	if (ACPI_FAILURE(status)) {
178*4882a593Smuzhiyun 		return_ACPI_STATUS(status);
179*4882a593Smuzhiyun 	}
180*4882a593Smuzhiyun 
181*4882a593Smuzhiyun 	/* Create and init a Root Node */
182*4882a593Smuzhiyun 
183*4882a593Smuzhiyun 	parse_root = acpi_ps_create_scope_op(aml_start);
184*4882a593Smuzhiyun 	if (!parse_root) {
185*4882a593Smuzhiyun 		return_ACPI_STATUS(AE_NO_MEMORY);
186*4882a593Smuzhiyun 	}
187*4882a593Smuzhiyun 
188*4882a593Smuzhiyun 	/* Create and initialize a new walk state */
189*4882a593Smuzhiyun 
190*4882a593Smuzhiyun 	walk_state = acpi_ds_create_walk_state(owner_id, NULL, NULL, NULL);
191*4882a593Smuzhiyun 	if (!walk_state) {
192*4882a593Smuzhiyun 		acpi_ps_free_op(parse_root);
193*4882a593Smuzhiyun 		return_ACPI_STATUS(AE_NO_MEMORY);
194*4882a593Smuzhiyun 	}
195*4882a593Smuzhiyun 
196*4882a593Smuzhiyun 	status = acpi_ds_init_aml_walk(walk_state, parse_root, NULL,
197*4882a593Smuzhiyun 				       aml_start, aml_length, NULL,
198*4882a593Smuzhiyun 				       (u8)pass_number);
199*4882a593Smuzhiyun 	if (ACPI_FAILURE(status)) {
200*4882a593Smuzhiyun 		acpi_ds_delete_walk_state(walk_state);
201*4882a593Smuzhiyun 		goto cleanup;
202*4882a593Smuzhiyun 	}
203*4882a593Smuzhiyun 
204*4882a593Smuzhiyun 	/* Found OSDT table, enable the namespace override feature */
205*4882a593Smuzhiyun 
206*4882a593Smuzhiyun 	if (ACPI_COMPARE_NAMESEG(table->signature, ACPI_SIG_OSDT) &&
207*4882a593Smuzhiyun 	    pass_number == ACPI_IMODE_LOAD_PASS1) {
208*4882a593Smuzhiyun 		walk_state->namespace_override = TRUE;
209*4882a593Smuzhiyun 	}
210*4882a593Smuzhiyun 
211*4882a593Smuzhiyun 	/* start_node is the default location to load the table */
212*4882a593Smuzhiyun 
213*4882a593Smuzhiyun 	if (start_node && start_node != acpi_gbl_root_node) {
214*4882a593Smuzhiyun 		status =
215*4882a593Smuzhiyun 		    acpi_ds_scope_stack_push(start_node, ACPI_TYPE_METHOD,
216*4882a593Smuzhiyun 					     walk_state);
217*4882a593Smuzhiyun 		if (ACPI_FAILURE(status)) {
218*4882a593Smuzhiyun 			acpi_ds_delete_walk_state(walk_state);
219*4882a593Smuzhiyun 			goto cleanup;
220*4882a593Smuzhiyun 		}
221*4882a593Smuzhiyun 	}
222*4882a593Smuzhiyun 
223*4882a593Smuzhiyun 	/* Parse the AML */
224*4882a593Smuzhiyun 
225*4882a593Smuzhiyun 	ACPI_DEBUG_PRINT((ACPI_DB_PARSE,
226*4882a593Smuzhiyun 			  "*PARSE* pass %u parse\n", pass_number));
227*4882a593Smuzhiyun 	acpi_ex_enter_interpreter();
228*4882a593Smuzhiyun 	status = acpi_ps_parse_aml(walk_state);
229*4882a593Smuzhiyun 	acpi_ex_exit_interpreter();
230*4882a593Smuzhiyun 
231*4882a593Smuzhiyun cleanup:
232*4882a593Smuzhiyun 	acpi_ps_delete_parse_tree(parse_root);
233*4882a593Smuzhiyun 	return_ACPI_STATUS(status);
234*4882a593Smuzhiyun }
235*4882a593Smuzhiyun 
236*4882a593Smuzhiyun /*******************************************************************************
237*4882a593Smuzhiyun  *
238*4882a593Smuzhiyun  * FUNCTION:    acpi_ns_parse_table
239*4882a593Smuzhiyun  *
240*4882a593Smuzhiyun  * PARAMETERS:  table_desc      - An ACPI table descriptor for table to parse
241*4882a593Smuzhiyun  *              start_node      - Where to enter the table into the namespace
242*4882a593Smuzhiyun  *
243*4882a593Smuzhiyun  * RETURN:      Status
244*4882a593Smuzhiyun  *
245*4882a593Smuzhiyun  * DESCRIPTION: Parse AML within an ACPI table and return a tree of ops
246*4882a593Smuzhiyun  *
247*4882a593Smuzhiyun  ******************************************************************************/
248*4882a593Smuzhiyun 
249*4882a593Smuzhiyun acpi_status
acpi_ns_parse_table(u32 table_index,struct acpi_namespace_node * start_node)250*4882a593Smuzhiyun acpi_ns_parse_table(u32 table_index, struct acpi_namespace_node *start_node)
251*4882a593Smuzhiyun {
252*4882a593Smuzhiyun 	acpi_status status;
253*4882a593Smuzhiyun 
254*4882a593Smuzhiyun 	ACPI_FUNCTION_TRACE(ns_parse_table);
255*4882a593Smuzhiyun 
256*4882a593Smuzhiyun 	/*
257*4882a593Smuzhiyun 	 * Executes the AML table as one large control method.
258*4882a593Smuzhiyun 	 * The point of this is to execute any module-level code in-place
259*4882a593Smuzhiyun 	 * as the table is parsed. Some AML code depends on this behavior.
260*4882a593Smuzhiyun 	 *
261*4882a593Smuzhiyun 	 * Note: This causes the table to only have a single-pass parse.
262*4882a593Smuzhiyun 	 * However, this is compatible with other ACPI implementations.
263*4882a593Smuzhiyun 	 */
264*4882a593Smuzhiyun 	ACPI_DEBUG_PRINT_RAW((ACPI_DB_PARSE,
265*4882a593Smuzhiyun 			      "%s: **** Start table execution pass\n",
266*4882a593Smuzhiyun 			      ACPI_GET_FUNCTION_NAME));
267*4882a593Smuzhiyun 
268*4882a593Smuzhiyun 	status = acpi_ns_execute_table(table_index, start_node);
269*4882a593Smuzhiyun 
270*4882a593Smuzhiyun 	return_ACPI_STATUS(status);
271*4882a593Smuzhiyun }
272