1 /****************************************************************************** 2 * 3 * Module Name: dswload - Dispatcher first pass namespace load callbacks 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 "acinterp.h" 50 #include "acnamesp.h" 51 #ifdef ACPI_ASL_COMPILER 52 #include "acdisasm.h" 53 #endif 54 55 #define _COMPONENT ACPI_DISPATCHER 56 ACPI_MODULE_NAME ("dswload") 57 58 59 /******************************************************************************* 60 * 61 * FUNCTION: AcpiDsInitCallbacks 62 * 63 * PARAMETERS: WalkState - Current state of the parse tree walk 64 * PassNumber - 1, 2, or 3 65 * 66 * RETURN: Status 67 * 68 * DESCRIPTION: Init walk state callbacks 69 * 70 ******************************************************************************/ 71 72 ACPI_STATUS 73 AcpiDsInitCallbacks ( 74 ACPI_WALK_STATE *WalkState, 75 UINT32 PassNumber) 76 { 77 78 switch (PassNumber) 79 { 80 case 0: 81 82 /* Parse only - caller will setup callbacks */ 83 84 WalkState->ParseFlags = ACPI_PARSE_LOAD_PASS1 | 85 ACPI_PARSE_DELETE_TREE | 86 ACPI_PARSE_DISASSEMBLE; 87 WalkState->DescendingCallback = NULL; 88 WalkState->AscendingCallback = NULL; 89 break; 90 91 case 1: 92 93 /* Load pass 1 */ 94 95 WalkState->ParseFlags = ACPI_PARSE_LOAD_PASS1 | 96 ACPI_PARSE_DELETE_TREE; 97 WalkState->DescendingCallback = AcpiDsLoad1BeginOp; 98 WalkState->AscendingCallback = AcpiDsLoad1EndOp; 99 break; 100 101 case 2: 102 103 /* Load pass 2 */ 104 105 WalkState->ParseFlags = ACPI_PARSE_LOAD_PASS1 | 106 ACPI_PARSE_DELETE_TREE; 107 WalkState->DescendingCallback = AcpiDsLoad2BeginOp; 108 WalkState->AscendingCallback = AcpiDsLoad2EndOp; 109 break; 110 111 case 3: 112 113 /* Execution pass */ 114 115 WalkState->ParseFlags |= ACPI_PARSE_EXECUTE | 116 ACPI_PARSE_DELETE_TREE; 117 WalkState->DescendingCallback = AcpiDsExecBeginOp; 118 WalkState->AscendingCallback = AcpiDsExecEndOp; 119 break; 120 121 default: 122 123 return (AE_BAD_PARAMETER); 124 } 125 126 return (AE_OK); 127 } 128 129 130 /******************************************************************************* 131 * 132 * FUNCTION: AcpiDsLoad1BeginOp 133 * 134 * PARAMETERS: WalkState - Current state of the parse tree walk 135 * OutOp - Where to return op if a new one is created 136 * 137 * RETURN: Status 138 * 139 * DESCRIPTION: Descending callback used during the loading of ACPI tables. 140 * 141 ******************************************************************************/ 142 143 ACPI_STATUS 144 AcpiDsLoad1BeginOp ( 145 ACPI_WALK_STATE *WalkState, 146 ACPI_PARSE_OBJECT **OutOp) 147 { 148 ACPI_PARSE_OBJECT *Op; 149 ACPI_NAMESPACE_NODE *Node; 150 ACPI_STATUS Status; 151 ACPI_OBJECT_TYPE ObjectType; 152 char *Path; 153 UINT32 Flags; 154 155 156 ACPI_FUNCTION_TRACE_PTR (DsLoad1BeginOp, WalkState->Op); 157 158 159 Op = WalkState->Op; 160 ACPI_DEBUG_PRINT ((ACPI_DB_DISPATCH, "Op=%p State=%p\n", Op, WalkState)); 161 162 /* We are only interested in opcodes that have an associated name */ 163 164 if (Op) 165 { 166 if (!(WalkState->OpInfo->Flags & AML_NAMED)) 167 { 168 *OutOp = Op; 169 return_ACPI_STATUS (AE_OK); 170 } 171 172 /* Check if this object has already been installed in the namespace */ 173 174 if (Op->Common.Node) 175 { 176 *OutOp = Op; 177 return_ACPI_STATUS (AE_OK); 178 } 179 } 180 181 Path = AcpiPsGetNextNamestring (&WalkState->ParserState); 182 183 /* Map the raw opcode into an internal object type */ 184 185 ObjectType = WalkState->OpInfo->ObjectType; 186 187 ACPI_DEBUG_PRINT ((ACPI_DB_DISPATCH, 188 "State=%p Op=%p [%s]\n", WalkState, Op, 189 AcpiUtGetTypeName (ObjectType))); 190 191 switch (WalkState->Opcode) 192 { 193 case AML_SCOPE_OP: 194 /* 195 * The target name of the Scope() operator must exist at this point so 196 * that we can actually open the scope to enter new names underneath it. 197 * Allow search-to-root for single namesegs. 198 */ 199 Status = AcpiNsLookup (WalkState->ScopeInfo, Path, ObjectType, 200 ACPI_IMODE_EXECUTE, ACPI_NS_SEARCH_PARENT, WalkState, &(Node)); 201 #ifdef ACPI_ASL_COMPILER 202 if (Status == AE_NOT_FOUND) 203 { 204 /* 205 * Table disassembly: 206 * Target of Scope() not found. Generate an External for it, and 207 * insert the name into the namespace. 208 */ 209 AcpiDmAddOpToExternalList (Op, Path, ACPI_TYPE_DEVICE, 0, 0); 210 Status = AcpiNsLookup (WalkState->ScopeInfo, Path, ObjectType, 211 ACPI_IMODE_LOAD_PASS1, ACPI_NS_SEARCH_PARENT, 212 WalkState, &Node); 213 } 214 #endif 215 if (ACPI_FAILURE (Status)) 216 { 217 ACPI_ERROR_NAMESPACE (WalkState->ScopeInfo, Path, Status); 218 return_ACPI_STATUS (Status); 219 } 220 221 /* 222 * Check to make sure that the target is 223 * one of the opcodes that actually opens a scope 224 */ 225 switch (Node->Type) 226 { 227 case ACPI_TYPE_ANY: 228 case ACPI_TYPE_LOCAL_SCOPE: /* Scope */ 229 case ACPI_TYPE_DEVICE: 230 case ACPI_TYPE_POWER: 231 case ACPI_TYPE_PROCESSOR: 232 case ACPI_TYPE_THERMAL: 233 234 /* These are acceptable types */ 235 break; 236 237 case ACPI_TYPE_INTEGER: 238 case ACPI_TYPE_STRING: 239 case ACPI_TYPE_BUFFER: 240 /* 241 * These types we will allow, but we will change the type. 242 * This enables some existing code of the form: 243 * 244 * Name (DEB, 0) 245 * Scope (DEB) { ... } 246 * 247 * Note: silently change the type here. On the second pass, 248 * we will report a warning 249 */ 250 ACPI_DEBUG_PRINT ((ACPI_DB_INFO, 251 "Type override - [%4.4s] had invalid type (%s) " 252 "for Scope operator, changed to type ANY\n", 253 AcpiUtGetNodeName (Node), AcpiUtGetTypeName (Node->Type))); 254 255 Node->Type = ACPI_TYPE_ANY; 256 WalkState->ScopeInfo->Common.Value = ACPI_TYPE_ANY; 257 break; 258 259 case ACPI_TYPE_METHOD: 260 /* 261 * Allow scope change to root during execution of module-level 262 * code. Root is typed METHOD during this time. 263 */ 264 if ((Node == AcpiGbl_RootNode) && 265 (WalkState->ParseFlags & ACPI_PARSE_MODULE_LEVEL)) 266 { 267 break; 268 } 269 270 ACPI_FALLTHROUGH; 271 272 default: 273 274 /* All other types are an error */ 275 276 ACPI_ERROR ((AE_INFO, 277 "Invalid type (%s) for target of " 278 "Scope operator [%4.4s] (Cannot override)", 279 AcpiUtGetTypeName (Node->Type), AcpiUtGetNodeName (Node))); 280 281 return_ACPI_STATUS (AE_AML_OPERAND_TYPE); 282 } 283 break; 284 285 default: 286 /* 287 * For all other named opcodes, we will enter the name into 288 * the namespace. 289 * 290 * Setup the search flags. 291 * Since we are entering a name into the namespace, we do not want to 292 * enable the search-to-root upsearch. 293 * 294 * There are only two conditions where it is acceptable that the name 295 * already exists: 296 * 1) the Scope() operator can reopen a scoping object that was 297 * previously defined (Scope, Method, Device, etc.) 298 * 2) Whenever we are parsing a deferred opcode (OpRegion, Buffer, 299 * BufferField, or Package), the name of the object is already 300 * in the namespace. 301 */ 302 if (WalkState->DeferredNode) 303 { 304 /* This name is already in the namespace, get the node */ 305 306 Node = WalkState->DeferredNode; 307 Status = AE_OK; 308 break; 309 } 310 311 /* 312 * If we are executing a method, do not create any namespace objects 313 * during the load phase, only during execution. 314 */ 315 if (WalkState->MethodNode) 316 { 317 Node = NULL; 318 Status = AE_OK; 319 break; 320 } 321 322 Flags = ACPI_NS_NO_UPSEARCH; 323 if ((WalkState->Opcode != AML_SCOPE_OP) && 324 (!(WalkState->ParseFlags & ACPI_PARSE_DEFERRED_OP))) 325 { 326 if (WalkState->NamespaceOverride) 327 { 328 Flags |= ACPI_NS_OVERRIDE_IF_FOUND; 329 ACPI_DEBUG_PRINT ((ACPI_DB_DISPATCH, "[%s] Override allowed\n", 330 AcpiUtGetTypeName (ObjectType))); 331 } 332 else 333 { 334 Flags |= ACPI_NS_ERROR_IF_FOUND; 335 ACPI_DEBUG_PRINT ((ACPI_DB_DISPATCH, "[%s] Cannot already exist\n", 336 AcpiUtGetTypeName (ObjectType))); 337 } 338 } 339 else 340 { 341 ACPI_DEBUG_PRINT ((ACPI_DB_DISPATCH, 342 "[%s] Both Find or Create allowed\n", 343 AcpiUtGetTypeName (ObjectType))); 344 } 345 346 /* 347 * Enter the named type into the internal namespace. We enter the name 348 * as we go downward in the parse tree. Any necessary subobjects that 349 * involve arguments to the opcode must be created as we go back up the 350 * parse tree later. 351 */ 352 Status = AcpiNsLookup (WalkState->ScopeInfo, Path, ObjectType, 353 ACPI_IMODE_LOAD_PASS1, Flags, WalkState, &Node); 354 if (ACPI_FAILURE (Status)) 355 { 356 if (Status == AE_ALREADY_EXISTS) 357 { 358 /* The name already exists in this scope */ 359 360 if (Node->Flags & ANOBJ_IS_EXTERNAL) 361 { 362 /* 363 * Allow one create on an object or segment that was 364 * previously declared External 365 */ 366 Node->Flags &= ~ANOBJ_IS_EXTERNAL; 367 Node->Type = (UINT8) ObjectType; 368 369 /* Just retyped a node, probably will need to open a scope */ 370 371 if (AcpiNsOpensScope (ObjectType)) 372 { 373 Status = AcpiDsScopeStackPush ( 374 Node, ObjectType, WalkState); 375 if (ACPI_FAILURE (Status)) 376 { 377 return_ACPI_STATUS (Status); 378 } 379 } 380 381 Status = AE_OK; 382 } 383 } 384 385 if (ACPI_FAILURE (Status)) 386 { 387 ACPI_ERROR_NAMESPACE (WalkState->ScopeInfo, Path, Status); 388 return_ACPI_STATUS (Status); 389 } 390 } 391 break; 392 } 393 394 /* Common exit */ 395 396 if (!Op) 397 { 398 /* Create a new op */ 399 400 Op = AcpiPsAllocOp (WalkState->Opcode, WalkState->Aml); 401 if (!Op) 402 { 403 return_ACPI_STATUS (AE_NO_MEMORY); 404 } 405 } 406 407 /* Initialize the op */ 408 409 #ifdef ACPI_CONSTANT_EVAL_ONLY 410 Op->Named.Path = Path; 411 #endif 412 413 if (Node) 414 { 415 /* 416 * Put the Node in the "op" object that the parser uses, so we 417 * can get it again quickly when this scope is closed 418 */ 419 Op->Common.Node = Node; 420 Op->Named.Name = Node->Name.Integer; 421 } 422 423 AcpiPsAppendArg (AcpiPsGetParentScope (&WalkState->ParserState), Op); 424 *OutOp = Op; 425 return_ACPI_STATUS (Status); 426 } 427 428 429 /******************************************************************************* 430 * 431 * FUNCTION: AcpiDsLoad1EndOp 432 * 433 * PARAMETERS: WalkState - Current state of the parse tree walk 434 * 435 * RETURN: Status 436 * 437 * DESCRIPTION: Ascending callback used during the loading of the namespace, 438 * both control methods and everything else. 439 * 440 ******************************************************************************/ 441 442 ACPI_STATUS 443 AcpiDsLoad1EndOp ( 444 ACPI_WALK_STATE *WalkState) 445 { 446 ACPI_PARSE_OBJECT *Op; 447 ACPI_OBJECT_TYPE ObjectType; 448 ACPI_STATUS Status = AE_OK; 449 #ifdef ACPI_ASL_COMPILER 450 UINT8 ParamCount; 451 #endif 452 453 454 ACPI_FUNCTION_TRACE (DsLoad1EndOp); 455 456 457 Op = WalkState->Op; 458 ACPI_DEBUG_PRINT ((ACPI_DB_DISPATCH, "Op=%p State=%p\n", Op, WalkState)); 459 460 /* 461 * Disassembler: handle create field operators here. 462 * 463 * CreateBufferField is a deferred op that is typically processed in load 464 * pass 2. However, disassembly of control method contents walk the parse 465 * tree with ACPI_PARSE_LOAD_PASS1 and AML_CREATE operators are processed 466 * in a later walk. This is a problem when there is a control method that 467 * has the same name as the AML_CREATE object. In this case, any use of the 468 * name segment will be detected as a method call rather than a reference 469 * to a buffer field. 470 * 471 * This earlier creation during disassembly solves this issue by inserting 472 * the named object in the ACPI namespace so that references to this name 473 * would be a name string rather than a method call. 474 */ 475 if ((WalkState->ParseFlags & ACPI_PARSE_DISASSEMBLE) && 476 (WalkState->OpInfo->Flags & AML_CREATE)) 477 { 478 Status = AcpiDsCreateBufferField (Op, WalkState); 479 return_ACPI_STATUS (Status); 480 } 481 482 /* We are only interested in opcodes that have an associated name */ 483 484 if (!(WalkState->OpInfo->Flags & (AML_NAMED | AML_FIELD))) 485 { 486 return_ACPI_STATUS (AE_OK); 487 } 488 489 /* Get the object type to determine if we should pop the scope */ 490 491 ObjectType = WalkState->OpInfo->ObjectType; 492 493 if (WalkState->OpInfo->Flags & AML_FIELD) 494 { 495 /* 496 * If we are executing a method, do not create any namespace objects 497 * during the load phase, only during execution. 498 */ 499 if (!WalkState->MethodNode) 500 { 501 if (WalkState->Opcode == AML_FIELD_OP || 502 WalkState->Opcode == AML_BANK_FIELD_OP || 503 WalkState->Opcode == AML_INDEX_FIELD_OP) 504 { 505 Status = AcpiDsInitFieldObjects (Op, WalkState); 506 } 507 } 508 return_ACPI_STATUS (Status); 509 } 510 511 /* 512 * If we are executing a method, do not create any namespace objects 513 * during the load phase, only during execution. 514 */ 515 if (!WalkState->MethodNode) 516 { 517 if (Op->Common.AmlOpcode == AML_REGION_OP) 518 { 519 Status = AcpiExCreateRegion (Op->Named.Data, Op->Named.Length, 520 (ACPI_ADR_SPACE_TYPE) 521 ((Op->Common.Value.Arg)->Common.Value.Integer), 522 WalkState); 523 if (ACPI_FAILURE (Status)) 524 { 525 return_ACPI_STATUS (Status); 526 } 527 } 528 else if (Op->Common.AmlOpcode == AML_DATA_REGION_OP) 529 { 530 Status = AcpiExCreateRegion (Op->Named.Data, Op->Named.Length, 531 ACPI_ADR_SPACE_DATA_TABLE, WalkState); 532 if (ACPI_FAILURE (Status)) 533 { 534 return_ACPI_STATUS (Status); 535 } 536 } 537 } 538 539 if (Op->Common.AmlOpcode == AML_NAME_OP) 540 { 541 /* For Name opcode, get the object type from the argument */ 542 543 if (Op->Common.Value.Arg) 544 { 545 ObjectType = (AcpiPsGetOpcodeInfo ( 546 (Op->Common.Value.Arg)->Common.AmlOpcode))->ObjectType; 547 548 /* Set node type if we have a namespace node */ 549 550 if (Op->Common.Node) 551 { 552 Op->Common.Node->Type = (UINT8) ObjectType; 553 } 554 } 555 } 556 557 #ifdef ACPI_ASL_COMPILER 558 /* 559 * For external opcode, get the object type from the argument and 560 * get the parameter count from the argument's next. 561 */ 562 if (AcpiGbl_DisasmFlag && 563 Op->Common.Node && 564 Op->Common.AmlOpcode == AML_EXTERNAL_OP) 565 { 566 /* 567 * Note, if this external is not a method 568 * Op->Common.Value.Arg->Common.Next->Common.Value.Integer == 0 569 * Therefore, ParamCount will be 0. 570 */ 571 ParamCount = (UINT8) Op->Common.Value.Arg->Common.Next->Common.Value.Integer; 572 ObjectType = (UINT8) Op->Common.Value.Arg->Common.Value.Integer; 573 Op->Common.Node->Flags |= ANOBJ_IS_EXTERNAL; 574 Op->Common.Node->Type = (UINT8) ObjectType; 575 576 AcpiDmCreateSubobjectForExternal ((UINT8)ObjectType, 577 &Op->Common.Node, ParamCount); 578 579 /* 580 * Add the external to the external list because we may be 581 * emitting code based off of the items within the external list. 582 */ 583 AcpiDmAddOpToExternalList (Op, Op->Named.Path, (UINT8)ObjectType, ParamCount, 584 ACPI_EXT_ORIGIN_FROM_OPCODE | ACPI_EXT_RESOLVED_REFERENCE); 585 } 586 #endif 587 588 /* 589 * If we are executing a method, do not create any namespace objects 590 * during the load phase, only during execution. 591 */ 592 if (!WalkState->MethodNode) 593 { 594 if (Op->Common.AmlOpcode == AML_METHOD_OP) 595 { 596 /* 597 * MethodOp PkgLength NameString MethodFlags TermList 598 * 599 * Note: We must create the method node/object pair as soon as we 600 * see the method declaration. This allows later pass1 parsing 601 * of invocations of the method (need to know the number of 602 * arguments.) 603 */ 604 ACPI_DEBUG_PRINT ((ACPI_DB_DISPATCH, 605 "LOADING-Method: State=%p Op=%p NamedObj=%p\n", 606 WalkState, Op, Op->Named.Node)); 607 608 if (!AcpiNsGetAttachedObject (Op->Named.Node)) 609 { 610 WalkState->Operands[0] = ACPI_CAST_PTR (void, Op->Named.Node); 611 WalkState->NumOperands = 1; 612 613 Status = AcpiDsCreateOperands ( 614 WalkState, Op->Common.Value.Arg); 615 if (ACPI_SUCCESS (Status)) 616 { 617 Status = AcpiExCreateMethod (Op->Named.Data, 618 Op->Named.Length, WalkState); 619 } 620 621 WalkState->Operands[0] = NULL; 622 WalkState->NumOperands = 0; 623 624 if (ACPI_FAILURE (Status)) 625 { 626 return_ACPI_STATUS (Status); 627 } 628 } 629 } 630 } 631 632 /* Pop the scope stack (only if loading a table) */ 633 634 if (!WalkState->MethodNode && 635 Op->Common.AmlOpcode != AML_EXTERNAL_OP && 636 AcpiNsOpensScope (ObjectType)) 637 { 638 ACPI_DEBUG_PRINT ((ACPI_DB_DISPATCH, "(%s): Popping scope for Op %p\n", 639 AcpiUtGetTypeName (ObjectType), Op)); 640 641 Status = AcpiDsScopeStackPop (WalkState); 642 } 643 644 return_ACPI_STATUS (Status); 645 } 646