xref: /freebsd/sys/contrib/dev/acpica/compiler/aslutils.c (revision fba3cde907930eed2adb8a320524bc250338c729)
1 /******************************************************************************
2  *
3  * Module Name: aslutils -- compiler utilities
4  *
5  *****************************************************************************/
6 
7 /*
8  * Copyright (C) 2000 - 2013, 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 
45 #include <contrib/dev/acpica/compiler/aslcompiler.h>
46 #include "aslcompiler.y.h"
47 #include <contrib/dev/acpica/include/acdisasm.h>
48 #include <contrib/dev/acpica/include/acnamesp.h>
49 #include <contrib/dev/acpica/include/amlcode.h>
50 #include <contrib/dev/acpica/include/acapps.h>
51 
52 #define _COMPONENT          ACPI_COMPILER
53         ACPI_MODULE_NAME    ("aslutils")
54 
55 
56 /* Local prototypes */
57 
58 static void
59 UtPadNameWithUnderscores (
60     char                    *NameSeg,
61     char                    *PaddedNameSeg);
62 
63 static void
64 UtAttachNameseg (
65     ACPI_PARSE_OBJECT       *Op,
66     char                    *Name);
67 
68 
69 /*******************************************************************************
70  *
71  * FUNCTION:    UtDisplaySupportedTables
72  *
73  * PARAMETERS:  None
74  *
75  * RETURN:      None
76  *
77  * DESCRIPTION: Print all supported ACPI table names.
78  *
79  ******************************************************************************/
80 
81 #define ACPI_TABLE_HELP_FORMAT  "%8u) %s    %s\n"
82 
83 void
84 UtDisplaySupportedTables (
85     void)
86 {
87     ACPI_DMTABLE_DATA       *TableData;
88     UINT32                  i;
89 
90 
91     printf ("\nACPI tables supported by iASL version %8.8X:\n"
92         "  (Compiler, Disassembler, Template Generator)\n\n",
93         ACPI_CA_VERSION);
94 
95     /* Special tables */
96 
97     printf ("  Special tables and AML tables:\n");
98     printf (ACPI_TABLE_HELP_FORMAT, 1, ACPI_RSDP_NAME, "Root System Description Pointer");
99     printf (ACPI_TABLE_HELP_FORMAT, 2, ACPI_SIG_FACS, "Firmware ACPI Control Structure");
100     printf (ACPI_TABLE_HELP_FORMAT, 3, ACPI_SIG_DSDT, "Differentiated System Description Table");
101     printf (ACPI_TABLE_HELP_FORMAT, 4, ACPI_SIG_SSDT, "Secondary System Description Table");
102 
103     /* All data tables with common table header */
104 
105     printf ("\n  Standard ACPI data tables:\n");
106     for (TableData = AcpiDmTableData, i = 5; TableData->Signature; TableData++, i++)
107     {
108         printf (ACPI_TABLE_HELP_FORMAT, i, TableData->Signature, TableData->Name);
109     }
110 }
111 
112 
113 /*******************************************************************************
114  *
115  * FUNCTION:    UtDisplayConstantOpcodes
116  *
117  * PARAMETERS:  None
118  *
119  * RETURN:      None
120  *
121  * DESCRIPTION: Print AML opcodes that can be used in constant expressions.
122  *
123  ******************************************************************************/
124 
125 void
126 UtDisplayConstantOpcodes (
127     void)
128 {
129     UINT32                  i;
130 
131 
132     printf ("Constant expression opcode information\n\n");
133 
134     for (i = 0; i < sizeof (AcpiGbl_AmlOpInfo) / sizeof (ACPI_OPCODE_INFO); i++)
135     {
136         if (AcpiGbl_AmlOpInfo[i].Flags & AML_CONSTANT)
137         {
138             printf ("%s\n", AcpiGbl_AmlOpInfo[i].Name);
139         }
140     }
141 }
142 
143 
144 /*******************************************************************************
145  *
146  * FUNCTION:    UtLocalCalloc
147  *
148  * PARAMETERS:  Size                - Bytes to be allocated
149  *
150  * RETURN:      Pointer to the allocated memory. Guaranteed to be valid.
151  *
152  * DESCRIPTION: Allocate zero-initialized memory. Aborts the compile on an
153  *              allocation failure, on the assumption that nothing more can be
154  *              accomplished.
155  *
156  ******************************************************************************/
157 
158 void *
159 UtLocalCalloc (
160     UINT32                  Size)
161 {
162     void                    *Allocated;
163 
164 
165     Allocated = ACPI_ALLOCATE_ZEROED (Size);
166     if (!Allocated)
167     {
168         AslCommonError (ASL_ERROR, ASL_MSG_MEMORY_ALLOCATION,
169             Gbl_CurrentLineNumber, Gbl_LogicalLineNumber,
170             Gbl_InputByteCount, Gbl_CurrentColumn,
171             Gbl_Files[ASL_FILE_INPUT].Filename, NULL);
172 
173         CmCleanupAndExit ();
174         exit (1);
175     }
176 
177     TotalAllocations++;
178     TotalAllocated += Size;
179     return (Allocated);
180 }
181 
182 
183 /*******************************************************************************
184  *
185  * FUNCTION:    UtBeginEvent
186  *
187  * PARAMETERS:  Name                - Ascii name of this event
188  *
189  * RETURN:      Event number (integer index)
190  *
191  * DESCRIPTION: Saves the current time with this event
192  *
193  ******************************************************************************/
194 
195 UINT8
196 UtBeginEvent (
197     char                    *Name)
198 {
199 
200     if (AslGbl_NextEvent >= ASL_NUM_EVENTS)
201     {
202         AcpiOsPrintf ("Ran out of compiler event structs!\n");
203         return (AslGbl_NextEvent);
204     }
205 
206     /* Init event with current (start) time */
207 
208     AslGbl_Events[AslGbl_NextEvent].StartTime = AcpiOsGetTimer ();
209     AslGbl_Events[AslGbl_NextEvent].EventName = Name;
210     AslGbl_Events[AslGbl_NextEvent].Valid = TRUE;
211 
212     return (AslGbl_NextEvent++);
213 }
214 
215 
216 /*******************************************************************************
217  *
218  * FUNCTION:    UtEndEvent
219  *
220  * PARAMETERS:  Event               - Event number (integer index)
221  *
222  * RETURN:      None
223  *
224  * DESCRIPTION: Saves the current time (end time) with this event
225  *
226  ******************************************************************************/
227 
228 void
229 UtEndEvent (
230     UINT8                   Event)
231 {
232 
233     if (Event >= ASL_NUM_EVENTS)
234     {
235         return;
236     }
237 
238     /* Insert end time for event */
239 
240     AslGbl_Events[Event].EndTime = AcpiOsGetTimer ();
241 }
242 
243 
244 /*******************************************************************************
245  *
246  * FUNCTION:    UtHexCharToValue
247  *
248  * PARAMETERS:  HexChar             - Hex character in Ascii
249  *
250  * RETURN:      The binary value of the hex character
251  *
252  * DESCRIPTION: Perform ascii-to-hex translation
253  *
254  ******************************************************************************/
255 
256 UINT8
257 UtHexCharToValue (
258     int                     HexChar)
259 {
260 
261     if (HexChar <= 0x39)
262     {
263         return ((UINT8) (HexChar - 0x30));
264     }
265 
266     if (HexChar <= 0x46)
267     {
268         return ((UINT8) (HexChar - 0x37));
269     }
270 
271     return ((UINT8) (HexChar - 0x57));
272 }
273 
274 
275 /*******************************************************************************
276  *
277  * FUNCTION:    UtConvertByteToHex
278  *
279  * PARAMETERS:  RawByte             - Binary data
280  *              Buffer              - Pointer to where the hex bytes will be
281  *                                    stored
282  *
283  * RETURN:      Ascii hex byte is stored in Buffer.
284  *
285  * DESCRIPTION: Perform hex-to-ascii translation. The return data is prefixed
286  *              with "0x"
287  *
288  ******************************************************************************/
289 
290 void
291 UtConvertByteToHex (
292     UINT8                   RawByte,
293     UINT8                   *Buffer)
294 {
295 
296     Buffer[0] = '0';
297     Buffer[1] = 'x';
298 
299     Buffer[2] = (UINT8) AslHexLookup[(RawByte >> 4) & 0xF];
300     Buffer[3] = (UINT8) AslHexLookup[RawByte & 0xF];
301 }
302 
303 
304 /*******************************************************************************
305  *
306  * FUNCTION:    UtConvertByteToAsmHex
307  *
308  * PARAMETERS:  RawByte             - Binary data
309  *              Buffer              - Pointer to where the hex bytes will be
310  *                                    stored
311  *
312  * RETURN:      Ascii hex byte is stored in Buffer.
313  *
314  * DESCRIPTION: Perform hex-to-ascii translation. The return data is prefixed
315  *              with "0x"
316  *
317  ******************************************************************************/
318 
319 void
320 UtConvertByteToAsmHex (
321     UINT8                   RawByte,
322     UINT8                   *Buffer)
323 {
324 
325     Buffer[0] = '0';
326     Buffer[1] = (UINT8) AslHexLookup[(RawByte >> 4) & 0xF];
327     Buffer[2] = (UINT8) AslHexLookup[RawByte & 0xF];
328     Buffer[3] = 'h';
329 }
330 
331 
332 /*******************************************************************************
333  *
334  * FUNCTION:    DbgPrint
335  *
336  * PARAMETERS:  Type                - Type of output
337  *              Fmt                 - Printf format string
338  *              ...                 - variable printf list
339  *
340  * RETURN:      None
341  *
342  * DESCRIPTION: Conditional print statement. Prints to stderr only if the
343  *              debug flag is set.
344  *
345  ******************************************************************************/
346 
347 void
348 DbgPrint (
349     UINT32                  Type,
350     char                    *Fmt,
351     ...)
352 {
353     va_list                 Args;
354 
355 
356     if (!Gbl_DebugFlag)
357     {
358         return;
359     }
360 
361     if ((Type == ASL_PARSE_OUTPUT) &&
362         (!(AslCompilerdebug)))
363     {
364         return;
365     }
366 
367     va_start (Args, Fmt);
368     (void) vfprintf (stderr, Fmt, Args);
369     va_end (Args);
370     return;
371 }
372 
373 
374 /*******************************************************************************
375  *
376  * FUNCTION:    UtPrintFormattedName
377  *
378  * PARAMETERS:  ParseOpcode         - Parser keyword ID
379  *              Level               - Indentation level
380  *
381  * RETURN:      None
382  *
383  * DESCRIPTION: Print the ascii name of the parse opcode.
384  *
385  ******************************************************************************/
386 
387 #define TEXT_OFFSET 10
388 
389 void
390 UtPrintFormattedName (
391     UINT16                  ParseOpcode,
392     UINT32                  Level)
393 {
394 
395     if (Level)
396     {
397         DbgPrint (ASL_TREE_OUTPUT,
398             "%*s", (3 * Level), " ");
399     }
400     DbgPrint (ASL_TREE_OUTPUT,
401         " %-20.20s", UtGetOpName (ParseOpcode));
402 
403     if (Level < TEXT_OFFSET)
404     {
405         DbgPrint (ASL_TREE_OUTPUT,
406             "%*s", (TEXT_OFFSET - Level) * 3, " ");
407     }
408 }
409 
410 
411 /*******************************************************************************
412  *
413  * FUNCTION:    UtSetParseOpName
414  *
415  * PARAMETERS:  Op                  - Parse op to be named.
416  *
417  * RETURN:      None
418  *
419  * DESCRIPTION: Insert the ascii name of the parse opcode
420  *
421  ******************************************************************************/
422 
423 void
424 UtSetParseOpName (
425     ACPI_PARSE_OBJECT       *Op)
426 {
427 
428     strncpy (Op->Asl.ParseOpName, UtGetOpName (Op->Asl.ParseOpcode),
429         ACPI_MAX_PARSEOP_NAME);
430 }
431 
432 
433 /*******************************************************************************
434  *
435  * FUNCTION:    UtDisplaySummary
436  *
437  * PARAMETERS:  FileID              - ID of outpout file
438  *
439  * RETURN:      None
440  *
441  * DESCRIPTION: Display compilation statistics
442  *
443  ******************************************************************************/
444 
445 void
446 UtDisplaySummary (
447     UINT32                  FileId)
448 {
449     UINT32                  i;
450 
451 
452     if (FileId != ASL_FILE_STDOUT)
453     {
454         /* Compiler name and version number */
455 
456         FlPrintFile (FileId, "%s version %X%s\n\n",
457             ASL_COMPILER_NAME, (UINT32) ACPI_CA_VERSION, ACPI_WIDTH);
458     }
459 
460     /* Summary of main input and output files */
461 
462     if (Gbl_FileType == ASL_INPUT_TYPE_ASCII_DATA)
463     {
464         FlPrintFile (FileId,
465             "%-14s %s - %u lines, %u bytes, %u fields\n",
466             "Table Input:",
467             Gbl_Files[ASL_FILE_INPUT].Filename, Gbl_CurrentLineNumber,
468             Gbl_InputByteCount, Gbl_InputFieldCount);
469 
470         if ((Gbl_ExceptionCount[ASL_ERROR] == 0) || (Gbl_IgnoreErrors))
471         {
472             FlPrintFile (FileId,
473                 "%-14s %s - %u bytes\n",
474                 "Binary Output:",
475                 Gbl_Files[ASL_FILE_AML_OUTPUT].Filename, Gbl_TableLength);
476         }
477     }
478     else
479     {
480         FlPrintFile (FileId,
481             "%-14s %s - %u lines, %u bytes, %u keywords\n",
482             "ASL Input:",
483             Gbl_Files[ASL_FILE_INPUT].Filename, Gbl_CurrentLineNumber,
484             Gbl_InputByteCount, TotalKeywords);
485 
486         /* AML summary */
487 
488         if ((Gbl_ExceptionCount[ASL_ERROR] == 0) || (Gbl_IgnoreErrors))
489         {
490             FlPrintFile (FileId,
491                 "%-14s %s - %u bytes, %u named objects, %u executable opcodes\n",
492                 "AML Output:",
493                 Gbl_Files[ASL_FILE_AML_OUTPUT].Filename, Gbl_TableLength,
494                 TotalNamedObjects, TotalExecutableOpcodes);
495         }
496     }
497 
498     /* Display summary of any optional files */
499 
500     for (i = ASL_FILE_SOURCE_OUTPUT; i <= ASL_MAX_FILE_TYPE; i++)
501     {
502         if (!Gbl_Files[i].Filename || !Gbl_Files[i].Handle)
503         {
504             continue;
505         }
506 
507         /* .SRC is a temp file unless specifically requested */
508 
509         if ((i == ASL_FILE_SOURCE_OUTPUT) && (!Gbl_SourceOutputFlag))
510         {
511             continue;
512         }
513 
514         /* .I is a temp file unless specifically requested */
515 
516         if ((i == ASL_FILE_PREPROCESSOR) && (!Gbl_PreprocessorOutputFlag))
517         {
518             continue;
519         }
520 
521         FlPrintFile (FileId, "%14s %s - %u bytes\n",
522             Gbl_Files[i].ShortDescription,
523             Gbl_Files[i].Filename, FlGetFileSize (i));
524     }
525 
526     /* Error summary */
527 
528     FlPrintFile (FileId,
529         "\nCompilation complete. %u Errors, %u Warnings, %u Remarks",
530         Gbl_ExceptionCount[ASL_ERROR],
531         Gbl_ExceptionCount[ASL_WARNING] +
532             Gbl_ExceptionCount[ASL_WARNING2] +
533             Gbl_ExceptionCount[ASL_WARNING3],
534         Gbl_ExceptionCount[ASL_REMARK]);
535 
536     if (Gbl_FileType != ASL_INPUT_TYPE_ASCII_DATA)
537     {
538         FlPrintFile (FileId,
539             ", %u Optimizations", Gbl_ExceptionCount[ASL_OPTIMIZATION]);
540     }
541 
542     FlPrintFile (FileId, "\n");
543 }
544 
545 
546 /*******************************************************************************
547  *
548  * FUNCTION:    UtCheckIntegerRange
549  *
550  * PARAMETERS:  Op                  - Integer parse node
551  *              LowValue            - Smallest allowed value
552  *              HighValue           - Largest allowed value
553  *
554  * RETURN:      Op if OK, otherwise NULL
555  *
556  * DESCRIPTION: Check integer for an allowable range
557  *
558  ******************************************************************************/
559 
560 ACPI_PARSE_OBJECT *
561 UtCheckIntegerRange (
562     ACPI_PARSE_OBJECT       *Op,
563     UINT32                  LowValue,
564     UINT32                  HighValue)
565 {
566 
567     if (!Op)
568     {
569         return (NULL);
570     }
571 
572     if ((Op->Asl.Value.Integer < LowValue) ||
573         (Op->Asl.Value.Integer > HighValue))
574     {
575         sprintf (MsgBuffer, "0x%X, allowable: 0x%X-0x%X",
576             (UINT32) Op->Asl.Value.Integer, LowValue, HighValue);
577 
578         AslError (ASL_ERROR, ASL_MSG_RANGE, Op, MsgBuffer);
579         return (NULL);
580     }
581 
582     return (Op);
583 }
584 
585 
586 /*******************************************************************************
587  *
588  * FUNCTION:    UtGetStringBuffer
589  *
590  * PARAMETERS:  Length              - Size of buffer requested
591  *
592  * RETURN:      Pointer to the buffer. Aborts on allocation failure
593  *
594  * DESCRIPTION: Allocate a string buffer. Bypass the local
595  *              dynamic memory manager for performance reasons (This has a
596  *              major impact on the speed of the compiler.)
597  *
598  ******************************************************************************/
599 
600 char *
601 UtGetStringBuffer (
602     UINT32                  Length)
603 {
604     char                    *Buffer;
605 
606 
607     if ((Gbl_StringCacheNext + Length) >= Gbl_StringCacheLast)
608     {
609         Gbl_StringCacheNext = UtLocalCalloc (ASL_STRING_CACHE_SIZE + Length);
610         Gbl_StringCacheLast = Gbl_StringCacheNext + ASL_STRING_CACHE_SIZE +
611                                 Length;
612     }
613 
614     Buffer = Gbl_StringCacheNext;
615     Gbl_StringCacheNext += Length;
616 
617     return (Buffer);
618 }
619 
620 
621 /******************************************************************************
622  *
623  * FUNCTION:    UtExpandLineBuffers
624  *
625  * PARAMETERS:  None. Updates global line buffer pointers.
626  *
627  * RETURN:      None. Reallocates the global line buffers
628  *
629  * DESCRIPTION: Called if the current line buffer becomes filled. Reallocates
630  *              all global line buffers and updates Gbl_LineBufferSize. NOTE:
631  *              Also used for the initial allocation of the buffers, when
632  *              all of the buffer pointers are NULL. Initial allocations are
633  *              of size ASL_DEFAULT_LINE_BUFFER_SIZE
634  *
635  *****************************************************************************/
636 
637 void
638 UtExpandLineBuffers (
639     void)
640 {
641     UINT32                  NewSize;
642 
643 
644     /* Attempt to double the size of all line buffers */
645 
646     NewSize = Gbl_LineBufferSize * 2;
647     if (Gbl_CurrentLineBuffer)
648     {
649         DbgPrint (ASL_DEBUG_OUTPUT,"Increasing line buffer size from %u to %u\n",
650             Gbl_LineBufferSize, NewSize);
651     }
652 
653     Gbl_CurrentLineBuffer = realloc (Gbl_CurrentLineBuffer, NewSize);
654     Gbl_LineBufPtr = Gbl_CurrentLineBuffer;
655     if (!Gbl_CurrentLineBuffer)
656     {
657         goto ErrorExit;
658     }
659 
660     Gbl_MainTokenBuffer = realloc (Gbl_MainTokenBuffer, NewSize);
661     if (!Gbl_MainTokenBuffer)
662     {
663         goto ErrorExit;
664     }
665 
666     Gbl_MacroTokenBuffer = realloc (Gbl_MacroTokenBuffer, NewSize);
667     if (!Gbl_MacroTokenBuffer)
668     {
669         goto ErrorExit;
670     }
671 
672     Gbl_ExpressionTokenBuffer = realloc (Gbl_ExpressionTokenBuffer, NewSize);
673     if (!Gbl_ExpressionTokenBuffer)
674     {
675         goto ErrorExit;
676     }
677 
678     Gbl_LineBufferSize = NewSize;
679     return;
680 
681 
682     /* On error above, simply issue error messages and abort, cannot continue */
683 
684 ErrorExit:
685     printf ("Could not increase line buffer size from %u to %u\n",
686         Gbl_LineBufferSize, Gbl_LineBufferSize * 2);
687 
688     AslError (ASL_ERROR, ASL_MSG_BUFFER_ALLOCATION,
689         NULL, NULL);
690     AslAbort ();
691 }
692 
693 
694 /*******************************************************************************
695  *
696  * FUNCTION:    UtInternalizeName
697  *
698  * PARAMETERS:  ExternalName        - Name to convert
699  *              ConvertedName       - Where the converted name is returned
700  *
701  * RETURN:      Status
702  *
703  * DESCRIPTION: Convert an external (ASL) name to an internal (AML) name
704  *
705  ******************************************************************************/
706 
707 ACPI_STATUS
708 UtInternalizeName (
709     char                    *ExternalName,
710     char                    **ConvertedName)
711 {
712     ACPI_NAMESTRING_INFO    Info;
713     ACPI_STATUS             Status;
714 
715 
716     if (!ExternalName)
717     {
718         return (AE_OK);
719     }
720 
721     /* Get the length of the new internal name */
722 
723     Info.ExternalName = ExternalName;
724     AcpiNsGetInternalNameLength (&Info);
725 
726     /* We need a segment to store the internal  name */
727 
728     Info.InternalName = UtGetStringBuffer (Info.Length);
729     if (!Info.InternalName)
730     {
731         return (AE_NO_MEMORY);
732     }
733 
734     /* Build the name */
735 
736     Status = AcpiNsBuildInternalName (&Info);
737     if (ACPI_FAILURE (Status))
738     {
739         return (Status);
740     }
741 
742     *ConvertedName = Info.InternalName;
743     return (AE_OK);
744 }
745 
746 
747 /*******************************************************************************
748  *
749  * FUNCTION:    UtPadNameWithUnderscores
750  *
751  * PARAMETERS:  NameSeg             - Input nameseg
752  *              PaddedNameSeg       - Output padded nameseg
753  *
754  * RETURN:      Padded nameseg.
755  *
756  * DESCRIPTION: Pads a NameSeg with underscores if necessary to form a full
757  *              ACPI_NAME.
758  *
759  ******************************************************************************/
760 
761 static void
762 UtPadNameWithUnderscores (
763     char                    *NameSeg,
764     char                    *PaddedNameSeg)
765 {
766     UINT32                  i;
767 
768 
769     for (i = 0; (i < ACPI_NAME_SIZE); i++)
770     {
771         if (*NameSeg)
772         {
773             *PaddedNameSeg = *NameSeg;
774             NameSeg++;
775         }
776         else
777         {
778             *PaddedNameSeg = '_';
779         }
780         PaddedNameSeg++;
781     }
782 }
783 
784 
785 /*******************************************************************************
786  *
787  * FUNCTION:    UtAttachNameseg
788  *
789  * PARAMETERS:  Op                  - Parent parse node
790  *              Name                - Full ExternalName
791  *
792  * RETURN:      None; Sets the NameSeg field in parent node
793  *
794  * DESCRIPTION: Extract the last nameseg of the ExternalName and store it
795  *              in the NameSeg field of the Op.
796  *
797  ******************************************************************************/
798 
799 static void
800 UtAttachNameseg (
801     ACPI_PARSE_OBJECT       *Op,
802     char                    *Name)
803 {
804     char                    *NameSeg;
805     char                    PaddedNameSeg[4];
806 
807 
808     if (!Name)
809     {
810         return;
811     }
812 
813     /* Look for the last dot in the namepath */
814 
815     NameSeg = strrchr (Name, '.');
816     if (NameSeg)
817     {
818         /* Found last dot, we have also found the final nameseg */
819 
820         NameSeg++;
821         UtPadNameWithUnderscores (NameSeg, PaddedNameSeg);
822     }
823     else
824     {
825         /* No dots in the namepath, there is only a single nameseg. */
826         /* Handle prefixes */
827 
828         while (ACPI_IS_ROOT_PREFIX (*Name) ||
829                ACPI_IS_PARENT_PREFIX (*Name))
830         {
831             Name++;
832         }
833 
834         /* Remaining string should be one single nameseg */
835 
836         UtPadNameWithUnderscores (Name, PaddedNameSeg);
837     }
838 
839     ACPI_MOVE_NAME (Op->Asl.NameSeg, PaddedNameSeg);
840 }
841 
842 
843 /*******************************************************************************
844  *
845  * FUNCTION:    UtAttachNamepathToOwner
846  *
847  * PARAMETERS:  Op                  - Parent parse node
848  *              NameOp              - Node that contains the name
849  *
850  * RETURN:      Sets the ExternalName and Namepath in the parent node
851  *
852  * DESCRIPTION: Store the name in two forms in the parent node: The original
853  *              (external) name, and the internalized name that is used within
854  *              the ACPI namespace manager.
855  *
856  ******************************************************************************/
857 
858 void
859 UtAttachNamepathToOwner (
860     ACPI_PARSE_OBJECT       *Op,
861     ACPI_PARSE_OBJECT       *NameOp)
862 {
863     ACPI_STATUS             Status;
864 
865 
866     /* Full external path */
867 
868     Op->Asl.ExternalName = NameOp->Asl.Value.String;
869 
870     /* Save the NameOp for possible error reporting later */
871 
872     Op->Asl.ParentMethod = (void *) NameOp;
873 
874     /* Last nameseg of the path */
875 
876     UtAttachNameseg (Op, Op->Asl.ExternalName);
877 
878     /* Create internalized path */
879 
880     Status = UtInternalizeName (NameOp->Asl.Value.String, &Op->Asl.Namepath);
881     if (ACPI_FAILURE (Status))
882     {
883         /* TBD: abort on no memory */
884     }
885 }
886 
887 
888 /*******************************************************************************
889  *
890  * FUNCTION:    UtDoConstant
891  *
892  * PARAMETERS:  String              - Hex, Octal, or Decimal string
893  *
894  * RETURN:      Converted Integer
895  *
896  * DESCRIPTION: Convert a string to an integer, with error checking.
897  *
898  ******************************************************************************/
899 
900 UINT64
901 UtDoConstant (
902     char                    *String)
903 {
904     ACPI_STATUS             Status;
905     UINT64                  Converted;
906     char                    ErrBuf[64];
907 
908 
909     Status = UtStrtoul64 (String, 0, &Converted);
910     if (ACPI_FAILURE (Status))
911     {
912         sprintf (ErrBuf, "%s %s\n", "Conversion error:",
913             AcpiFormatException (Status));
914         AslCompilererror (ErrBuf);
915     }
916 
917     return (Converted);
918 }
919 
920 
921 /* TBD: use version in ACPI CA main code base? */
922 
923 /*******************************************************************************
924  *
925  * FUNCTION:    UtStrtoul64
926  *
927  * PARAMETERS:  String              - Null terminated string
928  *              Terminater          - Where a pointer to the terminating byte
929  *                                    is returned
930  *              Base                - Radix of the string
931  *
932  * RETURN:      Converted value
933  *
934  * DESCRIPTION: Convert a string into an unsigned value.
935  *
936  ******************************************************************************/
937 
938 ACPI_STATUS
939 UtStrtoul64 (
940     char                    *String,
941     UINT32                  Base,
942     UINT64                  *RetInteger)
943 {
944     UINT32                  Index;
945     UINT32                  Sign;
946     UINT64                  ReturnValue = 0;
947     ACPI_STATUS             Status = AE_OK;
948 
949 
950     *RetInteger = 0;
951 
952     switch (Base)
953     {
954     case 0:
955     case 8:
956     case 10:
957     case 16:
958 
959         break;
960 
961     default:
962         /*
963          * The specified Base parameter is not in the domain of
964          * this function:
965          */
966         return (AE_BAD_PARAMETER);
967     }
968 
969     /* Skip over any white space in the buffer: */
970 
971     while (isspace ((int) *String) || *String == '\t')
972     {
973         ++String;
974     }
975 
976     /*
977      * The buffer may contain an optional plus or minus sign.
978      * If it does, then skip over it but remember what is was:
979      */
980     if (*String == '-')
981     {
982         Sign = NEGATIVE;
983         ++String;
984     }
985     else if (*String == '+')
986     {
987         ++String;
988         Sign = POSITIVE;
989     }
990     else
991     {
992         Sign = POSITIVE;
993     }
994 
995     /*
996      * If the input parameter Base is zero, then we need to
997      * determine if it is octal, decimal, or hexadecimal:
998      */
999     if (Base == 0)
1000     {
1001         if (*String == '0')
1002         {
1003             if (tolower ((int) *(++String)) == 'x')
1004             {
1005                 Base = 16;
1006                 ++String;
1007             }
1008             else
1009             {
1010                 Base = 8;
1011             }
1012         }
1013         else
1014         {
1015             Base = 10;
1016         }
1017     }
1018 
1019     /*
1020      * For octal and hexadecimal bases, skip over the leading
1021      * 0 or 0x, if they are present.
1022      */
1023     if (Base == 8 && *String == '0')
1024     {
1025         String++;
1026     }
1027 
1028     if (Base == 16 &&
1029         *String == '0' &&
1030         tolower ((int) *(++String)) == 'x')
1031     {
1032         String++;
1033     }
1034 
1035     /* Main loop: convert the string to an unsigned long */
1036 
1037     while (*String)
1038     {
1039         if (isdigit ((int) *String))
1040         {
1041             Index = ((UINT8) *String) - '0';
1042         }
1043         else
1044         {
1045             Index = (UINT8) toupper ((int) *String);
1046             if (isupper ((int) Index))
1047             {
1048                 Index = Index - 'A' + 10;
1049             }
1050             else
1051             {
1052                 goto ErrorExit;
1053             }
1054         }
1055 
1056         if (Index >= Base)
1057         {
1058             goto ErrorExit;
1059         }
1060 
1061         /* Check to see if value is out of range: */
1062 
1063         if (ReturnValue > ((ACPI_UINT64_MAX - (UINT64) Index) /
1064                             (UINT64) Base))
1065         {
1066             goto ErrorExit;
1067         }
1068         else
1069         {
1070             ReturnValue *= Base;
1071             ReturnValue += Index;
1072         }
1073 
1074         ++String;
1075     }
1076 
1077 
1078     /* If a minus sign was present, then "the conversion is negated": */
1079 
1080     if (Sign == NEGATIVE)
1081     {
1082         ReturnValue = (ACPI_UINT32_MAX - ReturnValue) + 1;
1083     }
1084 
1085     *RetInteger = ReturnValue;
1086     return (Status);
1087 
1088 
1089 ErrorExit:
1090     switch (Base)
1091     {
1092     case 8:
1093 
1094         Status = AE_BAD_OCTAL_CONSTANT;
1095         break;
1096 
1097     case 10:
1098 
1099         Status = AE_BAD_DECIMAL_CONSTANT;
1100         break;
1101 
1102     case 16:
1103 
1104         Status = AE_BAD_HEX_CONSTANT;
1105         break;
1106 
1107     default:
1108 
1109         /* Base validated above */
1110 
1111         break;
1112     }
1113 
1114     return (Status);
1115 }
1116