xref: /freebsd/sys/contrib/dev/acpica/compiler/asltransform.c (revision 69c5fa5cd1ec9b09ed88a086607a8a0993818db9)
1 /******************************************************************************
2  *
3  * Module Name: asltransform - Parse tree transforms
4  *
5  *****************************************************************************/
6 
7 /******************************************************************************
8  *
9  * 1. Copyright Notice
10  *
11  * Some or all of this work - Copyright (c) 1999 - 2021, Intel Corp.
12  * All rights reserved.
13  *
14  * 2. License
15  *
16  * 2.1. This is your license from Intel Corp. under its intellectual property
17  * rights. You may have additional license terms from the party that provided
18  * you this software, covering your right to use that party's intellectual
19  * property rights.
20  *
21  * 2.2. Intel grants, free of charge, to any person ("Licensee") obtaining a
22  * copy of the source code appearing in this file ("Covered Code") an
23  * irrevocable, perpetual, worldwide license under Intel's copyrights in the
24  * base code distributed originally by Intel ("Original Intel Code") to copy,
25  * make derivatives, distribute, use and display any portion of the Covered
26  * Code in any form, with the right to sublicense such rights; and
27  *
28  * 2.3. Intel grants Licensee a non-exclusive and non-transferable patent
29  * license (with the right to sublicense), under only those claims of Intel
30  * patents that are infringed by the Original Intel Code, to make, use, sell,
31  * offer to sell, and import the Covered Code and derivative works thereof
32  * solely to the minimum extent necessary to exercise the above copyright
33  * license, and in no event shall the patent license extend to any additions
34  * to or modifications of the Original Intel Code. No other license or right
35  * is granted directly or by implication, estoppel or otherwise;
36  *
37  * The above copyright and patent license is granted only if the following
38  * conditions are met:
39  *
40  * 3. Conditions
41  *
42  * 3.1. Redistribution of Source with Rights to Further Distribute Source.
43  * Redistribution of source code of any substantial portion of the Covered
44  * Code or modification with rights to further distribute source must include
45  * the above Copyright Notice, the above License, this list of Conditions,
46  * and the following Disclaimer and Export Compliance provision. In addition,
47  * Licensee must cause all Covered Code to which Licensee contributes to
48  * contain a file documenting the changes Licensee made to create that Covered
49  * Code and the date of any change. Licensee must include in that file the
50  * documentation of any changes made by any predecessor Licensee. Licensee
51  * must include a prominent statement that the modification is derived,
52  * directly or indirectly, from Original Intel Code.
53  *
54  * 3.2. Redistribution of Source with no Rights to Further Distribute Source.
55  * Redistribution of source code of any substantial portion of the Covered
56  * Code or modification without rights to further distribute source must
57  * include the following Disclaimer and Export Compliance provision in the
58  * documentation and/or other materials provided with distribution. In
59  * addition, Licensee may not authorize further sublicense of source of any
60  * portion of the Covered Code, and must include terms to the effect that the
61  * license from Licensee to its licensee is limited to the intellectual
62  * property embodied in the software Licensee provides to its licensee, and
63  * not to intellectual property embodied in modifications its licensee may
64  * make.
65  *
66  * 3.3. Redistribution of Executable. Redistribution in executable form of any
67  * substantial portion of the Covered Code or modification must reproduce the
68  * above Copyright Notice, and the following Disclaimer and Export Compliance
69  * provision in the documentation and/or other materials provided with the
70  * distribution.
71  *
72  * 3.4. Intel retains all right, title, and interest in and to the Original
73  * Intel Code.
74  *
75  * 3.5. Neither the name Intel nor any other trademark owned or controlled by
76  * Intel shall be used in advertising or otherwise to promote the sale, use or
77  * other dealings in products derived from or relating to the Covered Code
78  * without prior written authorization from Intel.
79  *
80  * 4. Disclaimer and Export Compliance
81  *
82  * 4.1. INTEL MAKES NO WARRANTY OF ANY KIND REGARDING ANY SOFTWARE PROVIDED
83  * HERE. ANY SOFTWARE ORIGINATING FROM INTEL OR DERIVED FROM INTEL SOFTWARE
84  * IS PROVIDED "AS IS," AND INTEL WILL NOT PROVIDE ANY SUPPORT, ASSISTANCE,
85  * INSTALLATION, TRAINING OR OTHER SERVICES. INTEL WILL NOT PROVIDE ANY
86  * UPDATES, ENHANCEMENTS OR EXTENSIONS. INTEL SPECIFICALLY DISCLAIMS ANY
87  * IMPLIED WARRANTIES OF MERCHANTABILITY, NONINFRINGEMENT AND FITNESS FOR A
88  * PARTICULAR PURPOSE.
89  *
90  * 4.2. IN NO EVENT SHALL INTEL HAVE ANY LIABILITY TO LICENSEE, ITS LICENSEES
91  * OR ANY OTHER THIRD PARTY, FOR ANY LOST PROFITS, LOST DATA, LOSS OF USE OR
92  * COSTS OF PROCUREMENT OF SUBSTITUTE GOODS OR SERVICES, OR FOR ANY INDIRECT,
93  * SPECIAL OR CONSEQUENTIAL DAMAGES ARISING OUT OF THIS AGREEMENT, UNDER ANY
94  * CAUSE OF ACTION OR THEORY OF LIABILITY, AND IRRESPECTIVE OF WHETHER INTEL
95  * HAS ADVANCE NOTICE OF THE POSSIBILITY OF SUCH DAMAGES. THESE LIMITATIONS
96  * SHALL APPLY NOTWITHSTANDING THE FAILURE OF THE ESSENTIAL PURPOSE OF ANY
97  * LIMITED REMEDY.
98  *
99  * 4.3. Licensee shall not export, either directly or indirectly, any of this
100  * software or system incorporating such software without first obtaining any
101  * required license or other approval from the U. S. Department of Commerce or
102  * any other agency or department of the United States Government. In the
103  * event Licensee exports any such software from the United States or
104  * re-exports any such software from a foreign destination, Licensee shall
105  * ensure that the distribution and export/re-export of the software is in
106  * compliance with all laws, regulations, orders, or other restrictions of the
107  * U.S. Export Administration Regulations. Licensee agrees that neither it nor
108  * any of its subsidiaries will export/re-export any technical data, process,
109  * software, or service, directly or indirectly, to any country for which the
110  * United States government or any agency thereof requires an export license,
111  * other governmental approval, or letter of assurance, without first obtaining
112  * such license, approval or letter.
113  *
114  *****************************************************************************
115  *
116  * Alternatively, you may choose to be licensed under the terms of the
117  * following license:
118  *
119  * Redistribution and use in source and binary forms, with or without
120  * modification, are permitted provided that the following conditions
121  * are met:
122  * 1. Redistributions of source code must retain the above copyright
123  *    notice, this list of conditions, and the following disclaimer,
124  *    without modification.
125  * 2. Redistributions in binary form must reproduce at minimum a disclaimer
126  *    substantially similar to the "NO WARRANTY" disclaimer below
127  *    ("Disclaimer") and any redistribution must be conditioned upon
128  *    including a substantially similar Disclaimer requirement for further
129  *    binary redistribution.
130  * 3. Neither the names of the above-listed copyright holders nor the names
131  *    of any contributors may be used to endorse or promote products derived
132  *    from this software without specific prior written permission.
133  *
134  * THIS SOFTWARE IS PROVIDED BY THE COPYRIGHT HOLDERS AND CONTRIBUTORS
135  * "AS IS" AND ANY EXPRESS OR IMPLIED WARRANTIES, INCLUDING, BUT NOT
136  * LIMITED TO, THE IMPLIED WARRANTIES OF MERCHANTABILITY AND FITNESS FOR
137  * A PARTICULAR PURPOSE ARE DISCLAIMED. IN NO EVENT SHALL THE COPYRIGHT
138  * OWNER OR CONTRIBUTORS BE LIABLE FOR ANY DIRECT, INDIRECT, INCIDENTAL,
139  * SPECIAL, EXEMPLARY, OR CONSEQUENTIAL DAMAGES (INCLUDING, BUT NOT
140  * LIMITED TO, PROCUREMENT OF SUBSTITUTE GOODS OR SERVICES; LOSS OF USE,
141  * DATA, OR PROFITS; OR BUSINESS INTERRUPTION) HOWEVER CAUSED AND ON ANY
142  * THEORY OF LIABILITY, WHETHER IN CONTRACT, STRICT LIABILITY, OR TORT
143  * (INCLUDING NEGLIGENCE OR OTHERWISE) ARISING IN ANY WAY OUT OF THE USE
144  * OF THIS SOFTWARE, EVEN IF ADVISED OF THE POSSIBILITY OF SUCH DAMAGE.
145  *
146  * Alternatively, you may choose to be licensed under the terms of the
147  * GNU General Public License ("GPL") version 2 as published by the Free
148  * Software Foundation.
149  *
150  *****************************************************************************/
151 
152 #include <contrib/dev/acpica/compiler/aslcompiler.h>
153 #include "aslcompiler.y.h"
154 #include <contrib/dev/acpica/include/acnamesp.h>
155 
156 #define _COMPONENT          ACPI_COMPILER
157         ACPI_MODULE_NAME    ("asltransform")
158 
159 /* Local prototypes */
160 
161 static void
162 TrTransformSubtree (
163     ACPI_PARSE_OBJECT       *Op);
164 
165 static char *
166 TrAmlGetNextTempName (
167     ACPI_PARSE_OBJECT       *Op,
168     UINT8                   *TempCount);
169 
170 static void
171 TrAmlInitLineNumbers (
172     ACPI_PARSE_OBJECT       *Op,
173     ACPI_PARSE_OBJECT       *Neighbor);
174 
175 static void
176 TrAmlInitNode (
177     ACPI_PARSE_OBJECT       *Op,
178     UINT16                  ParseOpcode);
179 
180 static void
181 TrAmlSetSubtreeParent (
182     ACPI_PARSE_OBJECT       *Op,
183     ACPI_PARSE_OBJECT       *Parent);
184 
185 static void
186 TrAmlInsertPeer (
187     ACPI_PARSE_OBJECT       *Op,
188     ACPI_PARSE_OBJECT       *NewPeer);
189 
190 static void
191 TrDoDefinitionBlock (
192     ACPI_PARSE_OBJECT       *Op);
193 
194 static void
195 TrDoSwitch (
196     ACPI_PARSE_OBJECT       *StartNode);
197 
198 static void
199 TrCheckForDuplicateCase (
200     ACPI_PARSE_OBJECT       *CaseOp,
201     ACPI_PARSE_OBJECT       *Predicate1);
202 
203 static BOOLEAN
204 TrCheckForBufferMatch (
205     ACPI_PARSE_OBJECT       *Next1,
206     ACPI_PARSE_OBJECT       *Next2);
207 
208 static void
209 TrDoMethod (
210     ACPI_PARSE_OBJECT       *Op);
211 
212 
213 /*******************************************************************************
214  *
215  * FUNCTION:    TrAmlGetNextTempName
216  *
217  * PARAMETERS:  Op              - Current parse op
218  *              TempCount       - Current temporary counter. Was originally
219  *                                per-module; Currently per method, could be
220  *                                expanded to per-scope.
221  *
222  * RETURN:      A pointer to name (allocated here).
223  *
224  * DESCRIPTION: Generate an ACPI name of the form _T_x. These names are
225  *              reserved for use by the ASL compiler. (_T_0 through _T_Z)
226  *
227  ******************************************************************************/
228 
229 static char *
230 TrAmlGetNextTempName (
231     ACPI_PARSE_OBJECT       *Op,
232     UINT8                   *TempCount)
233 {
234     char                    *TempName;
235 
236 
237     if (*TempCount >= (10 + 26))  /* 0-35 valid: 0-9 and A-Z for TempName[3] */
238     {
239         /* Too many temps */
240 
241         AslError (ASL_ERROR, ASL_MSG_TOO_MANY_TEMPS, Op, NULL);
242         return (NULL);
243     }
244 
245     TempName = UtLocalCalloc (5);
246 
247     if (*TempCount < 10)    /* 0-9 */
248     {
249         TempName[3] = (char) (*TempCount + '0');
250     }
251     else                    /* 10-35: A-Z */
252     {
253         TempName[3] = (char) (*TempCount + ('A' - 10));
254     }
255 
256     (*TempCount)++;
257 
258     /* First three characters are always "_T_" */
259 
260     TempName[0] = '_';
261     TempName[1] = 'T';
262     TempName[2] = '_';
263 
264     return (TempName);
265 }
266 
267 
268 /*******************************************************************************
269  *
270  * FUNCTION:    TrAmlInitLineNumbers
271  *
272  * PARAMETERS:  Op              - Op to be initialized
273  *              Neighbor        - Op used for initialization values
274  *
275  * RETURN:      None
276  *
277  * DESCRIPTION: Initialized the various line numbers for a parse node.
278  *
279  ******************************************************************************/
280 
281 static void
282 TrAmlInitLineNumbers (
283     ACPI_PARSE_OBJECT       *Op,
284     ACPI_PARSE_OBJECT       *Neighbor)
285 {
286 
287     Op->Asl.EndLine           = Neighbor->Asl.EndLine;
288     Op->Asl.EndLogicalLine    = Neighbor->Asl.EndLogicalLine;
289     Op->Asl.LineNumber        = Neighbor->Asl.LineNumber;
290     Op->Asl.LogicalByteOffset = Neighbor->Asl.LogicalByteOffset;
291     Op->Asl.LogicalLineNumber = Neighbor->Asl.LogicalLineNumber;
292 }
293 
294 
295 /*******************************************************************************
296  *
297  * FUNCTION:    TrAmlInitNode
298  *
299  * PARAMETERS:  Op              - Op to be initialized
300  *              ParseOpcode     - Opcode for this node
301  *
302  * RETURN:      None
303  *
304  * DESCRIPTION: Initialize a node with the parse opcode and opcode name.
305  *
306  ******************************************************************************/
307 
308 static void
309 TrAmlInitNode (
310     ACPI_PARSE_OBJECT       *Op,
311     UINT16                  ParseOpcode)
312 {
313 
314     Op->Asl.ParseOpcode = ParseOpcode;
315     UtSetParseOpName (Op);
316 }
317 
318 
319 /*******************************************************************************
320  *
321  * FUNCTION:    TrAmlSetSubtreeParent
322  *
323  * PARAMETERS:  Op              - First node in a list of peer nodes
324  *              Parent          - Parent of the subtree
325  *
326  * RETURN:      None
327  *
328  * DESCRIPTION: Set the parent for all peer nodes in a subtree
329  *
330  ******************************************************************************/
331 
332 static void
333 TrAmlSetSubtreeParent (
334     ACPI_PARSE_OBJECT       *Op,
335     ACPI_PARSE_OBJECT       *Parent)
336 {
337     ACPI_PARSE_OBJECT       *Next;
338 
339 
340     Next = Op;
341     while (Next)
342     {
343         Next->Asl.Parent = Parent;
344         Next = Next->Asl.Next;
345     }
346 }
347 
348 
349 /*******************************************************************************
350  *
351  * FUNCTION:    TrAmlInsertPeer
352  *
353  * PARAMETERS:  Op              - First node in a list of peer nodes
354  *              NewPeer         - Peer node to insert
355  *
356  * RETURN:      None
357  *
358  * DESCRIPTION: Insert a new peer node into a list of peers.
359  *
360  ******************************************************************************/
361 
362 static void
363 TrAmlInsertPeer (
364     ACPI_PARSE_OBJECT       *Op,
365     ACPI_PARSE_OBJECT       *NewPeer)
366 {
367 
368     NewPeer->Asl.Next = Op->Asl.Next;
369     Op->Asl.Next = NewPeer;
370 }
371 
372 
373 /*******************************************************************************
374  *
375  * FUNCTION:    TrAmlTransformWalkBegin
376  *
377  * PARAMETERS:  ASL_WALK_CALLBACK
378  *
379  * RETURN:      None
380  *
381  * DESCRIPTION: Parse tree walk to generate both the AML opcodes and the AML
382  *              operands.
383  *
384  ******************************************************************************/
385 
386 ACPI_STATUS
387 TrAmlTransformWalkBegin (
388     ACPI_PARSE_OBJECT       *Op,
389     UINT32                  Level,
390     void                    *Context)
391 {
392 
393     TrTransformSubtree (Op);
394     return (AE_OK);
395 }
396 
397 
398 /*******************************************************************************
399  *
400  * FUNCTION:    TrAmlTransformWalkEnd
401  *
402  * PARAMETERS:  ASL_WALK_CALLBACK
403  *
404  * RETURN:      None
405  *
406  * DESCRIPTION: Parse tree walk to generate both the AML opcodes and the AML
407  *              operands.
408  *
409  ******************************************************************************/
410 
411 ACPI_STATUS
412 TrAmlTransformWalkEnd (
413     ACPI_PARSE_OBJECT       *Op,
414     UINT32                  Level,
415     void                    *Context)
416 {
417 
418     /* Save possible Externals list in the DefintionBlock Op */
419 
420     if (Op->Asl.ParseOpcode == PARSEOP_DEFINITION_BLOCK)
421     {
422         Op->Asl.Value.Arg = AslGbl_ExternalsListHead;
423         AslGbl_ExternalsListHead = NULL;
424     }
425 
426     return (AE_OK);
427 }
428 
429 
430 /*******************************************************************************
431  *
432  * FUNCTION:    TrTransformSubtree
433  *
434  * PARAMETERS:  Op        - The parent parse node
435  *
436  * RETURN:      None
437  *
438  * DESCRIPTION: Prepare nodes to be output as AML data and operands. The more
439  *              complex AML opcodes require processing of the child nodes
440  *              (arguments/operands).
441  *
442  ******************************************************************************/
443 
444 static void
445 TrTransformSubtree (
446     ACPI_PARSE_OBJECT           *Op)
447 {
448     ACPI_PARSE_OBJECT           *MethodOp;
449     ACPI_NAMESTRING_INFO        Info;
450 
451 
452     if (Op->Asl.AmlOpcode == AML_RAW_DATA_BYTE)
453     {
454         return;
455     }
456 
457     switch (Op->Asl.ParseOpcode)
458     {
459     case PARSEOP_DEFINITION_BLOCK:
460 
461         TrDoDefinitionBlock (Op);
462         break;
463 
464     case PARSEOP_SWITCH:
465 
466         TrDoSwitch (Op);
467         break;
468 
469     case PARSEOP_METHOD:
470 
471         TrDoMethod (Op);
472         break;
473 
474     case PARSEOP_EXTERNAL:
475 
476         ExDoExternal (Op);
477         break;
478 
479     case PARSEOP___METHOD__:
480 
481         /* Transform to a string op containing the parent method name */
482 
483         Op->Asl.ParseOpcode = PARSEOP_STRING_LITERAL;
484         UtSetParseOpName (Op);
485 
486         /* Find the parent control method op */
487 
488         MethodOp = Op;
489         while (MethodOp)
490         {
491             if (MethodOp->Asl.ParseOpcode == PARSEOP_METHOD)
492             {
493                 /* First child contains the method name */
494 
495                 MethodOp = MethodOp->Asl.Child;
496                 Op->Asl.Value.String = MethodOp->Asl.Value.String;
497                 return;
498             }
499 
500             MethodOp = MethodOp->Asl.Parent;
501         }
502 
503         /* At the root, invocation not within a control method */
504 
505         Op->Asl.Value.String = "\\";
506         break;
507 
508     case PARSEOP_NAMESTRING:
509         /*
510          * A NameString can be up to 255 (0xFF) individual NameSegs maximum
511          * (with 254 dot separators) - as per the ACPI specification. Note:
512          * Cannot check for NumSegments == 0 because things like
513          * Scope(\) are legal and OK.
514          */
515         Info.ExternalName = Op->Asl.Value.String;
516         AcpiNsGetInternalNameLength (&Info);
517 
518         if (Info.NumSegments > 255)
519         {
520             AslError (ASL_ERROR, ASL_MSG_NAMESTRING_LENGTH, Op, NULL);
521         }
522         break;
523 
524     case PARSEOP_UNLOAD:
525 
526         AslError (ASL_WARNING, ASL_MSG_UNLOAD, Op, NULL);
527         break;
528 
529     case PARSEOP_SLEEP:
530 
531         /* Remark for very long sleep values */
532 
533         if (Op->Asl.Child->Asl.Value.Integer > 1000)
534         {
535             AslError (ASL_REMARK, ASL_MSG_LONG_SLEEP, Op, NULL);
536         }
537         break;
538 
539     case PARSEOP_PROCESSOR:
540 
541         AslError (ASL_WARNING, ASL_MSG_LEGACY_PROCESSOR_OP, Op, Op->Asl.ExternalName);
542         break;
543 
544     default:
545 
546         /* Nothing to do here for other opcodes */
547 
548         break;
549     }
550 }
551 
552 
553 /*******************************************************************************
554  *
555  * FUNCTION:    TrDoDefinitionBlock
556  *
557  * PARAMETERS:  Op        - Parse node
558  *
559  * RETURN:      None
560  *
561  * DESCRIPTION: Find the end of the definition block and set a global to this
562  *              node. It is used by the compiler to insert compiler-generated
563  *              names at the root level of the namespace.
564  *
565  ******************************************************************************/
566 
567 static void
568 TrDoDefinitionBlock (
569     ACPI_PARSE_OBJECT       *Op)
570 {
571     ACPI_PARSE_OBJECT       *Next;
572     UINT32                  i;
573 
574 
575     /* Reset external list when starting a definition block */
576 
577     AslGbl_ExternalsListHead = NULL;
578 
579     Next = Op->Asl.Child;
580     for (i = 0; i < 5; i++)
581     {
582         Next = Next->Asl.Next;
583         if (i == 0)
584         {
585             /*
586              * This is the table signature. Only the DSDT can be assumed
587              * to be at the root of the namespace;  Therefore, namepath
588              * optimization can only be performed on the DSDT.
589              */
590             if (!ACPI_COMPARE_NAMESEG (Next->Asl.Value.String, ACPI_SIG_DSDT))
591             {
592                 AslGbl_ReferenceOptimizationFlag = FALSE;
593             }
594         }
595     }
596 
597     AslGbl_FirstLevelInsertionNode = Next;
598 }
599 
600 
601 /*******************************************************************************
602  *
603  * FUNCTION:    TrDoSwitch
604  *
605  * PARAMETERS:  StartNode        - Parse node for SWITCH
606  *
607  * RETURN:      None
608  *
609  * DESCRIPTION: Translate ASL SWITCH statement to if/else pairs. There is
610  *              no actual AML opcode for SWITCH -- it must be simulated.
611  *
612  ******************************************************************************/
613 
614 static void
615 TrDoSwitch (
616     ACPI_PARSE_OBJECT       *StartNode)
617 {
618     ACPI_PARSE_OBJECT       *Next;
619     ACPI_PARSE_OBJECT       *CaseOp = NULL;
620     ACPI_PARSE_OBJECT       *CaseBlock = NULL;
621     ACPI_PARSE_OBJECT       *DefaultOp = NULL;
622     ACPI_PARSE_OBJECT       *CurrentParentNode;
623     ACPI_PARSE_OBJECT       *Conditional = NULL;
624     ACPI_PARSE_OBJECT       *Predicate;
625     ACPI_PARSE_OBJECT       *Peer;
626     ACPI_PARSE_OBJECT       *NewOp;
627     ACPI_PARSE_OBJECT       *NewOp2;
628     ACPI_PARSE_OBJECT       *MethodOp;
629     ACPI_PARSE_OBJECT       *StoreOp;
630     ACPI_PARSE_OBJECT       *BreakOp;
631     ACPI_PARSE_OBJECT       *BufferOp;
632     char                    *PredicateValueName;
633     UINT16                  Index;
634     UINT32                  Btype;
635 
636 
637     /* Start node is the Switch() node */
638 
639     CurrentParentNode  = StartNode;
640 
641     /* Create a new temp name of the form _T_x */
642 
643     PredicateValueName = TrAmlGetNextTempName (StartNode, &AslGbl_TempCount);
644     if (!PredicateValueName)
645     {
646         return;
647     }
648 
649     /* First child is the Switch() predicate */
650 
651     Next = StartNode->Asl.Child;
652 
653     /*
654      * Examine the return type of the Switch Value -
655      * must be Integer/Buffer/String
656      */
657     Index = (UINT16) (Next->Asl.ParseOpcode - ASL_PARSE_OPCODE_BASE);
658     Btype = AslKeywordMapping[Index].AcpiBtype;
659     if ((Btype != ACPI_BTYPE_INTEGER) &&
660         (Btype != ACPI_BTYPE_STRING)  &&
661         (Btype != ACPI_BTYPE_BUFFER))
662     {
663         AslError (ASL_WARNING, ASL_MSG_SWITCH_TYPE, Next, NULL);
664         Btype = ACPI_BTYPE_INTEGER;
665     }
666 
667     /* CASE statements start at next child */
668 
669     Peer = Next->Asl.Next;
670     while (Peer)
671     {
672         Next = Peer;
673         Peer = Next->Asl.Next;
674 
675         if (Next->Asl.ParseOpcode == PARSEOP_CASE)
676         {
677             TrCheckForDuplicateCase (Next, Next->Asl.Child);
678 
679             if (CaseOp)
680             {
681                 /* Add an ELSE to complete the previous CASE */
682 
683                 NewOp = TrCreateLeafOp (PARSEOP_ELSE);
684                 NewOp->Asl.Parent = Conditional->Asl.Parent;
685                 TrAmlInitLineNumbers (NewOp, NewOp->Asl.Parent);
686 
687                 /* Link ELSE node as a peer to the previous IF */
688 
689                 TrAmlInsertPeer (Conditional, NewOp);
690                 CurrentParentNode = NewOp;
691             }
692 
693             CaseOp = Next;
694             Conditional = CaseOp;
695             CaseBlock = CaseOp->Asl.Child->Asl.Next;
696             Conditional->Asl.Child->Asl.Next = NULL;
697             Predicate = CaseOp->Asl.Child;
698 
699             if ((Predicate->Asl.ParseOpcode == PARSEOP_PACKAGE) ||
700                 (Predicate->Asl.ParseOpcode == PARSEOP_VAR_PACKAGE))
701             {
702                 /*
703                  * Convert the package declaration to this form:
704                  *
705                  * If (LNotEqual (Match (Package(<size>){<data>},
706                  *                       MEQ, _T_x, MTR, Zero, Zero), Ones))
707                  */
708                 NewOp2              = TrCreateLeafOp (PARSEOP_MATCHTYPE_MEQ);
709                 Predicate->Asl.Next = NewOp2;
710                 TrAmlInitLineNumbers (NewOp2, Conditional);
711 
712                 NewOp               = NewOp2;
713                 NewOp2              = TrCreateValuedLeafOp (PARSEOP_NAMESTRING,
714                                         (UINT64) ACPI_TO_INTEGER (PredicateValueName));
715                 NewOp->Asl.Next     = NewOp2;
716                 TrAmlInitLineNumbers (NewOp2, Predicate);
717 
718                 NewOp               = NewOp2;
719                 NewOp2              = TrCreateLeafOp (PARSEOP_MATCHTYPE_MTR);
720                 NewOp->Asl.Next     = NewOp2;
721                 TrAmlInitLineNumbers (NewOp2, Predicate);
722 
723                 NewOp               = NewOp2;
724                 NewOp2              = TrCreateLeafOp (PARSEOP_ZERO);
725                 NewOp->Asl.Next     = NewOp2;
726                 TrAmlInitLineNumbers (NewOp2, Predicate);
727 
728                 NewOp               = NewOp2;
729                 NewOp2              = TrCreateLeafOp (PARSEOP_ZERO);
730                 NewOp->Asl.Next     = NewOp2;
731                 TrAmlInitLineNumbers (NewOp2, Predicate);
732 
733                 NewOp2              = TrCreateLeafOp (PARSEOP_MATCH);
734                 NewOp2->Asl.Child   = Predicate;  /* PARSEOP_PACKAGE */
735                 TrAmlInitLineNumbers (NewOp2, Conditional);
736                 TrAmlSetSubtreeParent (Predicate, NewOp2);
737 
738                 NewOp               = NewOp2;
739                 NewOp2              = TrCreateLeafOp (PARSEOP_ONES);
740                 NewOp->Asl.Next     = NewOp2;
741                 TrAmlInitLineNumbers (NewOp2, Conditional);
742 
743                 NewOp2              = TrCreateLeafOp (PARSEOP_LEQUAL);
744                 NewOp2->Asl.Child   = NewOp;
745                 NewOp->Asl.Parent   = NewOp2;
746                 TrAmlInitLineNumbers (NewOp2, Conditional);
747                 TrAmlSetSubtreeParent (NewOp, NewOp2);
748 
749                 NewOp               = NewOp2;
750                 NewOp2              = TrCreateLeafOp (PARSEOP_LNOT);
751                 NewOp2->Asl.Child   = NewOp;
752                 NewOp2->Asl.Parent  = Conditional;
753                 NewOp->Asl.Parent   = NewOp2;
754                 TrAmlInitLineNumbers (NewOp2, Conditional);
755 
756                 Conditional->Asl.Child = NewOp2;
757                 NewOp2->Asl.Next = CaseBlock;
758             }
759             else
760             {
761                 /*
762                  * Integer and Buffer case.
763                  *
764                  * Change CaseOp() to:  If (LEqual (SwitchValue, CaseValue)) {...}
765                  * Note: SwitchValue is first to allow the CaseValue to be implicitly
766                  * converted to the type of SwitchValue if necessary.
767                  *
768                  * CaseOp->Child is the case value
769                  * CaseOp->Child->Peer is the beginning of the case block
770                  */
771                 NewOp = TrCreateValuedLeafOp (PARSEOP_NAMESTRING,
772                     (UINT64) ACPI_TO_INTEGER (PredicateValueName));
773                 NewOp->Asl.Next = Predicate;
774                 TrAmlInitLineNumbers (NewOp, Predicate);
775 
776                 NewOp2              = TrCreateLeafOp (PARSEOP_LEQUAL);
777                 NewOp2->Asl.Parent  = Conditional;
778                 NewOp2->Asl.Child   = NewOp;
779                 TrAmlInitLineNumbers (NewOp2, Conditional);
780 
781                 TrAmlSetSubtreeParent (NewOp, NewOp2);
782 
783                 Predicate           = NewOp2;
784                 Predicate->Asl.Next = CaseBlock;
785 
786                 TrAmlSetSubtreeParent (Predicate, Conditional);
787                 Conditional->Asl.Child = Predicate;
788             }
789 
790             /* Reinitialize the CASE node to an IF node */
791 
792             TrAmlInitNode (Conditional, PARSEOP_IF);
793 
794             /*
795              * The first CASE(IF) is not nested under an ELSE.
796              * All other CASEs are children of a parent ELSE.
797              */
798             if (CurrentParentNode == StartNode)
799             {
800                 Conditional->Asl.Next = NULL;
801             }
802             else
803             {
804                 /*
805                  * The IF is a child of previous IF/ELSE. It
806                  * is therefore without peer.
807                  */
808                 CurrentParentNode->Asl.Child = Conditional;
809                 Conditional->Asl.Parent      = CurrentParentNode;
810                 Conditional->Asl.Next        = NULL;
811             }
812         }
813         else if (Next->Asl.ParseOpcode == PARSEOP_DEFAULT)
814         {
815             if (DefaultOp)
816             {
817                 /*
818                  * More than one Default
819                  * (Parser does not catch this, must check here)
820                  */
821                 AslError (ASL_ERROR, ASL_MSG_MULTIPLE_DEFAULT, Next, NULL);
822             }
823             else
824             {
825                 /* Save the DEFAULT node for later, after CASEs */
826 
827                 DefaultOp = Next;
828             }
829         }
830         else
831         {
832             /* Unknown peer opcode */
833 
834             AcpiOsPrintf ("Unknown parse opcode for switch statement: %s (%u)\n",
835                 Next->Asl.ParseOpName, Next->Asl.ParseOpcode);
836         }
837     }
838 
839     /* Add the default case at the end of the if/else construct */
840 
841     if (DefaultOp)
842     {
843         /* If no CASE statements, this is an error - see below */
844 
845         if (CaseOp)
846         {
847             /* Convert the DEFAULT node to an ELSE */
848 
849             TrAmlInitNode (DefaultOp, PARSEOP_ELSE);
850             DefaultOp->Asl.Parent = Conditional->Asl.Parent;
851 
852             /* Link ELSE node as a peer to the previous IF */
853 
854             TrAmlInsertPeer (Conditional, DefaultOp);
855         }
856     }
857 
858     if (!CaseOp)
859     {
860         AslError (ASL_ERROR, ASL_MSG_NO_CASES, StartNode, NULL);
861     }
862 
863 
864     /*
865      * Create a Name(_T_x, ...) statement. This statement must appear at the
866      * method level, in case a loop surrounds the switch statement and could
867      * cause the name to be created twice (error).
868      */
869 
870     /* Create the Name node */
871 
872     Predicate = StartNode->Asl.Child;
873     NewOp = TrCreateLeafOp (PARSEOP_NAME);
874     TrAmlInitLineNumbers (NewOp, StartNode);
875 
876     /* Find the parent method */
877 
878     Next = StartNode;
879     while ((Next->Asl.ParseOpcode != PARSEOP_METHOD) &&
880            (Next->Asl.ParseOpcode != PARSEOP_DEFINITION_BLOCK))
881     {
882         Next = Next->Asl.Parent;
883     }
884     MethodOp = Next;
885 
886     NewOp->Asl.CompileFlags |= OP_COMPILER_EMITTED;
887     NewOp->Asl.Parent = Next;
888 
889     /* Insert name after the method name and arguments */
890 
891     Next = Next->Asl.Child; /* Name */
892     Next = Next->Asl.Next;  /* NumArgs */
893     Next = Next->Asl.Next;  /* SerializeRule */
894 
895     /*
896      * If method is not Serialized, we must make is so, because of the way
897      * that Switch() must be implemented -- we cannot allow multiple threads
898      * to execute this method concurrently since we need to create local
899      * temporary name(s).
900      */
901     if (Next->Asl.ParseOpcode != PARSEOP_SERIALIZERULE_SERIAL)
902     {
903         AslError (ASL_REMARK, ASL_MSG_SERIALIZED, MethodOp,
904             "Due to use of Switch operator");
905         Next->Asl.ParseOpcode = PARSEOP_SERIALIZERULE_SERIAL;
906     }
907 
908     Next = Next->Asl.Next;  /* SyncLevel */
909     Next = Next->Asl.Next;  /* ReturnType */
910     Next = Next->Asl.Next;  /* ParameterTypes */
911 
912     TrAmlInsertPeer (Next, NewOp);
913     TrAmlInitLineNumbers (NewOp, Next);
914 
915     /* Create the NameSeg child for the Name node */
916 
917     NewOp2 = TrCreateValuedLeafOp (PARSEOP_NAMESEG,
918         (UINT64) ACPI_TO_INTEGER (PredicateValueName));
919     TrAmlInitLineNumbers (NewOp2, NewOp);
920     NewOp2->Asl.CompileFlags |= OP_IS_NAME_DECLARATION;
921     NewOp->Asl.Child  = NewOp2;
922 
923     /* Create the initial value for the Name. Btype was already validated above */
924 
925     switch (Btype)
926     {
927     case ACPI_BTYPE_INTEGER:
928 
929         NewOp2->Asl.Next = TrCreateValuedLeafOp (PARSEOP_ZERO,
930             (UINT64) 0);
931         TrAmlInitLineNumbers (NewOp2->Asl.Next, NewOp);
932         break;
933 
934     case ACPI_BTYPE_STRING:
935 
936         NewOp2->Asl.Next = TrCreateValuedLeafOp (PARSEOP_STRING_LITERAL,
937             (UINT64) ACPI_TO_INTEGER (""));
938         TrAmlInitLineNumbers (NewOp2->Asl.Next, NewOp);
939         break;
940 
941     case ACPI_BTYPE_BUFFER:
942 
943         (void) TrLinkPeerOp (NewOp2, TrCreateValuedLeafOp (PARSEOP_BUFFER,
944             (UINT64) 0));
945         Next = NewOp2->Asl.Next;
946         TrAmlInitLineNumbers (Next, NewOp2);
947 
948         (void) TrLinkOpChildren (Next, 1, TrCreateValuedLeafOp (PARSEOP_ZERO,
949             (UINT64) 1));
950         TrAmlInitLineNumbers (Next->Asl.Child, Next);
951 
952         BufferOp = TrCreateValuedLeafOp (PARSEOP_DEFAULT_ARG, (UINT64) 0);
953         TrAmlInitLineNumbers (BufferOp, Next->Asl.Child);
954         (void) TrLinkPeerOp (Next->Asl.Child, BufferOp);
955 
956         TrAmlSetSubtreeParent (Next->Asl.Child, Next);
957         break;
958 
959     default:
960 
961         break;
962     }
963 
964     TrAmlSetSubtreeParent (NewOp2, NewOp);
965 
966     /*
967      * Transform the Switch() into a While(One)-Break node.
968      * And create a Store() node which will be used to save the
969      * Switch() value. The store is of the form: Store (Value, _T_x)
970      * where _T_x is the temp variable.
971      */
972     TrAmlInitNode (StartNode, PARSEOP_WHILE);
973     NewOp = TrCreateLeafOp (PARSEOP_ONE);
974     TrAmlInitLineNumbers (NewOp, StartNode);
975     NewOp->Asl.Next = Predicate->Asl.Next;
976     NewOp->Asl.Parent = StartNode;
977     StartNode->Asl.Child = NewOp;
978 
979     /* Create a Store() node */
980 
981     StoreOp = TrCreateLeafOp (PARSEOP_STORE);
982     TrAmlInitLineNumbers (StoreOp, NewOp);
983     StoreOp->Asl.Parent = StartNode;
984     TrAmlInsertPeer (NewOp, StoreOp);
985 
986     /* Complete the Store subtree */
987 
988     StoreOp->Asl.Child = Predicate;
989     Predicate->Asl.Parent = StoreOp;
990 
991     NewOp = TrCreateValuedLeafOp (PARSEOP_NAMESEG,
992         (UINT64) ACPI_TO_INTEGER (PredicateValueName));
993     TrAmlInitLineNumbers (NewOp, StoreOp);
994     NewOp->Asl.Parent    = StoreOp;
995     Predicate->Asl.Next  = NewOp;
996 
997     /* Create a Break() node and insert it into the end of While() */
998 
999     Conditional = StartNode->Asl.Child;
1000     while (Conditional->Asl.Next)
1001     {
1002         Conditional = Conditional->Asl.Next;
1003     }
1004 
1005     BreakOp = TrCreateLeafOp (PARSEOP_BREAK);
1006     TrAmlInitLineNumbers (BreakOp, NewOp);
1007     BreakOp->Asl.Parent = StartNode;
1008     TrAmlInsertPeer (Conditional, BreakOp);
1009 }
1010 
1011 
1012 /*******************************************************************************
1013  *
1014  * FUNCTION:    TrCheckForDuplicateCase
1015  *
1016  * PARAMETERS:  CaseOp          - Parse node for first Case statement in list
1017  *              Predicate1      - Case value for the input CaseOp
1018  *
1019  * RETURN:      None
1020  *
1021  * DESCRIPTION: Check for duplicate case values. Currently, only handles
1022  *              Integers, Strings and Buffers. No support for Package objects.
1023  *
1024  ******************************************************************************/
1025 
1026 static void
1027 TrCheckForDuplicateCase (
1028     ACPI_PARSE_OBJECT       *CaseOp,
1029     ACPI_PARSE_OBJECT       *Predicate1)
1030 {
1031     ACPI_PARSE_OBJECT       *Next;
1032     ACPI_PARSE_OBJECT       *Predicate2;
1033 
1034 
1035     /* Walk the list of CASE opcodes */
1036 
1037     Next = CaseOp->Asl.Next;
1038     while (Next)
1039     {
1040         if (Next->Asl.ParseOpcode == PARSEOP_CASE)
1041         {
1042             /* Emit error only once */
1043 
1044             if (Next->Asl.CompileFlags & OP_IS_DUPLICATE)
1045             {
1046                 goto NextCase;
1047             }
1048 
1049             /* Check for a duplicate plain integer */
1050 
1051             Predicate2 = Next->Asl.Child;
1052             if ((Predicate1->Asl.ParseOpcode == PARSEOP_INTEGER) &&
1053                 (Predicate2->Asl.ParseOpcode == PARSEOP_INTEGER))
1054             {
1055                 if (Predicate1->Asl.Value.Integer == Predicate2->Asl.Value.Integer)
1056                 {
1057                     goto FoundDuplicate;
1058                 }
1059             }
1060 
1061             /* Check for pairs of the constants ZERO, ONE, ONES */
1062 
1063             else if (((Predicate1->Asl.ParseOpcode == PARSEOP_ZERO) &&
1064                 (Predicate2->Asl.ParseOpcode == PARSEOP_ZERO)) ||
1065                 ((Predicate1->Asl.ParseOpcode == PARSEOP_ONE) &&
1066                 (Predicate2->Asl.ParseOpcode == PARSEOP_ONE)) ||
1067                 ((Predicate1->Asl.ParseOpcode == PARSEOP_ONES) &&
1068                 (Predicate2->Asl.ParseOpcode == PARSEOP_ONES)))
1069             {
1070                 goto FoundDuplicate;
1071             }
1072 
1073             /* Check for a duplicate string constant (literal) */
1074 
1075             else if ((Predicate1->Asl.ParseOpcode == PARSEOP_STRING_LITERAL) &&
1076                 (Predicate2->Asl.ParseOpcode == PARSEOP_STRING_LITERAL))
1077             {
1078                 if (!strcmp (Predicate1->Asl.Value.String,
1079                         Predicate2->Asl.Value.String))
1080                 {
1081                     goto FoundDuplicate;
1082                 }
1083             }
1084 
1085             /* Check for a duplicate buffer constant */
1086 
1087             else if ((Predicate1->Asl.ParseOpcode == PARSEOP_BUFFER) &&
1088                 (Predicate2->Asl.ParseOpcode == PARSEOP_BUFFER))
1089             {
1090                 if (TrCheckForBufferMatch (Predicate1->Asl.Child,
1091                         Predicate2->Asl.Child))
1092                 {
1093                     goto FoundDuplicate;
1094                 }
1095             }
1096         }
1097         goto NextCase;
1098 
1099 FoundDuplicate:
1100         /* Emit error message only once */
1101 
1102         Next->Asl.CompileFlags |= OP_IS_DUPLICATE;
1103 
1104         AslDualParseOpError (ASL_ERROR, ASL_MSG_DUPLICATE_CASE, Next,
1105             Next->Asl.Value.String, ASL_MSG_CASE_FOUND_HERE, CaseOp,
1106             CaseOp->Asl.ExternalName);
1107 
1108 NextCase:
1109         Next = Next->Asl.Next;
1110     }
1111 }
1112 
1113 /*******************************************************************************
1114  *
1115  * FUNCTION:    TrBufferIsAllZero
1116  *
1117  * PARAMETERS:  Op          - Parse node for first opcode in buffer initializer
1118  *                            list
1119  *
1120  * RETURN:      TRUE if buffer contains all zeros or a DEFAULT_ARG
1121  *
1122  * DESCRIPTION: Check for duplicate Buffer case values.
1123  *
1124  ******************************************************************************/
1125 
1126 static BOOLEAN
1127 TrBufferIsAllZero (
1128     ACPI_PARSE_OBJECT       *Op)
1129 {
1130     while (Op)
1131     {
1132         if (Op->Asl.ParseOpcode == PARSEOP_DEFAULT_ARG)
1133         {
1134             return (TRUE);
1135         }
1136         else if (Op->Asl.Value.Integer != 0)
1137         {
1138             return (FALSE);
1139         }
1140 
1141         Op = Op->Asl.Next;
1142     }
1143 
1144     return (TRUE);
1145 }
1146 
1147 
1148 /*******************************************************************************
1149  *
1150  * FUNCTION:    TrCheckForBufferMatch
1151  *
1152  * PARAMETERS:  Next1       - Parse node for first opcode in first buffer list
1153  *                              (The DEFAULT_ARG or INTEGER node)
1154  *              Next2       - Parse node for first opcode in second buffer list
1155  *                              (The DEFAULT_ARG or INTEGER node)
1156  *
1157  * RETURN:      TRUE if buffers match, FALSE otherwise
1158  *
1159  * DESCRIPTION: Check for duplicate Buffer case values.
1160  *
1161  ******************************************************************************/
1162 
1163 static BOOLEAN
1164 TrCheckForBufferMatch (
1165     ACPI_PARSE_OBJECT       *NextOp1,
1166     ACPI_PARSE_OBJECT       *NextOp2)
1167 {
1168     /*
1169      * The buffer length can be a DEFAULT_ARG or INTEGER. If any of the nodes
1170      * are DEFAULT_ARG, it means that the length has yet to be computed.
1171      * However, the initializer list can be compared to determine if these two
1172      * buffers match.
1173      */
1174     if ((NextOp1->Asl.ParseOpcode == PARSEOP_INTEGER &&
1175         NextOp2->Asl.ParseOpcode == PARSEOP_INTEGER) &&
1176         NextOp1->Asl.Value.Integer != NextOp2->Asl.Value.Integer)
1177     {
1178         return (FALSE);
1179     }
1180 
1181     /*
1182      * Buffers that have explicit lengths but no initializer lists are
1183      * filled with zeros at runtime. This is equivalent to buffers that have the
1184      * same length that are filled with zeros.
1185      *
1186      * In other words, the following buffers are equivalent:
1187      *
1188      * Buffer(0x4) {}
1189      * Buffer() {0x0, 0x0, 0x0, 0x0}
1190      *
1191      * This statement checks for matches where one buffer does not have an
1192      * initializer list and another buffer contains all zeros.
1193      */
1194     if (NextOp1->Asl.ParseOpcode != NextOp2->Asl.ParseOpcode &&
1195         TrBufferIsAllZero (NextOp1->Asl.Next) &&
1196         TrBufferIsAllZero (NextOp2->Asl.Next))
1197     {
1198         return (TRUE);
1199     }
1200 
1201     /* Start at the BYTECONST initializer node list */
1202 
1203     NextOp1 = NextOp1->Asl.Next;
1204     NextOp2 = NextOp2->Asl.Next;
1205 
1206     /*
1207      * Walk both lists until either a mismatch is found, or one or more
1208      * end-of-lists are found
1209      */
1210     while (NextOp1 && NextOp2)
1211     {
1212         if ((NextOp1->Asl.ParseOpcode == PARSEOP_STRING_LITERAL) &&
1213             (NextOp2->Asl.ParseOpcode == PARSEOP_STRING_LITERAL))
1214         {
1215             if (!strcmp (NextOp1->Asl.Value.String, NextOp2->Asl.Value.String))
1216             {
1217                 return (TRUE);
1218             }
1219             else
1220             {
1221                 return (FALSE);
1222             }
1223         }
1224         if ((UINT8) NextOp1->Asl.Value.Integer != (UINT8) NextOp2->Asl.Value.Integer)
1225         {
1226             return (FALSE);
1227         }
1228 
1229         NextOp1 = NextOp1->Asl.Next;
1230         NextOp2 = NextOp2->Asl.Next;
1231     }
1232 
1233     /* Not a match if one of the lists is not at end-of-list */
1234 
1235     if (NextOp1 || NextOp2)
1236     {
1237         return (FALSE);
1238     }
1239 
1240     /* Otherwise, the buffers match */
1241 
1242     return (TRUE);
1243 }
1244 
1245 
1246 /*******************************************************************************
1247  *
1248  * FUNCTION:    TrDoMethod
1249  *
1250  * PARAMETERS:  Op               - Parse node for SWITCH
1251  *
1252  * RETURN:      None
1253  *
1254  * DESCRIPTION: Determine that parameter count of an ASL method node by
1255  *              translating the parameter count parse node from
1256  *              PARSEOP_DEFAULT_ARG to PARSEOP_BYTECONST.
1257  *
1258  ******************************************************************************/
1259 
1260 static void
1261 TrDoMethod (
1262     ACPI_PARSE_OBJECT       *Op)
1263 {
1264     ACPI_PARSE_OBJECT           *ArgCountOp;
1265     UINT8                       ArgCount;
1266     ACPI_PARSE_OBJECT           *ParameterOp;
1267 
1268 
1269     /*
1270      * TBD: Zero the tempname (_T_x) count. Probably shouldn't be a global,
1271      * however
1272      */
1273     AslGbl_TempCount = 0;
1274 
1275     ArgCountOp = Op->Asl.Child->Asl.Next;
1276     if (ArgCountOp->Asl.ParseOpcode == PARSEOP_BYTECONST)
1277     {
1278         /*
1279          * Parameter count for this method has already been recorded in the
1280          * method declaration.
1281          */
1282         return;
1283     }
1284 
1285     /*
1286      * Parameter count has been omitted in the method declaration.
1287      * Count the amount of arguments here.
1288      */
1289     ParameterOp = ArgCountOp->Asl.Next->Asl.Next->Asl.Next->Asl.Next;
1290     if (ParameterOp->Asl.ParseOpcode == PARSEOP_DEFAULT_ARG)
1291     {
1292         ArgCount = 0;
1293         ParameterOp = ParameterOp->Asl.Child;
1294 
1295         while (ParameterOp)
1296         {
1297             ParameterOp = ParameterOp->Asl.Next;
1298             ArgCount++;
1299         }
1300 
1301         ArgCountOp->Asl.Value.Integer = ArgCount;
1302         ArgCountOp->Asl.ParseOpcode = PARSEOP_BYTECONST;
1303     }
1304     else
1305     {
1306         /*
1307          * Method parameters can be counted by analyzing the Parameter type
1308          * list. If the Parameter list contains more than 1 parameter, it
1309          * is nested under PARSEOP_DEFAULT_ARG. When there is only 1
1310          * parameter, the parse tree contains a single node representing
1311          * that type.
1312          */
1313         ArgCountOp->Asl.Value.Integer = 1;
1314         ArgCountOp->Asl.ParseOpcode = PARSEOP_BYTECONST;
1315     }
1316 }
1317