dtexpress.c revision 1.1.1.1.8.2 1 /******************************************************************************
2 *
3 * Module Name: dtexpress.c - Support for integer expressions and labels
4 *
5 *****************************************************************************/
6
7 /*
8 * Copyright (C) 2000 - 2011, 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 #define __DTEXPRESS_C__
45
46 #include "aslcompiler.h"
47 #include "dtcompiler.h"
48
49 #define _COMPONENT DT_COMPILER
50 ACPI_MODULE_NAME ("dtexpress")
51
52
53 /* Local prototypes */
54
55 static UINT64
56 DtResolveInteger (
57 DT_FIELD *Field,
58 char *IntegerString);
59
60 static void
61 DtInsertLabelField (
62 DT_FIELD *Field);
63
64 static DT_FIELD *
65 DtLookupLabel (
66 char *Name);
67
68
69 /******************************************************************************
70 *
71 * FUNCTION: DtResolveIntegerExpression
72 *
73 * PARAMETERS: Field - Field object with Integer expression
74 *
75 * RETURN: A 64-bit integer value
76 *
77 * DESCRIPTION: Resolve an integer expression to a single value. Supports
78 * both integer constants and labels. Supported operators are:
79 * +,-,*,/,%,|,&,^
80 *
81 *****************************************************************************/
82
83 UINT64
84 DtResolveIntegerExpression (
85 DT_FIELD *Field)
86 {
87 char *IntegerString;
88 char *Operator;
89 UINT64 Value;
90 UINT64 Value2;
91
92
93 DbgPrint (ASL_DEBUG_OUTPUT, "Full Integer expression: %s\n",
94 Field->Value);
95
96 strcpy (MsgBuffer, Field->Value); /* Must take a copy for strtok() */
97
98 /* Obtain and resolve the first operand */
99
100 IntegerString = strtok (MsgBuffer, " ");
101 if (!IntegerString)
102 {
103 DtError (ASL_ERROR, ASL_MSG_INVALID_EXPRESSION, Field, Field->Value);
104 return (0);
105 }
106
107 Value = DtResolveInteger (Field, IntegerString);
108 DbgPrint (ASL_DEBUG_OUTPUT, "Integer resolved to V1: %8.8X%8.8X\n",
109 ACPI_FORMAT_UINT64 (Value));
110
111 /*
112 * Consume the entire expression string. For the rest of the
113 * expression string, values are of the form:
114 * <operator> <integer>
115 */
116 while (1)
117 {
118 Operator = strtok (NULL, " ");
119 if (!Operator)
120 {
121 /* Normal exit */
122
123 DbgPrint (ASL_DEBUG_OUTPUT, "Expression Resolved to: %8.8X%8.8X\n",
124 ACPI_FORMAT_UINT64 (Value));
125
126 return (Value);
127 }
128
129 IntegerString = strtok (NULL, " ");
130 if (!IntegerString ||
131 (strlen (Operator) > 1))
132 {
133 /* No corresponding operand for operator or invalid operator */
134
135 DtError (ASL_ERROR, ASL_MSG_INVALID_EXPRESSION, Field, Field->Value);
136 return (0);
137 }
138
139 Value2 = DtResolveInteger (Field, IntegerString);
140 DbgPrint (ASL_DEBUG_OUTPUT, "Integer resolved to V2: %8.8X%8.8X\n",
141 ACPI_FORMAT_UINT64 (Value2));
142
143 /* Perform the requested operation */
144
145 switch (*Operator)
146 {
147 case '-':
148 Value -= Value2;
149 break;
150
151 case '+':
152 Value += Value2;
153 break;
154
155 case '*':
156 Value *= Value2;
157 break;
158
159 case '|':
160 Value |= Value2;
161 break;
162
163 case '&':
164 Value &= Value2;
165 break;
166
167 case '^':
168 Value ^= Value2;
169 break;
170
171 case '/':
172 if (!Value2)
173 {
174 DtError (ASL_ERROR, ASL_MSG_DIVIDE_BY_ZERO, Field, Field->Value);
175 return (0);
176 }
177 Value /= Value2;
178 break;
179
180 case '%':
181 if (!Value2)
182 {
183 DtError (ASL_ERROR, ASL_MSG_DIVIDE_BY_ZERO, Field, Field->Value);
184 return (0);
185 }
186 Value %= Value2;
187 break;
188
189 default:
190
191 /* Unknown operator */
192
193 DtFatal (ASL_MSG_INVALID_EXPRESSION, Field, Field->Value);
194 break;
195 }
196 }
197
198 return (Value);
199 }
200
201
202 /******************************************************************************
203 *
204 * FUNCTION: DtResolveInteger
205 *
206 * PARAMETERS: Field - Field object with string to be resolved
207 * IntegerString - Integer to be resolved
208 *
209 * RETURN: A 64-bit integer value
210 *
211 * DESCRIPTION: Resolve a single integer string to a value. Supports both
212 * integer constants and labels.
213 *
214 * NOTE: References to labels must begin with a dollar sign ($)
215 *
216 *****************************************************************************/
217
218 static UINT64
219 DtResolveInteger (
220 DT_FIELD *Field,
221 char *IntegerString)
222 {
223 DT_FIELD *LabelField;
224 UINT64 Value = 0;
225 char *Message = NULL;
226 ACPI_STATUS Status;
227
228
229 DbgPrint (ASL_DEBUG_OUTPUT, "Resolve Integer: %s\n", IntegerString);
230
231 /* Resolve a label reference to an integer (table offset) */
232
233 if (*IntegerString == '$')
234 {
235 LabelField = DtLookupLabel (IntegerString);
236 if (!LabelField)
237 {
238 DtError (ASL_ERROR, ASL_MSG_UNKNOWN_LABEL, Field, IntegerString);
239 return (0);
240 }
241
242 /* All we need from the label is the offset in the table */
243
244 Value = LabelField->TableOffset;
245 return (Value);
246 }
247
248 /* Convert string to an actual integer */
249
250 Status = DtStrtoul64 (IntegerString, &Value);
251 if (ACPI_FAILURE (Status))
252 {
253 if (Status == AE_LIMIT)
254 {
255 Message = "Constant larger than 64 bits";
256 }
257 else if (Status == AE_BAD_CHARACTER)
258 {
259 Message = "Invalid character in constant";
260 }
261
262 DtError (ASL_ERROR, ASL_MSG_INVALID_HEX_INTEGER, Field, Message);
263 }
264
265 return (Value);
266 }
267
268
269 /******************************************************************************
270 *
271 * FUNCTION: DtDetectAllLabels
272 *
273 * PARAMETERS: FieldList - Field object at start of generic list
274 *
275 * RETURN: None
276 *
277 * DESCRIPTION: Detect all labels in a list of "generic" opcodes (such as
278 * a UEFI table.) and insert them into the global label list.
279 *
280 *****************************************************************************/
281
282 void
283 DtDetectAllLabels (
284 DT_FIELD *FieldList)
285 {
286 ACPI_DMTABLE_INFO *Info;
287 DT_FIELD *GenericField;
288 UINT32 TableOffset;
289
290
291 TableOffset = Gbl_CurrentTableOffset;
292 GenericField = FieldList;
293
294 /*
295 * Process all "Label:" fields within the parse tree. We need
296 * to know the offsets for all labels before we can compile
297 * the parse tree in order to handle forward references. Traverse
298 * tree and get/set all field lengths of all operators in order to
299 * determine the label offsets.
300 */
301 while (GenericField)
302 {
303 Info = DtGetGenericTableInfo (GenericField->Name);
304 if (Info)
305 {
306 /* Maintain table offsets */
307
308 GenericField->TableOffset = TableOffset;
309 TableOffset += DtGetFieldLength (GenericField, Info);
310
311 /* Insert all labels in the global label list */
312
313 if (Info->Opcode == ACPI_DMT_LABEL)
314 {
315 DtInsertLabelField (GenericField);
316 }
317 }
318
319 GenericField = GenericField->Next;
320 }
321 }
322
323
324 /******************************************************************************
325 *
326 * FUNCTION: DtInsertLabelField
327 *
328 * PARAMETERS: Field - Field object with Label to be inserted
329 *
330 * RETURN: None
331 *
332 * DESCRIPTION: Insert a label field into the global label list
333 *
334 *****************************************************************************/
335
336 static void
337 DtInsertLabelField (
338 DT_FIELD *Field)
339 {
340
341 DbgPrint (ASL_DEBUG_OUTPUT,
342 "DtInsertLabelField: Found Label : %s at output table offset %X\n",
343 Field->Value, Field->TableOffset);
344
345 Field->NextLabel = Gbl_LabelList;
346 Gbl_LabelList = Field;
347 }
348
349
350 /******************************************************************************
351 *
352 * FUNCTION: DtLookupLabel
353 *
354 * PARAMETERS: Name - Label to be resolved
355 *
356 * RETURN: Field object associated with the label
357 *
358 * DESCRIPTION: Lookup a label in the global label list. Used during the
359 * resolution of integer expressions.
360 *
361 *****************************************************************************/
362
363 static DT_FIELD *
364 DtLookupLabel (
365 char *Name)
366 {
367 DT_FIELD *LabelField;
368
369
370 /* Skip a leading $ */
371
372 if (*Name == '$')
373 {
374 Name++;
375 }
376
377 /* Search global list */
378
379 LabelField = Gbl_LabelList;
380 while (LabelField)
381 {
382 if (!ACPI_STRCMP (Name, LabelField->Value))
383 {
384 return (LabelField);
385 }
386 LabelField = LabelField->NextLabel;
387 }
388
389 return (NULL);
390 }
391