1 /****************************************************************************** 2 * 3 * Module Name: exconvrt - Object conversion 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 "acinterp.h" 47 #include "amlcode.h" 48 49 50 #define _COMPONENT ACPI_EXECUTER 51 ACPI_MODULE_NAME ("exconvrt") 52 53 /* Local prototypes */ 54 55 static UINT32 56 AcpiExConvertToAscii ( 57 UINT64 Integer, 58 UINT16 Base, 59 UINT8 *String, 60 UINT8 MaxLength); 61 62 63 /******************************************************************************* 64 * 65 * FUNCTION: AcpiExConvertToInteger 66 * 67 * PARAMETERS: ObjDesc - Object to be converted. Must be an 68 * Integer, Buffer, or String 69 * ResultDesc - Where the new Integer object is returned 70 * ImplicitConversion - Used for string conversion 71 * 72 * RETURN: Status 73 * 74 * DESCRIPTION: Convert an ACPI Object to an integer. 75 * 76 ******************************************************************************/ 77 78 ACPI_STATUS 79 AcpiExConvertToInteger ( 80 ACPI_OPERAND_OBJECT *ObjDesc, 81 ACPI_OPERAND_OBJECT **ResultDesc, 82 UINT32 ImplicitConversion) 83 { 84 ACPI_OPERAND_OBJECT *ReturnDesc; 85 UINT8 *Pointer; 86 UINT64 Result; 87 UINT32 i; 88 UINT32 Count; 89 90 91 ACPI_FUNCTION_TRACE_PTR (ExConvertToInteger, ObjDesc); 92 93 94 switch (ObjDesc->Common.Type) 95 { 96 case ACPI_TYPE_INTEGER: 97 98 /* No conversion necessary */ 99 100 *ResultDesc = ObjDesc; 101 return_ACPI_STATUS (AE_OK); 102 103 case ACPI_TYPE_BUFFER: 104 case ACPI_TYPE_STRING: 105 106 /* Note: Takes advantage of common buffer/string fields */ 107 108 Pointer = ObjDesc->Buffer.Pointer; 109 Count = ObjDesc->Buffer.Length; 110 break; 111 112 default: 113 114 return_ACPI_STATUS (AE_TYPE); 115 } 116 117 /* 118 * Convert the buffer/string to an integer. Note that both buffers and 119 * strings are treated as raw data - we don't convert ascii to hex for 120 * strings. 121 * 122 * There are two terminating conditions for the loop: 123 * 1) The size of an integer has been reached, or 124 * 2) The end of the buffer or string has been reached 125 */ 126 Result = 0; 127 128 /* String conversion is different than Buffer conversion */ 129 130 switch (ObjDesc->Common.Type) 131 { 132 case ACPI_TYPE_STRING: 133 /* 134 * Convert string to an integer - for most cases, the string must be 135 * hexadecimal as per the ACPI specification. The only exception (as 136 * of ACPI 3.0) is that the ToInteger() operator allows both decimal 137 * and hexadecimal strings (hex prefixed with "0x"). 138 * 139 * Explicit conversion is used only by ToInteger. 140 * All other string-to-integer conversions are implicit conversions. 141 */ 142 if (ImplicitConversion) 143 { 144 Result = AcpiUtImplicitStrtoul64 (ACPI_CAST_PTR (char, Pointer)); 145 } 146 else 147 { 148 Result = AcpiUtExplicitStrtoul64 (ACPI_CAST_PTR (char, Pointer)); 149 } 150 break; 151 152 case ACPI_TYPE_BUFFER: 153 154 /* Check for zero-length buffer */ 155 156 if (!Count) 157 { 158 return_ACPI_STATUS (AE_AML_BUFFER_LIMIT); 159 } 160 161 /* Transfer no more than an integer's worth of data */ 162 163 if (Count > AcpiGbl_IntegerByteWidth) 164 { 165 Count = AcpiGbl_IntegerByteWidth; 166 } 167 168 /* 169 * Convert buffer to an integer - we simply grab enough raw data 170 * from the buffer to fill an integer 171 */ 172 for (i = 0; i < Count; i++) 173 { 174 /* 175 * Get next byte and shift it into the Result. 176 * Little endian is used, meaning that the first byte of the buffer 177 * is the LSB of the integer 178 */ 179 Result |= (((UINT64) Pointer[i]) << (i * 8)); 180 } 181 break; 182 183 default: 184 185 /* No other types can get here */ 186 187 break; 188 } 189 190 /* Create a new integer */ 191 192 ReturnDesc = AcpiUtCreateIntegerObject (Result); 193 if (!ReturnDesc) 194 { 195 return_ACPI_STATUS (AE_NO_MEMORY); 196 } 197 198 ACPI_DEBUG_PRINT ((ACPI_DB_EXEC, "Converted value: %8.8X%8.8X\n", 199 ACPI_FORMAT_UINT64 (Result))); 200 201 /* Save the Result */ 202 203 (void) AcpiExTruncateFor32bitTable (ReturnDesc); 204 *ResultDesc = ReturnDesc; 205 return_ACPI_STATUS (AE_OK); 206 } 207 208 209 /******************************************************************************* 210 * 211 * FUNCTION: AcpiExConvertToBuffer 212 * 213 * PARAMETERS: ObjDesc - Object to be converted. Must be an 214 * Integer, Buffer, or String 215 * ResultDesc - Where the new buffer object is returned 216 * 217 * RETURN: Status 218 * 219 * DESCRIPTION: Convert an ACPI Object to a Buffer 220 * 221 ******************************************************************************/ 222 223 ACPI_STATUS 224 AcpiExConvertToBuffer ( 225 ACPI_OPERAND_OBJECT *ObjDesc, 226 ACPI_OPERAND_OBJECT **ResultDesc) 227 { 228 ACPI_OPERAND_OBJECT *ReturnDesc; 229 UINT8 *NewBuf; 230 231 232 ACPI_FUNCTION_TRACE_PTR (ExConvertToBuffer, ObjDesc); 233 234 235 switch (ObjDesc->Common.Type) 236 { 237 case ACPI_TYPE_BUFFER: 238 239 /* No conversion necessary */ 240 241 *ResultDesc = ObjDesc; 242 return_ACPI_STATUS (AE_OK); 243 244 245 case ACPI_TYPE_INTEGER: 246 /* 247 * Create a new Buffer object. 248 * Need enough space for one integer 249 */ 250 ReturnDesc = AcpiUtCreateBufferObject (AcpiGbl_IntegerByteWidth); 251 if (!ReturnDesc) 252 { 253 return_ACPI_STATUS (AE_NO_MEMORY); 254 } 255 256 /* Copy the integer to the buffer, LSB first */ 257 258 NewBuf = ReturnDesc->Buffer.Pointer; 259 memcpy (NewBuf, &ObjDesc->Integer.Value, AcpiGbl_IntegerByteWidth); 260 break; 261 262 case ACPI_TYPE_STRING: 263 /* 264 * Create a new Buffer object 265 * Size will be the string length 266 * 267 * NOTE: Add one to the string length to include the null terminator. 268 * The ACPI spec is unclear on this subject, but there is existing 269 * ASL/AML code that depends on the null being transferred to the new 270 * buffer. 271 */ 272 ReturnDesc = AcpiUtCreateBufferObject ((ACPI_SIZE) 273 ObjDesc->String.Length + 1); 274 if (!ReturnDesc) 275 { 276 return_ACPI_STATUS (AE_NO_MEMORY); 277 } 278 279 /* Copy the string to the buffer */ 280 281 NewBuf = ReturnDesc->Buffer.Pointer; 282 strncpy ((char *) NewBuf, (char *) ObjDesc->String.Pointer, 283 ObjDesc->String.Length); 284 break; 285 286 default: 287 288 return_ACPI_STATUS (AE_TYPE); 289 } 290 291 /* Mark buffer initialized */ 292 293 ReturnDesc->Common.Flags |= AOPOBJ_DATA_VALID; 294 *ResultDesc = ReturnDesc; 295 return_ACPI_STATUS (AE_OK); 296 } 297 298 299 /******************************************************************************* 300 * 301 * FUNCTION: AcpiExConvertToAscii 302 * 303 * PARAMETERS: Integer - Value to be converted 304 * Base - ACPI_STRING_DECIMAL or ACPI_STRING_HEX 305 * String - Where the string is returned 306 * DataWidth - Size of data item to be converted, in bytes 307 * 308 * RETURN: Actual string length 309 * 310 * DESCRIPTION: Convert an ACPI Integer to a hex or decimal string 311 * 312 ******************************************************************************/ 313 314 static UINT32 315 AcpiExConvertToAscii ( 316 UINT64 Integer, 317 UINT16 Base, 318 UINT8 *String, 319 UINT8 DataWidth) 320 { 321 UINT64 Digit; 322 UINT32 i; 323 UINT32 j; 324 UINT32 k = 0; 325 UINT32 HexLength; 326 UINT32 DecimalLength; 327 UINT32 Remainder; 328 BOOLEAN SupressZeros; 329 330 331 ACPI_FUNCTION_ENTRY (); 332 333 334 switch (Base) 335 { 336 case 10: 337 338 /* Setup max length for the decimal number */ 339 340 switch (DataWidth) 341 { 342 case 1: 343 344 DecimalLength = ACPI_MAX8_DECIMAL_DIGITS; 345 break; 346 347 case 4: 348 349 DecimalLength = ACPI_MAX32_DECIMAL_DIGITS; 350 break; 351 352 case 8: 353 default: 354 355 DecimalLength = ACPI_MAX64_DECIMAL_DIGITS; 356 break; 357 } 358 359 SupressZeros = TRUE; /* No leading zeros */ 360 Remainder = 0; 361 362 for (i = DecimalLength; i > 0; i--) 363 { 364 /* Divide by nth factor of 10 */ 365 366 Digit = Integer; 367 for (j = 0; j < i; j++) 368 { 369 (void) AcpiUtShortDivide (Digit, 10, &Digit, &Remainder); 370 } 371 372 /* Handle leading zeros */ 373 374 if (Remainder != 0) 375 { 376 SupressZeros = FALSE; 377 } 378 379 if (!SupressZeros) 380 { 381 String[k] = (UINT8) (ACPI_ASCII_ZERO + Remainder); 382 k++; 383 } 384 } 385 break; 386 387 case 16: 388 389 /* HexLength: 2 ascii hex chars per data byte */ 390 391 HexLength = (DataWidth * 2); 392 for (i = 0, j = (HexLength-1); i < HexLength; i++, j--) 393 { 394 /* Get one hex digit, most significant digits first */ 395 396 String[k] = (UINT8) 397 AcpiUtHexToAsciiChar (Integer, ACPI_MUL_4 (j)); 398 k++; 399 } 400 break; 401 402 default: 403 return (0); 404 } 405 406 /* 407 * Since leading zeros are suppressed, we must check for the case where 408 * the integer equals 0 409 * 410 * Finally, null terminate the string and return the length 411 */ 412 if (!k) 413 { 414 String [0] = ACPI_ASCII_ZERO; 415 k = 1; 416 } 417 418 String [k] = 0; 419 return ((UINT32) k); 420 } 421 422 423 /******************************************************************************* 424 * 425 * FUNCTION: AcpiExConvertToString 426 * 427 * PARAMETERS: ObjDesc - Object to be converted. Must be an 428 * Integer, Buffer, or String 429 * ResultDesc - Where the string object is returned 430 * Type - String flags (base and conversion type) 431 * 432 * RETURN: Status 433 * 434 * DESCRIPTION: Convert an ACPI Object to a string. Supports both implicit 435 * and explicit conversions and related rules. 436 * 437 ******************************************************************************/ 438 439 ACPI_STATUS 440 AcpiExConvertToString ( 441 ACPI_OPERAND_OBJECT *ObjDesc, 442 ACPI_OPERAND_OBJECT **ResultDesc, 443 UINT32 Type) 444 { 445 ACPI_OPERAND_OBJECT *ReturnDesc; 446 UINT8 *NewBuf; 447 UINT32 i; 448 UINT32 StringLength = 0; 449 UINT16 Base = 16; 450 UINT8 Separator = ','; 451 452 453 ACPI_FUNCTION_TRACE_PTR (ExConvertToString, ObjDesc); 454 455 456 switch (ObjDesc->Common.Type) 457 { 458 case ACPI_TYPE_STRING: 459 460 /* No conversion necessary */ 461 462 *ResultDesc = ObjDesc; 463 return_ACPI_STATUS (AE_OK); 464 465 case ACPI_TYPE_INTEGER: 466 467 switch (Type) 468 { 469 case ACPI_EXPLICIT_CONVERT_DECIMAL: 470 /* 471 * From ToDecimalString, integer source. 472 * 473 * Make room for the maximum decimal number size 474 */ 475 StringLength = ACPI_MAX_DECIMAL_DIGITS; 476 Base = 10; 477 break; 478 479 default: 480 481 /* Two hex string characters for each integer byte */ 482 483 StringLength = ACPI_MUL_2 (AcpiGbl_IntegerByteWidth); 484 break; 485 } 486 487 /* 488 * Create a new String 489 * Need enough space for one ASCII integer (plus null terminator) 490 */ 491 ReturnDesc = AcpiUtCreateStringObject ((ACPI_SIZE) StringLength); 492 if (!ReturnDesc) 493 { 494 return_ACPI_STATUS (AE_NO_MEMORY); 495 } 496 497 NewBuf = ReturnDesc->Buffer.Pointer; 498 499 /* Convert integer to string */ 500 501 StringLength = AcpiExConvertToAscii ( 502 ObjDesc->Integer.Value, Base, NewBuf, AcpiGbl_IntegerByteWidth); 503 504 /* Null terminate at the correct place */ 505 506 ReturnDesc->String.Length = StringLength; 507 NewBuf [StringLength] = 0; 508 break; 509 510 case ACPI_TYPE_BUFFER: 511 512 /* Setup string length, base, and separator */ 513 514 switch (Type) 515 { 516 case ACPI_EXPLICIT_CONVERT_DECIMAL: /* Used by ToDecimalString */ 517 /* 518 * Explicit conversion from the ToDecimalString ASL operator. 519 * 520 * From ACPI: "If the input is a buffer, it is converted to a 521 * a string of decimal values separated by commas." 522 */ 523 Base = 10; 524 525 /* 526 * Calculate the final string length. Individual string values 527 * are variable length (include separator for each) 528 */ 529 for (i = 0; i < ObjDesc->Buffer.Length; i++) 530 { 531 if (ObjDesc->Buffer.Pointer[i] >= 100) 532 { 533 StringLength += 4; 534 } 535 else if (ObjDesc->Buffer.Pointer[i] >= 10) 536 { 537 StringLength += 3; 538 } 539 else 540 { 541 StringLength += 2; 542 } 543 } 544 break; 545 546 case ACPI_IMPLICIT_CONVERT_HEX: 547 /* 548 * Implicit buffer-to-string conversion 549 * 550 * From the ACPI spec: 551 * "The entire contents of the buffer are converted to a string of 552 * two-character hexadecimal numbers, each separated by a space." 553 * 554 * Each hex number is prefixed with 0x (11/2018) 555 */ 556 Separator = ' '; 557 StringLength = (ObjDesc->Buffer.Length * 5); 558 break; 559 560 case ACPI_EXPLICIT_CONVERT_HEX: 561 /* 562 * Explicit conversion from the ToHexString ASL operator. 563 * 564 * From ACPI: "If Data is a buffer, it is converted to a string of 565 * hexadecimal values separated by commas." 566 * 567 * Each hex number is prefixed with 0x (11/2018) 568 */ 569 Separator = ','; 570 StringLength = (ObjDesc->Buffer.Length * 5); 571 break; 572 573 default: 574 return_ACPI_STATUS (AE_BAD_PARAMETER); 575 } 576 577 /* 578 * Create a new string object and string buffer 579 * (-1 because of extra separator included in StringLength from above) 580 * Allow creation of zero-length strings from zero-length buffers. 581 */ 582 if (StringLength) 583 { 584 StringLength--; 585 } 586 587 ReturnDesc = AcpiUtCreateStringObject ((ACPI_SIZE) StringLength); 588 if (!ReturnDesc) 589 { 590 return_ACPI_STATUS (AE_NO_MEMORY); 591 } 592 593 NewBuf = ReturnDesc->Buffer.Pointer; 594 595 /* 596 * Convert buffer bytes to hex or decimal values 597 * (separated by commas or spaces) 598 */ 599 for (i = 0; i < ObjDesc->Buffer.Length; i++) 600 { 601 if (Base == 16) 602 { 603 /* Emit 0x prefix for explicit/implicit hex conversion */ 604 605 *NewBuf++ = '0'; 606 *NewBuf++ = 'x'; 607 } 608 609 NewBuf += AcpiExConvertToAscii ( 610 (UINT64) ObjDesc->Buffer.Pointer[i], Base, NewBuf, 1); 611 612 /* Each digit is separated by either a comma or space */ 613 614 *NewBuf++ = Separator; 615 } 616 617 /* 618 * Null terminate the string 619 * (overwrites final comma/space from above) 620 */ 621 if (ObjDesc->Buffer.Length) 622 { 623 NewBuf--; 624 } 625 *NewBuf = 0; 626 break; 627 628 default: 629 630 return_ACPI_STATUS (AE_TYPE); 631 } 632 633 *ResultDesc = ReturnDesc; 634 return_ACPI_STATUS (AE_OK); 635 } 636 637 638 /******************************************************************************* 639 * 640 * FUNCTION: AcpiExConvertToTargetType 641 * 642 * PARAMETERS: DestinationType - Current type of the destination 643 * SourceDesc - Source object to be converted. 644 * ResultDesc - Where the converted object is returned 645 * WalkState - Current method state 646 * 647 * RETURN: Status 648 * 649 * DESCRIPTION: Implements "implicit conversion" rules for storing an object. 650 * 651 ******************************************************************************/ 652 653 ACPI_STATUS 654 AcpiExConvertToTargetType ( 655 ACPI_OBJECT_TYPE DestinationType, 656 ACPI_OPERAND_OBJECT *SourceDesc, 657 ACPI_OPERAND_OBJECT **ResultDesc, 658 ACPI_WALK_STATE *WalkState) 659 { 660 ACPI_STATUS Status = AE_OK; 661 662 663 ACPI_FUNCTION_TRACE (ExConvertToTargetType); 664 665 666 /* Default behavior */ 667 668 *ResultDesc = SourceDesc; 669 670 /* 671 * If required by the target, 672 * perform implicit conversion on the source before we store it. 673 */ 674 switch (GET_CURRENT_ARG_TYPE (WalkState->OpInfo->RuntimeArgs)) 675 { 676 case ARGI_SIMPLE_TARGET: 677 case ARGI_FIXED_TARGET: 678 case ARGI_INTEGER_REF: /* Handles Increment, Decrement cases */ 679 680 switch (DestinationType) 681 { 682 case ACPI_TYPE_LOCAL_REGION_FIELD: 683 /* 684 * Named field can always handle conversions 685 */ 686 break; 687 688 default: 689 690 /* No conversion allowed for these types */ 691 692 if (DestinationType != SourceDesc->Common.Type) 693 { 694 ACPI_DEBUG_PRINT ((ACPI_DB_INFO, 695 "Explicit operator, will store (%s) over existing type (%s)\n", 696 AcpiUtGetObjectTypeName (SourceDesc), 697 AcpiUtGetTypeName (DestinationType))); 698 Status = AE_TYPE; 699 } 700 } 701 break; 702 703 case ARGI_TARGETREF: 704 case ARGI_STORE_TARGET: 705 706 switch (DestinationType) 707 { 708 case ACPI_TYPE_INTEGER: 709 case ACPI_TYPE_BUFFER_FIELD: 710 case ACPI_TYPE_LOCAL_BANK_FIELD: 711 case ACPI_TYPE_LOCAL_INDEX_FIELD: 712 /* 713 * These types require an Integer operand. We can convert 714 * a Buffer or a String to an Integer if necessary. 715 */ 716 Status = AcpiExConvertToInteger (SourceDesc, ResultDesc, 717 ACPI_IMPLICIT_CONVERSION); 718 break; 719 720 case ACPI_TYPE_STRING: 721 /* 722 * The operand must be a String. We can convert an 723 * Integer or Buffer if necessary 724 */ 725 Status = AcpiExConvertToString (SourceDesc, ResultDesc, 726 ACPI_IMPLICIT_CONVERT_HEX); 727 break; 728 729 case ACPI_TYPE_BUFFER: 730 /* 731 * The operand must be a Buffer. We can convert an 732 * Integer or String if necessary 733 */ 734 Status = AcpiExConvertToBuffer (SourceDesc, ResultDesc); 735 break; 736 737 default: 738 739 ACPI_ERROR ((AE_INFO, 740 "Bad destination type during conversion: 0x%X", 741 DestinationType)); 742 Status = AE_AML_INTERNAL; 743 break; 744 } 745 break; 746 747 case ARGI_REFERENCE: 748 /* 749 * CreateXxxxField cases - we are storing the field object into the name 750 */ 751 break; 752 753 default: 754 755 ACPI_ERROR ((AE_INFO, 756 "Unknown Target type ID 0x%X AmlOpcode 0x%X DestType %s", 757 GET_CURRENT_ARG_TYPE (WalkState->OpInfo->RuntimeArgs), 758 WalkState->Opcode, AcpiUtGetTypeName (DestinationType))); 759 Status = AE_AML_INTERNAL; 760 } 761 762 /* 763 * Source-to-Target conversion semantics: 764 * 765 * If conversion to the target type cannot be performed, then simply 766 * overwrite the target with the new object and type. 767 */ 768 if (Status == AE_TYPE) 769 { 770 Status = AE_OK; 771 } 772 773 return_ACPI_STATUS (Status); 774 } 775