1 /******************************************************************************* 2 * 3 * Module Name: utresrc - Resource management utilities 4 * 5 ******************************************************************************/ 6 7 /* 8 * Copyright (C) 2000 - 2020, 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 "acpi.h" 45 #include "accommon.h" 46 #include "acresrc.h" 47 48 49 #define _COMPONENT ACPI_UTILITIES 50 ACPI_MODULE_NAME ("utresrc") 51 52 53 /* 54 * Base sizes of the raw AML resource descriptors, indexed by resource type. 55 * Zero indicates a reserved (and therefore invalid) resource type. 56 */ 57 const UINT8 AcpiGbl_ResourceAmlSizes[] = 58 { 59 /* Small descriptors */ 60 61 0, 62 0, 63 0, 64 0, 65 ACPI_AML_SIZE_SMALL (AML_RESOURCE_IRQ), 66 ACPI_AML_SIZE_SMALL (AML_RESOURCE_DMA), 67 ACPI_AML_SIZE_SMALL (AML_RESOURCE_START_DEPENDENT), 68 ACPI_AML_SIZE_SMALL (AML_RESOURCE_END_DEPENDENT), 69 ACPI_AML_SIZE_SMALL (AML_RESOURCE_IO), 70 ACPI_AML_SIZE_SMALL (AML_RESOURCE_FIXED_IO), 71 ACPI_AML_SIZE_SMALL (AML_RESOURCE_FIXED_DMA), 72 0, 73 0, 74 0, 75 ACPI_AML_SIZE_SMALL (AML_RESOURCE_VENDOR_SMALL), 76 ACPI_AML_SIZE_SMALL (AML_RESOURCE_END_TAG), 77 78 /* Large descriptors */ 79 80 0, 81 ACPI_AML_SIZE_LARGE (AML_RESOURCE_MEMORY24), 82 ACPI_AML_SIZE_LARGE (AML_RESOURCE_GENERIC_REGISTER), 83 0, 84 ACPI_AML_SIZE_LARGE (AML_RESOURCE_VENDOR_LARGE), 85 ACPI_AML_SIZE_LARGE (AML_RESOURCE_MEMORY32), 86 ACPI_AML_SIZE_LARGE (AML_RESOURCE_FIXED_MEMORY32), 87 ACPI_AML_SIZE_LARGE (AML_RESOURCE_ADDRESS32), 88 ACPI_AML_SIZE_LARGE (AML_RESOURCE_ADDRESS16), 89 ACPI_AML_SIZE_LARGE (AML_RESOURCE_EXTENDED_IRQ), 90 ACPI_AML_SIZE_LARGE (AML_RESOURCE_ADDRESS64), 91 ACPI_AML_SIZE_LARGE (AML_RESOURCE_EXTENDED_ADDRESS64), 92 ACPI_AML_SIZE_LARGE (AML_RESOURCE_GPIO), 93 ACPI_AML_SIZE_LARGE (AML_RESOURCE_PIN_FUNCTION), 94 ACPI_AML_SIZE_LARGE (AML_RESOURCE_COMMON_SERIALBUS), 95 ACPI_AML_SIZE_LARGE (AML_RESOURCE_PIN_CONFIG), 96 ACPI_AML_SIZE_LARGE (AML_RESOURCE_PIN_GROUP), 97 ACPI_AML_SIZE_LARGE (AML_RESOURCE_PIN_GROUP_FUNCTION), 98 ACPI_AML_SIZE_LARGE (AML_RESOURCE_PIN_GROUP_CONFIG), 99 }; 100 101 const UINT8 AcpiGbl_ResourceAmlSerialBusSizes[] = 102 { 103 0, 104 ACPI_AML_SIZE_LARGE (AML_RESOURCE_I2C_SERIALBUS), 105 ACPI_AML_SIZE_LARGE (AML_RESOURCE_SPI_SERIALBUS), 106 ACPI_AML_SIZE_LARGE (AML_RESOURCE_UART_SERIALBUS), 107 }; 108 109 110 /* 111 * Resource types, used to validate the resource length field. 112 * The length of fixed-length types must match exactly, variable 113 * lengths must meet the minimum required length, etc. 114 * Zero indicates a reserved (and therefore invalid) resource type. 115 */ 116 static const UINT8 AcpiGbl_ResourceTypes[] = 117 { 118 /* Small descriptors */ 119 120 0, 121 0, 122 0, 123 0, 124 ACPI_SMALL_VARIABLE_LENGTH, /* 04 IRQ */ 125 ACPI_FIXED_LENGTH, /* 05 DMA */ 126 ACPI_SMALL_VARIABLE_LENGTH, /* 06 StartDependentFunctions */ 127 ACPI_FIXED_LENGTH, /* 07 EndDependentFunctions */ 128 ACPI_FIXED_LENGTH, /* 08 IO */ 129 ACPI_FIXED_LENGTH, /* 09 FixedIO */ 130 ACPI_FIXED_LENGTH, /* 0A FixedDMA */ 131 0, 132 0, 133 0, 134 ACPI_VARIABLE_LENGTH, /* 0E VendorShort */ 135 ACPI_FIXED_LENGTH, /* 0F EndTag */ 136 137 /* Large descriptors */ 138 139 0, 140 ACPI_FIXED_LENGTH, /* 01 Memory24 */ 141 ACPI_FIXED_LENGTH, /* 02 GenericRegister */ 142 0, 143 ACPI_VARIABLE_LENGTH, /* 04 VendorLong */ 144 ACPI_FIXED_LENGTH, /* 05 Memory32 */ 145 ACPI_FIXED_LENGTH, /* 06 Memory32Fixed */ 146 ACPI_VARIABLE_LENGTH, /* 07 Dword* address */ 147 ACPI_VARIABLE_LENGTH, /* 08 Word* address */ 148 ACPI_VARIABLE_LENGTH, /* 09 ExtendedIRQ */ 149 ACPI_VARIABLE_LENGTH, /* 0A Qword* address */ 150 ACPI_FIXED_LENGTH, /* 0B Extended* address */ 151 ACPI_VARIABLE_LENGTH, /* 0C Gpio* */ 152 ACPI_VARIABLE_LENGTH, /* 0D PinFunction */ 153 ACPI_VARIABLE_LENGTH, /* 0E *SerialBus */ 154 ACPI_VARIABLE_LENGTH, /* 0F PinConfig */ 155 ACPI_VARIABLE_LENGTH, /* 10 PinGroup */ 156 ACPI_VARIABLE_LENGTH, /* 11 PinGroupFunction */ 157 ACPI_VARIABLE_LENGTH, /* 12 PinGroupConfig */ 158 }; 159 160 161 /******************************************************************************* 162 * 163 * FUNCTION: AcpiUtWalkAmlResources 164 * 165 * PARAMETERS: WalkState - Current walk info 166 * PARAMETERS: Aml - Pointer to the raw AML resource template 167 * AmlLength - Length of the entire template 168 * UserFunction - Called once for each descriptor found. If 169 * NULL, a pointer to the EndTag is returned 170 * Context - Passed to UserFunction 171 * 172 * RETURN: Status 173 * 174 * DESCRIPTION: Walk a raw AML resource list(buffer). User function called 175 * once for each resource found. 176 * 177 ******************************************************************************/ 178 179 ACPI_STATUS 180 AcpiUtWalkAmlResources ( 181 ACPI_WALK_STATE *WalkState, 182 UINT8 *Aml, 183 ACPI_SIZE AmlLength, 184 ACPI_WALK_AML_CALLBACK UserFunction, 185 void **Context) 186 { 187 ACPI_STATUS Status; 188 UINT8 *EndAml; 189 UINT8 ResourceIndex; 190 UINT32 Length; 191 UINT32 Offset = 0; 192 UINT8 EndTag[2] = {0x79, 0x00}; 193 194 195 ACPI_FUNCTION_TRACE (UtWalkAmlResources); 196 197 198 /* The absolute minimum resource template is one EndTag descriptor */ 199 200 if (AmlLength < sizeof (AML_RESOURCE_END_TAG)) 201 { 202 return_ACPI_STATUS (AE_AML_NO_RESOURCE_END_TAG); 203 } 204 205 /* Point to the end of the resource template buffer */ 206 207 EndAml = Aml + AmlLength; 208 209 /* Walk the byte list, abort on any invalid descriptor type or length */ 210 211 while (Aml < EndAml) 212 { 213 /* Validate the Resource Type and Resource Length */ 214 215 Status = AcpiUtValidateResource (WalkState, Aml, &ResourceIndex); 216 if (ACPI_FAILURE (Status)) 217 { 218 /* 219 * Exit on failure. Cannot continue because the descriptor 220 * length may be bogus also. 221 */ 222 return_ACPI_STATUS (Status); 223 } 224 225 /* Get the length of this descriptor */ 226 227 Length = AcpiUtGetDescriptorLength (Aml); 228 229 /* Invoke the user function */ 230 231 if (UserFunction) 232 { 233 Status = UserFunction ( 234 Aml, Length, Offset, ResourceIndex, Context); 235 if (ACPI_FAILURE (Status)) 236 { 237 return_ACPI_STATUS (Status); 238 } 239 } 240 241 /* An EndTag descriptor terminates this resource template */ 242 243 if (AcpiUtGetResourceType (Aml) == ACPI_RESOURCE_NAME_END_TAG) 244 { 245 /* 246 * There must be at least one more byte in the buffer for 247 * the 2nd byte of the EndTag 248 */ 249 if ((Aml + 1) >= EndAml) 250 { 251 return_ACPI_STATUS (AE_AML_NO_RESOURCE_END_TAG); 252 } 253 254 /* 255 * Don't attempt to perform any validation on the 2nd byte. 256 * Although all known ASL compilers insert a zero for the 2nd 257 * byte, it can also be a checksum (as per the ACPI spec), 258 * and this is occasionally seen in the field. July 2017. 259 */ 260 261 /* Return the pointer to the EndTag if requested */ 262 263 if (!UserFunction) 264 { 265 *Context = Aml; 266 } 267 268 /* Normal exit */ 269 270 return_ACPI_STATUS (AE_OK); 271 } 272 273 Aml += Length; 274 Offset += Length; 275 } 276 277 /* Did not find an EndTag descriptor */ 278 279 if (UserFunction) 280 { 281 /* Insert an EndTag anyway. AcpiRsGetListLength always leaves room */ 282 283 (void) AcpiUtValidateResource (WalkState, EndTag, &ResourceIndex); 284 Status = UserFunction (EndTag, 2, Offset, ResourceIndex, Context); 285 if (ACPI_FAILURE (Status)) 286 { 287 return_ACPI_STATUS (Status); 288 } 289 } 290 291 return_ACPI_STATUS (AE_AML_NO_RESOURCE_END_TAG); 292 } 293 294 295 /******************************************************************************* 296 * 297 * FUNCTION: AcpiUtValidateResource 298 * 299 * PARAMETERS: WalkState - Current walk info 300 * Aml - Pointer to the raw AML resource descriptor 301 * ReturnIndex - Where the resource index is returned. NULL 302 * if the index is not required. 303 * 304 * RETURN: Status, and optionally the Index into the global resource tables 305 * 306 * DESCRIPTION: Validate an AML resource descriptor by checking the Resource 307 * Type and Resource Length. Returns an index into the global 308 * resource information/dispatch tables for later use. 309 * 310 ******************************************************************************/ 311 312 ACPI_STATUS 313 AcpiUtValidateResource ( 314 ACPI_WALK_STATE *WalkState, 315 void *Aml, 316 UINT8 *ReturnIndex) 317 { 318 AML_RESOURCE *AmlResource; 319 UINT8 ResourceType; 320 UINT8 ResourceIndex; 321 ACPI_RS_LENGTH ResourceLength; 322 ACPI_RS_LENGTH MinimumResourceLength; 323 324 325 ACPI_FUNCTION_ENTRY (); 326 327 328 /* 329 * 1) Validate the ResourceType field (Byte 0) 330 */ 331 ResourceType = ACPI_GET8 (Aml); 332 333 /* 334 * Byte 0 contains the descriptor name (Resource Type) 335 * Examine the large/small bit in the resource header 336 */ 337 if (ResourceType & ACPI_RESOURCE_NAME_LARGE) 338 { 339 /* Verify the large resource type (name) against the max */ 340 341 if (ResourceType > ACPI_RESOURCE_NAME_LARGE_MAX) 342 { 343 goto InvalidResource; 344 } 345 346 /* 347 * Large Resource Type -- bits 6:0 contain the name 348 * Translate range 0x80-0x8B to index range 0x10-0x1B 349 */ 350 ResourceIndex = (UINT8) (ResourceType - 0x70); 351 } 352 else 353 { 354 /* 355 * Small Resource Type -- bits 6:3 contain the name 356 * Shift range to index range 0x00-0x0F 357 */ 358 ResourceIndex = (UINT8) 359 ((ResourceType & ACPI_RESOURCE_NAME_SMALL_MASK) >> 3); 360 } 361 362 /* 363 * Check validity of the resource type, via AcpiGbl_ResourceTypes. 364 * Zero indicates an invalid resource. 365 */ 366 if (!AcpiGbl_ResourceTypes[ResourceIndex]) 367 { 368 goto InvalidResource; 369 } 370 371 /* 372 * Validate the ResourceLength field. This ensures that the length 373 * is at least reasonable, and guarantees that it is non-zero. 374 */ 375 ResourceLength = AcpiUtGetResourceLength (Aml); 376 MinimumResourceLength = AcpiGbl_ResourceAmlSizes[ResourceIndex]; 377 378 /* Validate based upon the type of resource - fixed length or variable */ 379 380 switch (AcpiGbl_ResourceTypes[ResourceIndex]) 381 { 382 case ACPI_FIXED_LENGTH: 383 384 /* Fixed length resource, length must match exactly */ 385 386 if (ResourceLength != MinimumResourceLength) 387 { 388 goto BadResourceLength; 389 } 390 break; 391 392 case ACPI_VARIABLE_LENGTH: 393 394 /* Variable length resource, length must be at least the minimum */ 395 396 if (ResourceLength < MinimumResourceLength) 397 { 398 goto BadResourceLength; 399 } 400 break; 401 402 case ACPI_SMALL_VARIABLE_LENGTH: 403 404 /* Small variable length resource, length can be (Min) or (Min-1) */ 405 406 if ((ResourceLength > MinimumResourceLength) || 407 (ResourceLength < (MinimumResourceLength - 1))) 408 { 409 goto BadResourceLength; 410 } 411 break; 412 413 default: 414 415 /* Shouldn't happen (because of validation earlier), but be sure */ 416 417 goto InvalidResource; 418 } 419 420 AmlResource = ACPI_CAST_PTR (AML_RESOURCE, Aml); 421 if (ResourceType == ACPI_RESOURCE_NAME_SERIAL_BUS) 422 { 423 /* Validate the BusType field */ 424 425 if ((AmlResource->CommonSerialBus.Type == 0) || 426 (AmlResource->CommonSerialBus.Type > AML_RESOURCE_MAX_SERIALBUSTYPE)) 427 { 428 if (WalkState) 429 { 430 ACPI_ERROR ((AE_INFO, 431 "Invalid/unsupported SerialBus resource descriptor: BusType 0x%2.2X", 432 AmlResource->CommonSerialBus.Type)); 433 } 434 return (AE_AML_INVALID_RESOURCE_TYPE); 435 } 436 } 437 438 /* Optionally return the resource table index */ 439 440 if (ReturnIndex) 441 { 442 *ReturnIndex = ResourceIndex; 443 } 444 445 return (AE_OK); 446 447 448 InvalidResource: 449 450 if (WalkState) 451 { 452 ACPI_ERROR ((AE_INFO, 453 "Invalid/unsupported resource descriptor: Type 0x%2.2X", 454 ResourceType)); 455 } 456 return (AE_AML_INVALID_RESOURCE_TYPE); 457 458 BadResourceLength: 459 460 if (WalkState) 461 { 462 ACPI_ERROR ((AE_INFO, 463 "Invalid resource descriptor length: Type " 464 "0x%2.2X, Length 0x%4.4X, MinLength 0x%4.4X", 465 ResourceType, ResourceLength, MinimumResourceLength)); 466 } 467 return (AE_AML_BAD_RESOURCE_LENGTH); 468 } 469 470 471 /******************************************************************************* 472 * 473 * FUNCTION: AcpiUtGetResourceType 474 * 475 * PARAMETERS: Aml - Pointer to the raw AML resource descriptor 476 * 477 * RETURN: The Resource Type with no extraneous bits (except the 478 * Large/Small descriptor bit -- this is left alone) 479 * 480 * DESCRIPTION: Extract the Resource Type/Name from the first byte of 481 * a resource descriptor. 482 * 483 ******************************************************************************/ 484 485 UINT8 486 AcpiUtGetResourceType ( 487 void *Aml) 488 { 489 ACPI_FUNCTION_ENTRY (); 490 491 492 /* 493 * Byte 0 contains the descriptor name (Resource Type) 494 * Examine the large/small bit in the resource header 495 */ 496 if (ACPI_GET8 (Aml) & ACPI_RESOURCE_NAME_LARGE) 497 { 498 /* Large Resource Type -- bits 6:0 contain the name */ 499 500 return (ACPI_GET8 (Aml)); 501 } 502 else 503 { 504 /* Small Resource Type -- bits 6:3 contain the name */ 505 506 return ((UINT8) (ACPI_GET8 (Aml) & ACPI_RESOURCE_NAME_SMALL_MASK)); 507 } 508 } 509 510 511 /******************************************************************************* 512 * 513 * FUNCTION: AcpiUtGetResourceLength 514 * 515 * PARAMETERS: Aml - Pointer to the raw AML resource descriptor 516 * 517 * RETURN: Byte Length 518 * 519 * DESCRIPTION: Get the "Resource Length" of a raw AML descriptor. By 520 * definition, this does not include the size of the descriptor 521 * header or the length field itself. 522 * 523 ******************************************************************************/ 524 525 UINT16 526 AcpiUtGetResourceLength ( 527 void *Aml) 528 { 529 ACPI_RS_LENGTH ResourceLength; 530 531 532 ACPI_FUNCTION_ENTRY (); 533 534 535 /* 536 * Byte 0 contains the descriptor name (Resource Type) 537 * Examine the large/small bit in the resource header 538 */ 539 if (ACPI_GET8 (Aml) & ACPI_RESOURCE_NAME_LARGE) 540 { 541 /* Large Resource type -- bytes 1-2 contain the 16-bit length */ 542 543 ACPI_MOVE_16_TO_16 (&ResourceLength, ACPI_ADD_PTR (UINT8, Aml, 1)); 544 545 } 546 else 547 { 548 /* Small Resource type -- bits 2:0 of byte 0 contain the length */ 549 550 ResourceLength = (UINT16) (ACPI_GET8 (Aml) & 551 ACPI_RESOURCE_NAME_SMALL_LENGTH_MASK); 552 } 553 554 return (ResourceLength); 555 } 556 557 558 /******************************************************************************* 559 * 560 * FUNCTION: AcpiUtGetResourceHeaderLength 561 * 562 * PARAMETERS: Aml - Pointer to the raw AML resource descriptor 563 * 564 * RETURN: Length of the AML header (depends on large/small descriptor) 565 * 566 * DESCRIPTION: Get the length of the header for this resource. 567 * 568 ******************************************************************************/ 569 570 UINT8 571 AcpiUtGetResourceHeaderLength ( 572 void *Aml) 573 { 574 ACPI_FUNCTION_ENTRY (); 575 576 577 /* Examine the large/small bit in the resource header */ 578 579 if (ACPI_GET8 (Aml) & ACPI_RESOURCE_NAME_LARGE) 580 { 581 return (sizeof (AML_RESOURCE_LARGE_HEADER)); 582 } 583 else 584 { 585 return (sizeof (AML_RESOURCE_SMALL_HEADER)); 586 } 587 } 588 589 590 /******************************************************************************* 591 * 592 * FUNCTION: AcpiUtGetDescriptorLength 593 * 594 * PARAMETERS: Aml - Pointer to the raw AML resource descriptor 595 * 596 * RETURN: Byte length 597 * 598 * DESCRIPTION: Get the total byte length of a raw AML descriptor, including the 599 * length of the descriptor header and the length field itself. 600 * Used to walk descriptor lists. 601 * 602 ******************************************************************************/ 603 604 UINT32 605 AcpiUtGetDescriptorLength ( 606 void *Aml) 607 { 608 ACPI_FUNCTION_ENTRY (); 609 610 611 /* 612 * Get the Resource Length (does not include header length) and add 613 * the header length (depends on if this is a small or large resource) 614 */ 615 return (AcpiUtGetResourceLength (Aml) + 616 AcpiUtGetResourceHeaderLength (Aml)); 617 } 618 619 620 /******************************************************************************* 621 * 622 * FUNCTION: AcpiUtGetResourceEndTag 623 * 624 * PARAMETERS: ObjDesc - The resource template buffer object 625 * EndTag - Where the pointer to the EndTag is returned 626 * 627 * RETURN: Status, pointer to the end tag 628 * 629 * DESCRIPTION: Find the EndTag resource descriptor in an AML resource template 630 * Note: allows a buffer length of zero. 631 * 632 ******************************************************************************/ 633 634 ACPI_STATUS 635 AcpiUtGetResourceEndTag ( 636 ACPI_OPERAND_OBJECT *ObjDesc, 637 UINT8 **EndTag) 638 { 639 ACPI_STATUS Status; 640 641 642 ACPI_FUNCTION_TRACE (UtGetResourceEndTag); 643 644 645 /* Allow a buffer length of zero */ 646 647 if (!ObjDesc->Buffer.Length) 648 { 649 *EndTag = ObjDesc->Buffer.Pointer; 650 return_ACPI_STATUS (AE_OK); 651 } 652 653 /* Validate the template and get a pointer to the EndTag */ 654 655 Status = AcpiUtWalkAmlResources (NULL, ObjDesc->Buffer.Pointer, 656 ObjDesc->Buffer.Length, NULL, (void **) EndTag); 657 658 return_ACPI_STATUS (Status); 659 } 660