xref: /freebsd/sys/contrib/dev/acpica/compiler/aslprepkg.c (revision ce3adf4362fcca6a43e500b2531f0038adbfbd21)
1 /******************************************************************************
2  *
3  * Module Name: aslprepkg - support for ACPI predefined name package objects
4  *
5  *****************************************************************************/
6 
7 /*
8  * Copyright (C) 2000 - 2013, Intel Corp.
9  * All rights reserved.
10  *
11  * Redistribution and use in source and binary forms, with or without
12  * modification, are permitted provided that the following conditions
13  * are met:
14  * 1. Redistributions of source code must retain the above copyright
15  *    notice, this list of conditions, and the following disclaimer,
16  *    without modification.
17  * 2. Redistributions in binary form must reproduce at minimum a disclaimer
18  *    substantially similar to the "NO WARRANTY" disclaimer below
19  *    ("Disclaimer") and any redistribution must be conditioned upon
20  *    including a substantially similar Disclaimer requirement for further
21  *    binary redistribution.
22  * 3. Neither the names of the above-listed copyright holders nor the names
23  *    of any contributors may be used to endorse or promote products derived
24  *    from this software without specific prior written permission.
25  *
26  * Alternatively, this software may be distributed under the terms of the
27  * GNU General Public License ("GPL") version 2 as published by the Free
28  * Software Foundation.
29  *
30  * NO WARRANTY
31  * THIS SOFTWARE IS PROVIDED BY THE COPYRIGHT HOLDERS AND CONTRIBUTORS
32  * "AS IS" AND ANY EXPRESS OR IMPLIED WARRANTIES, INCLUDING, BUT NOT
33  * LIMITED TO, THE IMPLIED WARRANTIES OF MERCHANTIBILITY AND FITNESS FOR
34  * A PARTICULAR PURPOSE ARE DISCLAIMED. IN NO EVENT SHALL THE COPYRIGHT
35  * HOLDERS OR CONTRIBUTORS BE LIABLE FOR SPECIAL, EXEMPLARY, OR CONSEQUENTIAL
36  * DAMAGES (INCLUDING, BUT NOT LIMITED TO, PROCUREMENT OF SUBSTITUTE GOODS
37  * OR SERVICES; LOSS OF USE, DATA, OR PROFITS; OR BUSINESS INTERRUPTION)
38  * HOWEVER CAUSED AND ON ANY THEORY OF LIABILITY, WHETHER IN CONTRACT,
39  * STRICT LIABILITY, OR TORT (INCLUDING NEGLIGENCE OR OTHERWISE) ARISING
40  * IN ANY WAY OUT OF THE USE OF THIS SOFTWARE, EVEN IF ADVISED OF THE
41  * POSSIBILITY OF SUCH DAMAGES.
42  */
43 
44 #include <contrib/dev/acpica/compiler/aslcompiler.h>
45 #include "aslcompiler.y.h"
46 #include <contrib/dev/acpica/include/acpredef.h>
47 
48 
49 #define _COMPONENT          ACPI_COMPILER
50         ACPI_MODULE_NAME    ("aslprepkg")
51 
52 
53 /* Local prototypes */
54 
55 static void
56 ApCheckPackageElements (
57     const char                  *PredefinedName,
58     ACPI_PARSE_OBJECT           *Op,
59     UINT8                       Type1,
60     UINT32                      Count1,
61     UINT8                       Type2,
62     UINT32                      Count2);
63 
64 static void
65 ApCheckPackageList (
66     const char                  *PredefinedName,
67     ACPI_PARSE_OBJECT           *ParentOp,
68     const ACPI_PREDEFINED_INFO  *Package,
69     UINT32                      StartIndex,
70     UINT32                      Count);
71 
72 static void
73 ApPackageTooSmall (
74     const char                  *PredefinedName,
75     ACPI_PARSE_OBJECT           *Op,
76     UINT32                      Count,
77     UINT32                      ExpectedCount);
78 
79 static void
80 ApZeroLengthPackage (
81     const char                  *PredefinedName,
82     ACPI_PARSE_OBJECT           *Op);
83 
84 static void
85 ApPackageTooLarge (
86     const char                  *PredefinedName,
87     ACPI_PARSE_OBJECT           *Op,
88     UINT32                      Count,
89     UINT32                      ExpectedCount);
90 
91 
92 /*******************************************************************************
93  *
94  * FUNCTION:    ApCheckPackage
95  *
96  * PARAMETERS:  ParentOp            - Parser op for the package
97  *              Predefined          - Pointer to package-specific info for
98  *                                    the method
99  *
100  * RETURN:      None
101  *
102  * DESCRIPTION: Top-level validation for predefined name return package
103  *              objects.
104  *
105  ******************************************************************************/
106 
107 void
108 ApCheckPackage (
109     ACPI_PARSE_OBJECT           *ParentOp,
110     const ACPI_PREDEFINED_INFO  *Predefined)
111 {
112     ACPI_PARSE_OBJECT           *Op;
113     const ACPI_PREDEFINED_INFO  *Package;
114     ACPI_STATUS                 Status;
115     UINT32                      ExpectedCount;
116     UINT32                      Count;
117     UINT32                      i;
118 
119 
120     /* The package info for this name is in the next table entry */
121 
122     Package = Predefined + 1;
123 
124     /* First child is the package length */
125 
126     Op = ParentOp->Asl.Child;
127     Count = (UINT32) Op->Asl.Value.Integer;
128 
129     /*
130      * Many of the variable-length top-level packages are allowed to simply
131      * have zero elements. This allows the BIOS to tell the host that even
132      * though the predefined name/method exists, the feature is not supported.
133      * Other package types require one or more elements. In any case, there
134      * is no need to continue validation.
135      */
136     if (!Count)
137     {
138         switch (Package->RetInfo.Type)
139         {
140         case ACPI_PTYPE1_FIXED:
141         case ACPI_PTYPE1_OPTION:
142         case ACPI_PTYPE2_PKG_COUNT:
143         case ACPI_PTYPE2_REV_FIXED:
144 
145             ApZeroLengthPackage (Predefined->Info.Name, ParentOp);
146             break;
147 
148         case ACPI_PTYPE1_VAR:
149         case ACPI_PTYPE2:
150         case ACPI_PTYPE2_COUNT:
151         case ACPI_PTYPE2_FIXED:
152         case ACPI_PTYPE2_MIN:
153         case ACPI_PTYPE2_FIX_VAR:
154         default:
155 
156             break;
157         }
158 
159         return;
160     }
161 
162     /* Get the first element of the package */
163 
164     Op = Op->Asl.Next;
165 
166     /* Decode the package type */
167 
168     switch (Package->RetInfo.Type)
169     {
170     case ACPI_PTYPE1_FIXED:
171         /*
172          * The package count is fixed and there are no sub-packages
173          *
174          * If package is too small, exit.
175          * If package is larger than expected, issue warning but continue
176          */
177         ExpectedCount = Package->RetInfo.Count1 + Package->RetInfo.Count2;
178         if (Count < ExpectedCount)
179         {
180             goto PackageTooSmall;
181         }
182         else if (Count > ExpectedCount)
183         {
184             ApPackageTooLarge (Predefined->Info.Name, ParentOp,
185                 Count, ExpectedCount);
186         }
187 
188         /* Validate all elements of the package */
189 
190         ApCheckPackageElements (Predefined->Info.Name, Op,
191             Package->RetInfo.ObjectType1, Package->RetInfo.Count1,
192             Package->RetInfo.ObjectType2, Package->RetInfo.Count2);
193         break;
194 
195     case ACPI_PTYPE1_VAR:
196         /*
197          * The package count is variable, there are no sub-packages,
198          * and all elements must be of the same type
199          */
200         for (i = 0; i < Count; i++)
201         {
202             ApCheckObjectType (Predefined->Info.Name, Op,
203                 Package->RetInfo.ObjectType1, i);
204             Op = Op->Asl.Next;
205         }
206         break;
207 
208     case ACPI_PTYPE1_OPTION:
209         /*
210          * The package count is variable, there are no sub-packages.
211          * There are a fixed number of required elements, and a variable
212          * number of optional elements.
213          *
214          * Check if package is at least as large as the minimum required
215          */
216         ExpectedCount = Package->RetInfo3.Count;
217         if (Count < ExpectedCount)
218         {
219             goto PackageTooSmall;
220         }
221 
222         /* Variable number of sub-objects */
223 
224         for (i = 0; i < Count; i++)
225         {
226             if (i < Package->RetInfo3.Count)
227             {
228                 /* These are the required package elements (0, 1, or 2) */
229 
230                 ApCheckObjectType (Predefined->Info.Name, Op,
231                     Package->RetInfo3.ObjectType[i], i);
232             }
233             else
234             {
235                 /* These are the optional package elements */
236 
237                 ApCheckObjectType (Predefined->Info.Name, Op,
238                     Package->RetInfo3.TailObjectType, i);
239             }
240             Op = Op->Asl.Next;
241         }
242         break;
243 
244     case ACPI_PTYPE2_REV_FIXED:
245 
246         /* First element is the (Integer) revision */
247 
248         ApCheckObjectType (Predefined->Info.Name, Op,
249             ACPI_RTYPE_INTEGER, 0);
250 
251         Op = Op->Asl.Next;
252         Count--;
253 
254         /* Examine the sub-packages */
255 
256         ApCheckPackageList (Predefined->Info.Name, Op,
257             Package, 1, Count);
258         break;
259 
260     case ACPI_PTYPE2_PKG_COUNT:
261 
262         /* First element is the (Integer) count of sub-packages to follow */
263 
264         Status = ApCheckObjectType (Predefined->Info.Name, Op,
265             ACPI_RTYPE_INTEGER, 0);
266 
267         /* We must have an integer count from above (otherwise, use Count) */
268 
269         if (ACPI_SUCCESS (Status))
270         {
271             /*
272              * Count cannot be larger than the parent package length, but
273              * allow it to be smaller. The >= accounts for the Integer above.
274              */
275             ExpectedCount = (UINT32) Op->Asl.Value.Integer;
276             if (ExpectedCount >= Count)
277             {
278                 goto PackageTooSmall;
279             }
280 
281             Count = ExpectedCount;
282         }
283 
284         Op = Op->Asl.Next;
285 
286         /* Examine the sub-packages */
287 
288         ApCheckPackageList (Predefined->Info.Name, Op,
289             Package, 1, Count);
290         break;
291 
292     case ACPI_PTYPE2:
293     case ACPI_PTYPE2_FIXED:
294     case ACPI_PTYPE2_MIN:
295     case ACPI_PTYPE2_COUNT:
296     case ACPI_PTYPE2_FIX_VAR:
297         /*
298          * These types all return a single Package that consists of a
299          * variable number of sub-Packages.
300          */
301 
302         /* Examine the sub-packages */
303 
304         ApCheckPackageList (Predefined->Info.Name, Op,
305             Package, 0, Count);
306         break;
307 
308     default:
309         return;
310     }
311 
312     return;
313 
314 PackageTooSmall:
315     ApPackageTooSmall (Predefined->Info.Name, ParentOp,
316         Count, ExpectedCount);
317 }
318 
319 
320 /*******************************************************************************
321  *
322  * FUNCTION:    ApCheckPackageElements
323  *
324  * PARAMETERS:  PredefinedName      - Name of the predefined object
325  *              Op                  - Parser op for the package
326  *              Type1               - Object type for first group
327  *              Count1              - Count for first group
328  *              Type2               - Object type for second group
329  *              Count2              - Count for second group
330  *
331  * RETURN:      None
332  *
333  * DESCRIPTION: Validate all elements of a package. Works with packages that
334  *              are defined to contain up to two groups of different object
335  *              types.
336  *
337  ******************************************************************************/
338 
339 static void
340 ApCheckPackageElements (
341     const char              *PredefinedName,
342     ACPI_PARSE_OBJECT       *Op,
343     UINT8                   Type1,
344     UINT32                  Count1,
345     UINT8                   Type2,
346     UINT32                  Count2)
347 {
348     UINT32                  i;
349 
350 
351     /*
352      * Up to two groups of package elements are supported by the data
353      * structure. All elements in each group must be of the same type.
354      * The second group can have a count of zero.
355      *
356      * Aborts check upon a NULL package element, as this means (at compile
357      * time) that the remainder of the package elements are also NULL
358      * (This is the only way to create NULL package elements.)
359      */
360     for (i = 0; (i < Count1) && Op; i++)
361     {
362         ApCheckObjectType (PredefinedName, Op, Type1, i);
363         Op = Op->Asl.Next;
364     }
365 
366     for (i = 0; (i < Count2) && Op; i++)
367     {
368         ApCheckObjectType (PredefinedName, Op, Type2, (i + Count1));
369         Op = Op->Asl.Next;
370     }
371 }
372 
373 
374 /*******************************************************************************
375  *
376  * FUNCTION:    ApCheckPackageList
377  *
378  * PARAMETERS:  PredefinedName      - Name of the predefined object
379  *              ParentOp            - Parser op of the parent package
380  *              Package             - Package info for this predefined name
381  *              StartIndex          - Index in parent package where list begins
382  *              ParentCount         - Element count of parent package
383  *
384  * RETURN:      None
385  *
386  * DESCRIPTION: Validate the individual package elements for a predefined name.
387  *              Handles the cases where the predefined name is defined as a
388  *              Package of Packages (subpackages). These are the types:
389  *
390  *              ACPI_PTYPE2
391  *              ACPI_PTYPE2_FIXED
392  *              ACPI_PTYPE2_MIN
393  *              ACPI_PTYPE2_COUNT
394  *              ACPI_PTYPE2_FIX_VAR
395  *
396  ******************************************************************************/
397 
398 static void
399 ApCheckPackageList (
400     const char                  *PredefinedName,
401     ACPI_PARSE_OBJECT           *ParentOp,
402     const ACPI_PREDEFINED_INFO  *Package,
403     UINT32                      StartIndex,
404     UINT32                      ParentCount)
405 {
406     ACPI_PARSE_OBJECT           *SubPackageOp = ParentOp;
407     ACPI_PARSE_OBJECT           *Op;
408     ACPI_STATUS                 Status;
409     UINT32                      Count;
410     UINT32                      ExpectedCount;
411     UINT32                      i;
412     UINT32                      j;
413 
414 
415     /*
416      * Validate each subpackage in the parent Package
417      *
418      * Note: We ignore NULL package elements on the assumption that
419      * they will be initialized by the BIOS or other ASL code.
420      */
421     for (i = 0; (i < ParentCount) && SubPackageOp; i++)
422     {
423         /* Each object in the list must be of type Package */
424 
425         Status = ApCheckObjectType (PredefinedName, SubPackageOp,
426             ACPI_RTYPE_PACKAGE, i + StartIndex);
427         if (ACPI_FAILURE (Status))
428         {
429             goto NextSubpackage;
430         }
431 
432         /* Examine the different types of expected subpackages */
433 
434         Op = SubPackageOp->Asl.Child;
435 
436         /* First child is the package length */
437 
438         Count = (UINT32) Op->Asl.Value.Integer;
439         Op = Op->Asl.Next;
440 
441         /* The subpackage must have at least one element */
442 
443         if (!Count)
444         {
445             ApZeroLengthPackage (PredefinedName, SubPackageOp);
446             goto NextSubpackage;
447         }
448 
449         /*
450          * Decode the package type.
451          * PTYPE2 indicates that a "package of packages" is expected for
452          * this name. The various flavors of PTYPE2 indicate the number
453          * and format of the subpackages.
454          */
455         switch (Package->RetInfo.Type)
456         {
457         case ACPI_PTYPE2:
458         case ACPI_PTYPE2_PKG_COUNT:
459         case ACPI_PTYPE2_REV_FIXED:
460 
461             /* Each subpackage has a fixed number of elements */
462 
463             ExpectedCount = Package->RetInfo.Count1 + Package->RetInfo.Count2;
464             if (Count < ExpectedCount)
465             {
466                 ApPackageTooSmall (PredefinedName, SubPackageOp,
467                     Count, ExpectedCount);
468                 break;
469             }
470 
471             ApCheckPackageElements (PredefinedName, Op,
472                 Package->RetInfo.ObjectType1, Package->RetInfo.Count1,
473                 Package->RetInfo.ObjectType2, Package->RetInfo.Count2);
474             break;
475 
476         case ACPI_PTYPE2_FIX_VAR:
477             /*
478              * Each subpackage has a fixed number of elements and an
479              * optional element
480              */
481             ExpectedCount = Package->RetInfo.Count1 + Package->RetInfo.Count2;
482             if (Count < ExpectedCount)
483             {
484                 ApPackageTooSmall (PredefinedName, SubPackageOp,
485                     Count, ExpectedCount);
486                 break;
487             }
488 
489             ApCheckPackageElements (PredefinedName, Op,
490                 Package->RetInfo.ObjectType1, Package->RetInfo.Count1,
491                 Package->RetInfo.ObjectType2,
492                 Count - Package->RetInfo.Count1);
493             break;
494 
495         case ACPI_PTYPE2_FIXED:
496 
497             /* Each sub-package has a fixed length */
498 
499             ExpectedCount = Package->RetInfo2.Count;
500             if (Count < ExpectedCount)
501             {
502                 ApPackageTooSmall (PredefinedName, SubPackageOp,
503                     Count, ExpectedCount);
504                 break;
505             }
506 
507             /* Check each object/type combination */
508 
509             for (j = 0; j < ExpectedCount; j++)
510             {
511                 ApCheckObjectType (PredefinedName, Op,
512                     Package->RetInfo2.ObjectType[j], j);
513 
514                 Op = Op->Asl.Next;
515             }
516             break;
517 
518         case ACPI_PTYPE2_MIN:
519 
520             /* Each sub-package has a variable but minimum length */
521 
522             ExpectedCount = Package->RetInfo.Count1;
523             if (Count < ExpectedCount)
524             {
525                 ApPackageTooSmall (PredefinedName, SubPackageOp,
526                     Count, ExpectedCount);
527                 break;
528             }
529 
530             /* Check the type of each sub-package element */
531 
532             ApCheckPackageElements (PredefinedName, Op,
533                 Package->RetInfo.ObjectType1, Count, 0, 0);
534             break;
535 
536         case ACPI_PTYPE2_COUNT:
537             /*
538              * First element is the (Integer) count of elements, including
539              * the count field (the ACPI name is NumElements)
540              */
541             Status = ApCheckObjectType (PredefinedName, Op,
542                 ACPI_RTYPE_INTEGER, 0);
543 
544             /* We must have an integer count from above (otherwise, use Count) */
545 
546             if (ACPI_SUCCESS (Status))
547             {
548                 /*
549                  * Make sure package is large enough for the Count and is
550                  * is as large as the minimum size
551                  */
552                 ExpectedCount = (UINT32) Op->Asl.Value.Integer;
553 
554                 if (Count < ExpectedCount)
555                 {
556                     ApPackageTooSmall (PredefinedName, SubPackageOp,
557                         Count, ExpectedCount);
558                     break;
559                 }
560                 else if (Count > ExpectedCount)
561                 {
562                     ApPackageTooLarge (PredefinedName, SubPackageOp,
563                         Count, ExpectedCount);
564                 }
565 
566                 /* Some names of this type have a minimum length */
567 
568                 if (Count < Package->RetInfo.Count1)
569                 {
570                     ExpectedCount = Package->RetInfo.Count1;
571                     ApPackageTooSmall (PredefinedName, SubPackageOp,
572                         Count, ExpectedCount);
573                     break;
574                 }
575 
576                 Count = ExpectedCount;
577             }
578 
579             /* Check the type of each sub-package element */
580 
581             Op = Op->Asl.Next;
582             ApCheckPackageElements (PredefinedName, Op,
583                 Package->RetInfo.ObjectType1, (Count - 1), 0, 0);
584             break;
585 
586         default:
587             break;
588         }
589 
590 NextSubpackage:
591         SubPackageOp = SubPackageOp->Asl.Next;
592     }
593 }
594 
595 
596 /*******************************************************************************
597  *
598  * FUNCTION:    ApPackageTooSmall
599  *
600  * PARAMETERS:  PredefinedName      - Name of the predefined object
601  *              Op                  - Current parser op
602  *              Count               - Actual package element count
603  *              ExpectedCount       - Expected package element count
604  *
605  * RETURN:      None
606  *
607  * DESCRIPTION: Issue error message for a package that is smaller than
608  *              required.
609  *
610  ******************************************************************************/
611 
612 static void
613 ApPackageTooSmall (
614     const char                  *PredefinedName,
615     ACPI_PARSE_OBJECT           *Op,
616     UINT32                      Count,
617     UINT32                      ExpectedCount)
618 {
619 
620     sprintf (MsgBuffer, "%s: length %u, required minimum is %u",
621         PredefinedName, Count, ExpectedCount);
622 
623     AslError (ASL_ERROR, ASL_MSG_RESERVED_PACKAGE_LENGTH, Op, MsgBuffer);
624 }
625 
626 
627 /*******************************************************************************
628  *
629  * FUNCTION:    ApZeroLengthPackage
630  *
631  * PARAMETERS:  PredefinedName      - Name of the predefined object
632  *              Op                  - Current parser op
633  *
634  * RETURN:      None
635  *
636  * DESCRIPTION: Issue error message for a zero-length package (a package that
637  *              is required to have a non-zero length). Variable length
638  *              packages seem to be allowed to have zero length, however.
639  *              Even if not allowed, BIOS code does it.
640  *
641  ******************************************************************************/
642 
643 static void
644 ApZeroLengthPackage (
645     const char                  *PredefinedName,
646     ACPI_PARSE_OBJECT           *Op)
647 {
648 
649     sprintf (MsgBuffer, "%s: length is zero", PredefinedName);
650 
651     AslError (ASL_ERROR, ASL_MSG_RESERVED_PACKAGE_LENGTH, Op, MsgBuffer);
652 }
653 
654 
655 /*******************************************************************************
656  *
657  * FUNCTION:    ApPackageTooLarge
658  *
659  * PARAMETERS:  PredefinedName      - Name of the predefined object
660  *              Op                  - Current parser op
661  *              Count               - Actual package element count
662  *              ExpectedCount       - Expected package element count
663  *
664  * RETURN:      None
665  *
666  * DESCRIPTION: Issue a remark for a package that is larger than expected.
667  *
668  ******************************************************************************/
669 
670 static void
671 ApPackageTooLarge (
672     const char                  *PredefinedName,
673     ACPI_PARSE_OBJECT           *Op,
674     UINT32                      Count,
675     UINT32                      ExpectedCount)
676 {
677 
678     sprintf (MsgBuffer, "%s: length is %u, only %u required",
679         PredefinedName, Count, ExpectedCount);
680 
681     AslError (ASL_REMARK, ASL_MSG_RESERVED_PACKAGE_LENGTH, Op, MsgBuffer);
682 }
683