xref: /freebsd/sys/contrib/dev/acpica/components/parser/psxface.c (revision 0e97acdf58fe27b09c4824a474b0344daf997c5f)
1 /******************************************************************************
2  *
3  * Module Name: psxface - Parser external interfaces
4  *
5  *****************************************************************************/
6 
7 /*
8  * Copyright (C) 2000 - 2014, 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 __PSXFACE_C__
45 
46 #include <contrib/dev/acpica/include/acpi.h>
47 #include <contrib/dev/acpica/include/accommon.h>
48 #include <contrib/dev/acpica/include/acparser.h>
49 #include <contrib/dev/acpica/include/acdispat.h>
50 #include <contrib/dev/acpica/include/acinterp.h>
51 #include <contrib/dev/acpica/include/actables.h>
52 
53 
54 #define _COMPONENT          ACPI_PARSER
55         ACPI_MODULE_NAME    ("psxface")
56 
57 /* Local Prototypes */
58 
59 static void
60 AcpiPsStartTrace (
61     ACPI_EVALUATE_INFO      *Info);
62 
63 static void
64 AcpiPsStopTrace (
65     ACPI_EVALUATE_INFO      *Info);
66 
67 static void
68 AcpiPsUpdateParameterList (
69     ACPI_EVALUATE_INFO      *Info,
70     UINT16                  Action);
71 
72 
73 /*******************************************************************************
74  *
75  * FUNCTION:    AcpiDebugTrace
76  *
77  * PARAMETERS:  MethodName      - Valid ACPI name string
78  *              DebugLevel      - Optional level mask. 0 to use default
79  *              DebugLayer      - Optional layer mask. 0 to use default
80  *              Flags           - bit 1: one shot(1) or persistent(0)
81  *
82  * RETURN:      Status
83  *
84  * DESCRIPTION: External interface to enable debug tracing during control
85  *              method execution
86  *
87  ******************************************************************************/
88 
89 ACPI_STATUS
90 AcpiDebugTrace (
91     char                    *Name,
92     UINT32                  DebugLevel,
93     UINT32                  DebugLayer,
94     UINT32                  Flags)
95 {
96     ACPI_STATUS             Status;
97 
98 
99     Status = AcpiUtAcquireMutex (ACPI_MTX_NAMESPACE);
100     if (ACPI_FAILURE (Status))
101     {
102         return (Status);
103     }
104 
105     /* TBDs: Validate name, allow full path or just nameseg */
106 
107     AcpiGbl_TraceMethodName = *ACPI_CAST_PTR (UINT32, Name);
108     AcpiGbl_TraceFlags = Flags;
109 
110     if (DebugLevel)
111     {
112         AcpiGbl_TraceDbgLevel = DebugLevel;
113     }
114     if (DebugLayer)
115     {
116         AcpiGbl_TraceDbgLayer = DebugLayer;
117     }
118 
119     (void) AcpiUtReleaseMutex (ACPI_MTX_NAMESPACE);
120     return (AE_OK);
121 }
122 
123 
124 /*******************************************************************************
125  *
126  * FUNCTION:    AcpiPsStartTrace
127  *
128  * PARAMETERS:  Info        - Method info struct
129  *
130  * RETURN:      None
131  *
132  * DESCRIPTION: Start control method execution trace
133  *
134  ******************************************************************************/
135 
136 static void
137 AcpiPsStartTrace (
138     ACPI_EVALUATE_INFO      *Info)
139 {
140     ACPI_STATUS             Status;
141 
142 
143     ACPI_FUNCTION_ENTRY ();
144 
145 
146     Status = AcpiUtAcquireMutex (ACPI_MTX_NAMESPACE);
147     if (ACPI_FAILURE (Status))
148     {
149         return;
150     }
151 
152     if ((!AcpiGbl_TraceMethodName) ||
153         (AcpiGbl_TraceMethodName != Info->Node->Name.Integer))
154     {
155         goto Exit;
156     }
157 
158     AcpiGbl_OriginalDbgLevel = AcpiDbgLevel;
159     AcpiGbl_OriginalDbgLayer = AcpiDbgLayer;
160 
161     AcpiDbgLevel = 0x00FFFFFF;
162     AcpiDbgLayer = ACPI_UINT32_MAX;
163 
164     if (AcpiGbl_TraceDbgLevel)
165     {
166         AcpiDbgLevel = AcpiGbl_TraceDbgLevel;
167     }
168     if (AcpiGbl_TraceDbgLayer)
169     {
170         AcpiDbgLayer = AcpiGbl_TraceDbgLayer;
171     }
172 
173 
174 Exit:
175     (void) AcpiUtReleaseMutex (ACPI_MTX_NAMESPACE);
176 }
177 
178 
179 /*******************************************************************************
180  *
181  * FUNCTION:    AcpiPsStopTrace
182  *
183  * PARAMETERS:  Info        - Method info struct
184  *
185  * RETURN:      None
186  *
187  * DESCRIPTION: Stop control method execution trace
188  *
189  ******************************************************************************/
190 
191 static void
192 AcpiPsStopTrace (
193     ACPI_EVALUATE_INFO      *Info)
194 {
195     ACPI_STATUS             Status;
196 
197 
198     ACPI_FUNCTION_ENTRY ();
199 
200 
201     Status = AcpiUtAcquireMutex (ACPI_MTX_NAMESPACE);
202     if (ACPI_FAILURE (Status))
203     {
204         return;
205     }
206 
207     if ((!AcpiGbl_TraceMethodName) ||
208         (AcpiGbl_TraceMethodName != Info->Node->Name.Integer))
209     {
210         goto Exit;
211     }
212 
213     /* Disable further tracing if type is one-shot */
214 
215     if (AcpiGbl_TraceFlags & 1)
216     {
217         AcpiGbl_TraceMethodName = 0;
218         AcpiGbl_TraceDbgLevel = 0;
219         AcpiGbl_TraceDbgLayer = 0;
220     }
221 
222     AcpiDbgLevel = AcpiGbl_OriginalDbgLevel;
223     AcpiDbgLayer = AcpiGbl_OriginalDbgLayer;
224 
225 Exit:
226     (void) AcpiUtReleaseMutex (ACPI_MTX_NAMESPACE);
227 }
228 
229 
230 /*******************************************************************************
231  *
232  * FUNCTION:    AcpiPsExecuteMethod
233  *
234  * PARAMETERS:  Info            - Method info block, contains:
235  *                  Node            - Method Node to execute
236  *                  ObjDesc         - Method object
237  *                  Parameters      - List of parameters to pass to the method,
238  *                                    terminated by NULL. Params itself may be
239  *                                    NULL if no parameters are being passed.
240  *                  ReturnObject    - Where to put method's return value (if
241  *                                    any). If NULL, no value is returned.
242  *                  ParameterType   - Type of Parameter list
243  *                  ReturnObject    - Where to put method's return value (if
244  *                                    any). If NULL, no value is returned.
245  *                  PassNumber      - Parse or execute pass
246  *
247  * RETURN:      Status
248  *
249  * DESCRIPTION: Execute a control method
250  *
251  ******************************************************************************/
252 
253 ACPI_STATUS
254 AcpiPsExecuteMethod (
255     ACPI_EVALUATE_INFO      *Info)
256 {
257     ACPI_STATUS             Status;
258     ACPI_PARSE_OBJECT       *Op;
259     ACPI_WALK_STATE         *WalkState;
260 
261 
262     ACPI_FUNCTION_TRACE (PsExecuteMethod);
263 
264 
265     /* Quick validation of DSDT header */
266 
267     AcpiTbCheckDsdtHeader ();
268 
269     /* Validate the Info and method Node */
270 
271     if (!Info || !Info->Node)
272     {
273         return_ACPI_STATUS (AE_NULL_ENTRY);
274     }
275 
276     /* Init for new method, wait on concurrency semaphore */
277 
278     Status = AcpiDsBeginMethodExecution (Info->Node, Info->ObjDesc, NULL);
279     if (ACPI_FAILURE (Status))
280     {
281         return_ACPI_STATUS (Status);
282     }
283 
284     /*
285      * The caller "owns" the parameters, so give each one an extra reference
286      */
287     AcpiPsUpdateParameterList (Info, REF_INCREMENT);
288 
289     /* Begin tracing if requested */
290 
291     AcpiPsStartTrace (Info);
292 
293     /*
294      * Execute the method. Performs parse simultaneously
295      */
296     ACPI_DEBUG_PRINT ((ACPI_DB_PARSE,
297         "**** Begin Method Parse/Execute [%4.4s] **** Node=%p Obj=%p\n",
298         Info->Node->Name.Ascii, Info->Node, Info->ObjDesc));
299 
300     /* Create and init a Root Node */
301 
302     Op = AcpiPsCreateScopeOp ();
303     if (!Op)
304     {
305         Status = AE_NO_MEMORY;
306         goto Cleanup;
307     }
308 
309     /* Create and initialize a new walk state */
310 
311     Info->PassNumber = ACPI_IMODE_EXECUTE;
312     WalkState = AcpiDsCreateWalkState (
313                     Info->ObjDesc->Method.OwnerId, NULL, NULL, NULL);
314     if (!WalkState)
315     {
316         Status = AE_NO_MEMORY;
317         goto Cleanup;
318     }
319 
320     Status = AcpiDsInitAmlWalk (WalkState, Op, Info->Node,
321                 Info->ObjDesc->Method.AmlStart,
322                 Info->ObjDesc->Method.AmlLength, Info, Info->PassNumber);
323     if (ACPI_FAILURE (Status))
324     {
325         AcpiDsDeleteWalkState (WalkState);
326         goto Cleanup;
327     }
328 
329     if (Info->ObjDesc->Method.InfoFlags & ACPI_METHOD_MODULE_LEVEL)
330     {
331         WalkState->ParseFlags |= ACPI_PARSE_MODULE_LEVEL;
332     }
333 
334     /* Invoke an internal method if necessary */
335 
336     if (Info->ObjDesc->Method.InfoFlags & ACPI_METHOD_INTERNAL_ONLY)
337     {
338         Status = Info->ObjDesc->Method.Dispatch.Implementation (WalkState);
339         Info->ReturnObject = WalkState->ReturnDesc;
340 
341         /* Cleanup states */
342 
343         AcpiDsScopeStackClear (WalkState);
344         AcpiPsCleanupScope (&WalkState->ParserState);
345         AcpiDsTerminateControlMethod (WalkState->MethodDesc, WalkState);
346         AcpiDsDeleteWalkState (WalkState);
347         goto Cleanup;
348     }
349 
350     /*
351      * Start method evaluation with an implicit return of zero.
352      * This is done for Windows compatibility.
353      */
354     if (AcpiGbl_EnableInterpreterSlack)
355     {
356         WalkState->ImplicitReturnObj =
357             AcpiUtCreateIntegerObject ((UINT64) 0);
358         if (!WalkState->ImplicitReturnObj)
359         {
360             Status = AE_NO_MEMORY;
361             AcpiDsDeleteWalkState (WalkState);
362             goto Cleanup;
363         }
364     }
365 
366     /* Parse the AML */
367 
368     Status = AcpiPsParseAml (WalkState);
369 
370     /* WalkState was deleted by ParseAml */
371 
372 Cleanup:
373     AcpiPsDeleteParseTree (Op);
374 
375     /* End optional tracing */
376 
377     AcpiPsStopTrace (Info);
378 
379     /* Take away the extra reference that we gave the parameters above */
380 
381     AcpiPsUpdateParameterList (Info, REF_DECREMENT);
382 
383     /* Exit now if error above */
384 
385     if (ACPI_FAILURE (Status))
386     {
387         return_ACPI_STATUS (Status);
388     }
389 
390     /*
391      * If the method has returned an object, signal this to the caller with
392      * a control exception code
393      */
394     if (Info->ReturnObject)
395     {
396         ACPI_DEBUG_PRINT ((ACPI_DB_PARSE, "Method returned ObjDesc=%p\n",
397             Info->ReturnObject));
398         ACPI_DUMP_STACK_ENTRY (Info->ReturnObject);
399 
400         Status = AE_CTRL_RETURN_VALUE;
401     }
402 
403     return_ACPI_STATUS (Status);
404 }
405 
406 
407 /*******************************************************************************
408  *
409  * FUNCTION:    AcpiPsUpdateParameterList
410  *
411  * PARAMETERS:  Info            - See ACPI_EVALUATE_INFO
412  *                                (Used: ParameterType and Parameters)
413  *              Action          - Add or Remove reference
414  *
415  * RETURN:      Status
416  *
417  * DESCRIPTION: Update reference count on all method parameter objects
418  *
419  ******************************************************************************/
420 
421 static void
422 AcpiPsUpdateParameterList (
423     ACPI_EVALUATE_INFO      *Info,
424     UINT16                  Action)
425 {
426     UINT32                  i;
427 
428 
429     if (Info->Parameters)
430     {
431         /* Update reference count for each parameter */
432 
433         for (i = 0; Info->Parameters[i]; i++)
434         {
435             /* Ignore errors, just do them all */
436 
437             (void) AcpiUtUpdateObjectReference (Info->Parameters[i], Action);
438         }
439     }
440 }
441