prexpress.c revision 281075
1115013Smarcel/******************************************************************************
2160157Smarcel *
3121642Smarcel * Module Name: prexpress - Preprocessor #if expression support
4121642Smarcel *
5121642Smarcel *****************************************************************************/
6121642Smarcel
7121642Smarcel/*
8121642Smarcel * Copyright (C) 2000 - 2015, Intel Corp.
9121642Smarcel * All rights reserved.
10121642Smarcel *
11115013Smarcel * Redistribution and use in source and binary forms, with or without
12121642Smarcel * modification, are permitted provided that the following conditions
13121642Smarcel * are met:
14121642Smarcel * 1. Redistributions of source code must retain the above copyright
15121642Smarcel *    notice, this list of conditions, and the following disclaimer,
16121642Smarcel *    without modification.
17121642Smarcel * 2. Redistributions in binary form must reproduce at minimum a disclaimer
18121642Smarcel *    substantially similar to the "NO WARRANTY" disclaimer below
19121642Smarcel *    ("Disclaimer") and any redistribution must be conditioned upon
20121642Smarcel *    including a substantially similar Disclaimer requirement for further
21121642Smarcel *    binary redistribution.
22121642Smarcel * 3. Neither the names of the above-listed copyright holders nor the names
23121642Smarcel *    of any contributors may be used to endorse or promote products derived
24121642Smarcel *    from this software without specific prior written permission.
25115013Smarcel *
26115013Smarcel * Alternatively, this software may be distributed under the terms of the
27115013Smarcel * GNU General Public License ("GPL") version 2 as published by the Free
28115013Smarcel * Software Foundation.
29115013Smarcel *
30115013Smarcel * NO WARRANTY
31115013Smarcel * THIS SOFTWARE IS PROVIDED BY THE COPYRIGHT HOLDERS AND CONTRIBUTORS
32115013Smarcel * "AS IS" AND ANY EXPRESS OR IMPLIED WARRANTIES, INCLUDING, BUT NOT
33115013Smarcel * LIMITED TO, THE IMPLIED WARRANTIES OF MERCHANTIBILITY AND FITNESS FOR
34115013Smarcel * A PARTICULAR PURPOSE ARE DISCLAIMED. IN NO EVENT SHALL THE COPYRIGHT
35115013Smarcel * HOLDERS OR CONTRIBUTORS BE LIABLE FOR SPECIAL, EXEMPLARY, OR CONSEQUENTIAL
36115013Smarcel * DAMAGES (INCLUDING, BUT NOT LIMITED TO, PROCUREMENT OF SUBSTITUTE GOODS
37115013Smarcel * OR SERVICES; LOSS OF USE, DATA, OR PROFITS; OR BUSINESS INTERRUPTION)
38115013Smarcel * HOWEVER CAUSED AND ON ANY THEORY OF LIABILITY, WHETHER IN CONTRACT,
39115013Smarcel * STRICT LIABILITY, OR TORT (INCLUDING NEGLIGENCE OR OTHERWISE) ARISING
40115013Smarcel * IN ANY WAY OUT OF THE USE OF THIS SOFTWARE, EVEN IF ADVISED OF THE
41115013Smarcel * POSSIBILITY OF SUCH DAMAGES.
42115013Smarcel */
43115013Smarcel
44115013Smarcel#include <contrib/dev/acpica/compiler/aslcompiler.h>
45115013Smarcel#include <contrib/dev/acpica/compiler/dtcompiler.h>
46115013Smarcel
47115013Smarcel
48115013Smarcel#define _COMPONENT          ASL_PREPROCESSOR
49115013Smarcel        ACPI_MODULE_NAME    ("prexpress")
50115013Smarcel
51115013Smarcel/* Local prototypes */
52115013Smarcel
53115013Smarcelstatic char *
54160157SmarcelPrExpandMacros (
55115013Smarcel    char                    *Line);
56115013Smarcel
57160157Smarcel
58160157Smarcel#ifdef _UNDER_DEVELOPMENT
59160157Smarcel/******************************************************************************
60160157Smarcel *
61115013Smarcel * FUNCTION:    PrUnTokenize
62115013Smarcel *
63115013Smarcel * PARAMETERS:  Buffer              - Token Buffer
64115013Smarcel *              Next                - "Next" buffer from GetNextToken
65115013Smarcel *
66115013Smarcel * RETURN:      None
67115013Smarcel *
68115013Smarcel * DESCRIPTION: Un-tokenized the current token buffer. The implementation is
69115013Smarcel *              to simply set the null inserted by GetNextToken to a blank.
70115013Smarcel *              If Next is NULL, there were no tokens found in the Buffer,
71115013Smarcel *              so there is nothing to do.
72115013Smarcel *
73115013Smarcel *****************************************************************************/
74115013Smarcel
75115013Smarcelstatic void
76115013SmarcelPrUnTokenize (
77115013Smarcel    char                    *Buffer,
78115013Smarcel    char                    *Next)
79115013Smarcel{
80115013Smarcel    UINT32                  Length = strlen (Buffer);
81115013Smarcel
82115013Smarcel
83115013Smarcel    if (!Next)
84115013Smarcel    {
85        return;
86    }
87    if (Buffer[Length] != '\n')
88    {
89        Buffer[strlen(Buffer)] = ' ';
90    }
91}
92#endif
93
94
95/******************************************************************************
96 *
97 * FUNCTION:    PrExpandMacros
98 *
99 * PARAMETERS:  Line                - Pointer into the current line
100 *
101 * RETURN:      Updated pointer into the current line
102 *
103 * DESCRIPTION: Expand any macros found in the current line buffer.
104 *
105 *****************************************************************************/
106
107static char *
108PrExpandMacros (
109    char                    *Line)
110{
111    char                    *Token;
112    char                    *ReplaceString;
113    PR_DEFINE_INFO          *DefineInfo;
114    ACPI_SIZE               TokenOffset;
115    char                    *Next;
116    int                     OffsetAdjust;
117
118
119    strcpy (Gbl_ExpressionTokenBuffer, Gbl_CurrentLineBuffer);
120    Token = PrGetNextToken (Gbl_ExpressionTokenBuffer, PR_EXPR_SEPARATORS, &Next);
121    OffsetAdjust = 0;
122
123    while (Token)
124    {
125        DefineInfo = PrMatchDefine (Token);
126        if (DefineInfo)
127        {
128            if (DefineInfo->Body)
129            {
130                /* This is a macro. TBD: Is this allowed? */
131
132                DbgPrint (ASL_DEBUG_OUTPUT, PR_PREFIX_ID
133                    "Matched Macro: %s->%s\n",
134                    Gbl_CurrentLineNumber, DefineInfo->Identifier,
135                    DefineInfo->Replacement);
136
137                PrDoMacroInvocation (Gbl_ExpressionTokenBuffer, Token,
138                    DefineInfo, &Next);
139            }
140            else
141            {
142                ReplaceString = DefineInfo->Replacement;
143
144                /* Replace the name in the original line buffer */
145
146                TokenOffset = Token - Gbl_ExpressionTokenBuffer + OffsetAdjust;
147                PrReplaceData (
148                    &Gbl_CurrentLineBuffer[TokenOffset], strlen (Token),
149                    ReplaceString, strlen (ReplaceString));
150
151                /* Adjust for length difference between old and new name length */
152
153                OffsetAdjust += strlen (ReplaceString) - strlen (Token);
154
155                DbgPrint (ASL_DEBUG_OUTPUT, PR_PREFIX_ID
156                    "Matched #define within expression: %s->%s\n",
157                    Gbl_CurrentLineNumber, Token,
158                    *ReplaceString ? ReplaceString : "(NULL STRING)");
159            }
160        }
161
162        Token = PrGetNextToken (NULL, PR_EXPR_SEPARATORS, &Next);
163    }
164
165    return (Line);
166}
167
168
169/******************************************************************************
170 *
171 * FUNCTION:    PrIsDefined
172 *
173 * PARAMETERS:  Identifier          - Name to be resolved
174 *
175 * RETURN:      64-bit boolean integer value
176 *
177 * DESCRIPTION: Returns TRUE if the name is defined, FALSE otherwise (0).
178 *
179 *****************************************************************************/
180
181UINT64
182PrIsDefined (
183    char                    *Identifier)
184{
185    UINT64                  Value;
186    PR_DEFINE_INFO          *DefineInfo;
187
188
189    DbgPrint (ASL_DEBUG_OUTPUT, PR_PREFIX_ID
190        "**** Is defined?:  %s\n", Gbl_CurrentLineNumber, Identifier);
191
192    Value = 0; /* Default is "Not defined" -- FALSE */
193
194    DefineInfo = PrMatchDefine (Identifier);
195    if (DefineInfo)
196    {
197        Value = ACPI_UINT64_MAX; /* TRUE */
198    }
199
200    DbgPrint (ASL_DEBUG_OUTPUT, PR_PREFIX_ID
201        "[#if defined %s] resolved to: %8.8X%8.8X\n",
202        Gbl_CurrentLineNumber, Identifier, ACPI_FORMAT_UINT64 (Value));
203
204    return (Value);
205}
206
207
208/******************************************************************************
209 *
210 * FUNCTION:    PrResolveDefine
211 *
212 * PARAMETERS:  Identifier          - Name to be resolved
213 *
214 * RETURN:      A 64-bit boolean integer value
215 *
216 * DESCRIPTION: Returns TRUE if the name is defined, FALSE otherwise (0).
217 *
218 *****************************************************************************/
219
220UINT64
221PrResolveDefine (
222    char                    *Identifier)
223{
224    UINT64                  Value;
225    PR_DEFINE_INFO          *DefineInfo;
226
227
228    DbgPrint (ASL_DEBUG_OUTPUT, PR_PREFIX_ID
229        "**** Resolve #define:  %s\n", Gbl_CurrentLineNumber, Identifier);
230
231    Value = 0; /* Default is "Not defined" -- FALSE */
232
233    DefineInfo = PrMatchDefine (Identifier);
234    if (DefineInfo)
235    {
236        Value = ACPI_UINT64_MAX; /* TRUE */
237    }
238
239    DbgPrint (ASL_DEBUG_OUTPUT, PR_PREFIX_ID
240        "[#if defined %s] resolved to: %8.8X%8.8X\n",
241        Gbl_CurrentLineNumber, Identifier, ACPI_FORMAT_UINT64 (Value));
242
243    return (Value);
244}
245
246
247/******************************************************************************
248 *
249 * FUNCTION:    PrResolveIntegerExpression
250 *
251 * PARAMETERS:  Line                - Pointer to integer expression
252 *              ReturnValue         - Where the resolved 64-bit integer is
253 *                                    returned.
254 *
255 * RETURN:      Status
256 *
257 * DESCRIPTION: Resolve an integer expression to a single value. Supports
258 *              both integer constants and labels.
259 *
260 *****************************************************************************/
261
262ACPI_STATUS
263PrResolveIntegerExpression (
264    char                    *Line,
265    UINT64                  *ReturnValue)
266{
267    UINT64                  Result;
268    char                    *ExpandedLine;
269
270
271    DbgPrint (ASL_DEBUG_OUTPUT, PR_PREFIX_ID
272        "**** Resolve #if:  %s\n", Gbl_CurrentLineNumber, Line);
273
274    /* Expand all macros within the expression first */
275
276    ExpandedLine = PrExpandMacros (Line);
277
278    /* Now we can evaluate the expression */
279
280    Result = PrEvaluateExpression (ExpandedLine);
281    DbgPrint (ASL_DEBUG_OUTPUT, PR_PREFIX_ID
282        "**** Expression Resolved to: %8.8X%8.8X\n",
283        Gbl_CurrentLineNumber, ACPI_FORMAT_UINT64 (Result));
284
285    *ReturnValue = Result;
286    return (AE_OK);
287
288#if 0
289InvalidExpression:
290
291    ACPI_FREE (EvalBuffer);
292    PrError (ASL_ERROR, ASL_MSG_INVALID_EXPRESSION, 0);
293    return (AE_ERROR);
294
295
296NormalExit:
297
298    DbgPrint (ASL_DEBUG_OUTPUT, PR_PREFIX_ID
299        "**** Expression Resolved to: %8.8X%8.8X\n",
300        Gbl_CurrentLineNumber, ACPI_FORMAT_UINT64 (Value1));
301
302    *ReturnValue = Value1;
303    return (AE_OK);
304#endif
305}
306