1 /****************************************************************************** 2 * 3 * Module Name: dswstate - Dispatcher parse tree walk management routines 4 * 5 *****************************************************************************/ 6 7 /* 8 * Copyright (C) 2000 - 2015, Intel Corp. 9 * All rights reserved. 10 * 11 * Redistribution and use in source and binary forms, with or without 12 * modification, are permitted provided that the following conditions 13 * are met: 14 * 1. Redistributions of source code must retain the above copyright 15 * notice, this list of conditions, and the following disclaimer, 16 * without modification. 17 * 2. Redistributions in binary form must reproduce at minimum a disclaimer 18 * substantially similar to the "NO WARRANTY" disclaimer below 19 * ("Disclaimer") and any redistribution must be conditioned upon 20 * including a substantially similar Disclaimer requirement for further 21 * binary redistribution. 22 * 3. Neither the names of the above-listed copyright holders nor the names 23 * of any contributors may be used to endorse or promote products derived 24 * from this software without specific prior written permission. 25 * 26 * Alternatively, this software may be distributed under the terms of the 27 * GNU General Public License ("GPL") version 2 as published by the Free 28 * Software Foundation. 29 * 30 * NO WARRANTY 31 * THIS SOFTWARE IS PROVIDED BY THE COPYRIGHT HOLDERS AND CONTRIBUTORS 32 * "AS IS" AND ANY EXPRESS OR IMPLIED WARRANTIES, INCLUDING, BUT NOT 33 * LIMITED TO, THE IMPLIED WARRANTIES OF MERCHANTIBILITY AND FITNESS FOR 34 * A PARTICULAR PURPOSE ARE DISCLAIMED. IN NO EVENT SHALL THE COPYRIGHT 35 * HOLDERS OR CONTRIBUTORS BE LIABLE FOR SPECIAL, EXEMPLARY, OR CONSEQUENTIAL 36 * DAMAGES (INCLUDING, BUT NOT LIMITED TO, PROCUREMENT OF SUBSTITUTE GOODS 37 * OR SERVICES; LOSS OF USE, DATA, OR PROFITS; OR BUSINESS INTERRUPTION) 38 * HOWEVER CAUSED AND ON ANY THEORY OF LIABILITY, WHETHER IN CONTRACT, 39 * STRICT LIABILITY, OR TORT (INCLUDING NEGLIGENCE OR OTHERWISE) ARISING 40 * IN ANY WAY OUT OF THE USE OF THIS SOFTWARE, EVEN IF ADVISED OF THE 41 * POSSIBILITY OF SUCH DAMAGES. 42 */ 43 44 #include <contrib/dev/acpica/include/acpi.h> 45 #include <contrib/dev/acpica/include/accommon.h> 46 #include <contrib/dev/acpica/include/acparser.h> 47 #include <contrib/dev/acpica/include/acdispat.h> 48 #include <contrib/dev/acpica/include/acnamesp.h> 49 50 #define _COMPONENT ACPI_DISPATCHER 51 ACPI_MODULE_NAME ("dswstate") 52 53 /* Local prototypes */ 54 55 static ACPI_STATUS 56 AcpiDsResultStackPush ( 57 ACPI_WALK_STATE *WalkState); 58 59 static ACPI_STATUS 60 AcpiDsResultStackPop ( 61 ACPI_WALK_STATE *WalkState); 62 63 64 /******************************************************************************* 65 * 66 * FUNCTION: AcpiDsResultPop 67 * 68 * PARAMETERS: Object - Where to return the popped object 69 * WalkState - Current Walk state 70 * 71 * RETURN: Status 72 * 73 * DESCRIPTION: Pop an object off the top of this walk's result stack 74 * 75 ******************************************************************************/ 76 77 ACPI_STATUS 78 AcpiDsResultPop ( 79 ACPI_OPERAND_OBJECT **Object, 80 ACPI_WALK_STATE *WalkState) 81 { 82 UINT32 Index; 83 ACPI_GENERIC_STATE *State; 84 ACPI_STATUS Status; 85 86 87 ACPI_FUNCTION_NAME (DsResultPop); 88 89 90 State = WalkState->Results; 91 92 /* Incorrect state of result stack */ 93 94 if (State && !WalkState->ResultCount) 95 { 96 ACPI_ERROR ((AE_INFO, "No results on result stack")); 97 return (AE_AML_INTERNAL); 98 } 99 100 if (!State && WalkState->ResultCount) 101 { 102 ACPI_ERROR ((AE_INFO, "No result state for result stack")); 103 return (AE_AML_INTERNAL); 104 } 105 106 /* Empty result stack */ 107 108 if (!State) 109 { 110 ACPI_ERROR ((AE_INFO, "Result stack is empty! State=%p", WalkState)); 111 return (AE_AML_NO_RETURN_VALUE); 112 } 113 114 /* Return object of the top element and clean that top element result stack */ 115 116 WalkState->ResultCount--; 117 Index = (UINT32) WalkState->ResultCount % ACPI_RESULTS_FRAME_OBJ_NUM; 118 119 *Object = State->Results.ObjDesc [Index]; 120 if (!*Object) 121 { 122 ACPI_ERROR ((AE_INFO, "No result objects on result stack, State=%p", 123 WalkState)); 124 return (AE_AML_NO_RETURN_VALUE); 125 } 126 127 State->Results.ObjDesc [Index] = NULL; 128 if (Index == 0) 129 { 130 Status = AcpiDsResultStackPop (WalkState); 131 if (ACPI_FAILURE (Status)) 132 { 133 return (Status); 134 } 135 } 136 137 ACPI_DEBUG_PRINT ((ACPI_DB_EXEC, 138 "Obj=%p [%s] Index=%X State=%p Num=%X\n", *Object, 139 AcpiUtGetObjectTypeName (*Object), 140 Index, WalkState, WalkState->ResultCount)); 141 142 return (AE_OK); 143 } 144 145 146 /******************************************************************************* 147 * 148 * FUNCTION: AcpiDsResultPush 149 * 150 * PARAMETERS: Object - Where to return the popped object 151 * WalkState - Current Walk state 152 * 153 * RETURN: Status 154 * 155 * DESCRIPTION: Push an object onto the current result stack 156 * 157 ******************************************************************************/ 158 159 ACPI_STATUS 160 AcpiDsResultPush ( 161 ACPI_OPERAND_OBJECT *Object, 162 ACPI_WALK_STATE *WalkState) 163 { 164 ACPI_GENERIC_STATE *State; 165 ACPI_STATUS Status; 166 UINT32 Index; 167 168 169 ACPI_FUNCTION_NAME (DsResultPush); 170 171 172 if (WalkState->ResultCount > WalkState->ResultSize) 173 { 174 ACPI_ERROR ((AE_INFO, "Result stack is full")); 175 return (AE_AML_INTERNAL); 176 } 177 else if (WalkState->ResultCount == WalkState->ResultSize) 178 { 179 /* Extend the result stack */ 180 181 Status = AcpiDsResultStackPush (WalkState); 182 if (ACPI_FAILURE (Status)) 183 { 184 ACPI_ERROR ((AE_INFO, "Failed to extend the result stack")); 185 return (Status); 186 } 187 } 188 189 if (!(WalkState->ResultCount < WalkState->ResultSize)) 190 { 191 ACPI_ERROR ((AE_INFO, "No free elements in result stack")); 192 return (AE_AML_INTERNAL); 193 } 194 195 State = WalkState->Results; 196 if (!State) 197 { 198 ACPI_ERROR ((AE_INFO, "No result stack frame during push")); 199 return (AE_AML_INTERNAL); 200 } 201 202 if (!Object) 203 { 204 ACPI_ERROR ((AE_INFO, 205 "Null Object! Obj=%p State=%p Num=%u", 206 Object, WalkState, WalkState->ResultCount)); 207 return (AE_BAD_PARAMETER); 208 } 209 210 /* Assign the address of object to the top free element of result stack */ 211 212 Index = (UINT32) WalkState->ResultCount % ACPI_RESULTS_FRAME_OBJ_NUM; 213 State->Results.ObjDesc [Index] = Object; 214 WalkState->ResultCount++; 215 216 ACPI_DEBUG_PRINT ((ACPI_DB_EXEC, "Obj=%p [%s] State=%p Num=%X Cur=%X\n", 217 Object, AcpiUtGetObjectTypeName ((ACPI_OPERAND_OBJECT *) Object), 218 WalkState, WalkState->ResultCount, WalkState->CurrentResult)); 219 220 return (AE_OK); 221 } 222 223 224 /******************************************************************************* 225 * 226 * FUNCTION: AcpiDsResultStackPush 227 * 228 * PARAMETERS: WalkState - Current Walk state 229 * 230 * RETURN: Status 231 * 232 * DESCRIPTION: Push an object onto the WalkState result stack 233 * 234 ******************************************************************************/ 235 236 static ACPI_STATUS 237 AcpiDsResultStackPush ( 238 ACPI_WALK_STATE *WalkState) 239 { 240 ACPI_GENERIC_STATE *State; 241 242 243 ACPI_FUNCTION_NAME (DsResultStackPush); 244 245 246 /* Check for stack overflow */ 247 248 if (((UINT32) WalkState->ResultSize + ACPI_RESULTS_FRAME_OBJ_NUM) > 249 ACPI_RESULTS_OBJ_NUM_MAX) 250 { 251 ACPI_ERROR ((AE_INFO, "Result stack overflow: State=%p Num=%u", 252 WalkState, WalkState->ResultSize)); 253 return (AE_STACK_OVERFLOW); 254 } 255 256 State = AcpiUtCreateGenericState (); 257 if (!State) 258 { 259 return (AE_NO_MEMORY); 260 } 261 262 State->Common.DescriptorType = ACPI_DESC_TYPE_STATE_RESULT; 263 AcpiUtPushGenericState (&WalkState->Results, State); 264 265 /* Increase the length of the result stack by the length of frame */ 266 267 WalkState->ResultSize += ACPI_RESULTS_FRAME_OBJ_NUM; 268 269 ACPI_DEBUG_PRINT ((ACPI_DB_EXEC, "Results=%p State=%p\n", 270 State, WalkState)); 271 272 return (AE_OK); 273 } 274 275 276 /******************************************************************************* 277 * 278 * FUNCTION: AcpiDsResultStackPop 279 * 280 * PARAMETERS: WalkState - Current Walk state 281 * 282 * RETURN: Status 283 * 284 * DESCRIPTION: Pop an object off of the WalkState result stack 285 * 286 ******************************************************************************/ 287 288 static ACPI_STATUS 289 AcpiDsResultStackPop ( 290 ACPI_WALK_STATE *WalkState) 291 { 292 ACPI_GENERIC_STATE *State; 293 294 295 ACPI_FUNCTION_NAME (DsResultStackPop); 296 297 298 /* Check for stack underflow */ 299 300 if (WalkState->Results == NULL) 301 { 302 ACPI_DEBUG_PRINT ((ACPI_DB_EXEC, "Result stack underflow - State=%p\n", 303 WalkState)); 304 return (AE_AML_NO_OPERAND); 305 } 306 307 if (WalkState->ResultSize < ACPI_RESULTS_FRAME_OBJ_NUM) 308 { 309 ACPI_ERROR ((AE_INFO, "Insufficient result stack size")); 310 return (AE_AML_INTERNAL); 311 } 312 313 State = AcpiUtPopGenericState (&WalkState->Results); 314 AcpiUtDeleteGenericState (State); 315 316 /* Decrease the length of result stack by the length of frame */ 317 318 WalkState->ResultSize -= ACPI_RESULTS_FRAME_OBJ_NUM; 319 320 ACPI_DEBUG_PRINT ((ACPI_DB_EXEC, 321 "Result=%p RemainingResults=%X State=%p\n", 322 State, WalkState->ResultCount, WalkState)); 323 324 return (AE_OK); 325 } 326 327 328 /******************************************************************************* 329 * 330 * FUNCTION: AcpiDsObjStackPush 331 * 332 * PARAMETERS: Object - Object to push 333 * WalkState - Current Walk state 334 * 335 * RETURN: Status 336 * 337 * DESCRIPTION: Push an object onto this walk's object/operand stack 338 * 339 ******************************************************************************/ 340 341 ACPI_STATUS 342 AcpiDsObjStackPush ( 343 void *Object, 344 ACPI_WALK_STATE *WalkState) 345 { 346 ACPI_FUNCTION_NAME (DsObjStackPush); 347 348 349 /* Check for stack overflow */ 350 351 if (WalkState->NumOperands >= ACPI_OBJ_NUM_OPERANDS) 352 { 353 ACPI_ERROR ((AE_INFO, 354 "Object stack overflow! Obj=%p State=%p #Ops=%u", 355 Object, WalkState, WalkState->NumOperands)); 356 return (AE_STACK_OVERFLOW); 357 } 358 359 /* Put the object onto the stack */ 360 361 WalkState->Operands [WalkState->OperandIndex] = Object; 362 WalkState->NumOperands++; 363 364 /* For the usual order of filling the operand stack */ 365 366 WalkState->OperandIndex++; 367 368 ACPI_DEBUG_PRINT ((ACPI_DB_EXEC, "Obj=%p [%s] State=%p #Ops=%X\n", 369 Object, AcpiUtGetObjectTypeName ((ACPI_OPERAND_OBJECT *) Object), 370 WalkState, WalkState->NumOperands)); 371 372 return (AE_OK); 373 } 374 375 376 /******************************************************************************* 377 * 378 * FUNCTION: AcpiDsObjStackPop 379 * 380 * PARAMETERS: PopCount - Number of objects/entries to pop 381 * WalkState - Current Walk state 382 * 383 * RETURN: Status 384 * 385 * DESCRIPTION: Pop this walk's object stack. Objects on the stack are NOT 386 * deleted by this routine. 387 * 388 ******************************************************************************/ 389 390 ACPI_STATUS 391 AcpiDsObjStackPop ( 392 UINT32 PopCount, 393 ACPI_WALK_STATE *WalkState) 394 { 395 UINT32 i; 396 397 398 ACPI_FUNCTION_NAME (DsObjStackPop); 399 400 401 for (i = 0; i < PopCount; i++) 402 { 403 /* Check for stack underflow */ 404 405 if (WalkState->NumOperands == 0) 406 { 407 ACPI_ERROR ((AE_INFO, 408 "Object stack underflow! Count=%X State=%p #Ops=%u", 409 PopCount, WalkState, WalkState->NumOperands)); 410 return (AE_STACK_UNDERFLOW); 411 } 412 413 /* Just set the stack entry to null */ 414 415 WalkState->NumOperands--; 416 WalkState->Operands [WalkState->NumOperands] = NULL; 417 } 418 419 ACPI_DEBUG_PRINT ((ACPI_DB_EXEC, "Count=%X State=%p #Ops=%u\n", 420 PopCount, WalkState, WalkState->NumOperands)); 421 422 return (AE_OK); 423 } 424 425 426 /******************************************************************************* 427 * 428 * FUNCTION: AcpiDsObjStackPopAndDelete 429 * 430 * PARAMETERS: PopCount - Number of objects/entries to pop 431 * WalkState - Current Walk state 432 * 433 * RETURN: Status 434 * 435 * DESCRIPTION: Pop this walk's object stack and delete each object that is 436 * popped off. 437 * 438 ******************************************************************************/ 439 440 void 441 AcpiDsObjStackPopAndDelete ( 442 UINT32 PopCount, 443 ACPI_WALK_STATE *WalkState) 444 { 445 INT32 i; 446 ACPI_OPERAND_OBJECT *ObjDesc; 447 448 449 ACPI_FUNCTION_NAME (DsObjStackPopAndDelete); 450 451 452 if (PopCount == 0) 453 { 454 return; 455 } 456 457 for (i = (INT32) PopCount - 1; i >= 0; i--) 458 { 459 if (WalkState->NumOperands == 0) 460 { 461 return; 462 } 463 464 /* Pop the stack and delete an object if present in this stack entry */ 465 466 WalkState->NumOperands--; 467 ObjDesc = WalkState->Operands [i]; 468 if (ObjDesc) 469 { 470 AcpiUtRemoveReference (WalkState->Operands [i]); 471 WalkState->Operands [i] = NULL; 472 } 473 } 474 475 ACPI_DEBUG_PRINT ((ACPI_DB_EXEC, "Count=%X State=%p #Ops=%X\n", 476 PopCount, WalkState, WalkState->NumOperands)); 477 } 478 479 480 /******************************************************************************* 481 * 482 * FUNCTION: AcpiDsGetCurrentWalkState 483 * 484 * PARAMETERS: Thread - Get current active state for this Thread 485 * 486 * RETURN: Pointer to the current walk state 487 * 488 * DESCRIPTION: Get the walk state that is at the head of the list (the "current" 489 * walk state.) 490 * 491 ******************************************************************************/ 492 493 ACPI_WALK_STATE * 494 AcpiDsGetCurrentWalkState ( 495 ACPI_THREAD_STATE *Thread) 496 { 497 ACPI_FUNCTION_NAME (DsGetCurrentWalkState); 498 499 500 if (!Thread) 501 { 502 return (NULL); 503 } 504 505 ACPI_DEBUG_PRINT ((ACPI_DB_PARSE, "Current WalkState %p\n", 506 Thread->WalkStateList)); 507 508 return (Thread->WalkStateList); 509 } 510 511 512 /******************************************************************************* 513 * 514 * FUNCTION: AcpiDsPushWalkState 515 * 516 * PARAMETERS: WalkState - State to push 517 * Thread - Thread state object 518 * 519 * RETURN: None 520 * 521 * DESCRIPTION: Place the Thread state at the head of the state list 522 * 523 ******************************************************************************/ 524 525 void 526 AcpiDsPushWalkState ( 527 ACPI_WALK_STATE *WalkState, 528 ACPI_THREAD_STATE *Thread) 529 { 530 ACPI_FUNCTION_TRACE (DsPushWalkState); 531 532 533 WalkState->Next = Thread->WalkStateList; 534 Thread->WalkStateList = WalkState; 535 536 return_VOID; 537 } 538 539 540 /******************************************************************************* 541 * 542 * FUNCTION: AcpiDsPopWalkState 543 * 544 * PARAMETERS: Thread - Current thread state 545 * 546 * RETURN: A WalkState object popped from the thread's stack 547 * 548 * DESCRIPTION: Remove and return the walkstate object that is at the head of 549 * the walk stack for the given walk list. NULL indicates that 550 * the list is empty. 551 * 552 ******************************************************************************/ 553 554 ACPI_WALK_STATE * 555 AcpiDsPopWalkState ( 556 ACPI_THREAD_STATE *Thread) 557 { 558 ACPI_WALK_STATE *WalkState; 559 560 561 ACPI_FUNCTION_TRACE (DsPopWalkState); 562 563 564 WalkState = Thread->WalkStateList; 565 566 if (WalkState) 567 { 568 /* Next walk state becomes the current walk state */ 569 570 Thread->WalkStateList = WalkState->Next; 571 572 /* 573 * Don't clear the NEXT field, this serves as an indicator 574 * that there is a parent WALK STATE 575 * Do Not: WalkState->Next = NULL; 576 */ 577 } 578 579 return_PTR (WalkState); 580 } 581 582 583 /******************************************************************************* 584 * 585 * FUNCTION: AcpiDsCreateWalkState 586 * 587 * PARAMETERS: OwnerId - ID for object creation 588 * Origin - Starting point for this walk 589 * MethodDesc - Method object 590 * Thread - Current thread state 591 * 592 * RETURN: Pointer to the new walk state. 593 * 594 * DESCRIPTION: Allocate and initialize a new walk state. The current walk 595 * state is set to this new state. 596 * 597 ******************************************************************************/ 598 599 ACPI_WALK_STATE * 600 AcpiDsCreateWalkState ( 601 ACPI_OWNER_ID OwnerId, 602 ACPI_PARSE_OBJECT *Origin, 603 ACPI_OPERAND_OBJECT *MethodDesc, 604 ACPI_THREAD_STATE *Thread) 605 { 606 ACPI_WALK_STATE *WalkState; 607 608 609 ACPI_FUNCTION_TRACE (DsCreateWalkState); 610 611 612 WalkState = ACPI_ALLOCATE_ZEROED (sizeof (ACPI_WALK_STATE)); 613 if (!WalkState) 614 { 615 return_PTR (NULL); 616 } 617 618 WalkState->DescriptorType = ACPI_DESC_TYPE_WALK; 619 WalkState->MethodDesc = MethodDesc; 620 WalkState->OwnerId = OwnerId; 621 WalkState->Origin = Origin; 622 WalkState->Thread = Thread; 623 624 WalkState->ParserState.StartOp = Origin; 625 626 /* Init the method args/local */ 627 628 #if (!defined (ACPI_NO_METHOD_EXECUTION) && !defined (ACPI_CONSTANT_EVAL_ONLY)) 629 AcpiDsMethodDataInit (WalkState); 630 #endif 631 632 /* Put the new state at the head of the walk list */ 633 634 if (Thread) 635 { 636 AcpiDsPushWalkState (WalkState, Thread); 637 } 638 639 return_PTR (WalkState); 640 } 641 642 643 /******************************************************************************* 644 * 645 * FUNCTION: AcpiDsInitAmlWalk 646 * 647 * PARAMETERS: WalkState - New state to be initialized 648 * Op - Current parse op 649 * MethodNode - Control method NS node, if any 650 * AmlStart - Start of AML 651 * AmlLength - Length of AML 652 * Info - Method info block (params, etc.) 653 * PassNumber - 1, 2, or 3 654 * 655 * RETURN: Status 656 * 657 * DESCRIPTION: Initialize a walk state for a pass 1 or 2 parse tree walk 658 * 659 ******************************************************************************/ 660 661 ACPI_STATUS 662 AcpiDsInitAmlWalk ( 663 ACPI_WALK_STATE *WalkState, 664 ACPI_PARSE_OBJECT *Op, 665 ACPI_NAMESPACE_NODE *MethodNode, 666 UINT8 *AmlStart, 667 UINT32 AmlLength, 668 ACPI_EVALUATE_INFO *Info, 669 UINT8 PassNumber) 670 { 671 ACPI_STATUS Status; 672 ACPI_PARSE_STATE *ParserState = &WalkState->ParserState; 673 ACPI_PARSE_OBJECT *ExtraOp; 674 675 676 ACPI_FUNCTION_TRACE (DsInitAmlWalk); 677 678 679 WalkState->ParserState.Aml = 680 WalkState->ParserState.AmlStart = AmlStart; 681 WalkState->ParserState.AmlEnd = 682 WalkState->ParserState.PkgEnd = AmlStart + AmlLength; 683 684 /* The NextOp of the NextWalk will be the beginning of the method */ 685 686 WalkState->NextOp = NULL; 687 WalkState->PassNumber = PassNumber; 688 689 if (Info) 690 { 691 WalkState->Params = Info->Parameters; 692 WalkState->CallerReturnDesc = &Info->ReturnObject; 693 } 694 695 Status = AcpiPsInitScope (&WalkState->ParserState, Op); 696 if (ACPI_FAILURE (Status)) 697 { 698 return_ACPI_STATUS (Status); 699 } 700 701 if (MethodNode) 702 { 703 WalkState->ParserState.StartNode = MethodNode; 704 WalkState->WalkType = ACPI_WALK_METHOD; 705 WalkState->MethodNode = MethodNode; 706 WalkState->MethodDesc = AcpiNsGetAttachedObject (MethodNode); 707 708 /* Push start scope on scope stack and make it current */ 709 710 Status = AcpiDsScopeStackPush (MethodNode, ACPI_TYPE_METHOD, WalkState); 711 if (ACPI_FAILURE (Status)) 712 { 713 return_ACPI_STATUS (Status); 714 } 715 716 /* Init the method arguments */ 717 718 Status = AcpiDsMethodDataInitArgs (WalkState->Params, 719 ACPI_METHOD_NUM_ARGS, WalkState); 720 if (ACPI_FAILURE (Status)) 721 { 722 return_ACPI_STATUS (Status); 723 } 724 } 725 else 726 { 727 /* 728 * Setup the current scope. 729 * Find a Named Op that has a namespace node associated with it. 730 * search upwards from this Op. Current scope is the first 731 * Op with a namespace node. 732 */ 733 ExtraOp = ParserState->StartOp; 734 while (ExtraOp && !ExtraOp->Common.Node) 735 { 736 ExtraOp = ExtraOp->Common.Parent; 737 } 738 739 if (!ExtraOp) 740 { 741 ParserState->StartNode = NULL; 742 } 743 else 744 { 745 ParserState->StartNode = ExtraOp->Common.Node; 746 } 747 748 if (ParserState->StartNode) 749 { 750 /* Push start scope on scope stack and make it current */ 751 752 Status = AcpiDsScopeStackPush (ParserState->StartNode, 753 ParserState->StartNode->Type, WalkState); 754 if (ACPI_FAILURE (Status)) 755 { 756 return_ACPI_STATUS (Status); 757 } 758 } 759 } 760 761 Status = AcpiDsInitCallbacks (WalkState, PassNumber); 762 return_ACPI_STATUS (Status); 763 } 764 765 766 /******************************************************************************* 767 * 768 * FUNCTION: AcpiDsDeleteWalkState 769 * 770 * PARAMETERS: WalkState - State to delete 771 * 772 * RETURN: Status 773 * 774 * DESCRIPTION: Delete a walk state including all internal data structures 775 * 776 ******************************************************************************/ 777 778 void 779 AcpiDsDeleteWalkState ( 780 ACPI_WALK_STATE *WalkState) 781 { 782 ACPI_GENERIC_STATE *State; 783 784 785 ACPI_FUNCTION_TRACE_PTR (DsDeleteWalkState, WalkState); 786 787 788 if (!WalkState) 789 { 790 return_VOID; 791 } 792 793 if (WalkState->DescriptorType != ACPI_DESC_TYPE_WALK) 794 { 795 ACPI_ERROR ((AE_INFO, "%p is not a valid walk state", 796 WalkState)); 797 return_VOID; 798 } 799 800 /* There should not be any open scopes */ 801 802 if (WalkState->ParserState.Scope) 803 { 804 ACPI_ERROR ((AE_INFO, "%p walk still has a scope list", 805 WalkState)); 806 AcpiPsCleanupScope (&WalkState->ParserState); 807 } 808 809 /* Always must free any linked control states */ 810 811 while (WalkState->ControlState) 812 { 813 State = WalkState->ControlState; 814 WalkState->ControlState = State->Common.Next; 815 816 AcpiUtDeleteGenericState (State); 817 } 818 819 /* Always must free any linked parse states */ 820 821 while (WalkState->ScopeInfo) 822 { 823 State = WalkState->ScopeInfo; 824 WalkState->ScopeInfo = State->Common.Next; 825 826 AcpiUtDeleteGenericState (State); 827 } 828 829 /* Always must free any stacked result states */ 830 831 while (WalkState->Results) 832 { 833 State = WalkState->Results; 834 WalkState->Results = State->Common.Next; 835 836 AcpiUtDeleteGenericState (State); 837 } 838 839 ACPI_FREE (WalkState); 840 return_VOID; 841 } 842