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