1 /****************************************************************************** 2 * 3 * Module Name: nsrepair2 - Repair for objects returned by specific 4 * predefined methods 5 * 6 *****************************************************************************/ 7 8 /* 9 * Copyright (C) 2000 - 2015, Intel Corp. 10 * All rights reserved. 11 * 12 * Redistribution and use in source and binary forms, with or without 13 * modification, are permitted provided that the following conditions 14 * are met: 15 * 1. Redistributions of source code must retain the above copyright 16 * notice, this list of conditions, and the following disclaimer, 17 * without modification. 18 * 2. Redistributions in binary form must reproduce at minimum a disclaimer 19 * substantially similar to the "NO WARRANTY" disclaimer below 20 * ("Disclaimer") and any redistribution must be conditioned upon 21 * including a substantially similar Disclaimer requirement for further 22 * binary redistribution. 23 * 3. Neither the names of the above-listed copyright holders nor the names 24 * of any contributors may be used to endorse or promote products derived 25 * from this software without specific prior written permission. 26 * 27 * Alternatively, this software may be distributed under the terms of the 28 * GNU General Public License ("GPL") version 2 as published by the Free 29 * Software Foundation. 30 * 31 * NO WARRANTY 32 * THIS SOFTWARE IS PROVIDED BY THE COPYRIGHT HOLDERS AND CONTRIBUTORS 33 * "AS IS" AND ANY EXPRESS OR IMPLIED WARRANTIES, INCLUDING, BUT NOT 34 * LIMITED TO, THE IMPLIED WARRANTIES OF MERCHANTIBILITY AND FITNESS FOR 35 * A PARTICULAR PURPOSE ARE DISCLAIMED. IN NO EVENT SHALL THE COPYRIGHT 36 * HOLDERS OR CONTRIBUTORS BE LIABLE FOR SPECIAL, EXEMPLARY, OR CONSEQUENTIAL 37 * DAMAGES (INCLUDING, BUT NOT LIMITED TO, PROCUREMENT OF SUBSTITUTE GOODS 38 * OR SERVICES; LOSS OF USE, DATA, OR PROFITS; OR BUSINESS INTERRUPTION) 39 * HOWEVER CAUSED AND ON ANY THEORY OF LIABILITY, WHETHER IN CONTRACT, 40 * STRICT LIABILITY, OR TORT (INCLUDING NEGLIGENCE OR OTHERWISE) ARISING 41 * IN ANY WAY OUT OF THE USE OF THIS SOFTWARE, EVEN IF ADVISED OF THE 42 * POSSIBILITY OF SUCH DAMAGES. 43 */ 44 45 #include <contrib/dev/acpica/include/acpi.h> 46 #include <contrib/dev/acpica/include/accommon.h> 47 #include <contrib/dev/acpica/include/acnamesp.h> 48 49 #define _COMPONENT ACPI_NAMESPACE 50 ACPI_MODULE_NAME ("nsrepair2") 51 52 53 /* 54 * Information structure and handler for ACPI predefined names that can 55 * be repaired on a per-name basis. 56 */ 57 typedef 58 ACPI_STATUS (*ACPI_REPAIR_FUNCTION) ( 59 ACPI_EVALUATE_INFO *Info, 60 ACPI_OPERAND_OBJECT **ReturnObjectPtr); 61 62 typedef struct acpi_repair_info 63 { 64 char Name[ACPI_NAME_SIZE]; 65 ACPI_REPAIR_FUNCTION RepairFunction; 66 67 } ACPI_REPAIR_INFO; 68 69 70 /* Local prototypes */ 71 72 static const ACPI_REPAIR_INFO * 73 AcpiNsMatchComplexRepair ( 74 ACPI_NAMESPACE_NODE *Node); 75 76 static ACPI_STATUS 77 AcpiNsRepair_ALR ( 78 ACPI_EVALUATE_INFO *Info, 79 ACPI_OPERAND_OBJECT **ReturnObjectPtr); 80 81 static ACPI_STATUS 82 AcpiNsRepair_CID ( 83 ACPI_EVALUATE_INFO *Info, 84 ACPI_OPERAND_OBJECT **ReturnObjectPtr); 85 86 static ACPI_STATUS 87 AcpiNsRepair_CST ( 88 ACPI_EVALUATE_INFO *Info, 89 ACPI_OPERAND_OBJECT **ReturnObjectPtr); 90 91 static ACPI_STATUS 92 AcpiNsRepair_FDE ( 93 ACPI_EVALUATE_INFO *Info, 94 ACPI_OPERAND_OBJECT **ReturnObjectPtr); 95 96 static ACPI_STATUS 97 AcpiNsRepair_HID ( 98 ACPI_EVALUATE_INFO *Info, 99 ACPI_OPERAND_OBJECT **ReturnObjectPtr); 100 101 static ACPI_STATUS 102 AcpiNsRepair_PRT ( 103 ACPI_EVALUATE_INFO *Info, 104 ACPI_OPERAND_OBJECT **ReturnObjectPtr); 105 106 static ACPI_STATUS 107 AcpiNsRepair_PSS ( 108 ACPI_EVALUATE_INFO *Info, 109 ACPI_OPERAND_OBJECT **ReturnObjectPtr); 110 111 static ACPI_STATUS 112 AcpiNsRepair_TSS ( 113 ACPI_EVALUATE_INFO *Info, 114 ACPI_OPERAND_OBJECT **ReturnObjectPtr); 115 116 static ACPI_STATUS 117 AcpiNsCheckSortedList ( 118 ACPI_EVALUATE_INFO *Info, 119 ACPI_OPERAND_OBJECT *ReturnObject, 120 UINT32 StartIndex, 121 UINT32 ExpectedCount, 122 UINT32 SortIndex, 123 UINT8 SortDirection, 124 char *SortKeyName); 125 126 /* Values for SortDirection above */ 127 128 #define ACPI_SORT_ASCENDING 0 129 #define ACPI_SORT_DESCENDING 1 130 131 static void 132 AcpiNsRemoveElement ( 133 ACPI_OPERAND_OBJECT *ObjDesc, 134 UINT32 Index); 135 136 static void 137 AcpiNsSortList ( 138 ACPI_OPERAND_OBJECT **Elements, 139 UINT32 Count, 140 UINT32 Index, 141 UINT8 SortDirection); 142 143 144 /* 145 * This table contains the names of the predefined methods for which we can 146 * perform more complex repairs. 147 * 148 * As necessary: 149 * 150 * _ALR: Sort the list ascending by AmbientIlluminance 151 * _CID: Strings: uppercase all, remove any leading asterisk 152 * _CST: Sort the list ascending by C state type 153 * _FDE: Convert Buffer of BYTEs to a Buffer of DWORDs 154 * _GTM: Convert Buffer of BYTEs to a Buffer of DWORDs 155 * _HID: Strings: uppercase all, remove any leading asterisk 156 * _PRT: Fix reversed SourceName and SourceIndex 157 * _PSS: Sort the list descending by Power 158 * _TSS: Sort the list descending by Power 159 * 160 * Names that must be packages, but cannot be sorted: 161 * 162 * _BCL: Values are tied to the Package index where they appear, and cannot 163 * be moved or sorted. These index values are used for _BQC and _BCM. 164 * However, we can fix the case where a buffer is returned, by converting 165 * it to a Package of integers. 166 */ 167 static const ACPI_REPAIR_INFO AcpiNsRepairableNames[] = 168 { 169 {"_ALR", AcpiNsRepair_ALR}, 170 {"_CID", AcpiNsRepair_CID}, 171 {"_CST", AcpiNsRepair_CST}, 172 {"_FDE", AcpiNsRepair_FDE}, 173 {"_GTM", AcpiNsRepair_FDE}, /* _GTM has same repair as _FDE */ 174 {"_HID", AcpiNsRepair_HID}, 175 {"_PRT", AcpiNsRepair_PRT}, 176 {"_PSS", AcpiNsRepair_PSS}, 177 {"_TSS", AcpiNsRepair_TSS}, 178 {{0,0,0,0}, NULL} /* Table terminator */ 179 }; 180 181 182 #define ACPI_FDE_FIELD_COUNT 5 183 #define ACPI_FDE_BYTE_BUFFER_SIZE 5 184 #define ACPI_FDE_DWORD_BUFFER_SIZE (ACPI_FDE_FIELD_COUNT * sizeof (UINT32)) 185 186 187 /****************************************************************************** 188 * 189 * FUNCTION: AcpiNsComplexRepairs 190 * 191 * PARAMETERS: Info - Method execution information block 192 * Node - Namespace node for the method/object 193 * ValidateStatus - Original status of earlier validation 194 * ReturnObjectPtr - Pointer to the object returned from the 195 * evaluation of a method or object 196 * 197 * RETURN: Status. AE_OK if repair was successful. If name is not 198 * matched, ValidateStatus is returned. 199 * 200 * DESCRIPTION: Attempt to repair/convert a return object of a type that was 201 * not expected. 202 * 203 *****************************************************************************/ 204 205 ACPI_STATUS 206 AcpiNsComplexRepairs ( 207 ACPI_EVALUATE_INFO *Info, 208 ACPI_NAMESPACE_NODE *Node, 209 ACPI_STATUS ValidateStatus, 210 ACPI_OPERAND_OBJECT **ReturnObjectPtr) 211 { 212 const ACPI_REPAIR_INFO *Predefined; 213 ACPI_STATUS Status; 214 215 216 /* Check if this name is in the list of repairable names */ 217 218 Predefined = AcpiNsMatchComplexRepair (Node); 219 if (!Predefined) 220 { 221 return (ValidateStatus); 222 } 223 224 Status = Predefined->RepairFunction (Info, ReturnObjectPtr); 225 return (Status); 226 } 227 228 229 /****************************************************************************** 230 * 231 * FUNCTION: AcpiNsMatchComplexRepair 232 * 233 * PARAMETERS: Node - Namespace node for the method/object 234 * 235 * RETURN: Pointer to entry in repair table. NULL indicates not found. 236 * 237 * DESCRIPTION: Check an object name against the repairable object list. 238 * 239 *****************************************************************************/ 240 241 static const ACPI_REPAIR_INFO * 242 AcpiNsMatchComplexRepair ( 243 ACPI_NAMESPACE_NODE *Node) 244 { 245 const ACPI_REPAIR_INFO *ThisName; 246 247 248 /* Search info table for a repairable predefined method/object name */ 249 250 ThisName = AcpiNsRepairableNames; 251 while (ThisName->RepairFunction) 252 { 253 if (ACPI_COMPARE_NAME (Node->Name.Ascii, ThisName->Name)) 254 { 255 return (ThisName); 256 } 257 ThisName++; 258 } 259 260 return (NULL); /* Not found */ 261 } 262 263 264 /****************************************************************************** 265 * 266 * FUNCTION: AcpiNsRepair_ALR 267 * 268 * PARAMETERS: Info - Method execution information block 269 * ReturnObjectPtr - Pointer to the object returned from the 270 * evaluation of a method or object 271 * 272 * RETURN: Status. AE_OK if object is OK or was repaired successfully 273 * 274 * DESCRIPTION: Repair for the _ALR object. If necessary, sort the object list 275 * ascending by the ambient illuminance values. 276 * 277 *****************************************************************************/ 278 279 static ACPI_STATUS 280 AcpiNsRepair_ALR ( 281 ACPI_EVALUATE_INFO *Info, 282 ACPI_OPERAND_OBJECT **ReturnObjectPtr) 283 { 284 ACPI_OPERAND_OBJECT *ReturnObject = *ReturnObjectPtr; 285 ACPI_STATUS Status; 286 287 288 Status = AcpiNsCheckSortedList (Info, ReturnObject, 0, 2, 1, 289 ACPI_SORT_ASCENDING, "AmbientIlluminance"); 290 291 return (Status); 292 } 293 294 295 /****************************************************************************** 296 * 297 * FUNCTION: AcpiNsRepair_FDE 298 * 299 * PARAMETERS: Info - Method execution information block 300 * ReturnObjectPtr - Pointer to the object returned from the 301 * evaluation of a method or object 302 * 303 * RETURN: Status. AE_OK if object is OK or was repaired successfully 304 * 305 * DESCRIPTION: Repair for the _FDE and _GTM objects. The expected return 306 * value is a Buffer of 5 DWORDs. This function repairs a common 307 * problem where the return value is a Buffer of BYTEs, not 308 * DWORDs. 309 * 310 *****************************************************************************/ 311 312 static ACPI_STATUS 313 AcpiNsRepair_FDE ( 314 ACPI_EVALUATE_INFO *Info, 315 ACPI_OPERAND_OBJECT **ReturnObjectPtr) 316 { 317 ACPI_OPERAND_OBJECT *ReturnObject = *ReturnObjectPtr; 318 ACPI_OPERAND_OBJECT *BufferObject; 319 UINT8 *ByteBuffer; 320 UINT32 *DwordBuffer; 321 UINT32 i; 322 323 324 ACPI_FUNCTION_NAME (NsRepair_FDE); 325 326 327 switch (ReturnObject->Common.Type) 328 { 329 case ACPI_TYPE_BUFFER: 330 331 /* This is the expected type. Length should be (at least) 5 DWORDs */ 332 333 if (ReturnObject->Buffer.Length >= ACPI_FDE_DWORD_BUFFER_SIZE) 334 { 335 return (AE_OK); 336 } 337 338 /* We can only repair if we have exactly 5 BYTEs */ 339 340 if (ReturnObject->Buffer.Length != ACPI_FDE_BYTE_BUFFER_SIZE) 341 { 342 ACPI_WARN_PREDEFINED ((AE_INFO, Info->FullPathname, Info->NodeFlags, 343 "Incorrect return buffer length %u, expected %u", 344 ReturnObject->Buffer.Length, ACPI_FDE_DWORD_BUFFER_SIZE)); 345 346 return (AE_AML_OPERAND_TYPE); 347 } 348 349 /* Create the new (larger) buffer object */ 350 351 BufferObject = AcpiUtCreateBufferObject (ACPI_FDE_DWORD_BUFFER_SIZE); 352 if (!BufferObject) 353 { 354 return (AE_NO_MEMORY); 355 } 356 357 /* Expand each byte to a DWORD */ 358 359 ByteBuffer = ReturnObject->Buffer.Pointer; 360 DwordBuffer = ACPI_CAST_PTR (UINT32, BufferObject->Buffer.Pointer); 361 362 for (i = 0; i < ACPI_FDE_FIELD_COUNT; i++) 363 { 364 *DwordBuffer = (UINT32) *ByteBuffer; 365 DwordBuffer++; 366 ByteBuffer++; 367 } 368 369 ACPI_DEBUG_PRINT ((ACPI_DB_REPAIR, 370 "%s Expanded Byte Buffer to expected DWord Buffer\n", 371 Info->FullPathname)); 372 break; 373 374 default: 375 376 return (AE_AML_OPERAND_TYPE); 377 } 378 379 /* Delete the original return object, return the new buffer object */ 380 381 AcpiUtRemoveReference (ReturnObject); 382 *ReturnObjectPtr = BufferObject; 383 384 Info->ReturnFlags |= ACPI_OBJECT_REPAIRED; 385 return (AE_OK); 386 } 387 388 389 /****************************************************************************** 390 * 391 * FUNCTION: AcpiNsRepair_CID 392 * 393 * PARAMETERS: Info - Method execution information block 394 * ReturnObjectPtr - Pointer to the object returned from the 395 * evaluation of a method or object 396 * 397 * RETURN: Status. AE_OK if object is OK or was repaired successfully 398 * 399 * DESCRIPTION: Repair for the _CID object. If a string, ensure that all 400 * letters are uppercase and that there is no leading asterisk. 401 * If a Package, ensure same for all string elements. 402 * 403 *****************************************************************************/ 404 405 static ACPI_STATUS 406 AcpiNsRepair_CID ( 407 ACPI_EVALUATE_INFO *Info, 408 ACPI_OPERAND_OBJECT **ReturnObjectPtr) 409 { 410 ACPI_STATUS Status; 411 ACPI_OPERAND_OBJECT *ReturnObject = *ReturnObjectPtr; 412 ACPI_OPERAND_OBJECT **ElementPtr; 413 ACPI_OPERAND_OBJECT *OriginalElement; 414 UINT16 OriginalRefCount; 415 UINT32 i; 416 417 418 /* Check for _CID as a simple string */ 419 420 if (ReturnObject->Common.Type == ACPI_TYPE_STRING) 421 { 422 Status = AcpiNsRepair_HID (Info, ReturnObjectPtr); 423 return (Status); 424 } 425 426 /* Exit if not a Package */ 427 428 if (ReturnObject->Common.Type != ACPI_TYPE_PACKAGE) 429 { 430 return (AE_OK); 431 } 432 433 /* Examine each element of the _CID package */ 434 435 ElementPtr = ReturnObject->Package.Elements; 436 for (i = 0; i < ReturnObject->Package.Count; i++) 437 { 438 OriginalElement = *ElementPtr; 439 OriginalRefCount = OriginalElement->Common.ReferenceCount; 440 441 Status = AcpiNsRepair_HID (Info, ElementPtr); 442 if (ACPI_FAILURE (Status)) 443 { 444 return (Status); 445 } 446 447 /* Take care with reference counts */ 448 449 if (OriginalElement != *ElementPtr) 450 { 451 /* Element was replaced */ 452 453 (*ElementPtr)->Common.ReferenceCount = 454 OriginalRefCount; 455 456 AcpiUtRemoveReference (OriginalElement); 457 } 458 459 ElementPtr++; 460 } 461 462 return (AE_OK); 463 } 464 465 466 /****************************************************************************** 467 * 468 * FUNCTION: AcpiNsRepair_CST 469 * 470 * PARAMETERS: Info - Method execution information block 471 * ReturnObjectPtr - Pointer to the object returned from the 472 * evaluation of a method or object 473 * 474 * RETURN: Status. AE_OK if object is OK or was repaired successfully 475 * 476 * DESCRIPTION: Repair for the _CST object: 477 * 1. Sort the list ascending by C state type 478 * 2. Ensure type cannot be zero 479 * 3. A subpackage count of zero means _CST is meaningless 480 * 4. Count must match the number of C state subpackages 481 * 482 *****************************************************************************/ 483 484 static ACPI_STATUS 485 AcpiNsRepair_CST ( 486 ACPI_EVALUATE_INFO *Info, 487 ACPI_OPERAND_OBJECT **ReturnObjectPtr) 488 { 489 ACPI_OPERAND_OBJECT *ReturnObject = *ReturnObjectPtr; 490 ACPI_OPERAND_OBJECT **OuterElements; 491 UINT32 OuterElementCount; 492 ACPI_OPERAND_OBJECT *ObjDesc; 493 ACPI_STATUS Status; 494 BOOLEAN Removing; 495 UINT32 i; 496 497 498 ACPI_FUNCTION_NAME (NsRepair_CST); 499 500 501 /* 502 * Check if the C-state type values are proportional. 503 */ 504 OuterElementCount = ReturnObject->Package.Count - 1; 505 i = 0; 506 while (i < OuterElementCount) 507 { 508 OuterElements = &ReturnObject->Package.Elements[i + 1]; 509 Removing = FALSE; 510 511 if ((*OuterElements)->Package.Count == 0) 512 { 513 ACPI_WARN_PREDEFINED ((AE_INFO, Info->FullPathname, Info->NodeFlags, 514 "SubPackage[%u] - removing entry due to zero count", i)); 515 Removing = TRUE; 516 goto RemoveElement; 517 } 518 519 ObjDesc = (*OuterElements)->Package.Elements[1]; /* Index1 = Type */ 520 if ((UINT32) ObjDesc->Integer.Value == 0) 521 { 522 ACPI_WARN_PREDEFINED ((AE_INFO, Info->FullPathname, Info->NodeFlags, 523 "SubPackage[%u] - removing entry due to invalid Type(0)", i)); 524 Removing = TRUE; 525 } 526 527 RemoveElement: 528 if (Removing) 529 { 530 AcpiNsRemoveElement (ReturnObject, i + 1); 531 OuterElementCount--; 532 } 533 else 534 { 535 i++; 536 } 537 } 538 539 /* Update top-level package count, Type "Integer" checked elsewhere */ 540 541 ObjDesc = ReturnObject->Package.Elements[0]; 542 ObjDesc->Integer.Value = OuterElementCount; 543 544 /* 545 * Entries (subpackages) in the _CST Package must be sorted by the 546 * C-state type, in ascending order. 547 */ 548 Status = AcpiNsCheckSortedList (Info, ReturnObject, 1, 4, 1, 549 ACPI_SORT_ASCENDING, "C-State Type"); 550 if (ACPI_FAILURE (Status)) 551 { 552 return (Status); 553 } 554 555 return (AE_OK); 556 } 557 558 559 /****************************************************************************** 560 * 561 * FUNCTION: AcpiNsRepair_HID 562 * 563 * PARAMETERS: Info - Method execution information block 564 * ReturnObjectPtr - Pointer to the object returned from the 565 * evaluation of a method or object 566 * 567 * RETURN: Status. AE_OK if object is OK or was repaired successfully 568 * 569 * DESCRIPTION: Repair for the _HID object. If a string, ensure that all 570 * letters are uppercase and that there is no leading asterisk. 571 * 572 *****************************************************************************/ 573 574 static ACPI_STATUS 575 AcpiNsRepair_HID ( 576 ACPI_EVALUATE_INFO *Info, 577 ACPI_OPERAND_OBJECT **ReturnObjectPtr) 578 { 579 ACPI_OPERAND_OBJECT *ReturnObject = *ReturnObjectPtr; 580 ACPI_OPERAND_OBJECT *NewString; 581 char *Source; 582 char *Dest; 583 584 585 ACPI_FUNCTION_NAME (NsRepair_HID); 586 587 588 /* We only care about string _HID objects (not integers) */ 589 590 if (ReturnObject->Common.Type != ACPI_TYPE_STRING) 591 { 592 return (AE_OK); 593 } 594 595 if (ReturnObject->String.Length == 0) 596 { 597 ACPI_WARN_PREDEFINED ((AE_INFO, Info->FullPathname, Info->NodeFlags, 598 "Invalid zero-length _HID or _CID string")); 599 600 /* Return AE_OK anyway, let driver handle it */ 601 602 Info->ReturnFlags |= ACPI_OBJECT_REPAIRED; 603 return (AE_OK); 604 } 605 606 /* It is simplest to always create a new string object */ 607 608 NewString = AcpiUtCreateStringObject (ReturnObject->String.Length); 609 if (!NewString) 610 { 611 return (AE_NO_MEMORY); 612 } 613 614 /* 615 * Remove a leading asterisk if present. For some unknown reason, there 616 * are many machines in the field that contains IDs like this. 617 * 618 * Examples: "*PNP0C03", "*ACPI0003" 619 */ 620 Source = ReturnObject->String.Pointer; 621 if (*Source == '*') 622 { 623 Source++; 624 NewString->String.Length--; 625 626 ACPI_DEBUG_PRINT ((ACPI_DB_REPAIR, 627 "%s: Removed invalid leading asterisk\n", Info->FullPathname)); 628 } 629 630 /* 631 * Copy and uppercase the string. From the ACPI 5.0 specification: 632 * 633 * A valid PNP ID must be of the form "AAA####" where A is an uppercase 634 * letter and # is a hex digit. A valid ACPI ID must be of the form 635 * "NNNN####" where N is an uppercase letter or decimal digit, and 636 * # is a hex digit. 637 */ 638 for (Dest = NewString->String.Pointer; *Source; Dest++, Source++) 639 { 640 *Dest = (char) toupper ((int) *Source); 641 } 642 643 AcpiUtRemoveReference (ReturnObject); 644 *ReturnObjectPtr = NewString; 645 return (AE_OK); 646 } 647 648 649 /****************************************************************************** 650 * 651 * FUNCTION: AcpiNsRepair_PRT 652 * 653 * PARAMETERS: Info - Method execution information block 654 * ReturnObjectPtr - Pointer to the object returned from the 655 * evaluation of a method or object 656 * 657 * RETURN: Status. AE_OK if object is OK or was repaired successfully 658 * 659 * DESCRIPTION: Repair for the _PRT object. If necessary, fix reversed 660 * SourceName and SourceIndex field, a common BIOS bug. 661 * 662 *****************************************************************************/ 663 664 static ACPI_STATUS 665 AcpiNsRepair_PRT ( 666 ACPI_EVALUATE_INFO *Info, 667 ACPI_OPERAND_OBJECT **ReturnObjectPtr) 668 { 669 ACPI_OPERAND_OBJECT *PackageObject = *ReturnObjectPtr; 670 ACPI_OPERAND_OBJECT **TopObjectList; 671 ACPI_OPERAND_OBJECT **SubObjectList; 672 ACPI_OPERAND_OBJECT *ObjDesc; 673 ACPI_OPERAND_OBJECT *SubPackage; 674 UINT32 ElementCount; 675 UINT32 Index; 676 677 678 /* Each element in the _PRT package is a subpackage */ 679 680 TopObjectList = PackageObject->Package.Elements; 681 ElementCount = PackageObject->Package.Count; 682 683 /* Examine each subpackage */ 684 685 for (Index = 0; Index < ElementCount; Index++, TopObjectList++) 686 { 687 SubPackage = *TopObjectList; 688 SubObjectList = SubPackage->Package.Elements; 689 690 /* Check for minimum required element count */ 691 692 if (SubPackage->Package.Count < 4) 693 { 694 continue; 695 } 696 697 /* 698 * If the BIOS has erroneously reversed the _PRT SourceName (index 2) 699 * and the SourceIndex (index 3), fix it. _PRT is important enough to 700 * workaround this BIOS error. This also provides compatibility with 701 * other ACPI implementations. 702 */ 703 ObjDesc = SubObjectList[3]; 704 if (!ObjDesc || (ObjDesc->Common.Type != ACPI_TYPE_INTEGER)) 705 { 706 SubObjectList[3] = SubObjectList[2]; 707 SubObjectList[2] = ObjDesc; 708 Info->ReturnFlags |= ACPI_OBJECT_REPAIRED; 709 710 ACPI_WARN_PREDEFINED ((AE_INFO, 711 Info->FullPathname, Info->NodeFlags, 712 "PRT[%X]: Fixed reversed SourceName and SourceIndex", 713 Index)); 714 } 715 } 716 717 return (AE_OK); 718 } 719 720 721 /****************************************************************************** 722 * 723 * FUNCTION: AcpiNsRepair_PSS 724 * 725 * PARAMETERS: Info - Method execution information block 726 * ReturnObjectPtr - Pointer to the object returned from the 727 * evaluation of a method or object 728 * 729 * RETURN: Status. AE_OK if object is OK or was repaired successfully 730 * 731 * DESCRIPTION: Repair for the _PSS object. If necessary, sort the object list 732 * by the CPU frequencies. Check that the power dissipation values 733 * are all proportional to CPU frequency (i.e., sorting by 734 * frequency should be the same as sorting by power.) 735 * 736 *****************************************************************************/ 737 738 static ACPI_STATUS 739 AcpiNsRepair_PSS ( 740 ACPI_EVALUATE_INFO *Info, 741 ACPI_OPERAND_OBJECT **ReturnObjectPtr) 742 { 743 ACPI_OPERAND_OBJECT *ReturnObject = *ReturnObjectPtr; 744 ACPI_OPERAND_OBJECT **OuterElements; 745 UINT32 OuterElementCount; 746 ACPI_OPERAND_OBJECT **Elements; 747 ACPI_OPERAND_OBJECT *ObjDesc; 748 UINT32 PreviousValue; 749 ACPI_STATUS Status; 750 UINT32 i; 751 752 753 /* 754 * Entries (subpackages) in the _PSS Package must be sorted by power 755 * dissipation, in descending order. If it appears that the list is 756 * incorrectly sorted, sort it. We sort by CpuFrequency, since this 757 * should be proportional to the power. 758 */ 759 Status =AcpiNsCheckSortedList (Info, ReturnObject, 0, 6, 0, 760 ACPI_SORT_DESCENDING, "CpuFrequency"); 761 if (ACPI_FAILURE (Status)) 762 { 763 return (Status); 764 } 765 766 /* 767 * We now know the list is correctly sorted by CPU frequency. Check if 768 * the power dissipation values are proportional. 769 */ 770 PreviousValue = ACPI_UINT32_MAX; 771 OuterElements = ReturnObject->Package.Elements; 772 OuterElementCount = ReturnObject->Package.Count; 773 774 for (i = 0; i < OuterElementCount; i++) 775 { 776 Elements = (*OuterElements)->Package.Elements; 777 ObjDesc = Elements[1]; /* Index1 = PowerDissipation */ 778 779 if ((UINT32) ObjDesc->Integer.Value > PreviousValue) 780 { 781 ACPI_WARN_PREDEFINED ((AE_INFO, Info->FullPathname, Info->NodeFlags, 782 "SubPackage[%u,%u] - suspicious power dissipation values", 783 i-1, i)); 784 } 785 786 PreviousValue = (UINT32) ObjDesc->Integer.Value; 787 OuterElements++; 788 } 789 790 return (AE_OK); 791 } 792 793 794 /****************************************************************************** 795 * 796 * FUNCTION: AcpiNsRepair_TSS 797 * 798 * PARAMETERS: Info - Method execution information block 799 * ReturnObjectPtr - Pointer to the object returned from the 800 * evaluation of a method or object 801 * 802 * RETURN: Status. AE_OK if object is OK or was repaired successfully 803 * 804 * DESCRIPTION: Repair for the _TSS object. If necessary, sort the object list 805 * descending by the power dissipation values. 806 * 807 *****************************************************************************/ 808 809 static ACPI_STATUS 810 AcpiNsRepair_TSS ( 811 ACPI_EVALUATE_INFO *Info, 812 ACPI_OPERAND_OBJECT **ReturnObjectPtr) 813 { 814 ACPI_OPERAND_OBJECT *ReturnObject = *ReturnObjectPtr; 815 ACPI_STATUS Status; 816 ACPI_NAMESPACE_NODE *Node; 817 818 819 /* 820 * We can only sort the _TSS return package if there is no _PSS in the 821 * same scope. This is because if _PSS is present, the ACPI specification 822 * dictates that the _TSS Power Dissipation field is to be ignored, and 823 * therefore some BIOSs leave garbage values in the _TSS Power field(s). 824 * In this case, it is best to just return the _TSS package as-is. 825 * (May, 2011) 826 */ 827 Status = AcpiNsGetNode (Info->Node, "^_PSS", 828 ACPI_NS_NO_UPSEARCH, &Node); 829 if (ACPI_SUCCESS (Status)) 830 { 831 return (AE_OK); 832 } 833 834 Status = AcpiNsCheckSortedList (Info, ReturnObject, 0, 5, 1, 835 ACPI_SORT_DESCENDING, "PowerDissipation"); 836 837 return (Status); 838 } 839 840 841 /****************************************************************************** 842 * 843 * FUNCTION: AcpiNsCheckSortedList 844 * 845 * PARAMETERS: Info - Method execution information block 846 * ReturnObject - Pointer to the top-level returned object 847 * StartIndex - Index of the first subpackage 848 * ExpectedCount - Minimum length of each subpackage 849 * SortIndex - Subpackage entry to sort on 850 * SortDirection - Ascending or descending 851 * SortKeyName - Name of the SortIndex field 852 * 853 * RETURN: Status. AE_OK if the list is valid and is sorted correctly or 854 * has been repaired by sorting the list. 855 * 856 * DESCRIPTION: Check if the package list is valid and sorted correctly by the 857 * SortIndex. If not, then sort the list. 858 * 859 *****************************************************************************/ 860 861 static ACPI_STATUS 862 AcpiNsCheckSortedList ( 863 ACPI_EVALUATE_INFO *Info, 864 ACPI_OPERAND_OBJECT *ReturnObject, 865 UINT32 StartIndex, 866 UINT32 ExpectedCount, 867 UINT32 SortIndex, 868 UINT8 SortDirection, 869 char *SortKeyName) 870 { 871 UINT32 OuterElementCount; 872 ACPI_OPERAND_OBJECT **OuterElements; 873 ACPI_OPERAND_OBJECT **Elements; 874 ACPI_OPERAND_OBJECT *ObjDesc; 875 UINT32 i; 876 UINT32 PreviousValue; 877 878 879 ACPI_FUNCTION_NAME (NsCheckSortedList); 880 881 882 /* The top-level object must be a package */ 883 884 if (ReturnObject->Common.Type != ACPI_TYPE_PACKAGE) 885 { 886 return (AE_AML_OPERAND_TYPE); 887 } 888 889 /* 890 * NOTE: assumes list of subpackages contains no NULL elements. 891 * Any NULL elements should have been removed by earlier call 892 * to AcpiNsRemoveNullElements. 893 */ 894 OuterElementCount = ReturnObject->Package.Count; 895 if (!OuterElementCount || StartIndex >= OuterElementCount) 896 { 897 return (AE_AML_PACKAGE_LIMIT); 898 } 899 900 OuterElements = &ReturnObject->Package.Elements[StartIndex]; 901 OuterElementCount -= StartIndex; 902 903 PreviousValue = 0; 904 if (SortDirection == ACPI_SORT_DESCENDING) 905 { 906 PreviousValue = ACPI_UINT32_MAX; 907 } 908 909 /* Examine each subpackage */ 910 911 for (i = 0; i < OuterElementCount; i++) 912 { 913 /* Each element of the top-level package must also be a package */ 914 915 if ((*OuterElements)->Common.Type != ACPI_TYPE_PACKAGE) 916 { 917 return (AE_AML_OPERAND_TYPE); 918 } 919 920 /* Each subpackage must have the minimum length */ 921 922 if ((*OuterElements)->Package.Count < ExpectedCount) 923 { 924 return (AE_AML_PACKAGE_LIMIT); 925 } 926 927 Elements = (*OuterElements)->Package.Elements; 928 ObjDesc = Elements[SortIndex]; 929 930 if (ObjDesc->Common.Type != ACPI_TYPE_INTEGER) 931 { 932 return (AE_AML_OPERAND_TYPE); 933 } 934 935 /* 936 * The list must be sorted in the specified order. If we detect a 937 * discrepancy, sort the entire list. 938 */ 939 if (((SortDirection == ACPI_SORT_ASCENDING) && 940 (ObjDesc->Integer.Value < PreviousValue)) || 941 ((SortDirection == ACPI_SORT_DESCENDING) && 942 (ObjDesc->Integer.Value > PreviousValue))) 943 { 944 AcpiNsSortList (&ReturnObject->Package.Elements[StartIndex], 945 OuterElementCount, SortIndex, SortDirection); 946 947 Info->ReturnFlags |= ACPI_OBJECT_REPAIRED; 948 949 ACPI_DEBUG_PRINT ((ACPI_DB_REPAIR, 950 "%s: Repaired unsorted list - now sorted by %s\n", 951 Info->FullPathname, SortKeyName)); 952 return (AE_OK); 953 } 954 955 PreviousValue = (UINT32) ObjDesc->Integer.Value; 956 OuterElements++; 957 } 958 959 return (AE_OK); 960 } 961 962 963 /****************************************************************************** 964 * 965 * FUNCTION: AcpiNsSortList 966 * 967 * PARAMETERS: Elements - Package object element list 968 * Count - Element count for above 969 * Index - Sort by which package element 970 * SortDirection - Ascending or Descending sort 971 * 972 * RETURN: None 973 * 974 * DESCRIPTION: Sort the objects that are in a package element list. 975 * 976 * NOTE: Assumes that all NULL elements have been removed from the package, 977 * and that all elements have been verified to be of type Integer. 978 * 979 *****************************************************************************/ 980 981 static void 982 AcpiNsSortList ( 983 ACPI_OPERAND_OBJECT **Elements, 984 UINT32 Count, 985 UINT32 Index, 986 UINT8 SortDirection) 987 { 988 ACPI_OPERAND_OBJECT *ObjDesc1; 989 ACPI_OPERAND_OBJECT *ObjDesc2; 990 ACPI_OPERAND_OBJECT *TempObj; 991 UINT32 i; 992 UINT32 j; 993 994 995 /* Simple bubble sort */ 996 997 for (i = 1; i < Count; i++) 998 { 999 for (j = (Count - 1); j >= i; j--) 1000 { 1001 ObjDesc1 = Elements[j-1]->Package.Elements[Index]; 1002 ObjDesc2 = Elements[j]->Package.Elements[Index]; 1003 1004 if (((SortDirection == ACPI_SORT_ASCENDING) && 1005 (ObjDesc1->Integer.Value > ObjDesc2->Integer.Value)) || 1006 1007 ((SortDirection == ACPI_SORT_DESCENDING) && 1008 (ObjDesc1->Integer.Value < ObjDesc2->Integer.Value))) 1009 { 1010 TempObj = Elements[j-1]; 1011 Elements[j-1] = Elements[j]; 1012 Elements[j] = TempObj; 1013 } 1014 } 1015 } 1016 } 1017 1018 1019 /****************************************************************************** 1020 * 1021 * FUNCTION: AcpiNsRemoveElement 1022 * 1023 * PARAMETERS: ObjDesc - Package object element list 1024 * Index - Index of element to remove 1025 * 1026 * RETURN: None 1027 * 1028 * DESCRIPTION: Remove the requested element of a package and delete it. 1029 * 1030 *****************************************************************************/ 1031 1032 static void 1033 AcpiNsRemoveElement ( 1034 ACPI_OPERAND_OBJECT *ObjDesc, 1035 UINT32 Index) 1036 { 1037 ACPI_OPERAND_OBJECT **Source; 1038 ACPI_OPERAND_OBJECT **Dest; 1039 UINT32 Count; 1040 UINT32 NewCount; 1041 UINT32 i; 1042 1043 1044 ACPI_FUNCTION_NAME (NsRemoveElement); 1045 1046 1047 Count = ObjDesc->Package.Count; 1048 NewCount = Count - 1; 1049 1050 Source = ObjDesc->Package.Elements; 1051 Dest = Source; 1052 1053 /* Examine all elements of the package object, remove matched index */ 1054 1055 for (i = 0; i < Count; i++) 1056 { 1057 if (i == Index) 1058 { 1059 AcpiUtRemoveReference (*Source); /* Remove one ref for being in pkg */ 1060 AcpiUtRemoveReference (*Source); 1061 } 1062 else 1063 { 1064 *Dest = *Source; 1065 Dest++; 1066 } 1067 Source++; 1068 } 1069 1070 /* NULL terminate list and update the package count */ 1071 1072 *Dest = NULL; 1073 ObjDesc->Package.Count = NewCount; 1074 } 1075