1 /****************************************************************************** 2 * 3 * Module Name: dsobject - Dispatcher object management routines 4 * 5 *****************************************************************************/ 6 7 /* 8 * Copyright (C) 2000 - 2022, 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 MERCHANTABILITY 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 "acpi.h" 45 #include "accommon.h" 46 #include "acparser.h" 47 #include "amlcode.h" 48 #include "acdispat.h" 49 #include "acnamesp.h" 50 #include "acinterp.h" 51 52 #define _COMPONENT ACPI_DISPATCHER 53 ACPI_MODULE_NAME ("dsobject") 54 55 56 /******************************************************************************* 57 * 58 * FUNCTION: AcpiDsBuildInternalObject 59 * 60 * PARAMETERS: WalkState - Current walk state 61 * Op - Parser object to be translated 62 * ObjDescPtr - Where the ACPI internal object is returned 63 * 64 * RETURN: Status 65 * 66 * DESCRIPTION: Translate a parser Op object to the equivalent namespace object 67 * Simple objects are any objects other than a package object! 68 * 69 ******************************************************************************/ 70 71 ACPI_STATUS 72 AcpiDsBuildInternalObject ( 73 ACPI_WALK_STATE *WalkState, 74 ACPI_PARSE_OBJECT *Op, 75 ACPI_OPERAND_OBJECT **ObjDescPtr) 76 { 77 ACPI_OPERAND_OBJECT *ObjDesc; 78 ACPI_STATUS Status; 79 80 81 ACPI_FUNCTION_TRACE (DsBuildInternalObject); 82 83 84 *ObjDescPtr = NULL; 85 if (Op->Common.AmlOpcode == AML_INT_NAMEPATH_OP) 86 { 87 /* 88 * This is a named object reference. If this name was 89 * previously looked up in the namespace, it was stored in 90 * this op. Otherwise, go ahead and look it up now 91 */ 92 if (!Op->Common.Node) 93 { 94 /* Check if we are resolving a named reference within a package */ 95 96 if ((Op->Common.Parent->Common.AmlOpcode == AML_PACKAGE_OP) || 97 (Op->Common.Parent->Common.AmlOpcode == AML_VARIABLE_PACKAGE_OP)) 98 { 99 /* 100 * We won't resolve package elements here, we will do this 101 * after all ACPI tables are loaded into the namespace. This 102 * behavior supports both forward references to named objects 103 * and external references to objects in other tables. 104 */ 105 goto CreateNewObject; 106 } 107 else 108 { 109 Status = AcpiNsLookup (WalkState->ScopeInfo, 110 Op->Common.Value.String, 111 ACPI_TYPE_ANY, ACPI_IMODE_EXECUTE, 112 ACPI_NS_SEARCH_PARENT | ACPI_NS_DONT_OPEN_SCOPE, NULL, 113 ACPI_CAST_INDIRECT_PTR ( 114 ACPI_NAMESPACE_NODE, &(Op->Common.Node))); 115 if (ACPI_FAILURE (Status)) 116 { 117 ACPI_ERROR_NAMESPACE (WalkState->ScopeInfo, 118 Op->Common.Value.String, Status); 119 return_ACPI_STATUS (Status); 120 } 121 } 122 } 123 } 124 125 CreateNewObject: 126 127 /* Create and init a new internal ACPI object */ 128 129 ObjDesc = AcpiUtCreateInternalObject ( 130 (AcpiPsGetOpcodeInfo (Op->Common.AmlOpcode))->ObjectType); 131 if (!ObjDesc) 132 { 133 return_ACPI_STATUS (AE_NO_MEMORY); 134 } 135 136 Status = AcpiDsInitObjectFromOp ( 137 WalkState, Op, Op->Common.AmlOpcode, &ObjDesc); 138 if (ACPI_FAILURE (Status)) 139 { 140 AcpiUtRemoveReference (ObjDesc); 141 return_ACPI_STATUS (Status); 142 } 143 144 /* 145 * Handling for unresolved package reference elements. 146 * These are elements that are namepaths. 147 */ 148 if ((Op->Common.Parent->Common.AmlOpcode == AML_PACKAGE_OP) || 149 (Op->Common.Parent->Common.AmlOpcode == AML_VARIABLE_PACKAGE_OP)) 150 { 151 ObjDesc->Reference.Resolved = TRUE; 152 153 if ((Op->Common.AmlOpcode == AML_INT_NAMEPATH_OP) && 154 !ObjDesc->Reference.Node) 155 { 156 /* 157 * Name was unresolved above. 158 * Get the prefix node for later lookup 159 */ 160 ObjDesc->Reference.Node = WalkState->ScopeInfo->Scope.Node; 161 ObjDesc->Reference.Aml = Op->Common.Aml; 162 ObjDesc->Reference.Resolved = FALSE; 163 } 164 } 165 166 *ObjDescPtr = ObjDesc; 167 return_ACPI_STATUS (Status); 168 } 169 170 171 /******************************************************************************* 172 * 173 * FUNCTION: AcpiDsBuildInternalBufferObj 174 * 175 * PARAMETERS: WalkState - Current walk state 176 * Op - Parser object to be translated 177 * BufferLength - Length of the buffer 178 * ObjDescPtr - Where the ACPI internal object is returned 179 * 180 * RETURN: Status 181 * 182 * DESCRIPTION: Translate a parser Op package object to the equivalent 183 * namespace object 184 * 185 ******************************************************************************/ 186 187 ACPI_STATUS 188 AcpiDsBuildInternalBufferObj ( 189 ACPI_WALK_STATE *WalkState, 190 ACPI_PARSE_OBJECT *Op, 191 UINT32 BufferLength, 192 ACPI_OPERAND_OBJECT **ObjDescPtr) 193 { 194 ACPI_PARSE_OBJECT *Arg; 195 ACPI_OPERAND_OBJECT *ObjDesc; 196 ACPI_PARSE_OBJECT *ByteList; 197 UINT32 ByteListLength = 0; 198 199 200 ACPI_FUNCTION_TRACE (DsBuildInternalBufferObj); 201 202 203 /* 204 * If we are evaluating a Named buffer object "Name (xxxx, Buffer)". 205 * The buffer object already exists (from the NS node), otherwise it must 206 * be created. 207 */ 208 ObjDesc = *ObjDescPtr; 209 if (!ObjDesc) 210 { 211 /* Create a new buffer object */ 212 213 ObjDesc = AcpiUtCreateInternalObject (ACPI_TYPE_BUFFER); 214 *ObjDescPtr = ObjDesc; 215 if (!ObjDesc) 216 { 217 return_ACPI_STATUS (AE_NO_MEMORY); 218 } 219 } 220 221 /* 222 * Second arg is the buffer data (optional) ByteList can be either 223 * individual bytes or a string initializer. In either case, a 224 * ByteList appears in the AML. 225 */ 226 Arg = Op->Common.Value.Arg; /* skip first arg */ 227 228 ByteList = Arg->Named.Next; 229 if (ByteList) 230 { 231 if (ByteList->Common.AmlOpcode != AML_INT_BYTELIST_OP) 232 { 233 ACPI_ERROR ((AE_INFO, 234 "Expecting bytelist, found AML opcode 0x%X in op %p", 235 ByteList->Common.AmlOpcode, ByteList)); 236 237 AcpiUtRemoveReference (ObjDesc); 238 return (AE_TYPE); 239 } 240 241 ByteListLength = (UINT32) ByteList->Common.Value.Integer; 242 } 243 244 /* 245 * The buffer length (number of bytes) will be the larger of: 246 * 1) The specified buffer length and 247 * 2) The length of the initializer byte list 248 */ 249 ObjDesc->Buffer.Length = BufferLength; 250 if (ByteListLength > BufferLength) 251 { 252 ObjDesc->Buffer.Length = ByteListLength; 253 } 254 255 /* Allocate the buffer */ 256 257 if (ObjDesc->Buffer.Length == 0) 258 { 259 ObjDesc->Buffer.Pointer = NULL; 260 ACPI_DEBUG_PRINT ((ACPI_DB_EXEC, 261 "Buffer defined with zero length in AML, creating\n")); 262 } 263 else 264 { 265 ObjDesc->Buffer.Pointer = 266 ACPI_ALLOCATE_ZEROED (ObjDesc->Buffer.Length); 267 if (!ObjDesc->Buffer.Pointer) 268 { 269 AcpiUtDeleteObjectDesc (ObjDesc); 270 return_ACPI_STATUS (AE_NO_MEMORY); 271 } 272 273 /* Initialize buffer from the ByteList (if present) */ 274 275 if (ByteList) 276 { 277 memcpy (ObjDesc->Buffer.Pointer, ByteList->Named.Data, 278 ByteListLength); 279 } 280 } 281 282 ObjDesc->Buffer.Flags |= AOPOBJ_DATA_VALID; 283 Op->Common.Node = ACPI_CAST_PTR (ACPI_NAMESPACE_NODE, ObjDesc); 284 return_ACPI_STATUS (AE_OK); 285 } 286 287 /******************************************************************************* 288 * 289 * FUNCTION: AcpiDsCreateNode 290 * 291 * PARAMETERS: WalkState - Current walk state 292 * Node - NS Node to be initialized 293 * Op - Parser object to be translated 294 * 295 * RETURN: Status 296 * 297 * DESCRIPTION: Create the object to be associated with a namespace node 298 * 299 ******************************************************************************/ 300 301 ACPI_STATUS 302 AcpiDsCreateNode ( 303 ACPI_WALK_STATE *WalkState, 304 ACPI_NAMESPACE_NODE *Node, 305 ACPI_PARSE_OBJECT *Op) 306 { 307 ACPI_STATUS Status; 308 ACPI_OPERAND_OBJECT *ObjDesc; 309 310 311 ACPI_FUNCTION_TRACE_PTR (DsCreateNode, Op); 312 313 314 /* 315 * Because of the execution pass through the non-control-method 316 * parts of the table, we can arrive here twice. Only init 317 * the named object node the first time through 318 */ 319 if (AcpiNsGetAttachedObject (Node)) 320 { 321 return_ACPI_STATUS (AE_OK); 322 } 323 324 if (!Op->Common.Value.Arg) 325 { 326 /* No arguments, there is nothing to do */ 327 328 return_ACPI_STATUS (AE_OK); 329 } 330 331 /* Build an internal object for the argument(s) */ 332 333 Status = AcpiDsBuildInternalObject ( 334 WalkState, Op->Common.Value.Arg, &ObjDesc); 335 if (ACPI_FAILURE (Status)) 336 { 337 return_ACPI_STATUS (Status); 338 } 339 340 /* Re-type the object according to its argument */ 341 342 Node->Type = ObjDesc->Common.Type; 343 344 /* Attach obj to node */ 345 346 Status = AcpiNsAttachObject (Node, ObjDesc, Node->Type); 347 348 /* Remove local reference to the object */ 349 350 AcpiUtRemoveReference (ObjDesc); 351 return_ACPI_STATUS (Status); 352 } 353 354 355 /******************************************************************************* 356 * 357 * FUNCTION: AcpiDsInitObjectFromOp 358 * 359 * PARAMETERS: WalkState - Current walk state 360 * Op - Parser op used to init the internal object 361 * Opcode - AML opcode associated with the object 362 * RetObjDesc - Namespace object to be initialized 363 * 364 * RETURN: Status 365 * 366 * DESCRIPTION: Initialize a namespace object from a parser Op and its 367 * associated arguments. The namespace object is a more compact 368 * representation of the Op and its arguments. 369 * 370 ******************************************************************************/ 371 372 ACPI_STATUS 373 AcpiDsInitObjectFromOp ( 374 ACPI_WALK_STATE *WalkState, 375 ACPI_PARSE_OBJECT *Op, 376 UINT16 Opcode, 377 ACPI_OPERAND_OBJECT **RetObjDesc) 378 { 379 const ACPI_OPCODE_INFO *OpInfo; 380 ACPI_OPERAND_OBJECT *ObjDesc; 381 ACPI_STATUS Status = AE_OK; 382 383 384 ACPI_FUNCTION_TRACE (DsInitObjectFromOp); 385 386 387 ObjDesc = *RetObjDesc; 388 OpInfo = AcpiPsGetOpcodeInfo (Opcode); 389 if (OpInfo->Class == AML_CLASS_UNKNOWN) 390 { 391 /* Unknown opcode */ 392 393 return_ACPI_STATUS (AE_TYPE); 394 } 395 396 /* Perform per-object initialization */ 397 398 switch (ObjDesc->Common.Type) 399 { 400 case ACPI_TYPE_BUFFER: 401 /* 402 * Defer evaluation of Buffer TermArg operand 403 */ 404 ObjDesc->Buffer.Node = ACPI_CAST_PTR ( 405 ACPI_NAMESPACE_NODE, WalkState->Operands[0]); 406 ObjDesc->Buffer.AmlStart = Op->Named.Data; 407 ObjDesc->Buffer.AmlLength = Op->Named.Length; 408 break; 409 410 case ACPI_TYPE_PACKAGE: 411 /* 412 * Defer evaluation of Package TermArg operand and all 413 * package elements. (01/2017): We defer the element 414 * resolution to allow forward references from the package 415 * in order to provide compatibility with other ACPI 416 * implementations. 417 */ 418 ObjDesc->Package.Node = ACPI_CAST_PTR ( 419 ACPI_NAMESPACE_NODE, WalkState->Operands[0]); 420 421 if (!Op->Named.Data) 422 { 423 return_ACPI_STATUS (AE_OK); 424 } 425 426 ObjDesc->Package.AmlStart = Op->Named.Data; 427 ObjDesc->Package.AmlLength = Op->Named.Length; 428 break; 429 430 case ACPI_TYPE_INTEGER: 431 432 switch (OpInfo->Type) 433 { 434 case AML_TYPE_CONSTANT: 435 /* 436 * Resolve AML Constants here - AND ONLY HERE! 437 * All constants are integers. 438 * We mark the integer with a flag that indicates that it started 439 * life as a constant -- so that stores to constants will perform 440 * as expected (noop). ZeroOp is used as a placeholder for optional 441 * target operands. 442 */ 443 ObjDesc->Common.Flags = AOPOBJ_AML_CONSTANT; 444 445 switch (Opcode) 446 { 447 case AML_ZERO_OP: 448 449 ObjDesc->Integer.Value = 0; 450 break; 451 452 case AML_ONE_OP: 453 454 ObjDesc->Integer.Value = 1; 455 break; 456 457 case AML_ONES_OP: 458 459 ObjDesc->Integer.Value = ACPI_UINT64_MAX; 460 461 /* Truncate value if we are executing from a 32-bit ACPI table */ 462 463 (void) AcpiExTruncateFor32bitTable (ObjDesc); 464 break; 465 466 case AML_REVISION_OP: 467 468 ObjDesc->Integer.Value = ACPI_CA_VERSION; 469 break; 470 471 default: 472 473 ACPI_ERROR ((AE_INFO, 474 "Unknown constant opcode 0x%X", Opcode)); 475 Status = AE_AML_OPERAND_TYPE; 476 break; 477 } 478 break; 479 480 case AML_TYPE_LITERAL: 481 482 ObjDesc->Integer.Value = Op->Common.Value.Integer; 483 484 if (AcpiExTruncateFor32bitTable (ObjDesc)) 485 { 486 /* Warn if we found a 64-bit constant in a 32-bit table */ 487 488 ACPI_WARNING ((AE_INFO, 489 "Truncated 64-bit constant found in 32-bit table: %8.8X%8.8X => %8.8X", 490 ACPI_FORMAT_UINT64 (Op->Common.Value.Integer), 491 (UINT32) ObjDesc->Integer.Value)); 492 } 493 break; 494 495 default: 496 497 ACPI_ERROR ((AE_INFO, "Unknown Integer type 0x%X", 498 OpInfo->Type)); 499 Status = AE_AML_OPERAND_TYPE; 500 break; 501 } 502 break; 503 504 case ACPI_TYPE_STRING: 505 506 ObjDesc->String.Pointer = Op->Common.Value.String; 507 ObjDesc->String.Length = (UINT32) strlen (Op->Common.Value.String); 508 509 /* 510 * The string is contained in the ACPI table, don't ever try 511 * to delete it 512 */ 513 ObjDesc->Common.Flags |= AOPOBJ_STATIC_POINTER; 514 break; 515 516 case ACPI_TYPE_METHOD: 517 break; 518 519 case ACPI_TYPE_LOCAL_REFERENCE: 520 521 switch (OpInfo->Type) 522 { 523 case AML_TYPE_LOCAL_VARIABLE: 524 525 /* Local ID (0-7) is (AML opcode - base AML_FIRST_LOCAL_OP) */ 526 527 ObjDesc->Reference.Value = ((UINT32) Opcode) - AML_FIRST_LOCAL_OP; 528 ObjDesc->Reference.Class = ACPI_REFCLASS_LOCAL; 529 530 Status = AcpiDsMethodDataGetNode (ACPI_REFCLASS_LOCAL, 531 ObjDesc->Reference.Value, WalkState, 532 ACPI_CAST_INDIRECT_PTR (ACPI_NAMESPACE_NODE, 533 &ObjDesc->Reference.Object)); 534 break; 535 536 case AML_TYPE_METHOD_ARGUMENT: 537 538 /* Arg ID (0-6) is (AML opcode - base AML_FIRST_ARG_OP) */ 539 540 ObjDesc->Reference.Value = ((UINT32) Opcode) - AML_FIRST_ARG_OP; 541 ObjDesc->Reference.Class = ACPI_REFCLASS_ARG; 542 543 Status = AcpiDsMethodDataGetNode (ACPI_REFCLASS_ARG, 544 ObjDesc->Reference.Value, WalkState, 545 ACPI_CAST_INDIRECT_PTR (ACPI_NAMESPACE_NODE, 546 &ObjDesc->Reference.Object)); 547 break; 548 549 default: /* Object name or Debug object */ 550 551 switch (Op->Common.AmlOpcode) 552 { 553 case AML_INT_NAMEPATH_OP: 554 555 /* Node was saved in Op */ 556 557 ObjDesc->Reference.Node = Op->Common.Node; 558 ObjDesc->Reference.Class = ACPI_REFCLASS_NAME; 559 if (Op->Common.Node) 560 { 561 ObjDesc->Reference.Object = Op->Common.Node->Object; 562 } 563 break; 564 565 case AML_DEBUG_OP: 566 567 ObjDesc->Reference.Class = ACPI_REFCLASS_DEBUG; 568 break; 569 570 default: 571 572 ACPI_ERROR ((AE_INFO, 573 "Unimplemented reference type for AML opcode: 0x%4.4X", Opcode)); 574 return_ACPI_STATUS (AE_AML_OPERAND_TYPE); 575 } 576 break; 577 } 578 break; 579 580 default: 581 582 ACPI_ERROR ((AE_INFO, "Unimplemented data type: 0x%X", 583 ObjDesc->Common.Type)); 584 585 Status = AE_AML_OPERAND_TYPE; 586 break; 587 } 588 589 return_ACPI_STATUS (Status); 590 } 591