2005-06-29 20:55:00 +04:00
/******************************************************************************
*
2005-06-29 21:01:47 +04:00
* Module Name : dsopcode - Dispatcher Op Region support and handling of
2005-06-29 20:59:25 +04:00
* " control " opcodes
2005-06-29 21:02:05 +04:00
* $ Revision : 1.56 $
2005-06-29 20:55:00 +04:00
*
* * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * */
/******************************************************************************
*
* 1. Copyright Notice
*
2005-06-29 21:02:02 +04:00
* Some or all of this work - Copyright ( c ) 1999 , 2000 , 2001 , Intel Corp .
* All rights reserved .
2005-06-29 20:55:00 +04:00
*
* 2. License
*
* 2.1 . This is your license from Intel Corp . under its intellectual property
* rights . You may have additional license terms from the party that provided
* you this software , covering your right to use that party ' s intellectual
* property rights .
*
* 2.2 . Intel grants , free of charge , to any person ( " Licensee " ) obtaining a
* copy of the source code appearing in this file ( " Covered Code " ) an
* irrevocable , perpetual , worldwide license under Intel ' s copyrights in the
* base code distributed originally by Intel ( " Original Intel Code " ) to copy ,
* make derivatives , distribute , use and display any portion of the Covered
* Code in any form , with the right to sublicense such rights ; and
*
* 2.3 . Intel grants Licensee a non - exclusive and non - transferable patent
* license ( with the right to sublicense ) , under only those claims of Intel
* patents that are infringed by the Original Intel Code , to make , use , sell ,
* offer to sell , and import the Covered Code and derivative works thereof
* solely to the minimum extent necessary to exercise the above copyright
* license , and in no event shall the patent license extend to any additions
* to or modifications of the Original Intel Code . No other license or right
* is granted directly or by implication , estoppel or otherwise ;
*
* The above copyright and patent license is granted only if the following
* conditions are met :
*
* 3. Conditions
*
* 3.1 . Redistribution of Source with Rights to Further Distribute Source .
* Redistribution of source code of any substantial portion of the Covered
* Code or modification with rights to further distribute source must include
* the above Copyright Notice , the above License , this list of Conditions ,
* and the following Disclaimer and Export Compliance provision . In addition ,
* Licensee must cause all Covered Code to which Licensee contributes to
* contain a file documenting the changes Licensee made to create that Covered
* Code and the date of any change . Licensee must include in that file the
* documentation of any changes made by any predecessor Licensee . Licensee
* must include a prominent statement that the modification is derived ,
* directly or indirectly , from Original Intel Code .
*
* 3.2 . Redistribution of Source with no Rights to Further Distribute Source .
* Redistribution of source code of any substantial portion of the Covered
* Code or modification without rights to further distribute source must
* include the following Disclaimer and Export Compliance provision in the
* documentation and / or other materials provided with distribution . In
* addition , Licensee may not authorize further sublicense of source of any
* portion of the Covered Code , and must include terms to the effect that the
* license from Licensee to its licensee is limited to the intellectual
* property embodied in the software Licensee provides to its licensee , and
* not to intellectual property embodied in modifications its licensee may
* make .
*
* 3.3 . Redistribution of Executable . Redistribution in executable form of any
* substantial portion of the Covered Code or modification must reproduce the
* above Copyright Notice , and the following Disclaimer and Export Compliance
* provision in the documentation and / or other materials provided with the
* distribution .
*
* 3.4 . Intel retains all right , title , and interest in and to the Original
* Intel Code .
*
* 3.5 . Neither the name Intel nor any other trademark owned or controlled by
* Intel shall be used in advertising or otherwise to promote the sale , use or
* other dealings in products derived from or relating to the Covered Code
* without prior written authorization from Intel .
*
* 4. Disclaimer and Export Compliance
*
* 4.1 . INTEL MAKES NO WARRANTY OF ANY KIND REGARDING ANY SOFTWARE PROVIDED
* HERE . ANY SOFTWARE ORIGINATING FROM INTEL OR DERIVED FROM INTEL SOFTWARE
* IS PROVIDED " AS IS, " AND INTEL WILL NOT PROVIDE ANY SUPPORT , ASSISTANCE ,
* INSTALLATION , TRAINING OR OTHER SERVICES . INTEL WILL NOT PROVIDE ANY
* UPDATES , ENHANCEMENTS OR EXTENSIONS . INTEL SPECIFICALLY DISCLAIMS ANY
* IMPLIED WARRANTIES OF MERCHANTABILITY , NONINFRINGEMENT AND FITNESS FOR A
* PARTICULAR PURPOSE .
*
* 4.2 . IN NO EVENT SHALL INTEL HAVE ANY LIABILITY TO LICENSEE , ITS LICENSEES
* OR ANY OTHER THIRD PARTY , FOR ANY LOST PROFITS , LOST DATA , LOSS OF USE OR
* COSTS OF PROCUREMENT OF SUBSTITUTE GOODS OR SERVICES , OR FOR ANY INDIRECT ,
* SPECIAL OR CONSEQUENTIAL DAMAGES ARISING OUT OF THIS AGREEMENT , UNDER ANY
* CAUSE OF ACTION OR THEORY OF LIABILITY , AND IRRESPECTIVE OF WHETHER INTEL
* HAS ADVANCE NOTICE OF THE POSSIBILITY OF SUCH DAMAGES . THESE LIMITATIONS
* SHALL APPLY NOTWITHSTANDING THE FAILURE OF THE ESSENTIAL PURPOSE OF ANY
* LIMITED REMEDY .
*
* 4.3 . Licensee shall not export , either directly or indirectly , any of this
* software or system incorporating such software without first obtaining any
* required license or other approval from the U . S . Department of Commerce or
* any other agency or department of the United States Government . In the
* event Licensee exports any such software from the United States or
* re - exports any such software from a foreign destination , Licensee shall
* ensure that the distribution and export / re - export of the software is in
* compliance with all laws , regulations , orders , or other restrictions of the
* U . S . Export Administration Regulations . Licensee agrees that neither it nor
* any of its subsidiaries will export / re - export any technical data , process ,
* software , or service , directly or indirectly , to any country for which the
* United States government or any agency thereof requires an export license ,
* other governmental approval , or letter of assurance , without first obtaining
* such license , approval or letter .
*
* * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * */
# define __DSOPCODE_C__
# include "acpi.h"
2005-06-29 20:59:20 +04:00
# include "acparser.h"
2005-06-29 20:55:00 +04:00
# include "amlcode.h"
2005-06-29 20:59:20 +04:00
# include "acdispat.h"
# include "acinterp.h"
# include "acnamesp.h"
# include "acevents.h"
2005-06-29 21:01:41 +04:00
# include "actables.h"
2005-06-29 20:55:00 +04:00
2005-06-29 21:02:02 +04:00
# define _COMPONENT ACPI_DISPATCHER
2005-06-29 21:01:47 +04:00
MODULE_NAME ( " dsopcode " )
/*****************************************************************************
*
2005-06-29 21:02:02 +04:00
* FUNCTION : AcpiDsGetBufferFieldArguments
2005-06-29 21:01:47 +04:00
*
2005-06-29 21:02:02 +04:00
* PARAMETERS : ObjDesc - A valid BufferField object
2005-06-29 21:01:47 +04:00
*
* RETURN : Status .
*
2005-06-29 21:02:02 +04:00
* DESCRIPTION : Get BufferField Buffer and Index . This implements the late
2005-06-29 21:01:47 +04:00
* evaluation of these field attributes .
*
* * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * */
ACPI_STATUS
2005-06-29 21:02:02 +04:00
AcpiDsGetBufferFieldArguments (
2005-06-29 21:01:47 +04:00
ACPI_OPERAND_OBJECT * ObjDesc )
{
ACPI_OPERAND_OBJECT * ExtraDesc ;
ACPI_NAMESPACE_NODE * Node ;
ACPI_PARSE_OBJECT * Op ;
ACPI_PARSE_OBJECT * FieldOp ;
ACPI_STATUS Status ;
ACPI_TABLE_DESC * TableDesc ;
2005-06-29 21:02:02 +04:00
ACPI_WALK_STATE * WalkState ;
2005-06-29 21:01:47 +04:00
2005-06-29 21:02:02 +04:00
FUNCTION_TRACE_PTR ( " DsGetBufferFieldArguments " , ObjDesc ) ;
2005-06-29 21:01:47 +04:00
if ( ObjDesc - > Common . Flags & AOPOBJ_DATA_VALID )
{
return_ACPI_STATUS ( AE_OK ) ;
}
2005-06-29 21:02:02 +04:00
/* Get the AML pointer (method object) and BufferField node */
2005-06-29 21:01:47 +04:00
2005-06-29 21:02:02 +04:00
ExtraDesc = ObjDesc - > BufferField . Extra ;
Node = ObjDesc - > BufferField . Node ;
2005-06-29 21:01:47 +04:00
2005-06-29 21:02:02 +04:00
DEBUG_EXEC ( AcpiUtDisplayInitPathname ( Node , " [Field] " ) ) ;
ACPI_DEBUG_PRINT ( ( ACPI_DB_EXEC , " [%4.4s] BufferField JIT Init \n " ,
2005-06-29 21:02:05 +04:00
( char * ) & Node - > Name ) ) ;
2005-06-29 21:01:47 +04:00
/*
* Allocate a new parser op to be the root of the parsed
* OpRegion tree
*/
Op = AcpiPsAllocOp ( AML_SCOPE_OP ) ;
if ( ! Op )
{
return ( AE_NO_MEMORY ) ;
}
/* Save the Node for use in AcpiPsParseAml */
Op - > Node = AcpiNsGetParentObject ( Node ) ;
/* Get a handle to the parent ACPI table */
Status = AcpiTbHandleToObject ( Node - > OwnerId , & TableDesc ) ;
if ( ACPI_FAILURE ( Status ) )
{
return_ACPI_STATUS ( Status ) ;
}
2005-06-29 21:02:02 +04:00
/* Create and initialize a new parser state */
WalkState = AcpiDsCreateWalkState ( TABLE_ID_DSDT ,
NULL , NULL , NULL ) ;
if ( ! WalkState )
{
return_ACPI_STATUS ( AE_NO_MEMORY ) ;
}
2005-06-29 21:01:47 +04:00
2005-06-29 21:02:05 +04:00
Status = AcpiDsInitAmlWalk ( WalkState , Op , NULL , ExtraDesc - > Extra . AmlStart ,
ExtraDesc - > Extra . AmlLength , NULL , NULL , 1 ) ;
2005-06-29 21:01:47 +04:00
if ( ACPI_FAILURE ( Status ) )
{
2005-06-29 21:02:02 +04:00
/* TBD: delete walk state */
2005-06-29 21:01:47 +04:00
return_ACPI_STATUS ( Status ) ;
}
2005-06-29 21:02:02 +04:00
/* TBD: No Walk flags?? */
WalkState - > ParseFlags = 0 ;
2005-06-29 21:01:47 +04:00
2005-06-29 21:02:02 +04:00
/* Pass1: Parse the entire BufferField declaration */
Status = AcpiPsParseAml ( WalkState ) ;
if ( ACPI_FAILURE ( Status ) )
{
AcpiPsDeleteParseTree ( Op ) ;
return_ACPI_STATUS ( Status ) ;
}
/* Get and init the actual FieldUnit Op created above */
2005-06-29 21:01:47 +04:00
FieldOp = Op - > Value . Arg ;
Op - > Node = Node ;
FieldOp = Op - > Value . Arg ;
FieldOp - > Node = Node ;
AcpiPsDeleteParseTree ( Op ) ;
2005-06-29 21:02:02 +04:00
/* Evaluate the address and length arguments for the OpRegion */
2005-06-29 21:01:47 +04:00
Op = AcpiPsAllocOp ( AML_SCOPE_OP ) ;
if ( ! Op )
{
return ( AE_NO_MEMORY ) ;
}
Op - > Node = AcpiNsGetParentObject ( Node ) ;
2005-06-29 21:02:02 +04:00
/* Create and initialize a new parser state */
2005-06-29 21:01:47 +04:00
2005-06-29 21:02:02 +04:00
WalkState = AcpiDsCreateWalkState ( TABLE_ID_DSDT ,
NULL , NULL , NULL ) ;
if ( ! WalkState )
{
return_ACPI_STATUS ( AE_NO_MEMORY ) ;
}
2005-06-29 21:01:47 +04:00
2005-06-29 21:02:05 +04:00
Status = AcpiDsInitAmlWalk ( WalkState , Op , NULL , ExtraDesc - > Extra . AmlStart ,
ExtraDesc - > Extra . AmlLength , NULL , NULL , 3 ) ;
2005-06-29 21:02:02 +04:00
if ( ACPI_FAILURE ( Status ) )
{
/* TBD: delete walk state */
return_ACPI_STATUS ( Status ) ;
}
Status = AcpiPsParseAml ( WalkState ) ;
AcpiPsDeleteParseTree ( Op ) ;
2005-06-29 21:01:47 +04:00
/*
* The pseudo - method object is no longer needed since the region is
* now initialized
*/
2005-06-29 21:02:02 +04:00
AcpiUtRemoveReference ( ObjDesc - > BufferField . Extra ) ;
ObjDesc - > BufferField . Extra = NULL ;
2005-06-29 21:01:47 +04:00
return_ACPI_STATUS ( Status ) ;
}
2005-06-29 21:00:32 +04:00
2005-06-29 20:55:00 +04:00
2005-06-29 21:01:41 +04:00
/*****************************************************************************
2005-06-29 20:59:30 +04:00
*
2005-06-29 21:01:41 +04:00
* FUNCTION : AcpiDsGetRegionArguments
2005-06-29 20:59:30 +04:00
*
2005-06-29 21:01:47 +04:00
* PARAMETERS : ObjDesc - A valid region object
2005-06-29 20:59:30 +04:00
*
* RETURN : Status .
*
2005-06-29 21:01:41 +04:00
* DESCRIPTION : Get region address and length . This implements the late
* evaluation of these region attributes .
2005-06-29 20:59:30 +04:00
*
2005-06-29 21:01:41 +04:00
* * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * */
2005-06-29 20:59:30 +04:00
2005-06-29 21:01:41 +04:00
ACPI_STATUS
AcpiDsGetRegionArguments (
2005-06-29 21:01:47 +04:00
ACPI_OPERAND_OBJECT * ObjDesc )
2005-06-29 20:59:30 +04:00
{
2005-06-29 21:01:47 +04:00
ACPI_OPERAND_OBJECT * ExtraDesc = NULL ;
ACPI_NAMESPACE_NODE * Node ;
ACPI_PARSE_OBJECT * Op ;
ACPI_PARSE_OBJECT * RegionOp ;
2005-06-29 20:59:30 +04:00
ACPI_STATUS Status ;
2005-06-29 21:01:41 +04:00
ACPI_TABLE_DESC * TableDesc ;
2005-06-29 21:02:02 +04:00
ACPI_WALK_STATE * WalkState ;
2005-06-29 20:59:30 +04:00
2005-06-29 21:01:47 +04:00
FUNCTION_TRACE_PTR ( " DsGetRegionArguments " , ObjDesc ) ;
2005-06-29 20:59:30 +04:00
2005-06-29 21:01:47 +04:00
if ( ObjDesc - > Region . Flags & AOPOBJ_DATA_VALID )
2005-06-29 20:59:30 +04:00
{
2005-06-29 21:01:41 +04:00
return_ACPI_STATUS ( AE_OK ) ;
2005-06-29 20:59:30 +04:00
}
2005-06-29 21:01:47 +04:00
/* Get the AML pointer (method object) and region node */
2005-06-29 20:59:30 +04:00
2005-06-29 21:01:47 +04:00
ExtraDesc = ObjDesc - > Region . Extra ;
Node = ObjDesc - > Region . Node ;
2005-06-29 21:02:02 +04:00
DEBUG_EXEC ( AcpiUtDisplayInitPathname ( Node , " [Operation Region] " ) ) ;
2005-06-29 21:02:05 +04:00
ACPI_DEBUG_PRINT ( ( ACPI_DB_EXEC , " [%4.4s] OpRegion Init at AML %p \n " ,
( char * ) & Node - > Name , ExtraDesc - > Extra . AmlStart ) ) ;
2005-06-29 20:59:30 +04:00
2005-06-29 21:01:41 +04:00
/*
* Allocate a new parser op to be the root of the parsed
* OpRegion tree
*/
Op = AcpiPsAllocOp ( AML_SCOPE_OP ) ;
if ( ! Op )
2005-06-29 20:59:30 +04:00
{
2005-06-29 21:01:41 +04:00
return ( AE_NO_MEMORY ) ;
2005-06-29 20:59:30 +04:00
}
2005-06-29 21:01:47 +04:00
/* Save the Node for use in AcpiPsParseAml */
2005-06-29 21:00:23 +04:00
2005-06-29 21:01:47 +04:00
Op - > Node = AcpiNsGetParentObject ( Node ) ;
2005-06-29 21:00:02 +04:00
2005-06-29 21:01:41 +04:00
/* Get a handle to the parent ACPI table */
2005-06-29 20:59:30 +04:00
2005-06-29 21:01:47 +04:00
Status = AcpiTbHandleToObject ( Node - > OwnerId , & TableDesc ) ;
2005-06-29 20:59:30 +04:00
if ( ACPI_FAILURE ( Status ) )
{
return_ACPI_STATUS ( Status ) ;
}
2005-06-29 21:02:02 +04:00
/* Create and initialize a new parser state */
2005-06-29 20:59:30 +04:00
2005-06-29 21:02:02 +04:00
WalkState = AcpiDsCreateWalkState ( TABLE_ID_DSDT ,
Op , NULL , NULL ) ;
if ( ! WalkState )
{
return_ACPI_STATUS ( AE_NO_MEMORY ) ;
}
2005-06-29 20:59:30 +04:00
2005-06-29 21:02:05 +04:00
Status = AcpiDsInitAmlWalk ( WalkState , Op , NULL , ExtraDesc - > Extra . AmlStart ,
ExtraDesc - > Extra . AmlLength , NULL , NULL , 1 ) ;
2005-06-29 21:00:02 +04:00
if ( ACPI_FAILURE ( Status ) )
{
2005-06-29 21:02:02 +04:00
/* TBD: delete walk state */
2005-06-29 21:00:02 +04:00
return_ACPI_STATUS ( Status ) ;
}
2005-06-29 20:59:30 +04:00
2005-06-29 21:02:02 +04:00
/* TBD: No Walk flags?? */
WalkState - > ParseFlags = 0 ;
/* Parse the entire OpRegion declaration, creating a parse tree */
Status = AcpiPsParseAml ( WalkState ) ;
if ( ACPI_FAILURE ( Status ) )
{
AcpiPsDeleteParseTree ( Op ) ;
return_ACPI_STATUS ( Status ) ;
}
2005-06-29 21:00:05 +04:00
2005-06-29 21:01:41 +04:00
/* Get and init the actual RegionOp created above */
2005-06-29 21:00:05 +04:00
2005-06-29 21:01:47 +04:00
RegionOp = Op - > Value . Arg ;
Op - > Node = Node ;
2005-06-29 21:00:05 +04:00
2005-06-29 21:01:41 +04:00
RegionOp = Op - > Value . Arg ;
2005-06-29 21:01:47 +04:00
RegionOp - > Node = Node ;
2005-06-29 21:01:41 +04:00
AcpiPsDeleteParseTree ( Op ) ;
2005-06-29 21:00:05 +04:00
2005-06-29 21:02:02 +04:00
/* Evaluate the address and length arguments for the OpRegion */
2005-06-29 21:00:05 +04:00
2005-06-29 21:01:41 +04:00
Op = AcpiPsAllocOp ( AML_SCOPE_OP ) ;
if ( ! Op )
2005-06-29 21:00:05 +04:00
{
2005-06-29 21:01:41 +04:00
return ( AE_NO_MEMORY ) ;
2005-06-29 21:00:05 +04:00
}
2005-06-29 21:01:47 +04:00
Op - > Node = AcpiNsGetParentObject ( Node ) ;
2005-06-29 21:00:05 +04:00
2005-06-29 21:02:02 +04:00
/* Create and initialize a new parser state */
2005-06-29 21:01:47 +04:00
2005-06-29 21:02:02 +04:00
WalkState = AcpiDsCreateWalkState ( TABLE_ID_DSDT ,
Op , NULL , NULL ) ;
if ( ! WalkState )
{
return_ACPI_STATUS ( AE_NO_MEMORY ) ;
}
2005-06-29 21:02:05 +04:00
Status = AcpiDsInitAmlWalk ( WalkState , Op , NULL , ExtraDesc - > Extra . AmlStart ,
ExtraDesc - > Extra . AmlLength , NULL , NULL , 3 ) ;
2005-06-29 21:02:02 +04:00
if ( ACPI_FAILURE ( Status ) )
{
/* TBD: delete walk state */
return_ACPI_STATUS ( Status ) ;
}
2005-06-29 21:00:05 +04:00
2005-06-29 21:02:02 +04:00
Status = AcpiPsParseAml ( WalkState ) ;
2005-06-29 21:01:41 +04:00
AcpiPsDeleteParseTree ( Op ) ;
2005-06-29 21:00:05 +04:00
2005-06-29 21:00:02 +04:00
return_ACPI_STATUS ( Status ) ;
}
2005-06-29 20:55:39 +04:00
2005-06-29 21:00:02 +04:00
/*****************************************************************************
2005-06-29 20:55:00 +04:00
*
* FUNCTION : AcpiDsInitializeRegion
*
2005-06-29 21:01:41 +04:00
* PARAMETERS : Op - A valid region Op object
2005-06-29 20:55:00 +04:00
*
* RETURN : Status
*
2005-06-29 21:01:41 +04:00
* DESCRIPTION :
2005-06-29 20:55:00 +04:00
*
2005-06-29 21:01:41 +04:00
* * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * */
2005-06-29 20:55:00 +04:00
ACPI_STATUS
AcpiDsInitializeRegion (
ACPI_HANDLE ObjHandle )
{
2005-06-29 21:01:47 +04:00
ACPI_OPERAND_OBJECT * ObjDesc ;
2005-06-29 20:55:00 +04:00
ACPI_STATUS Status ;
ObjDesc = AcpiNsGetAttachedObject ( ObjHandle ) ;
2005-06-29 20:59:16 +04:00
/* Namespace is NOT locked */
Status = AcpiEvInitializeRegion ( ObjDesc , FALSE ) ;
2005-06-29 20:59:30 +04:00
2005-06-29 21:01:41 +04:00
return ( Status ) ;
2005-06-29 20:59:30 +04:00
}
2005-06-29 21:01:47 +04:00
/*****************************************************************************
*
2005-06-29 21:02:02 +04:00
* FUNCTION : AcpiDsEvalBufferFieldOperands
2005-06-29 21:01:47 +04:00
*
2005-06-29 21:02:02 +04:00
* PARAMETERS : Op - A valid BufferField Op object
2005-06-29 21:01:47 +04:00
*
* RETURN : Status
*
2005-06-29 21:02:02 +04:00
* DESCRIPTION : Get BufferField Buffer and Index
* Called from AcpiDsExecEndOp during BufferField parse tree walk
*
* ACPI SPECIFICATION REFERENCES :
* Each of the Buffer Field opcodes is defined as specified in in - line
* comments below . For each one , use the following definitions .
*
* DefBitField : = BitFieldOp SrcBuf BitIdx Destination
* DefByteField : = ByteFieldOp SrcBuf ByteIdx Destination
* DefCreateField : = CreateFieldOp SrcBuf BitIdx NumBits NameString
* DefDWordField : = DWordFieldOp SrcBuf ByteIdx Destination
* DefWordField : = WordFieldOp SrcBuf ByteIdx Destination
* BitIndex : = TermArg = > Integer
* ByteIndex : = TermArg = > Integer
* Destination : = NameString
* NumBits : = TermArg = > Integer
* SourceBuf : = TermArg = > Buffer
2005-06-29 21:01:47 +04:00
*
* * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * */
ACPI_STATUS
2005-06-29 21:02:02 +04:00
AcpiDsEvalBufferFieldOperands (
2005-06-29 21:01:47 +04:00
ACPI_WALK_STATE * WalkState ,
ACPI_PARSE_OBJECT * Op )
{
ACPI_STATUS Status ;
2005-06-29 21:02:02 +04:00
ACPI_OPERAND_OBJECT * ObjDesc ;
2005-06-29 21:01:47 +04:00
ACPI_NAMESPACE_NODE * Node ;
ACPI_PARSE_OBJECT * NextOp ;
UINT32 Offset ;
UINT32 BitOffset ;
2005-06-29 21:02:02 +04:00
UINT32 BitCount ;
UINT8 FieldFlags ;
2005-06-29 21:01:47 +04:00
ACPI_OPERAND_OBJECT * ResDesc = NULL ;
ACPI_OPERAND_OBJECT * CntDesc = NULL ;
ACPI_OPERAND_OBJECT * OffDesc = NULL ;
ACPI_OPERAND_OBJECT * SrcDesc = NULL ;
2005-06-29 21:02:02 +04:00
FUNCTION_TRACE_PTR ( " DsEvalBufferFieldOperands " , Op ) ;
2005-06-29 21:01:47 +04:00
/*
2005-06-29 21:02:02 +04:00
* This is where we evaluate the address and length fields of the
* CreateXxxField declaration
2005-06-29 21:01:47 +04:00
*/
Node = Op - > Node ;
/* NextOp points to the op that holds the Buffer */
2005-06-29 21:02:02 +04:00
2005-06-29 21:01:47 +04:00
NextOp = Op - > Value . Arg ;
/* AcpiEvaluate/create the address and length operands */
Status = AcpiDsCreateOperands ( WalkState , NextOp ) ;
if ( ACPI_FAILURE ( Status ) )
{
return_ACPI_STATUS ( Status ) ;
}
2005-06-29 21:02:02 +04:00
ObjDesc = AcpiNsGetAttachedObject ( Node ) ;
if ( ! ObjDesc )
2005-06-29 21:01:47 +04:00
{
return_ACPI_STATUS ( AE_NOT_EXIST ) ;
}
/* Resolve the operands */
2005-06-29 21:02:02 +04:00
Status = AcpiExResolveOperands ( Op - > Opcode , WALK_OPERANDS , WalkState ) ;
2005-06-29 21:01:47 +04:00
DUMP_OPERANDS ( WALK_OPERANDS , IMODE_EXECUTE , AcpiPsGetOpcodeName ( Op - > Opcode ) ,
2005-06-29 21:02:02 +04:00
WalkState - > NumOperands , " after AcpiExResolveOperands " ) ;
2005-06-29 21:01:47 +04:00
2005-06-29 21:02:02 +04:00
if ( ACPI_FAILURE ( Status ) )
{
ACPI_DEBUG_PRINT ( ( ACPI_DB_ERROR , " (%s) bad operand(s) (%X) \n " ,
AcpiPsGetOpcodeName ( Op - > Opcode ) , Status ) ) ;
goto Cleanup ;
}
2005-06-29 21:01:47 +04:00
/* Get the operands */
if ( AML_CREATE_FIELD_OP = = Op - > Opcode )
{
2005-06-29 21:02:02 +04:00
ResDesc = WalkState - > Operands [ 3 ] ;
CntDesc = WalkState - > Operands [ 2 ] ;
2005-06-29 21:01:47 +04:00
}
2005-06-29 21:02:02 +04:00
else
2005-06-29 21:01:47 +04:00
{
2005-06-29 21:02:02 +04:00
ResDesc = WalkState - > Operands [ 2 ] ;
2005-06-29 21:01:47 +04:00
}
2005-06-29 21:02:02 +04:00
OffDesc = WalkState - > Operands [ 1 ] ;
SrcDesc = WalkState - > Operands [ 0 ] ;
2005-06-29 21:01:47 +04:00
2005-06-29 21:02:02 +04:00
Offset = ( UINT32 ) OffDesc - > Integer . Value ;
2005-06-29 21:01:47 +04:00
/*
* If ResDesc is a Name , it will be a direct name pointer after
2005-06-29 21:02:02 +04:00
* AcpiExResolveOperands ( )
2005-06-29 21:01:47 +04:00
*/
if ( ! VALID_DESCRIPTOR_TYPE ( ResDesc , ACPI_DESC_TYPE_NAMED ) )
{
2005-06-29 21:02:02 +04:00
ACPI_DEBUG_PRINT ( ( ACPI_DB_ERROR , " (%s) destination must be a Node \n " ,
2005-06-29 21:01:47 +04:00
AcpiPsGetOpcodeName ( Op - > Opcode ) ) ) ;
Status = AE_AML_OPERAND_TYPE ;
goto Cleanup ;
}
/*
* Setup the Bit offsets and counts , according to the opcode
*/
switch ( Op - > Opcode )
{
2005-06-29 21:02:02 +04:00
/* DefCreateField */
case AML_CREATE_FIELD_OP :
/* Offset is in bits, count is in bits */
BitOffset = Offset ;
BitCount = ( UINT32 ) CntDesc - > Integer . Value ;
FieldFlags = ACCESS_BYTE_ACC ;
break ;
2005-06-29 21:01:47 +04:00
/* DefCreateBitField */
2005-06-29 21:02:02 +04:00
case AML_CREATE_BIT_FIELD_OP :
2005-06-29 21:01:47 +04:00
2005-06-29 21:02:02 +04:00
/* Offset is in bits, Field is one bit */
2005-06-29 21:01:47 +04:00
2005-06-29 21:02:02 +04:00
BitOffset = Offset ;
BitCount = 1 ;
FieldFlags = ACCESS_BYTE_ACC ;
2005-06-29 21:01:47 +04:00
break ;
/* DefCreateByteField */
2005-06-29 21:02:02 +04:00
case AML_CREATE_BYTE_FIELD_OP :
2005-06-29 21:01:47 +04:00
2005-06-29 21:02:02 +04:00
/* Offset is in bytes, field is one byte */
2005-06-29 21:01:47 +04:00
2005-06-29 21:02:02 +04:00
BitOffset = 8 * Offset ;
BitCount = 8 ;
FieldFlags = ACCESS_BYTE_ACC ;
2005-06-29 21:01:47 +04:00
break ;
/* DefCreateWordField */
2005-06-29 21:02:02 +04:00
case AML_CREATE_WORD_FIELD_OP :
2005-06-29 21:01:47 +04:00
2005-06-29 21:02:02 +04:00
/* Offset is in bytes, field is one word */
2005-06-29 21:01:47 +04:00
2005-06-29 21:02:02 +04:00
BitOffset = 8 * Offset ;
BitCount = 16 ;
FieldFlags = ACCESS_WORD_ACC ;
2005-06-29 21:01:47 +04:00
break ;
/* DefCreateDWordField */
2005-06-29 21:02:02 +04:00
case AML_CREATE_DWORD_FIELD_OP :
2005-06-29 21:01:47 +04:00
2005-06-29 21:02:02 +04:00
/* Offset is in bytes, field is one dword */
2005-06-29 21:01:47 +04:00
2005-06-29 21:02:02 +04:00
BitOffset = 8 * Offset ;
BitCount = 32 ;
FieldFlags = ACCESS_DWORD_ACC ;
2005-06-29 21:01:47 +04:00
break ;
2005-06-29 21:02:02 +04:00
/* DefCreateQWordField */
2005-06-29 21:01:47 +04:00
2005-06-29 21:02:02 +04:00
case AML_CREATE_QWORD_FIELD_OP :
2005-06-29 21:01:47 +04:00
2005-06-29 21:02:02 +04:00
/* Offset is in bytes, field is one qword */
2005-06-29 21:01:47 +04:00
2005-06-29 21:02:02 +04:00
BitOffset = 8 * Offset ;
BitCount = 64 ;
FieldFlags = ACCESS_QWORD_ACC ;
2005-06-29 21:01:47 +04:00
break ;
default :
2005-06-29 21:02:02 +04:00
ACPI_DEBUG_PRINT ( ( ACPI_DB_ERROR ,
" Internal error - unknown field creation opcode %02x \n " ,
2005-06-29 21:01:47 +04:00
Op - > Opcode ) ) ;
Status = AE_AML_BAD_OPCODE ;
goto Cleanup ;
}
/*
* Setup field according to the object type
*/
switch ( SrcDesc - > Common . Type )
{
/* SourceBuff := TermArg=>Buffer */
case ACPI_TYPE_BUFFER :
2005-06-29 21:02:02 +04:00
if ( ( BitOffset + BitCount ) >
2005-06-29 21:01:47 +04:00
( 8 * ( UINT32 ) SrcDesc - > Buffer . Length ) )
{
2005-06-29 21:02:02 +04:00
ACPI_DEBUG_PRINT ( ( ACPI_DB_ERROR ,
" Field size %d exceeds Buffer size %d (bits) \n " ,
BitOffset + BitCount , 8 * ( UINT32 ) SrcDesc - > Buffer . Length ) ) ;
2005-06-29 21:01:47 +04:00
Status = AE_AML_BUFFER_LIMIT ;
goto Cleanup ;
}
2005-06-29 21:02:02 +04:00
/*
* Initialize areas of the field object that are common to all fields
* For FieldFlags , use LOCK_RULE = 0 ( NO_LOCK ) , UPDATE_RULE = 0 ( UPDATE_PRESERVE )
*/
Status = AcpiExPrepCommonFieldObject ( ObjDesc , FieldFlags ,
BitOffset , BitCount ) ;
if ( ACPI_FAILURE ( Status ) )
{
return_ACPI_STATUS ( Status ) ;
}
2005-06-29 21:01:47 +04:00
2005-06-29 21:02:02 +04:00
ObjDesc - > BufferField . BufferObj = SrcDesc ;
2005-06-29 21:01:47 +04:00
2005-06-29 21:02:02 +04:00
/* Reference count for SrcDesc inherits ObjDesc count */
2005-06-29 21:01:47 +04:00
SrcDesc - > Common . ReferenceCount = ( UINT16 ) ( SrcDesc - > Common . ReferenceCount +
2005-06-29 21:02:02 +04:00
ObjDesc - > Common . ReferenceCount ) ;
2005-06-29 21:01:47 +04:00
break ;
/* Improper object type */
default :
2005-06-29 21:02:02 +04:00
if ( ( SrcDesc - > Common . Type > ( UINT8 ) INTERNAL_TYPE_REFERENCE ) | | ! AcpiUtValidObjectType ( SrcDesc - > Common . Type ) ) /* TBD: This line MUST be a single line until AcpiSrc can handle it (block deletion) */
2005-06-29 21:01:47 +04:00
{
2005-06-29 21:02:02 +04:00
ACPI_DEBUG_PRINT ( ( ACPI_DB_ERROR ,
" Tried to create field in invalid object type %X \n " ,
2005-06-29 21:01:47 +04:00
SrcDesc - > Common . Type ) ) ;
}
else
{
2005-06-29 21:02:02 +04:00
ACPI_DEBUG_PRINT ( ( ACPI_DB_ERROR ,
" Tried to create field in improper object type - %s \n " ,
AcpiUtGetTypeName ( SrcDesc - > Common . Type ) ) ) ;
2005-06-29 21:01:47 +04:00
}
Status = AE_AML_OPERAND_TYPE ;
goto Cleanup ;
}
if ( AML_CREATE_FIELD_OP = = Op - > Opcode )
{
/* Delete object descriptor unique to CreateField */
2005-06-29 21:02:02 +04:00
AcpiUtRemoveReference ( CntDesc ) ;
2005-06-29 21:01:47 +04:00
CntDesc = NULL ;
}
Cleanup :
/* Always delete the operands */
2005-06-29 21:02:02 +04:00
AcpiUtRemoveReference ( OffDesc ) ;
AcpiUtRemoveReference ( SrcDesc ) ;
2005-06-29 21:01:47 +04:00
if ( AML_CREATE_FIELD_OP = = Op - > Opcode )
{
2005-06-29 21:02:02 +04:00
AcpiUtRemoveReference ( CntDesc ) ;
2005-06-29 21:01:47 +04:00
}
/* On failure, delete the result descriptor */
if ( ACPI_FAILURE ( Status ) )
{
2005-06-29 21:02:02 +04:00
AcpiUtRemoveReference ( ResDesc ) ; /* Result descriptor */
2005-06-29 21:01:47 +04:00
}
else
{
2005-06-29 21:02:02 +04:00
/* Now the address and length are valid for this BufferField */
2005-06-29 21:01:47 +04:00
2005-06-29 21:02:02 +04:00
ObjDesc - > BufferField . Flags | = AOPOBJ_DATA_VALID ;
2005-06-29 21:01:47 +04:00
}
return_ACPI_STATUS ( Status ) ;
}
2005-06-29 21:01:41 +04:00
/*****************************************************************************
2005-06-29 20:55:00 +04:00
*
* FUNCTION : AcpiDsEvalRegionOperands
*
2005-06-29 21:01:41 +04:00
* PARAMETERS : Op - A valid region Op object
2005-06-29 20:55:00 +04:00
*
* RETURN : Status
*
* DESCRIPTION : Get region address and length
* Called from AcpiDsExecEndOp during OpRegion parse tree walk
*
2005-06-29 21:01:41 +04:00
* * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * */
2005-06-29 20:55:00 +04:00
ACPI_STATUS
AcpiDsEvalRegionOperands (
ACPI_WALK_STATE * WalkState ,
2005-06-29 21:01:47 +04:00
ACPI_PARSE_OBJECT * Op )
2005-06-29 20:55:00 +04:00
{
ACPI_STATUS Status ;
2005-06-29 21:01:47 +04:00
ACPI_OPERAND_OBJECT * ObjDesc ;
ACPI_OPERAND_OBJECT * OperandDesc ;
ACPI_NAMESPACE_NODE * Node ;
ACPI_PARSE_OBJECT * NextOp ;
2005-06-29 20:55:00 +04:00
2005-06-29 21:01:41 +04:00
FUNCTION_TRACE_PTR ( " DsEvalRegionOperands " , Op ) ;
2005-06-29 20:55:00 +04:00
/*
2005-06-29 21:01:41 +04:00
* This is where we evaluate the address and length fields of the OpRegion declaration
2005-06-29 20:55:00 +04:00
*/
2005-06-29 21:01:47 +04:00
Node = Op - > Node ;
2005-06-29 20:59:45 +04:00
2005-06-29 21:01:41 +04:00
/* NextOp points to the op that holds the SpaceID */
2005-06-29 21:02:02 +04:00
2005-06-29 21:01:41 +04:00
NextOp = Op - > Value . Arg ;
2005-06-29 20:59:20 +04:00
2005-06-29 20:59:16 +04:00
/* NextOp points to address op */
2005-06-29 21:02:02 +04:00
2005-06-29 21:01:41 +04:00
NextOp = NextOp - > Next ;
2005-06-29 20:59:45 +04:00
2005-06-29 21:01:41 +04:00
/* AcpiEvaluate/create the address and length operands */
2005-06-29 20:55:00 +04:00
Status = AcpiDsCreateOperands ( WalkState , NextOp ) ;
if ( ACPI_FAILURE ( Status ) )
{
return_ACPI_STATUS ( Status ) ;
}
2005-06-29 21:01:47 +04:00
/* Resolve the length and address operands to numbers */
2005-06-29 21:02:02 +04:00
Status = AcpiExResolveOperands ( Op - > Opcode , WALK_OPERANDS , WalkState ) ;
2005-06-29 21:01:47 +04:00
if ( ACPI_FAILURE ( Status ) )
2005-06-29 20:59:30 +04:00
{
2005-06-29 21:01:47 +04:00
return_ACPI_STATUS ( Status ) ;
2005-06-29 20:59:30 +04:00
}
2005-06-29 20:55:00 +04:00
2005-06-29 21:01:47 +04:00
DUMP_OPERANDS ( WALK_OPERANDS , IMODE_EXECUTE ,
AcpiPsGetOpcodeName ( Op - > Opcode ) ,
2005-06-29 21:02:02 +04:00
1 , " after AcpiExResolveOperands " ) ;
2005-06-29 21:01:47 +04:00
2005-06-29 21:00:05 +04:00
2005-06-29 21:01:47 +04:00
ObjDesc = AcpiNsGetAttachedObject ( Node ) ;
if ( ! ObjDesc )
{
return_ACPI_STATUS ( AE_NOT_EXIST ) ;
}
2005-06-29 21:00:05 +04:00
2005-06-29 21:01:47 +04:00
/*
* Get the length operand and save it
* ( at Top of stack )
*/
OperandDesc = WalkState - > Operands [ WalkState - > NumOperands - 1 ] ;
2005-06-29 21:00:05 +04:00
2005-06-29 21:02:02 +04:00
ObjDesc - > Region . Length = ( UINT32 ) OperandDesc - > Integer . Value ;
AcpiUtRemoveReference ( OperandDesc ) ;
2005-06-29 21:00:05 +04:00
2005-06-29 21:01:47 +04:00
/*
* Get the address and save it
* ( at top of stack - 1 )
*/
OperandDesc = WalkState - > Operands [ WalkState - > NumOperands - 2 ] ;
2005-06-29 21:00:10 +04:00
2005-06-29 21:02:02 +04:00
ObjDesc - > Region . Address = ( ACPI_PHYSICAL_ADDRESS ) OperandDesc - > Integer . Value ;
AcpiUtRemoveReference ( OperandDesc ) ;
2005-06-29 21:00:05 +04:00
2005-06-29 21:02:05 +04:00
ACPI_DEBUG_PRINT ( ( ACPI_DB_EXEC , " RgnObj %p Addr %8.8X%8.8X Len %X \n " ,
2005-06-29 21:02:02 +04:00
ObjDesc , HIDWORD ( ObjDesc - > Region . Address ) , LODWORD ( ObjDesc - > Region . Address ) ,
ObjDesc - > Region . Length ) ) ;
2005-06-29 21:00:05 +04:00
2005-06-29 21:01:41 +04:00
/* Now the address and length are valid for this opregion */
2005-06-29 21:00:05 +04:00
2005-06-29 21:01:47 +04:00
ObjDesc - > Region . Flags | = AOPOBJ_DATA_VALID ;
2005-06-29 21:00:05 +04:00
return_ACPI_STATUS ( Status ) ;
}
2005-06-29 20:55:00 +04:00
/*******************************************************************************
*
* FUNCTION : AcpiDsExecBeginControlOp
*
* PARAMETERS : WalkList - The list that owns the walk stack
* Op - The control Op
*
* RETURN : Status
*
* DESCRIPTION : Handles all control ops encountered during control method
* execution .
*
* * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * */
ACPI_STATUS
AcpiDsExecBeginControlOp (
ACPI_WALK_STATE * WalkState ,
2005-06-29 21:01:47 +04:00
ACPI_PARSE_OBJECT * Op )
2005-06-29 20:55:00 +04:00
{
ACPI_STATUS Status = AE_OK ;
ACPI_GENERIC_STATE * ControlState ;
2005-06-29 21:02:02 +04:00
PROC_NAME ( " DsExecBeginControlOp " ) ;
ACPI_DEBUG_PRINT ( ( ACPI_DB_DISPATCH , " Op=%p Opcode=%2.2X State=%p \n " , Op ,
2005-06-29 21:01:41 +04:00
Op - > Opcode , WalkState ) ) ;
2005-06-29 20:55:48 +04:00
2005-06-29 21:01:41 +04:00
switch ( Op - > Opcode )
2005-06-29 20:55:00 +04:00
{
case AML_IF_OP :
case AML_WHILE_OP :
/*
2005-06-29 20:59:16 +04:00
* IF / WHILE : Create a new control state to manage these
* constructs . We need to manage these as a stack , in order
* to handle nesting .
2005-06-29 20:55:00 +04:00
*/
2005-06-29 21:02:02 +04:00
ControlState = AcpiUtCreateControlState ( ) ;
2005-06-29 20:55:00 +04:00
if ( ! ControlState )
{
Status = AE_NO_MEMORY ;
break ;
}
2005-06-29 21:01:41 +04:00
2005-06-29 21:02:02 +04:00
AcpiUtPushGenericState ( & WalkState - > ControlState , ControlState ) ;
2005-06-29 21:01:41 +04:00
2005-06-29 21:01:47 +04:00
/*
2005-06-29 20:59:20 +04:00
* Save a pointer to the predicate for multiple executions
* of a loop
*/
2005-06-29 21:01:47 +04:00
WalkState - > ControlState - > Control . AmlPredicateStart =
2005-06-29 21:02:02 +04:00
WalkState - > ParserState . Aml - 1 ;
2005-06-29 21:01:47 +04:00
/* TBD: can this be removed? */
2005-06-29 21:01:41 +04:00
/*AcpiPsPkgLengthEncodingSize (GET8 (WalkState->ParserState->Aml));*/
2005-06-29 21:00:02 +04:00
break ;
2005-06-29 21:01:41 +04:00
2005-06-29 20:55:00 +04:00
case AML_ELSE_OP :
/* Predicate is in the state object */
/* If predicate is true, the IF was executed, ignore ELSE part */
if ( WalkState - > LastPredicate )
{
Status = AE_CTRL_TRUE ;
}
break ;
2005-06-29 21:01:41 +04:00
2005-06-29 20:55:00 +04:00
case AML_RETURN_OP :
break ;
2005-06-29 21:01:41 +04:00
2005-06-29 20:55:00 +04:00
default :
break ;
}
2005-06-29 20:59:20 +04:00
return ( Status ) ;
2005-06-29 20:55:00 +04:00
}
/*******************************************************************************
*
* FUNCTION : AcpiDsExecEndControlOp
*
* PARAMETERS : WalkList - The list that owns the walk stack
* Op - The control Op
*
* RETURN : Status
*
* DESCRIPTION : Handles all control ops encountered during control method
* execution .
*
2005-06-29 21:01:41 +04:00
*
2005-06-29 20:55:00 +04:00
* * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * */
ACPI_STATUS
AcpiDsExecEndControlOp (
ACPI_WALK_STATE * WalkState ,
2005-06-29 21:01:47 +04:00
ACPI_PARSE_OBJECT * Op )
2005-06-29 20:55:00 +04:00
{
ACPI_STATUS Status = AE_OK ;
ACPI_GENERIC_STATE * ControlState ;
2005-06-29 21:02:02 +04:00
PROC_NAME ( " DsExecEndControlOp " ) ;
2005-06-29 21:01:41 +04:00
switch ( Op - > Opcode )
2005-06-29 20:55:00 +04:00
{
case AML_IF_OP :
2005-06-29 21:02:02 +04:00
ACPI_DEBUG_PRINT ( ( ACPI_DB_DISPATCH , " [IF_OP] Op=%p \n " , Op ) ) ;
2005-06-29 20:55:03 +04:00
2005-06-29 20:59:16 +04:00
/*
* Save the result of the predicate in case there is an
* ELSE to come
*/
2005-06-29 21:00:02 +04:00
WalkState - > LastPredicate =
2005-06-29 21:02:02 +04:00
( BOOLEAN ) WalkState - > ControlState - > Common . Value ;
2005-06-29 20:59:14 +04:00
2005-06-29 20:59:16 +04:00
/*
* Pop the control state that was created at the start
* of the IF and free it
*/
2005-06-29 21:02:02 +04:00
ControlState = AcpiUtPopGenericState ( & WalkState - > ControlState ) ;
AcpiUtDeleteGenericState ( ControlState ) ;
2005-06-29 20:55:00 +04:00
break ;
case AML_ELSE_OP :
break ;
case AML_WHILE_OP :
2005-06-29 21:02:02 +04:00
ACPI_DEBUG_PRINT ( ( ACPI_DB_DISPATCH , " [WHILE_OP] Op=%p \n " , Op ) ) ;
2005-06-29 20:55:00 +04:00
if ( WalkState - > ControlState - > Common . Value )
{
/* Predicate was true, go back and evaluate it again! */
2005-06-29 20:59:20 +04:00
Status = AE_CTRL_PENDING ;
2005-06-29 20:55:00 +04:00
}
2005-06-29 21:02:02 +04:00
ACPI_DEBUG_PRINT ( ( ACPI_DB_DISPATCH , " [WHILE_OP] termination! Op=%p \n " , Op ) ) ;
2005-06-29 21:01:47 +04:00
/* Pop this control state and free it */
2005-06-29 20:55:39 +04:00
2005-06-29 21:02:02 +04:00
ControlState = AcpiUtPopGenericState ( & WalkState - > ControlState ) ;
2005-06-29 21:01:41 +04:00
2005-06-29 21:01:47 +04:00
WalkState - > AmlLastWhile = ControlState - > Control . AmlPredicateStart ;
2005-06-29 21:02:02 +04:00
AcpiUtDeleteGenericState ( ControlState ) ;
2005-06-29 20:55:00 +04:00
break ;
case AML_RETURN_OP :
2005-06-29 21:02:02 +04:00
ACPI_DEBUG_PRINT ( ( ACPI_DB_DISPATCH ,
" [RETURN_OP] Op=%p Arg=%p \n " , Op , Op - > Value . Arg ) ) ;
2005-06-29 21:01:41 +04:00
2005-06-29 21:01:47 +04:00
/*
2005-06-29 20:59:20 +04:00
* One optional operand - - the return value
* It can be either an immediate operand or a result that
* has been bubbled up the tree
*/
2005-06-29 21:01:41 +04:00
if ( Op - > Value . Arg )
2005-06-29 20:59:14 +04:00
{
2005-06-29 20:59:20 +04:00
/* Return statement has an immediate operand */
2005-06-29 21:01:41 +04:00
Status = AcpiDsCreateOperands ( WalkState , Op - > Value . Arg ) ;
2005-06-29 20:55:00 +04:00
if ( ACPI_FAILURE ( Status ) )
{
2005-06-29 20:59:20 +04:00
return ( Status ) ;
2005-06-29 20:55:00 +04:00
}
/*
2005-06-29 20:59:16 +04:00
* If value being returned is a Reference ( such as
* an arg or local ) , resolve it now because it may
* cease to exist at the end of the method .
2005-06-29 20:55:00 +04:00
*/
2005-06-29 21:02:02 +04:00
Status = AcpiExResolveToValue ( & WalkState - > Operands [ 0 ] , WalkState ) ;
2005-06-29 20:55:00 +04:00
if ( ACPI_FAILURE ( Status ) )
{
2005-06-29 20:59:20 +04:00
return ( Status ) ;
2005-06-29 20:55:00 +04:00
}
/*
2005-06-29 20:59:16 +04:00
* Get the return value and save as the last result
2005-06-29 20:59:20 +04:00
* value . This is the only place where WalkState - > ReturnDesc
2005-06-29 20:59:16 +04:00
* is set to anything other than zero !
2005-06-29 20:55:00 +04:00
*/
WalkState - > ReturnDesc = WalkState - > Operands [ 0 ] ;
}
2005-06-29 21:00:30 +04:00
2005-06-29 21:01:47 +04:00
else if ( ( WalkState - > Results ) & &
( WalkState - > Results - > Results . NumResults > 0 ) )
2005-06-29 21:01:41 +04:00
{
2005-06-29 20:59:20 +04:00
/*
* The return value has come from a previous calculation .
2005-06-29 21:01:47 +04:00
*
2005-06-29 20:59:20 +04:00
* If value being returned is a Reference ( such as
* an arg or local ) , resolve it now because it may
* cease to exist at the end of the method .
2005-06-29 21:02:02 +04:00
*
* Allow references created by the Index operator to return unchanged .
2005-06-29 20:59:20 +04:00
*/
2005-06-29 21:02:02 +04:00
if ( VALID_DESCRIPTOR_TYPE ( WalkState - > Results - > Results . ObjDesc [ 0 ] , ACPI_DESC_TYPE_INTERNAL ) & &
( ( WalkState - > Results - > Results . ObjDesc [ 0 ] ) - > Common . Type = = INTERNAL_TYPE_REFERENCE ) & &
( ( WalkState - > Results - > Results . ObjDesc [ 0 ] ) - > Reference . Opcode ! = AML_INDEX_OP ) )
2005-06-29 20:59:20 +04:00
{
2005-06-29 21:02:02 +04:00
Status = AcpiExResolveToValue ( & WalkState - > Results - > Results . ObjDesc [ 0 ] , WalkState ) ;
if ( ACPI_FAILURE ( Status ) )
{
return ( Status ) ;
}
2005-06-29 20:59:20 +04:00
}
2005-06-29 21:01:47 +04:00
WalkState - > ReturnDesc = WalkState - > Results - > Results . ObjDesc [ 0 ] ;
2005-06-29 20:59:20 +04:00
}
2005-06-29 21:01:41 +04:00
2005-06-29 20:55:00 +04:00
else
{
/* No return operand */
2005-06-29 20:59:20 +04:00
if ( WalkState - > NumOperands )
{
2005-06-29 21:02:02 +04:00
AcpiUtRemoveReference ( WalkState - > Operands [ 0 ] ) ;
2005-06-29 20:59:20 +04:00
}
2005-06-29 20:55:00 +04:00
WalkState - > Operands [ 0 ] = NULL ;
WalkState - > NumOperands = 0 ;
WalkState - > ReturnDesc = NULL ;
}
2005-06-29 21:02:02 +04:00
ACPI_DEBUG_PRINT ( ( ACPI_DB_DISPATCH ,
" Completed RETURN_OP State=%p, RetVal=%p \n " ,
2005-06-29 20:59:16 +04:00
WalkState , WalkState - > ReturnDesc ) ) ;
2005-06-29 20:55:00 +04:00
2005-06-29 20:59:16 +04:00
/* End the control method execution right now */
2005-06-29 21:02:02 +04:00
2005-06-29 20:59:16 +04:00
Status = AE_CTRL_TERMINATE ;
2005-06-29 20:55:00 +04:00
break ;
2005-06-29 20:59:20 +04:00
case AML_NOOP_OP :
2005-06-29 20:55:00 +04:00
/* Just do nothing! */
break ;
case AML_BREAK_POINT_OP :
2005-06-29 21:02:02 +04:00
/* Call up to the OS service layer to handle this */
2005-06-29 20:55:00 +04:00
2005-06-29 21:02:02 +04:00
AcpiOsSignal ( ACPI_SIGNAL_BREAKPOINT , " Executed AML Breakpoint opcode " ) ;
2005-06-29 20:55:00 +04:00
2005-06-29 21:02:02 +04:00
/* If and when it returns, all done. */
2005-06-29 20:55:00 +04:00
break ;
case AML_BREAK_OP :
2005-06-29 20:55:48 +04:00
2005-06-29 21:02:02 +04:00
ACPI_DEBUG_PRINT ( ( ACPI_DB_INFO ,
" Break to end of current package, Op=%p \n " , Op ) ) ;
/* TBD: update behavior for ACPI 2.0 */
2005-06-29 20:55:48 +04:00
2005-06-29 21:01:41 +04:00
/*
* As per the ACPI specification :
* " The break operation causes the current package
* execution to complete "
* " Break -- Stop executing the current code package
* at this point "
*
* Returning AE_FALSE here will cause termination of
* the current package , and execution will continue one
* level up , starting with the completion of the parent Op .
*/
Status = AE_CTRL_FALSE ;
2005-06-29 20:59:45 +04:00
break ;
2005-06-29 21:02:02 +04:00
case AML_CONTINUE_OP : /* ACPI 2.0 */
Status = AE_NOT_IMPLEMENTED ;
break ;
2005-06-29 20:55:00 +04:00
default :
2005-06-29 21:02:02 +04:00
ACPI_DEBUG_PRINT ( ( ACPI_DB_ERROR , " Unknown control opcode=%X Op=%p \n " ,
2005-06-29 21:01:41 +04:00
Op - > Opcode , Op ) ) ;
2005-06-29 20:55:00 +04:00
Status = AE_AML_BAD_OPCODE ;
break ;
}
2005-06-29 21:01:41 +04:00
2005-06-29 20:59:20 +04:00
return ( Status ) ;
2005-06-29 20:55:00 +04:00
}