Path: blob/main/sys/contrib/dev/acpica/compiler/asllength.c
48373 views
/******************************************************************************1*2* Module Name: asllength - Tree walk to determine package and opcode lengths3*4*****************************************************************************/56/******************************************************************************7*8* 1. Copyright Notice9*10* Some or all of this work - Copyright (c) 1999 - 2025, Intel Corp.11* All rights reserved.12*13* 2. License14*15* 2.1. This is your license from Intel Corp. under its intellectual property16* rights. You may have additional license terms from the party that provided17* you this software, covering your right to use that party's intellectual18* property rights.19*20* 2.2. Intel grants, free of charge, to any person ("Licensee") obtaining a21* copy of the source code appearing in this file ("Covered Code") an22* irrevocable, perpetual, worldwide license under Intel's copyrights in the23* base code distributed originally by Intel ("Original Intel Code") to copy,24* make derivatives, distribute, use and display any portion of the Covered25* Code in any form, with the right to sublicense such rights; and26*27* 2.3. Intel grants Licensee a non-exclusive and non-transferable patent28* license (with the right to sublicense), under only those claims of Intel29* patents that are infringed by the Original Intel Code, to make, use, sell,30* offer to sell, and import the Covered Code and derivative works thereof31* solely to the minimum extent necessary to exercise the above copyright32* license, and in no event shall the patent license extend to any additions33* to or modifications of the Original Intel Code. No other license or right34* is granted directly or by implication, estoppel or otherwise;35*36* The above copyright and patent license is granted only if the following37* conditions are met:38*39* 3. Conditions40*41* 3.1. Redistribution of Source with Rights to Further Distribute Source.42* Redistribution of source code of any substantial portion of the Covered43* Code or modification with rights to further distribute source must include44* the above Copyright Notice, the above License, this list of Conditions,45* and the following Disclaimer and Export Compliance provision. In addition,46* Licensee must cause all Covered Code to which Licensee contributes to47* contain a file documenting the changes Licensee made to create that Covered48* Code and the date of any change. Licensee must include in that file the49* documentation of any changes made by any predecessor Licensee. Licensee50* must include a prominent statement that the modification is derived,51* directly or indirectly, from Original Intel Code.52*53* 3.2. Redistribution of Source with no Rights to Further Distribute Source.54* Redistribution of source code of any substantial portion of the Covered55* Code or modification without rights to further distribute source must56* include the following Disclaimer and Export Compliance provision in the57* documentation and/or other materials provided with distribution. In58* addition, Licensee may not authorize further sublicense of source of any59* portion of the Covered Code, and must include terms to the effect that the60* license from Licensee to its licensee is limited to the intellectual61* property embodied in the software Licensee provides to its licensee, and62* not to intellectual property embodied in modifications its licensee may63* make.64*65* 3.3. Redistribution of Executable. Redistribution in executable form of any66* substantial portion of the Covered Code or modification must reproduce the67* above Copyright Notice, and the following Disclaimer and Export Compliance68* provision in the documentation and/or other materials provided with the69* distribution.70*71* 3.4. Intel retains all right, title, and interest in and to the Original72* Intel Code.73*74* 3.5. Neither the name Intel nor any other trademark owned or controlled by75* Intel shall be used in advertising or otherwise to promote the sale, use or76* other dealings in products derived from or relating to the Covered Code77* without prior written authorization from Intel.78*79* 4. Disclaimer and Export Compliance80*81* 4.1. INTEL MAKES NO WARRANTY OF ANY KIND REGARDING ANY SOFTWARE PROVIDED82* HERE. ANY SOFTWARE ORIGINATING FROM INTEL OR DERIVED FROM INTEL SOFTWARE83* IS PROVIDED "AS IS," AND INTEL WILL NOT PROVIDE ANY SUPPORT, ASSISTANCE,84* INSTALLATION, TRAINING OR OTHER SERVICES. INTEL WILL NOT PROVIDE ANY85* UPDATES, ENHANCEMENTS OR EXTENSIONS. INTEL SPECIFICALLY DISCLAIMS ANY86* IMPLIED WARRANTIES OF MERCHANTABILITY, NONINFRINGEMENT AND FITNESS FOR A87* PARTICULAR PURPOSE.88*89* 4.2. IN NO EVENT SHALL INTEL HAVE ANY LIABILITY TO LICENSEE, ITS LICENSEES90* OR ANY OTHER THIRD PARTY, FOR ANY LOST PROFITS, LOST DATA, LOSS OF USE OR91* COSTS OF PROCUREMENT OF SUBSTITUTE GOODS OR SERVICES, OR FOR ANY INDIRECT,92* SPECIAL OR CONSEQUENTIAL DAMAGES ARISING OUT OF THIS AGREEMENT, UNDER ANY93* CAUSE OF ACTION OR THEORY OF LIABILITY, AND IRRESPECTIVE OF WHETHER INTEL94* HAS ADVANCE NOTICE OF THE POSSIBILITY OF SUCH DAMAGES. THESE LIMITATIONS95* SHALL APPLY NOTWITHSTANDING THE FAILURE OF THE ESSENTIAL PURPOSE OF ANY96* LIMITED REMEDY.97*98* 4.3. Licensee shall not export, either directly or indirectly, any of this99* software or system incorporating such software without first obtaining any100* required license or other approval from the U. S. Department of Commerce or101* any other agency or department of the United States Government. In the102* event Licensee exports any such software from the United States or103* re-exports any such software from a foreign destination, Licensee shall104* ensure that the distribution and export/re-export of the software is in105* compliance with all laws, regulations, orders, or other restrictions of the106* U.S. Export Administration Regulations. Licensee agrees that neither it nor107* any of its subsidiaries will export/re-export any technical data, process,108* software, or service, directly or indirectly, to any country for which the109* United States government or any agency thereof requires an export license,110* other governmental approval, or letter of assurance, without first obtaining111* such license, approval or letter.112*113*****************************************************************************114*115* Alternatively, you may choose to be licensed under the terms of the116* following license:117*118* Redistribution and use in source and binary forms, with or without119* modification, are permitted provided that the following conditions120* are met:121* 1. Redistributions of source code must retain the above copyright122* notice, this list of conditions, and the following disclaimer,123* without modification.124* 2. Redistributions in binary form must reproduce at minimum a disclaimer125* substantially similar to the "NO WARRANTY" disclaimer below126* ("Disclaimer") and any redistribution must be conditioned upon127* including a substantially similar Disclaimer requirement for further128* binary redistribution.129* 3. Neither the names of the above-listed copyright holders nor the names130* of any contributors may be used to endorse or promote products derived131* from this software without specific prior written permission.132*133* THIS SOFTWARE IS PROVIDED BY THE COPYRIGHT HOLDERS AND CONTRIBUTORS134* "AS IS" AND ANY EXPRESS OR IMPLIED WARRANTIES, INCLUDING, BUT NOT135* LIMITED TO, THE IMPLIED WARRANTIES OF MERCHANTABILITY AND FITNESS FOR136* A PARTICULAR PURPOSE ARE DISCLAIMED. IN NO EVENT SHALL THE COPYRIGHT137* OWNER OR CONTRIBUTORS BE LIABLE FOR ANY DIRECT, INDIRECT, INCIDENTAL,138* SPECIAL, EXEMPLARY, OR CONSEQUENTIAL DAMAGES (INCLUDING, BUT NOT139* LIMITED TO, PROCUREMENT OF SUBSTITUTE GOODS OR SERVICES; LOSS OF USE,140* DATA, OR PROFITS; OR BUSINESS INTERRUPTION) HOWEVER CAUSED AND ON ANY141* THEORY OF LIABILITY, WHETHER IN CONTRACT, STRICT LIABILITY, OR TORT142* (INCLUDING NEGLIGENCE OR OTHERWISE) ARISING IN ANY WAY OUT OF THE USE143* OF THIS SOFTWARE, EVEN IF ADVISED OF THE POSSIBILITY OF SUCH DAMAGE.144*145* Alternatively, you may choose to be licensed under the terms of the146* GNU General Public License ("GPL") version 2 as published by the Free147* Software Foundation.148*149*****************************************************************************/150151#include <contrib/dev/acpica/compiler/aslcompiler.h>152#include "aslcompiler.y.h"153#include <contrib/dev/acpica/include/amlcode.h>154#include <contrib/dev/acpica/include/acconvert.h>155156157#define _COMPONENT ACPI_COMPILER158ACPI_MODULE_NAME ("asllength")159160/* Local prototypes */161162static UINT8163CgGetPackageLenByteCount (164ACPI_PARSE_OBJECT *Op,165UINT32 PackageLength);166167static void168CgGenerateAmlOpcodeLength (169ACPI_PARSE_OBJECT *Op);170171172#ifdef ACPI_OBSOLETE_FUNCTIONS173void174LnAdjustLengthToRoot (175ACPI_PARSE_OBJECT *Op,176UINT32 LengthDelta);177#endif178179180/*******************************************************************************181*182* FUNCTION: LnInitLengthsWalk183*184* PARAMETERS: ASL_WALK_CALLBACK185*186* RETURN: Status187*188* DESCRIPTION: Walk callback to initialize (and re-initialize) the node189* subtree length(s) to zero. The Subtree lengths are bubbled190* up to the root node in order to get a total AML length.191*192******************************************************************************/193194ACPI_STATUS195LnInitLengthsWalk (196ACPI_PARSE_OBJECT *Op,197UINT32 Level,198void *Context)199{200201Op->Asl.AmlSubtreeLength = 0;202return (AE_OK);203}204205206/*******************************************************************************207*208* FUNCTION: LnPackageLengthWalk209*210* PARAMETERS: ASL_WALK_CALLBACK211*212* RETURN: Status213*214* DESCRIPTION: Walk callback to calculate the total AML length.215* 1) Calculate the AML lengths (opcode, package length, etc.) for216* THIS node.217* 2) Bubbble up all of these lengths to the parent node by summing218* them all into the parent subtree length.219*220* Note: The SubtreeLength represents the total AML length of all child nodes221* in all subtrees under a given node. Therefore, once this walk is222* complete, the Root Node subtree length is the AML length of the entire223* tree (and thus, the entire ACPI table)224*225******************************************************************************/226227ACPI_STATUS228LnPackageLengthWalk (229ACPI_PARSE_OBJECT *Op,230UINT32 Level,231void *Context)232{233234/* Generate the AML lengths for this node */235236CgGenerateAmlLengths (Op);237238/* Bubble up all lengths (this node and all below it) to the parent */239240if ((Op->Asl.Parent) &&241(Op->Asl.ParseOpcode != PARSEOP_DEFAULT_ARG))242{243Op->Asl.Parent->Asl.AmlSubtreeLength += (244Op->Asl.AmlLength +245Op->Asl.AmlOpcodeLength +246Op->Asl.AmlPkgLenBytes +247Op->Asl.AmlSubtreeLength +248CvCalculateCommentLengths (Op)249);250}251return (AE_OK);252}253254255/*******************************************************************************256*257* FUNCTION: CgGetPackageLenByteCount258*259* PARAMETERS: Op - Parse node260* PackageLength - Length to be encoded261*262* RETURN: Required length of the package length encoding263*264* DESCRIPTION: Calculate the number of bytes required to encode the given265* package length.266*267******************************************************************************/268269static UINT8270CgGetPackageLenByteCount (271ACPI_PARSE_OBJECT *Op,272UINT32 PackageLength)273{274275/*276* Determine the number of bytes required to encode the package length277* Note: the package length includes the number of bytes used to encode278* the package length, so we must account for this also.279*/280if (PackageLength <= (0x0000003F - 1))281{282return (1);283}284else if (PackageLength <= (0x00000FFF - 2))285{286return (2);287}288else if (PackageLength <= (0x000FFFFF - 3))289{290return (3);291}292else if (PackageLength <= (0x0FFFFFFF - 4))293{294return (4);295}296else297{298/* Fatal error - the package length is too large to encode */299300AslError (ASL_ERROR, ASL_MSG_ENCODING_LENGTH, Op, NULL);301}302303return (0);304}305306307/*******************************************************************************308*309* FUNCTION: CgGenerateAmlOpcodeLength310*311* PARAMETERS: Op - Parse node whose AML opcode lengths will be312* calculated313*314* RETURN: None.315*316* DESCRIPTION: Calculate the AmlOpcodeLength, AmlPkgLenBytes, and AmlLength317* fields for this node.318*319******************************************************************************/320321static void322CgGenerateAmlOpcodeLength (323ACPI_PARSE_OBJECT *Op)324{325326/* Check for two-byte opcode */327328if (Op->Asl.AmlOpcode > 0x00FF)329{330Op->Asl.AmlOpcodeLength = 2;331}332else333{334Op->Asl.AmlOpcodeLength = 1;335}336337/* Does this opcode have an associated "PackageLength" field? */338339Op->Asl.AmlPkgLenBytes = 0;340if (Op->Asl.CompileFlags & OP_AML_PACKAGE)341{342Op->Asl.AmlPkgLenBytes = CgGetPackageLenByteCount (343Op, Op->Asl.AmlSubtreeLength);344}345346/* Data opcode lengths are easy */347348switch (Op->Asl.AmlOpcode)349{350case AML_BYTE_OP:351352Op->Asl.AmlLength = 1;353break;354355case AML_WORD_OP:356357Op->Asl.AmlLength = 2;358break;359360case AML_DWORD_OP:361362Op->Asl.AmlLength = 4;363break;364365case AML_QWORD_OP:366367Op->Asl.AmlLength = 8;368break;369370default:371372/* All data opcodes must be above */373break;374}375}376377378/*******************************************************************************379*380* FUNCTION: CgGenerateAmlLengths381*382* PARAMETERS: Op - Parse node383*384* RETURN: None.385*386* DESCRIPTION: Generate internal length fields based on the AML opcode or387* parse opcode.388*389******************************************************************************/390391void392CgGenerateAmlLengths (393ACPI_PARSE_OBJECT *Op)394{395char *Buffer;396ACPI_STATUS Status;397398399switch (Op->Asl.AmlOpcode)400{401case AML_RAW_DATA_BYTE:402403Op->Asl.AmlOpcodeLength = 0;404Op->Asl.AmlLength = 1;405return;406407case AML_RAW_DATA_WORD:408409Op->Asl.AmlOpcodeLength = 0;410Op->Asl.AmlLength = 2;411return;412413case AML_RAW_DATA_DWORD:414415Op->Asl.AmlOpcodeLength = 0;416Op->Asl.AmlLength = 4;417return;418419case AML_RAW_DATA_QWORD:420421Op->Asl.AmlOpcodeLength = 0;422Op->Asl.AmlLength = 8;423return;424425case AML_RAW_DATA_BUFFER:426427/* Aml length is/was set by creator */428429Op->Asl.AmlOpcodeLength = 0;430return;431432case AML_RAW_DATA_CHAIN:433434/* Aml length is/was set by creator */435436Op->Asl.AmlOpcodeLength = 0;437return;438439default:440441break;442}443444switch (Op->Asl.ParseOpcode)445{446case PARSEOP_DEFINITION_BLOCK:447448AslGbl_TableLength = sizeof (ACPI_TABLE_HEADER) + Op->Asl.AmlSubtreeLength;449break;450451case PARSEOP_NAMESEG:452453Op->Asl.AmlOpcodeLength = 0;454Op->Asl.AmlLength = 4;455Op->Asl.ExternalName = Op->Asl.Value.String;456break;457458case PARSEOP_NAMESTRING:459case PARSEOP_METHODCALL:460461if (Op->Asl.CompileFlags & OP_NAME_INTERNALIZED)462{463break;464}465466Op->Asl.AmlOpcodeLength = 0;467Status = UtInternalizeName (Op->Asl.Value.String, &Buffer);468if (ACPI_FAILURE (Status))469{470DbgPrint (ASL_DEBUG_OUTPUT,471"Failure from internalize name %X\n", Status);472break;473}474475Op->Asl.ExternalName = Op->Asl.Value.String;476Op->Asl.Value.String = Buffer;477Op->Asl.CompileFlags |= OP_NAME_INTERNALIZED;478Op->Asl.AmlLength = strlen (Buffer);479480/*481* Check for single backslash reference to root or reference to a name482* consisting of only prefix (^) characters. Make it a null terminated483* string in the AML.484*/485if (Op->Asl.AmlLength == 1 || UtNameContainsAllPrefix(Op))486{487Op->Asl.AmlLength++;488}489break;490491case PARSEOP_STRING_LITERAL:492493Op->Asl.AmlOpcodeLength = 1;494495/* Get null terminator */496497Op->Asl.AmlLength = strlen (Op->Asl.Value.String) + 1;498break;499500case PARSEOP_PACKAGE_LENGTH:501502Op->Asl.AmlOpcodeLength = 0;503Op->Asl.AmlPkgLenBytes = CgGetPackageLenByteCount (Op,504(UINT32) Op->Asl.Value.Integer);505break;506507case PARSEOP_RAW_DATA:508509Op->Asl.AmlOpcodeLength = 0;510break;511512case PARSEOP_DEFAULT_ARG:513case PARSEOP_INCLUDE:514case PARSEOP_INCLUDE_END:515516/* Ignore the "default arg" nodes, they are extraneous at this point */517518break;519520case PARSEOP_EXTERNAL:521522CgGenerateAmlOpcodeLength (Op);523break;524525default:526527CgGenerateAmlOpcodeLength (Op);528break;529}530}531532533#ifdef ACPI_OBSOLETE_FUNCTIONS534/*******************************************************************************535*536* FUNCTION: LnAdjustLengthToRoot537*538* PARAMETERS: Op - Node whose Length was changed539*540* RETURN: None.541*542* DESCRIPTION: Change the Subtree length of the given node, and bubble the543* change all the way up to the root node. This allows for544* last second changes to a package length (for example, if the545* package length encoding gets shorter or longer.)546*547******************************************************************************/548549void550LnAdjustLengthToRoot (551ACPI_PARSE_OBJECT *SubtreeOp,552UINT32 LengthDelta)553{554ACPI_PARSE_OBJECT *Op;555556557/* Adjust all subtree lengths up to the root */558559Op = SubtreeOp->Asl.Parent;560while (Op)561{562Op->Asl.AmlSubtreeLength -= LengthDelta;563Op = Op->Asl.Parent;564}565566/* Adjust the global table length */567568AslGbl_TableLength -= LengthDelta;569}570#endif571572573