1a9643ea8Slogwang /******************************************************************************
2a9643ea8Slogwang *
3a9643ea8Slogwang * Module Name: psparse - Parser top level AML parse routines
4a9643ea8Slogwang *
5a9643ea8Slogwang *****************************************************************************/
6a9643ea8Slogwang
7*22ce4affSfengbojiang /******************************************************************************
8*22ce4affSfengbojiang *
9*22ce4affSfengbojiang * 1. Copyright Notice
10*22ce4affSfengbojiang *
11*22ce4affSfengbojiang * Some or all of this work - Copyright (c) 1999 - 2020, Intel Corp.
12a9643ea8Slogwang * All rights reserved.
13a9643ea8Slogwang *
14*22ce4affSfengbojiang * 2. License
15*22ce4affSfengbojiang *
16*22ce4affSfengbojiang * 2.1. This is your license from Intel Corp. under its intellectual property
17*22ce4affSfengbojiang * rights. You may have additional license terms from the party that provided
18*22ce4affSfengbojiang * you this software, covering your right to use that party's intellectual
19*22ce4affSfengbojiang * property rights.
20*22ce4affSfengbojiang *
21*22ce4affSfengbojiang * 2.2. Intel grants, free of charge, to any person ("Licensee") obtaining a
22*22ce4affSfengbojiang * copy of the source code appearing in this file ("Covered Code") an
23*22ce4affSfengbojiang * irrevocable, perpetual, worldwide license under Intel's copyrights in the
24*22ce4affSfengbojiang * base code distributed originally by Intel ("Original Intel Code") to copy,
25*22ce4affSfengbojiang * make derivatives, distribute, use and display any portion of the Covered
26*22ce4affSfengbojiang * Code in any form, with the right to sublicense such rights; and
27*22ce4affSfengbojiang *
28*22ce4affSfengbojiang * 2.3. Intel grants Licensee a non-exclusive and non-transferable patent
29*22ce4affSfengbojiang * license (with the right to sublicense), under only those claims of Intel
30*22ce4affSfengbojiang * patents that are infringed by the Original Intel Code, to make, use, sell,
31*22ce4affSfengbojiang * offer to sell, and import the Covered Code and derivative works thereof
32*22ce4affSfengbojiang * solely to the minimum extent necessary to exercise the above copyright
33*22ce4affSfengbojiang * license, and in no event shall the patent license extend to any additions
34*22ce4affSfengbojiang * to or modifications of the Original Intel Code. No other license or right
35*22ce4affSfengbojiang * is granted directly or by implication, estoppel or otherwise;
36*22ce4affSfengbojiang *
37*22ce4affSfengbojiang * The above copyright and patent license is granted only if the following
38*22ce4affSfengbojiang * conditions are met:
39*22ce4affSfengbojiang *
40*22ce4affSfengbojiang * 3. Conditions
41*22ce4affSfengbojiang *
42*22ce4affSfengbojiang * 3.1. Redistribution of Source with Rights to Further Distribute Source.
43*22ce4affSfengbojiang * Redistribution of source code of any substantial portion of the Covered
44*22ce4affSfengbojiang * Code or modification with rights to further distribute source must include
45*22ce4affSfengbojiang * the above Copyright Notice, the above License, this list of Conditions,
46*22ce4affSfengbojiang * and the following Disclaimer and Export Compliance provision. In addition,
47*22ce4affSfengbojiang * Licensee must cause all Covered Code to which Licensee contributes to
48*22ce4affSfengbojiang * contain a file documenting the changes Licensee made to create that Covered
49*22ce4affSfengbojiang * Code and the date of any change. Licensee must include in that file the
50*22ce4affSfengbojiang * documentation of any changes made by any predecessor Licensee. Licensee
51*22ce4affSfengbojiang * must include a prominent statement that the modification is derived,
52*22ce4affSfengbojiang * directly or indirectly, from Original Intel Code.
53*22ce4affSfengbojiang *
54*22ce4affSfengbojiang * 3.2. Redistribution of Source with no Rights to Further Distribute Source.
55*22ce4affSfengbojiang * Redistribution of source code of any substantial portion of the Covered
56*22ce4affSfengbojiang * Code or modification without rights to further distribute source must
57*22ce4affSfengbojiang * include the following Disclaimer and Export Compliance provision in the
58*22ce4affSfengbojiang * documentation and/or other materials provided with distribution. In
59*22ce4affSfengbojiang * addition, Licensee may not authorize further sublicense of source of any
60*22ce4affSfengbojiang * portion of the Covered Code, and must include terms to the effect that the
61*22ce4affSfengbojiang * license from Licensee to its licensee is limited to the intellectual
62*22ce4affSfengbojiang * property embodied in the software Licensee provides to its licensee, and
63*22ce4affSfengbojiang * not to intellectual property embodied in modifications its licensee may
64*22ce4affSfengbojiang * make.
65*22ce4affSfengbojiang *
66*22ce4affSfengbojiang * 3.3. Redistribution of Executable. Redistribution in executable form of any
67*22ce4affSfengbojiang * substantial portion of the Covered Code or modification must reproduce the
68*22ce4affSfengbojiang * above Copyright Notice, and the following Disclaimer and Export Compliance
69*22ce4affSfengbojiang * provision in the documentation and/or other materials provided with the
70*22ce4affSfengbojiang * distribution.
71*22ce4affSfengbojiang *
72*22ce4affSfengbojiang * 3.4. Intel retains all right, title, and interest in and to the Original
73*22ce4affSfengbojiang * Intel Code.
74*22ce4affSfengbojiang *
75*22ce4affSfengbojiang * 3.5. Neither the name Intel nor any other trademark owned or controlled by
76*22ce4affSfengbojiang * Intel shall be used in advertising or otherwise to promote the sale, use or
77*22ce4affSfengbojiang * other dealings in products derived from or relating to the Covered Code
78*22ce4affSfengbojiang * without prior written authorization from Intel.
79*22ce4affSfengbojiang *
80*22ce4affSfengbojiang * 4. Disclaimer and Export Compliance
81*22ce4affSfengbojiang *
82*22ce4affSfengbojiang * 4.1. INTEL MAKES NO WARRANTY OF ANY KIND REGARDING ANY SOFTWARE PROVIDED
83*22ce4affSfengbojiang * HERE. ANY SOFTWARE ORIGINATING FROM INTEL OR DERIVED FROM INTEL SOFTWARE
84*22ce4affSfengbojiang * IS PROVIDED "AS IS," AND INTEL WILL NOT PROVIDE ANY SUPPORT, ASSISTANCE,
85*22ce4affSfengbojiang * INSTALLATION, TRAINING OR OTHER SERVICES. INTEL WILL NOT PROVIDE ANY
86*22ce4affSfengbojiang * UPDATES, ENHANCEMENTS OR EXTENSIONS. INTEL SPECIFICALLY DISCLAIMS ANY
87*22ce4affSfengbojiang * IMPLIED WARRANTIES OF MERCHANTABILITY, NONINFRINGEMENT AND FITNESS FOR A
88*22ce4affSfengbojiang * PARTICULAR PURPOSE.
89*22ce4affSfengbojiang *
90*22ce4affSfengbojiang * 4.2. IN NO EVENT SHALL INTEL HAVE ANY LIABILITY TO LICENSEE, ITS LICENSEES
91*22ce4affSfengbojiang * OR ANY OTHER THIRD PARTY, FOR ANY LOST PROFITS, LOST DATA, LOSS OF USE OR
92*22ce4affSfengbojiang * COSTS OF PROCUREMENT OF SUBSTITUTE GOODS OR SERVICES, OR FOR ANY INDIRECT,
93*22ce4affSfengbojiang * SPECIAL OR CONSEQUENTIAL DAMAGES ARISING OUT OF THIS AGREEMENT, UNDER ANY
94*22ce4affSfengbojiang * CAUSE OF ACTION OR THEORY OF LIABILITY, AND IRRESPECTIVE OF WHETHER INTEL
95*22ce4affSfengbojiang * HAS ADVANCE NOTICE OF THE POSSIBILITY OF SUCH DAMAGES. THESE LIMITATIONS
96*22ce4affSfengbojiang * SHALL APPLY NOTWITHSTANDING THE FAILURE OF THE ESSENTIAL PURPOSE OF ANY
97*22ce4affSfengbojiang * LIMITED REMEDY.
98*22ce4affSfengbojiang *
99*22ce4affSfengbojiang * 4.3. Licensee shall not export, either directly or indirectly, any of this
100*22ce4affSfengbojiang * software or system incorporating such software without first obtaining any
101*22ce4affSfengbojiang * required license or other approval from the U. S. Department of Commerce or
102*22ce4affSfengbojiang * any other agency or department of the United States Government. In the
103*22ce4affSfengbojiang * event Licensee exports any such software from the United States or
104*22ce4affSfengbojiang * re-exports any such software from a foreign destination, Licensee shall
105*22ce4affSfengbojiang * ensure that the distribution and export/re-export of the software is in
106*22ce4affSfengbojiang * compliance with all laws, regulations, orders, or other restrictions of the
107*22ce4affSfengbojiang * U.S. Export Administration Regulations. Licensee agrees that neither it nor
108*22ce4affSfengbojiang * any of its subsidiaries will export/re-export any technical data, process,
109*22ce4affSfengbojiang * software, or service, directly or indirectly, to any country for which the
110*22ce4affSfengbojiang * United States government or any agency thereof requires an export license,
111*22ce4affSfengbojiang * other governmental approval, or letter of assurance, without first obtaining
112*22ce4affSfengbojiang * such license, approval or letter.
113*22ce4affSfengbojiang *
114*22ce4affSfengbojiang *****************************************************************************
115*22ce4affSfengbojiang *
116*22ce4affSfengbojiang * Alternatively, you may choose to be licensed under the terms of the
117*22ce4affSfengbojiang * following license:
118*22ce4affSfengbojiang *
119a9643ea8Slogwang * Redistribution and use in source and binary forms, with or without
120a9643ea8Slogwang * modification, are permitted provided that the following conditions
121a9643ea8Slogwang * are met:
122a9643ea8Slogwang * 1. Redistributions of source code must retain the above copyright
123a9643ea8Slogwang * notice, this list of conditions, and the following disclaimer,
124a9643ea8Slogwang * without modification.
125a9643ea8Slogwang * 2. Redistributions in binary form must reproduce at minimum a disclaimer
126a9643ea8Slogwang * substantially similar to the "NO WARRANTY" disclaimer below
127a9643ea8Slogwang * ("Disclaimer") and any redistribution must be conditioned upon
128a9643ea8Slogwang * including a substantially similar Disclaimer requirement for further
129a9643ea8Slogwang * binary redistribution.
130a9643ea8Slogwang * 3. Neither the names of the above-listed copyright holders nor the names
131a9643ea8Slogwang * of any contributors may be used to endorse or promote products derived
132a9643ea8Slogwang * from this software without specific prior written permission.
133a9643ea8Slogwang *
134*22ce4affSfengbojiang * THIS SOFTWARE IS PROVIDED BY THE COPYRIGHT HOLDERS AND CONTRIBUTORS
135*22ce4affSfengbojiang * "AS IS" AND ANY EXPRESS OR IMPLIED WARRANTIES, INCLUDING, BUT NOT
136*22ce4affSfengbojiang * LIMITED TO, THE IMPLIED WARRANTIES OF MERCHANTABILITY AND FITNESS FOR
137*22ce4affSfengbojiang * A PARTICULAR PURPOSE ARE DISCLAIMED. IN NO EVENT SHALL THE COPYRIGHT
138*22ce4affSfengbojiang * OWNER OR CONTRIBUTORS BE LIABLE FOR ANY DIRECT, INDIRECT, INCIDENTAL,
139*22ce4affSfengbojiang * SPECIAL, EXEMPLARY, OR CONSEQUENTIAL DAMAGES (INCLUDING, BUT NOT
140*22ce4affSfengbojiang * LIMITED TO, PROCUREMENT OF SUBSTITUTE GOODS OR SERVICES; LOSS OF USE,
141*22ce4affSfengbojiang * DATA, OR PROFITS; OR BUSINESS INTERRUPTION) HOWEVER CAUSED AND ON ANY
142*22ce4affSfengbojiang * THEORY OF LIABILITY, WHETHER IN CONTRACT, STRICT LIABILITY, OR TORT
143*22ce4affSfengbojiang * (INCLUDING NEGLIGENCE OR OTHERWISE) ARISING IN ANY WAY OUT OF THE USE
144*22ce4affSfengbojiang * OF THIS SOFTWARE, EVEN IF ADVISED OF THE POSSIBILITY OF SUCH DAMAGE.
145*22ce4affSfengbojiang *
146*22ce4affSfengbojiang * Alternatively, you may choose to be licensed under the terms of the
147a9643ea8Slogwang * GNU General Public License ("GPL") version 2 as published by the Free
148a9643ea8Slogwang * Software Foundation.
149a9643ea8Slogwang *
150*22ce4affSfengbojiang *****************************************************************************/
151a9643ea8Slogwang
152a9643ea8Slogwang /*
153a9643ea8Slogwang * Parse the AML and build an operation tree as most interpreters,
154a9643ea8Slogwang * like Perl, do. Parsing is done by hand rather than with a YACC
155a9643ea8Slogwang * generated parser to tightly constrain stack and dynamic memory
156a9643ea8Slogwang * usage. At the same time, parsing is kept flexible and the code
157a9643ea8Slogwang * fairly compact by parsing based on a list of AML opcode
158a9643ea8Slogwang * templates in AmlOpInfo[]
159a9643ea8Slogwang */
160a9643ea8Slogwang
161a9643ea8Slogwang #include <contrib/dev/acpica/include/acpi.h>
162a9643ea8Slogwang #include <contrib/dev/acpica/include/accommon.h>
163a9643ea8Slogwang #include <contrib/dev/acpica/include/acparser.h>
164a9643ea8Slogwang #include <contrib/dev/acpica/include/acdispat.h>
165a9643ea8Slogwang #include <contrib/dev/acpica/include/amlcode.h>
166a9643ea8Slogwang #include <contrib/dev/acpica/include/acinterp.h>
167*22ce4affSfengbojiang #include <contrib/dev/acpica/include/acnamesp.h>
168a9643ea8Slogwang
169a9643ea8Slogwang #define _COMPONENT ACPI_PARSER
170a9643ea8Slogwang ACPI_MODULE_NAME ("psparse")
171a9643ea8Slogwang
172a9643ea8Slogwang
173a9643ea8Slogwang /*******************************************************************************
174a9643ea8Slogwang *
175a9643ea8Slogwang * FUNCTION: AcpiPsGetOpcodeSize
176a9643ea8Slogwang *
177a9643ea8Slogwang * PARAMETERS: Opcode - An AML opcode
178a9643ea8Slogwang *
179a9643ea8Slogwang * RETURN: Size of the opcode, in bytes (1 or 2)
180a9643ea8Slogwang *
181a9643ea8Slogwang * DESCRIPTION: Get the size of the current opcode.
182a9643ea8Slogwang *
183a9643ea8Slogwang ******************************************************************************/
184a9643ea8Slogwang
185a9643ea8Slogwang UINT32
AcpiPsGetOpcodeSize(UINT32 Opcode)186a9643ea8Slogwang AcpiPsGetOpcodeSize (
187a9643ea8Slogwang UINT32 Opcode)
188a9643ea8Slogwang {
189a9643ea8Slogwang
190a9643ea8Slogwang /* Extended (2-byte) opcode if > 255 */
191a9643ea8Slogwang
192a9643ea8Slogwang if (Opcode > 0x00FF)
193a9643ea8Slogwang {
194a9643ea8Slogwang return (2);
195a9643ea8Slogwang }
196a9643ea8Slogwang
197a9643ea8Slogwang /* Otherwise, just a single byte opcode */
198a9643ea8Slogwang
199a9643ea8Slogwang return (1);
200a9643ea8Slogwang }
201a9643ea8Slogwang
202a9643ea8Slogwang
203a9643ea8Slogwang /*******************************************************************************
204a9643ea8Slogwang *
205a9643ea8Slogwang * FUNCTION: AcpiPsPeekOpcode
206a9643ea8Slogwang *
207a9643ea8Slogwang * PARAMETERS: ParserState - A parser state object
208a9643ea8Slogwang *
209a9643ea8Slogwang * RETURN: Next AML opcode
210a9643ea8Slogwang *
211a9643ea8Slogwang * DESCRIPTION: Get next AML opcode (without incrementing AML pointer)
212a9643ea8Slogwang *
213a9643ea8Slogwang ******************************************************************************/
214a9643ea8Slogwang
215a9643ea8Slogwang UINT16
AcpiPsPeekOpcode(ACPI_PARSE_STATE * ParserState)216a9643ea8Slogwang AcpiPsPeekOpcode (
217a9643ea8Slogwang ACPI_PARSE_STATE *ParserState)
218a9643ea8Slogwang {
219a9643ea8Slogwang UINT8 *Aml;
220a9643ea8Slogwang UINT16 Opcode;
221a9643ea8Slogwang
222a9643ea8Slogwang
223a9643ea8Slogwang Aml = ParserState->Aml;
224a9643ea8Slogwang Opcode = (UINT16) ACPI_GET8 (Aml);
225a9643ea8Slogwang
226*22ce4affSfengbojiang if (Opcode == AML_EXTENDED_PREFIX)
227a9643ea8Slogwang {
228a9643ea8Slogwang /* Extended opcode, get the second opcode byte */
229a9643ea8Slogwang
230a9643ea8Slogwang Aml++;
231a9643ea8Slogwang Opcode = (UINT16) ((Opcode << 8) | ACPI_GET8 (Aml));
232a9643ea8Slogwang }
233a9643ea8Slogwang
234a9643ea8Slogwang return (Opcode);
235a9643ea8Slogwang }
236a9643ea8Slogwang
237a9643ea8Slogwang
238a9643ea8Slogwang /*******************************************************************************
239a9643ea8Slogwang *
240a9643ea8Slogwang * FUNCTION: AcpiPsCompleteThisOp
241a9643ea8Slogwang *
242a9643ea8Slogwang * PARAMETERS: WalkState - Current State
243a9643ea8Slogwang * Op - Op to complete
244a9643ea8Slogwang *
245a9643ea8Slogwang * RETURN: Status
246a9643ea8Slogwang *
247a9643ea8Slogwang * DESCRIPTION: Perform any cleanup at the completion of an Op.
248a9643ea8Slogwang *
249a9643ea8Slogwang ******************************************************************************/
250a9643ea8Slogwang
251a9643ea8Slogwang ACPI_STATUS
AcpiPsCompleteThisOp(ACPI_WALK_STATE * WalkState,ACPI_PARSE_OBJECT * Op)252a9643ea8Slogwang AcpiPsCompleteThisOp (
253a9643ea8Slogwang ACPI_WALK_STATE *WalkState,
254a9643ea8Slogwang ACPI_PARSE_OBJECT *Op)
255a9643ea8Slogwang {
256a9643ea8Slogwang ACPI_PARSE_OBJECT *Prev;
257a9643ea8Slogwang ACPI_PARSE_OBJECT *Next;
258a9643ea8Slogwang const ACPI_OPCODE_INFO *ParentInfo;
259a9643ea8Slogwang ACPI_PARSE_OBJECT *ReplacementOp = NULL;
260a9643ea8Slogwang ACPI_STATUS Status = AE_OK;
261a9643ea8Slogwang
262a9643ea8Slogwang
263a9643ea8Slogwang ACPI_FUNCTION_TRACE_PTR (PsCompleteThisOp, Op);
264a9643ea8Slogwang
265a9643ea8Slogwang
266a9643ea8Slogwang /* Check for null Op, can happen if AML code is corrupt */
267a9643ea8Slogwang
268a9643ea8Slogwang if (!Op)
269a9643ea8Slogwang {
270a9643ea8Slogwang return_ACPI_STATUS (AE_OK); /* OK for now */
271a9643ea8Slogwang }
272a9643ea8Slogwang
273a9643ea8Slogwang AcpiExStopTraceOpcode (Op, WalkState);
274a9643ea8Slogwang
275a9643ea8Slogwang /* Delete this op and the subtree below it if asked to */
276a9643ea8Slogwang
277a9643ea8Slogwang if (((WalkState->ParseFlags & ACPI_PARSE_TREE_MASK) != ACPI_PARSE_DELETE_TREE) ||
278a9643ea8Slogwang (WalkState->OpInfo->Class == AML_CLASS_ARGUMENT))
279a9643ea8Slogwang {
280a9643ea8Slogwang return_ACPI_STATUS (AE_OK);
281a9643ea8Slogwang }
282a9643ea8Slogwang
283a9643ea8Slogwang /* Make sure that we only delete this subtree */
284a9643ea8Slogwang
285a9643ea8Slogwang if (Op->Common.Parent)
286a9643ea8Slogwang {
287a9643ea8Slogwang Prev = Op->Common.Parent->Common.Value.Arg;
288a9643ea8Slogwang if (!Prev)
289a9643ea8Slogwang {
290a9643ea8Slogwang /* Nothing more to do */
291a9643ea8Slogwang
292a9643ea8Slogwang goto Cleanup;
293a9643ea8Slogwang }
294a9643ea8Slogwang
295a9643ea8Slogwang /*
296a9643ea8Slogwang * Check if we need to replace the operator and its subtree
297a9643ea8Slogwang * with a return value op (placeholder op)
298a9643ea8Slogwang */
299a9643ea8Slogwang ParentInfo = AcpiPsGetOpcodeInfo (Op->Common.Parent->Common.AmlOpcode);
300a9643ea8Slogwang
301a9643ea8Slogwang switch (ParentInfo->Class)
302a9643ea8Slogwang {
303a9643ea8Slogwang case AML_CLASS_CONTROL:
304a9643ea8Slogwang
305a9643ea8Slogwang break;
306a9643ea8Slogwang
307a9643ea8Slogwang case AML_CLASS_CREATE:
308a9643ea8Slogwang /*
309a9643ea8Slogwang * These opcodes contain TermArg operands. The current
310a9643ea8Slogwang * op must be replaced by a placeholder return op
311a9643ea8Slogwang */
312a9643ea8Slogwang ReplacementOp = AcpiPsAllocOp (
313a9643ea8Slogwang AML_INT_RETURN_VALUE_OP, Op->Common.Aml);
314a9643ea8Slogwang if (!ReplacementOp)
315a9643ea8Slogwang {
316a9643ea8Slogwang Status = AE_NO_MEMORY;
317a9643ea8Slogwang }
318a9643ea8Slogwang break;
319a9643ea8Slogwang
320a9643ea8Slogwang case AML_CLASS_NAMED_OBJECT:
321a9643ea8Slogwang /*
322a9643ea8Slogwang * These opcodes contain TermArg operands. The current
323a9643ea8Slogwang * op must be replaced by a placeholder return op
324a9643ea8Slogwang */
325a9643ea8Slogwang if ((Op->Common.Parent->Common.AmlOpcode == AML_REGION_OP) ||
326a9643ea8Slogwang (Op->Common.Parent->Common.AmlOpcode == AML_DATA_REGION_OP) ||
327a9643ea8Slogwang (Op->Common.Parent->Common.AmlOpcode == AML_BUFFER_OP) ||
328a9643ea8Slogwang (Op->Common.Parent->Common.AmlOpcode == AML_PACKAGE_OP) ||
329a9643ea8Slogwang (Op->Common.Parent->Common.AmlOpcode == AML_BANK_FIELD_OP) ||
330*22ce4affSfengbojiang (Op->Common.Parent->Common.AmlOpcode == AML_VARIABLE_PACKAGE_OP))
331a9643ea8Slogwang {
332a9643ea8Slogwang ReplacementOp = AcpiPsAllocOp (
333a9643ea8Slogwang AML_INT_RETURN_VALUE_OP, Op->Common.Aml);
334a9643ea8Slogwang if (!ReplacementOp)
335a9643ea8Slogwang {
336a9643ea8Slogwang Status = AE_NO_MEMORY;
337a9643ea8Slogwang }
338a9643ea8Slogwang }
339a9643ea8Slogwang else if ((Op->Common.Parent->Common.AmlOpcode == AML_NAME_OP) &&
340a9643ea8Slogwang (WalkState->PassNumber <= ACPI_IMODE_LOAD_PASS2))
341a9643ea8Slogwang {
342a9643ea8Slogwang if ((Op->Common.AmlOpcode == AML_BUFFER_OP) ||
343a9643ea8Slogwang (Op->Common.AmlOpcode == AML_PACKAGE_OP) ||
344*22ce4affSfengbojiang (Op->Common.AmlOpcode == AML_VARIABLE_PACKAGE_OP))
345a9643ea8Slogwang {
346a9643ea8Slogwang ReplacementOp = AcpiPsAllocOp (Op->Common.AmlOpcode,
347a9643ea8Slogwang Op->Common.Aml);
348a9643ea8Slogwang if (!ReplacementOp)
349a9643ea8Slogwang {
350a9643ea8Slogwang Status = AE_NO_MEMORY;
351a9643ea8Slogwang }
352a9643ea8Slogwang else
353a9643ea8Slogwang {
354a9643ea8Slogwang ReplacementOp->Named.Data = Op->Named.Data;
355a9643ea8Slogwang ReplacementOp->Named.Length = Op->Named.Length;
356a9643ea8Slogwang }
357a9643ea8Slogwang }
358a9643ea8Slogwang }
359a9643ea8Slogwang break;
360a9643ea8Slogwang
361a9643ea8Slogwang default:
362a9643ea8Slogwang
363a9643ea8Slogwang ReplacementOp = AcpiPsAllocOp (
364a9643ea8Slogwang AML_INT_RETURN_VALUE_OP, Op->Common.Aml);
365a9643ea8Slogwang if (!ReplacementOp)
366a9643ea8Slogwang {
367a9643ea8Slogwang Status = AE_NO_MEMORY;
368a9643ea8Slogwang }
369a9643ea8Slogwang }
370a9643ea8Slogwang
371a9643ea8Slogwang /* We must unlink this op from the parent tree */
372a9643ea8Slogwang
373a9643ea8Slogwang if (Prev == Op)
374a9643ea8Slogwang {
375a9643ea8Slogwang /* This op is the first in the list */
376a9643ea8Slogwang
377a9643ea8Slogwang if (ReplacementOp)
378a9643ea8Slogwang {
379a9643ea8Slogwang ReplacementOp->Common.Parent = Op->Common.Parent;
380a9643ea8Slogwang ReplacementOp->Common.Value.Arg = NULL;
381a9643ea8Slogwang ReplacementOp->Common.Node = Op->Common.Node;
382a9643ea8Slogwang Op->Common.Parent->Common.Value.Arg = ReplacementOp;
383a9643ea8Slogwang ReplacementOp->Common.Next = Op->Common.Next;
384a9643ea8Slogwang }
385a9643ea8Slogwang else
386a9643ea8Slogwang {
387a9643ea8Slogwang Op->Common.Parent->Common.Value.Arg = Op->Common.Next;
388a9643ea8Slogwang }
389a9643ea8Slogwang }
390a9643ea8Slogwang
391a9643ea8Slogwang /* Search the parent list */
392a9643ea8Slogwang
393a9643ea8Slogwang else while (Prev)
394a9643ea8Slogwang {
395a9643ea8Slogwang /* Traverse all siblings in the parent's argument list */
396a9643ea8Slogwang
397a9643ea8Slogwang Next = Prev->Common.Next;
398a9643ea8Slogwang if (Next == Op)
399a9643ea8Slogwang {
400a9643ea8Slogwang if (ReplacementOp)
401a9643ea8Slogwang {
402a9643ea8Slogwang ReplacementOp->Common.Parent = Op->Common.Parent;
403a9643ea8Slogwang ReplacementOp->Common.Value.Arg = NULL;
404a9643ea8Slogwang ReplacementOp->Common.Node = Op->Common.Node;
405a9643ea8Slogwang Prev->Common.Next = ReplacementOp;
406a9643ea8Slogwang ReplacementOp->Common.Next = Op->Common.Next;
407a9643ea8Slogwang Next = NULL;
408a9643ea8Slogwang }
409a9643ea8Slogwang else
410a9643ea8Slogwang {
411a9643ea8Slogwang Prev->Common.Next = Op->Common.Next;
412a9643ea8Slogwang Next = NULL;
413a9643ea8Slogwang }
414a9643ea8Slogwang }
415a9643ea8Slogwang Prev = Next;
416a9643ea8Slogwang }
417a9643ea8Slogwang }
418a9643ea8Slogwang
419a9643ea8Slogwang
420a9643ea8Slogwang Cleanup:
421a9643ea8Slogwang
422a9643ea8Slogwang /* Now we can actually delete the subtree rooted at Op */
423a9643ea8Slogwang
424a9643ea8Slogwang AcpiPsDeleteParseTree (Op);
425a9643ea8Slogwang return_ACPI_STATUS (Status);
426a9643ea8Slogwang }
427a9643ea8Slogwang
428a9643ea8Slogwang
429a9643ea8Slogwang /*******************************************************************************
430a9643ea8Slogwang *
431a9643ea8Slogwang * FUNCTION: AcpiPsNextParseState
432a9643ea8Slogwang *
433a9643ea8Slogwang * PARAMETERS: WalkState - Current state
434a9643ea8Slogwang * Op - Current parse op
435a9643ea8Slogwang * CallbackStatus - Status from previous operation
436a9643ea8Slogwang *
437a9643ea8Slogwang * RETURN: Status
438a9643ea8Slogwang *
439a9643ea8Slogwang * DESCRIPTION: Update the parser state based upon the return exception from
440a9643ea8Slogwang * the parser callback.
441a9643ea8Slogwang *
442a9643ea8Slogwang ******************************************************************************/
443a9643ea8Slogwang
444a9643ea8Slogwang ACPI_STATUS
AcpiPsNextParseState(ACPI_WALK_STATE * WalkState,ACPI_PARSE_OBJECT * Op,ACPI_STATUS CallbackStatus)445a9643ea8Slogwang AcpiPsNextParseState (
446a9643ea8Slogwang ACPI_WALK_STATE *WalkState,
447a9643ea8Slogwang ACPI_PARSE_OBJECT *Op,
448a9643ea8Slogwang ACPI_STATUS CallbackStatus)
449a9643ea8Slogwang {
450a9643ea8Slogwang ACPI_PARSE_STATE *ParserState = &WalkState->ParserState;
451a9643ea8Slogwang ACPI_STATUS Status = AE_CTRL_PENDING;
452a9643ea8Slogwang
453a9643ea8Slogwang
454a9643ea8Slogwang ACPI_FUNCTION_TRACE_PTR (PsNextParseState, Op);
455a9643ea8Slogwang
456a9643ea8Slogwang
457a9643ea8Slogwang switch (CallbackStatus)
458a9643ea8Slogwang {
459a9643ea8Slogwang case AE_CTRL_TERMINATE:
460a9643ea8Slogwang /*
461a9643ea8Slogwang * A control method was terminated via a RETURN statement.
462a9643ea8Slogwang * The walk of this method is complete.
463a9643ea8Slogwang */
464a9643ea8Slogwang ParserState->Aml = ParserState->AmlEnd;
465a9643ea8Slogwang Status = AE_CTRL_TERMINATE;
466a9643ea8Slogwang break;
467a9643ea8Slogwang
468a9643ea8Slogwang case AE_CTRL_BREAK:
469a9643ea8Slogwang
470a9643ea8Slogwang ParserState->Aml = WalkState->AmlLastWhile;
471a9643ea8Slogwang WalkState->ControlState->Common.Value = FALSE;
472a9643ea8Slogwang Status = AE_CTRL_BREAK;
473a9643ea8Slogwang break;
474a9643ea8Slogwang
475a9643ea8Slogwang case AE_CTRL_CONTINUE:
476a9643ea8Slogwang
477a9643ea8Slogwang ParserState->Aml = WalkState->AmlLastWhile;
478a9643ea8Slogwang Status = AE_CTRL_CONTINUE;
479a9643ea8Slogwang break;
480a9643ea8Slogwang
481a9643ea8Slogwang case AE_CTRL_PENDING:
482a9643ea8Slogwang
483a9643ea8Slogwang ParserState->Aml = WalkState->AmlLastWhile;
484a9643ea8Slogwang break;
485a9643ea8Slogwang
486a9643ea8Slogwang #if 0
487a9643ea8Slogwang case AE_CTRL_SKIP:
488a9643ea8Slogwang
489a9643ea8Slogwang ParserState->Aml = ParserState->Scope->ParseScope.PkgEnd;
490a9643ea8Slogwang Status = AE_OK;
491a9643ea8Slogwang break;
492a9643ea8Slogwang #endif
493a9643ea8Slogwang
494a9643ea8Slogwang case AE_CTRL_TRUE:
495a9643ea8Slogwang /*
496a9643ea8Slogwang * Predicate of an IF was true, and we are at the matching ELSE.
497a9643ea8Slogwang * Just close out this package
498a9643ea8Slogwang */
499a9643ea8Slogwang ParserState->Aml = AcpiPsGetNextPackageEnd (ParserState);
500a9643ea8Slogwang Status = AE_CTRL_PENDING;
501a9643ea8Slogwang break;
502a9643ea8Slogwang
503a9643ea8Slogwang case AE_CTRL_FALSE:
504a9643ea8Slogwang /*
505a9643ea8Slogwang * Either an IF/WHILE Predicate was false or we encountered a BREAK
506a9643ea8Slogwang * opcode. In both cases, we do not execute the rest of the
507a9643ea8Slogwang * package; We simply close out the parent (finishing the walk of
508a9643ea8Slogwang * this branch of the tree) and continue execution at the parent
509a9643ea8Slogwang * level.
510a9643ea8Slogwang */
511a9643ea8Slogwang ParserState->Aml = ParserState->Scope->ParseScope.PkgEnd;
512a9643ea8Slogwang
513a9643ea8Slogwang /* In the case of a BREAK, just force a predicate (if any) to FALSE */
514a9643ea8Slogwang
515a9643ea8Slogwang WalkState->ControlState->Common.Value = FALSE;
516a9643ea8Slogwang Status = AE_CTRL_END;
517a9643ea8Slogwang break;
518a9643ea8Slogwang
519a9643ea8Slogwang case AE_CTRL_TRANSFER:
520a9643ea8Slogwang
521a9643ea8Slogwang /* A method call (invocation) -- transfer control */
522a9643ea8Slogwang
523a9643ea8Slogwang Status = AE_CTRL_TRANSFER;
524a9643ea8Slogwang WalkState->PrevOp = Op;
525a9643ea8Slogwang WalkState->MethodCallOp = Op;
526a9643ea8Slogwang WalkState->MethodCallNode = (Op->Common.Value.Arg)->Common.Node;
527a9643ea8Slogwang
528a9643ea8Slogwang /* Will return value (if any) be used by the caller? */
529a9643ea8Slogwang
530a9643ea8Slogwang WalkState->ReturnUsed = AcpiDsIsResultUsed (Op, WalkState);
531a9643ea8Slogwang break;
532a9643ea8Slogwang
533a9643ea8Slogwang default:
534a9643ea8Slogwang
535a9643ea8Slogwang Status = CallbackStatus;
536a9643ea8Slogwang if ((CallbackStatus & AE_CODE_MASK) == AE_CODE_CONTROL)
537a9643ea8Slogwang {
538a9643ea8Slogwang Status = AE_OK;
539a9643ea8Slogwang }
540a9643ea8Slogwang break;
541a9643ea8Slogwang }
542a9643ea8Slogwang
543a9643ea8Slogwang return_ACPI_STATUS (Status);
544a9643ea8Slogwang }
545a9643ea8Slogwang
546a9643ea8Slogwang
547a9643ea8Slogwang /*******************************************************************************
548a9643ea8Slogwang *
549a9643ea8Slogwang * FUNCTION: AcpiPsParseAml
550a9643ea8Slogwang *
551a9643ea8Slogwang * PARAMETERS: WalkState - Current state
552a9643ea8Slogwang *
553a9643ea8Slogwang *
554a9643ea8Slogwang * RETURN: Status
555a9643ea8Slogwang *
556a9643ea8Slogwang * DESCRIPTION: Parse raw AML and return a tree of ops
557a9643ea8Slogwang *
558a9643ea8Slogwang ******************************************************************************/
559a9643ea8Slogwang
560a9643ea8Slogwang ACPI_STATUS
AcpiPsParseAml(ACPI_WALK_STATE * WalkState)561a9643ea8Slogwang AcpiPsParseAml (
562a9643ea8Slogwang ACPI_WALK_STATE *WalkState)
563a9643ea8Slogwang {
564a9643ea8Slogwang ACPI_STATUS Status;
565a9643ea8Slogwang ACPI_THREAD_STATE *Thread;
566a9643ea8Slogwang ACPI_THREAD_STATE *PrevWalkList = AcpiGbl_CurrentWalkList;
567a9643ea8Slogwang ACPI_WALK_STATE *PreviousWalkState;
568a9643ea8Slogwang
569a9643ea8Slogwang
570a9643ea8Slogwang ACPI_FUNCTION_TRACE (PsParseAml);
571a9643ea8Slogwang
572a9643ea8Slogwang ACPI_DEBUG_PRINT ((ACPI_DB_PARSE,
573a9643ea8Slogwang "Entered with WalkState=%p Aml=%p size=%X\n",
574a9643ea8Slogwang WalkState, WalkState->ParserState.Aml,
575a9643ea8Slogwang WalkState->ParserState.AmlSize));
576a9643ea8Slogwang
577a9643ea8Slogwang if (!WalkState->ParserState.Aml)
578a9643ea8Slogwang {
579*22ce4affSfengbojiang return_ACPI_STATUS (AE_BAD_ADDRESS);
580a9643ea8Slogwang }
581a9643ea8Slogwang
582a9643ea8Slogwang /* Create and initialize a new thread state */
583a9643ea8Slogwang
584a9643ea8Slogwang Thread = AcpiUtCreateThreadState ();
585a9643ea8Slogwang if (!Thread)
586a9643ea8Slogwang {
587a9643ea8Slogwang if (WalkState->MethodDesc)
588a9643ea8Slogwang {
589a9643ea8Slogwang /* Executing a control method - additional cleanup */
590a9643ea8Slogwang
591a9643ea8Slogwang AcpiDsTerminateControlMethod (WalkState->MethodDesc, WalkState);
592a9643ea8Slogwang }
593a9643ea8Slogwang
594a9643ea8Slogwang AcpiDsDeleteWalkState (WalkState);
595a9643ea8Slogwang return_ACPI_STATUS (AE_NO_MEMORY);
596a9643ea8Slogwang }
597a9643ea8Slogwang
598a9643ea8Slogwang WalkState->Thread = Thread;
599a9643ea8Slogwang
600a9643ea8Slogwang /*
601a9643ea8Slogwang * If executing a method, the starting SyncLevel is this method's
602a9643ea8Slogwang * SyncLevel
603a9643ea8Slogwang */
604a9643ea8Slogwang if (WalkState->MethodDesc)
605a9643ea8Slogwang {
606a9643ea8Slogwang WalkState->Thread->CurrentSyncLevel =
607a9643ea8Slogwang WalkState->MethodDesc->Method.SyncLevel;
608a9643ea8Slogwang }
609a9643ea8Slogwang
610a9643ea8Slogwang AcpiDsPushWalkState (WalkState, Thread);
611a9643ea8Slogwang
612a9643ea8Slogwang /*
613a9643ea8Slogwang * This global allows the AML debugger to get a handle to the currently
614a9643ea8Slogwang * executing control method.
615a9643ea8Slogwang */
616a9643ea8Slogwang AcpiGbl_CurrentWalkList = Thread;
617a9643ea8Slogwang
618a9643ea8Slogwang /*
619a9643ea8Slogwang * Execute the walk loop as long as there is a valid Walk State. This
620a9643ea8Slogwang * handles nested control method invocations without recursion.
621a9643ea8Slogwang */
622a9643ea8Slogwang ACPI_DEBUG_PRINT ((ACPI_DB_PARSE, "State=%p\n", WalkState));
623a9643ea8Slogwang
624a9643ea8Slogwang Status = AE_OK;
625a9643ea8Slogwang while (WalkState)
626a9643ea8Slogwang {
627a9643ea8Slogwang if (ACPI_SUCCESS (Status))
628a9643ea8Slogwang {
629a9643ea8Slogwang /*
630a9643ea8Slogwang * The ParseLoop executes AML until the method terminates
631a9643ea8Slogwang * or calls another method.
632a9643ea8Slogwang */
633a9643ea8Slogwang Status = AcpiPsParseLoop (WalkState);
634a9643ea8Slogwang }
635a9643ea8Slogwang
636a9643ea8Slogwang ACPI_DEBUG_PRINT ((ACPI_DB_PARSE,
637a9643ea8Slogwang "Completed one call to walk loop, %s State=%p\n",
638a9643ea8Slogwang AcpiFormatException (Status), WalkState));
639a9643ea8Slogwang
640*22ce4affSfengbojiang if (WalkState->MethodPathname && WalkState->MethodIsNested)
641*22ce4affSfengbojiang {
642*22ce4affSfengbojiang /* Optional object evaluation log */
643*22ce4affSfengbojiang
644*22ce4affSfengbojiang ACPI_DEBUG_PRINT_RAW ((ACPI_DB_EVALUATION, "%-26s: %*s%s\n",
645*22ce4affSfengbojiang " Exit nested method",
646*22ce4affSfengbojiang (WalkState->MethodNestingDepth + 1) * 3, " ",
647*22ce4affSfengbojiang &WalkState->MethodPathname[1]));
648*22ce4affSfengbojiang
649*22ce4affSfengbojiang ACPI_FREE (WalkState->MethodPathname);
650*22ce4affSfengbojiang WalkState->MethodIsNested = FALSE;
651*22ce4affSfengbojiang }
652a9643ea8Slogwang if (Status == AE_CTRL_TRANSFER)
653a9643ea8Slogwang {
654a9643ea8Slogwang /*
655a9643ea8Slogwang * A method call was detected.
656a9643ea8Slogwang * Transfer control to the called control method
657a9643ea8Slogwang */
658a9643ea8Slogwang Status = AcpiDsCallControlMethod (Thread, WalkState, NULL);
659a9643ea8Slogwang if (ACPI_FAILURE (Status))
660a9643ea8Slogwang {
661a9643ea8Slogwang Status = AcpiDsMethodError (Status, WalkState);
662a9643ea8Slogwang }
663a9643ea8Slogwang
664a9643ea8Slogwang /*
665*22ce4affSfengbojiang * If the transfer to the new method method call worked,
666*22ce4affSfengbojiang * a new walk state was created -- get it
667a9643ea8Slogwang */
668a9643ea8Slogwang WalkState = AcpiDsGetCurrentWalkState (Thread);
669a9643ea8Slogwang continue;
670a9643ea8Slogwang }
671a9643ea8Slogwang else if (Status == AE_CTRL_TERMINATE)
672a9643ea8Slogwang {
673a9643ea8Slogwang Status = AE_OK;
674a9643ea8Slogwang }
675a9643ea8Slogwang else if ((Status != AE_OK) && (WalkState->MethodDesc))
676a9643ea8Slogwang {
677a9643ea8Slogwang /* Either the method parse or actual execution failed */
678a9643ea8Slogwang
679*22ce4affSfengbojiang AcpiExExitInterpreter ();
680*22ce4affSfengbojiang if (Status == AE_ABORT_METHOD)
681*22ce4affSfengbojiang {
682*22ce4affSfengbojiang AcpiNsPrintNodePathname (
683*22ce4affSfengbojiang WalkState->MethodNode, "Aborting method");
684*22ce4affSfengbojiang AcpiOsPrintf ("\n");
685*22ce4affSfengbojiang }
686*22ce4affSfengbojiang else
687*22ce4affSfengbojiang {
688*22ce4affSfengbojiang ACPI_ERROR_METHOD ("Aborting method",
689a9643ea8Slogwang WalkState->MethodNode, NULL, Status);
690*22ce4affSfengbojiang }
691*22ce4affSfengbojiang AcpiExEnterInterpreter ();
692a9643ea8Slogwang
693a9643ea8Slogwang /* Check for possible multi-thread reentrancy problem */
694a9643ea8Slogwang
695a9643ea8Slogwang if ((Status == AE_ALREADY_EXISTS) &&
696a9643ea8Slogwang (!(WalkState->MethodDesc->Method.InfoFlags &
697a9643ea8Slogwang ACPI_METHOD_SERIALIZED)))
698a9643ea8Slogwang {
699a9643ea8Slogwang /*
700a9643ea8Slogwang * Method is not serialized and tried to create an object
701a9643ea8Slogwang * twice. The probable cause is that the method cannot
702a9643ea8Slogwang * handle reentrancy. Mark as "pending serialized" now, and
703a9643ea8Slogwang * then mark "serialized" when the last thread exits.
704a9643ea8Slogwang */
705a9643ea8Slogwang WalkState->MethodDesc->Method.InfoFlags |=
706a9643ea8Slogwang ACPI_METHOD_SERIALIZED_PENDING;
707a9643ea8Slogwang }
708a9643ea8Slogwang }
709a9643ea8Slogwang
710a9643ea8Slogwang /* We are done with this walk, move on to the parent if any */
711a9643ea8Slogwang
712a9643ea8Slogwang WalkState = AcpiDsPopWalkState (Thread);
713a9643ea8Slogwang
714a9643ea8Slogwang /* Reset the current scope to the beginning of scope stack */
715a9643ea8Slogwang
716a9643ea8Slogwang AcpiDsScopeStackClear (WalkState);
717a9643ea8Slogwang
718a9643ea8Slogwang /*
719a9643ea8Slogwang * If we just returned from the execution of a control method or if we
720a9643ea8Slogwang * encountered an error during the method parse phase, there's lots of
721a9643ea8Slogwang * cleanup to do
722a9643ea8Slogwang */
723a9643ea8Slogwang if (((WalkState->ParseFlags & ACPI_PARSE_MODE_MASK) ==
724*22ce4affSfengbojiang ACPI_PARSE_EXECUTE &&
725*22ce4affSfengbojiang !(WalkState->ParseFlags & ACPI_PARSE_MODULE_LEVEL)) ||
726a9643ea8Slogwang (ACPI_FAILURE (Status)))
727a9643ea8Slogwang {
728a9643ea8Slogwang AcpiDsTerminateControlMethod (WalkState->MethodDesc, WalkState);
729a9643ea8Slogwang }
730a9643ea8Slogwang
731a9643ea8Slogwang /* Delete this walk state and all linked control states */
732a9643ea8Slogwang
733a9643ea8Slogwang AcpiPsCleanupScope (&WalkState->ParserState);
734a9643ea8Slogwang PreviousWalkState = WalkState;
735a9643ea8Slogwang
736a9643ea8Slogwang ACPI_DEBUG_PRINT ((ACPI_DB_PARSE,
737a9643ea8Slogwang "ReturnValue=%p, ImplicitValue=%p State=%p\n",
738a9643ea8Slogwang WalkState->ReturnDesc, WalkState->ImplicitReturnObj, WalkState));
739a9643ea8Slogwang
740a9643ea8Slogwang /* Check if we have restarted a preempted walk */
741a9643ea8Slogwang
742a9643ea8Slogwang WalkState = AcpiDsGetCurrentWalkState (Thread);
743a9643ea8Slogwang if (WalkState)
744a9643ea8Slogwang {
745a9643ea8Slogwang if (ACPI_SUCCESS (Status))
746a9643ea8Slogwang {
747a9643ea8Slogwang /*
748a9643ea8Slogwang * There is another walk state, restart it.
749a9643ea8Slogwang * If the method return value is not used by the parent,
750a9643ea8Slogwang * The object is deleted
751a9643ea8Slogwang */
752a9643ea8Slogwang if (!PreviousWalkState->ReturnDesc)
753a9643ea8Slogwang {
754a9643ea8Slogwang /*
755a9643ea8Slogwang * In slack mode execution, if there is no return value
756a9643ea8Slogwang * we should implicitly return zero (0) as a default value.
757a9643ea8Slogwang */
758a9643ea8Slogwang if (AcpiGbl_EnableInterpreterSlack &&
759a9643ea8Slogwang !PreviousWalkState->ImplicitReturnObj)
760a9643ea8Slogwang {
761a9643ea8Slogwang PreviousWalkState->ImplicitReturnObj =
762a9643ea8Slogwang AcpiUtCreateIntegerObject ((UINT64) 0);
763a9643ea8Slogwang if (!PreviousWalkState->ImplicitReturnObj)
764a9643ea8Slogwang {
765a9643ea8Slogwang return_ACPI_STATUS (AE_NO_MEMORY);
766a9643ea8Slogwang }
767a9643ea8Slogwang }
768a9643ea8Slogwang
769a9643ea8Slogwang /* Restart the calling control method */
770a9643ea8Slogwang
771a9643ea8Slogwang Status = AcpiDsRestartControlMethod (WalkState,
772a9643ea8Slogwang PreviousWalkState->ImplicitReturnObj);
773a9643ea8Slogwang }
774a9643ea8Slogwang else
775a9643ea8Slogwang {
776a9643ea8Slogwang /*
777a9643ea8Slogwang * We have a valid return value, delete any implicit
778a9643ea8Slogwang * return value.
779a9643ea8Slogwang */
780a9643ea8Slogwang AcpiDsClearImplicitReturn (PreviousWalkState);
781a9643ea8Slogwang
782a9643ea8Slogwang Status = AcpiDsRestartControlMethod (WalkState,
783a9643ea8Slogwang PreviousWalkState->ReturnDesc);
784a9643ea8Slogwang }
785a9643ea8Slogwang if (ACPI_SUCCESS (Status))
786a9643ea8Slogwang {
787a9643ea8Slogwang WalkState->WalkType |= ACPI_WALK_METHOD_RESTART;
788a9643ea8Slogwang }
789a9643ea8Slogwang }
790a9643ea8Slogwang else
791a9643ea8Slogwang {
792a9643ea8Slogwang /* On error, delete any return object or implicit return */
793a9643ea8Slogwang
794a9643ea8Slogwang AcpiUtRemoveReference (PreviousWalkState->ReturnDesc);
795a9643ea8Slogwang AcpiDsClearImplicitReturn (PreviousWalkState);
796a9643ea8Slogwang }
797a9643ea8Slogwang }
798a9643ea8Slogwang
799a9643ea8Slogwang /*
800a9643ea8Slogwang * Just completed a 1st-level method, save the final internal return
801a9643ea8Slogwang * value (if any)
802a9643ea8Slogwang */
803a9643ea8Slogwang else if (PreviousWalkState->CallerReturnDesc)
804a9643ea8Slogwang {
805a9643ea8Slogwang if (PreviousWalkState->ImplicitReturnObj)
806a9643ea8Slogwang {
807a9643ea8Slogwang *(PreviousWalkState->CallerReturnDesc) =
808a9643ea8Slogwang PreviousWalkState->ImplicitReturnObj;
809a9643ea8Slogwang }
810a9643ea8Slogwang else
811a9643ea8Slogwang {
812a9643ea8Slogwang /* NULL if no return value */
813a9643ea8Slogwang
814a9643ea8Slogwang *(PreviousWalkState->CallerReturnDesc) =
815a9643ea8Slogwang PreviousWalkState->ReturnDesc;
816a9643ea8Slogwang }
817a9643ea8Slogwang }
818a9643ea8Slogwang else
819a9643ea8Slogwang {
820a9643ea8Slogwang if (PreviousWalkState->ReturnDesc)
821a9643ea8Slogwang {
822a9643ea8Slogwang /* Caller doesn't want it, must delete it */
823a9643ea8Slogwang
824a9643ea8Slogwang AcpiUtRemoveReference (PreviousWalkState->ReturnDesc);
825a9643ea8Slogwang }
826a9643ea8Slogwang if (PreviousWalkState->ImplicitReturnObj)
827a9643ea8Slogwang {
828a9643ea8Slogwang /* Caller doesn't want it, must delete it */
829a9643ea8Slogwang
830a9643ea8Slogwang AcpiUtRemoveReference (PreviousWalkState->ImplicitReturnObj);
831a9643ea8Slogwang }
832a9643ea8Slogwang }
833a9643ea8Slogwang
834a9643ea8Slogwang AcpiDsDeleteWalkState (PreviousWalkState);
835a9643ea8Slogwang }
836a9643ea8Slogwang
837a9643ea8Slogwang /* Normal exit */
838a9643ea8Slogwang
839a9643ea8Slogwang AcpiExReleaseAllMutexes (Thread);
840a9643ea8Slogwang AcpiUtDeleteGenericState (ACPI_CAST_PTR (ACPI_GENERIC_STATE, Thread));
841a9643ea8Slogwang AcpiGbl_CurrentWalkList = PrevWalkList;
842a9643ea8Slogwang return_ACPI_STATUS (Status);
843a9643ea8Slogwang }
844