xref: /minix3/minix/drivers/power/acpi/events/evxfgpe.c (revision 29492bb71c7148a089a5afafa0c99409161218df)
1*29492bb7SDavid van Moolenbroek /******************************************************************************
2*29492bb7SDavid van Moolenbroek  *
3*29492bb7SDavid van Moolenbroek  * Module Name: evxfgpe - External Interfaces for General Purpose Events (GPEs)
4*29492bb7SDavid van Moolenbroek  *
5*29492bb7SDavid van Moolenbroek  *****************************************************************************/
6*29492bb7SDavid van Moolenbroek 
7*29492bb7SDavid van Moolenbroek /*
8*29492bb7SDavid van Moolenbroek  * Copyright (C) 2000 - 2014, Intel Corp.
9*29492bb7SDavid van Moolenbroek  * All rights reserved.
10*29492bb7SDavid van Moolenbroek  *
11*29492bb7SDavid van Moolenbroek  * Redistribution and use in source and binary forms, with or without
12*29492bb7SDavid van Moolenbroek  * modification, are permitted provided that the following conditions
13*29492bb7SDavid van Moolenbroek  * are met:
14*29492bb7SDavid van Moolenbroek  * 1. Redistributions of source code must retain the above copyright
15*29492bb7SDavid van Moolenbroek  *    notice, this list of conditions, and the following disclaimer,
16*29492bb7SDavid van Moolenbroek  *    without modification.
17*29492bb7SDavid van Moolenbroek  * 2. Redistributions in binary form must reproduce at minimum a disclaimer
18*29492bb7SDavid van Moolenbroek  *    substantially similar to the "NO WARRANTY" disclaimer below
19*29492bb7SDavid van Moolenbroek  *    ("Disclaimer") and any redistribution must be conditioned upon
20*29492bb7SDavid van Moolenbroek  *    including a substantially similar Disclaimer requirement for further
21*29492bb7SDavid van Moolenbroek  *    binary redistribution.
22*29492bb7SDavid van Moolenbroek  * 3. Neither the names of the above-listed copyright holders nor the names
23*29492bb7SDavid van Moolenbroek  *    of any contributors may be used to endorse or promote products derived
24*29492bb7SDavid van Moolenbroek  *    from this software without specific prior written permission.
25*29492bb7SDavid van Moolenbroek  *
26*29492bb7SDavid van Moolenbroek  * Alternatively, this software may be distributed under the terms of the
27*29492bb7SDavid van Moolenbroek  * GNU General Public License ("GPL") version 2 as published by the Free
28*29492bb7SDavid van Moolenbroek  * Software Foundation.
29*29492bb7SDavid van Moolenbroek  *
30*29492bb7SDavid van Moolenbroek  * NO WARRANTY
31*29492bb7SDavid van Moolenbroek  * THIS SOFTWARE IS PROVIDED BY THE COPYRIGHT HOLDERS AND CONTRIBUTORS
32*29492bb7SDavid van Moolenbroek  * "AS IS" AND ANY EXPRESS OR IMPLIED WARRANTIES, INCLUDING, BUT NOT
33*29492bb7SDavid van Moolenbroek  * LIMITED TO, THE IMPLIED WARRANTIES OF MERCHANTIBILITY AND FITNESS FOR
34*29492bb7SDavid van Moolenbroek  * A PARTICULAR PURPOSE ARE DISCLAIMED. IN NO EVENT SHALL THE COPYRIGHT
35*29492bb7SDavid van Moolenbroek  * HOLDERS OR CONTRIBUTORS BE LIABLE FOR SPECIAL, EXEMPLARY, OR CONSEQUENTIAL
36*29492bb7SDavid van Moolenbroek  * DAMAGES (INCLUDING, BUT NOT LIMITED TO, PROCUREMENT OF SUBSTITUTE GOODS
37*29492bb7SDavid van Moolenbroek  * OR SERVICES; LOSS OF USE, DATA, OR PROFITS; OR BUSINESS INTERRUPTION)
38*29492bb7SDavid van Moolenbroek  * HOWEVER CAUSED AND ON ANY THEORY OF LIABILITY, WHETHER IN CONTRACT,
39*29492bb7SDavid van Moolenbroek  * STRICT LIABILITY, OR TORT (INCLUDING NEGLIGENCE OR OTHERWISE) ARISING
40*29492bb7SDavid van Moolenbroek  * IN ANY WAY OUT OF THE USE OF THIS SOFTWARE, EVEN IF ADVISED OF THE
41*29492bb7SDavid van Moolenbroek  * POSSIBILITY OF SUCH DAMAGES.
42*29492bb7SDavid van Moolenbroek  */
43*29492bb7SDavid van Moolenbroek 
44*29492bb7SDavid van Moolenbroek #define EXPORT_ACPI_INTERFACES
45*29492bb7SDavid van Moolenbroek 
46*29492bb7SDavid van Moolenbroek #include "acpi.h"
47*29492bb7SDavid van Moolenbroek #include "accommon.h"
48*29492bb7SDavid van Moolenbroek #include "acevents.h"
49*29492bb7SDavid van Moolenbroek #include "acnamesp.h"
50*29492bb7SDavid van Moolenbroek 
51*29492bb7SDavid van Moolenbroek #define _COMPONENT          ACPI_EVENTS
52*29492bb7SDavid van Moolenbroek         ACPI_MODULE_NAME    ("evxfgpe")
53*29492bb7SDavid van Moolenbroek 
54*29492bb7SDavid van Moolenbroek 
55*29492bb7SDavid van Moolenbroek #if (!ACPI_REDUCED_HARDWARE) /* Entire module */
56*29492bb7SDavid van Moolenbroek /*******************************************************************************
57*29492bb7SDavid van Moolenbroek  *
58*29492bb7SDavid van Moolenbroek  * FUNCTION:    AcpiUpdateAllGpes
59*29492bb7SDavid van Moolenbroek  *
60*29492bb7SDavid van Moolenbroek  * PARAMETERS:  None
61*29492bb7SDavid van Moolenbroek  *
62*29492bb7SDavid van Moolenbroek  * RETURN:      Status
63*29492bb7SDavid van Moolenbroek  *
64*29492bb7SDavid van Moolenbroek  * DESCRIPTION: Complete GPE initialization and enable all GPEs that have
65*29492bb7SDavid van Moolenbroek  *              associated _Lxx or _Exx methods and are not pointed to by any
66*29492bb7SDavid van Moolenbroek  *              device _PRW methods (this indicates that these GPEs are
67*29492bb7SDavid van Moolenbroek  *              generally intended for system or device wakeup. Such GPEs
68*29492bb7SDavid van Moolenbroek  *              have to be enabled directly when the devices whose _PRW
69*29492bb7SDavid van Moolenbroek  *              methods point to them are set up for wakeup signaling.)
70*29492bb7SDavid van Moolenbroek  *
71*29492bb7SDavid van Moolenbroek  * NOTE: Should be called after any GPEs are added to the system. Primarily,
72*29492bb7SDavid van Moolenbroek  * after the system _PRW methods have been run, but also after a GPE Block
73*29492bb7SDavid van Moolenbroek  * Device has been added or if any new GPE methods have been added via a
74*29492bb7SDavid van Moolenbroek  * dynamic table load.
75*29492bb7SDavid van Moolenbroek  *
76*29492bb7SDavid van Moolenbroek  ******************************************************************************/
77*29492bb7SDavid van Moolenbroek 
78*29492bb7SDavid van Moolenbroek ACPI_STATUS
AcpiUpdateAllGpes(void)79*29492bb7SDavid van Moolenbroek AcpiUpdateAllGpes (
80*29492bb7SDavid van Moolenbroek     void)
81*29492bb7SDavid van Moolenbroek {
82*29492bb7SDavid van Moolenbroek     ACPI_STATUS             Status;
83*29492bb7SDavid van Moolenbroek 
84*29492bb7SDavid van Moolenbroek 
85*29492bb7SDavid van Moolenbroek     ACPI_FUNCTION_TRACE (AcpiUpdateAllGpes);
86*29492bb7SDavid van Moolenbroek 
87*29492bb7SDavid van Moolenbroek 
88*29492bb7SDavid van Moolenbroek     Status = AcpiUtAcquireMutex (ACPI_MTX_EVENTS);
89*29492bb7SDavid van Moolenbroek     if (ACPI_FAILURE (Status))
90*29492bb7SDavid van Moolenbroek     {
91*29492bb7SDavid van Moolenbroek         return_ACPI_STATUS (Status);
92*29492bb7SDavid van Moolenbroek     }
93*29492bb7SDavid van Moolenbroek 
94*29492bb7SDavid van Moolenbroek     if (AcpiGbl_AllGpesInitialized)
95*29492bb7SDavid van Moolenbroek     {
96*29492bb7SDavid van Moolenbroek         goto UnlockAndExit;
97*29492bb7SDavid van Moolenbroek     }
98*29492bb7SDavid van Moolenbroek 
99*29492bb7SDavid van Moolenbroek     Status = AcpiEvWalkGpeList (AcpiEvInitializeGpeBlock, NULL);
100*29492bb7SDavid van Moolenbroek     if (ACPI_SUCCESS (Status))
101*29492bb7SDavid van Moolenbroek     {
102*29492bb7SDavid van Moolenbroek         AcpiGbl_AllGpesInitialized = TRUE;
103*29492bb7SDavid van Moolenbroek     }
104*29492bb7SDavid van Moolenbroek 
105*29492bb7SDavid van Moolenbroek UnlockAndExit:
106*29492bb7SDavid van Moolenbroek     (void) AcpiUtReleaseMutex (ACPI_MTX_EVENTS);
107*29492bb7SDavid van Moolenbroek     return_ACPI_STATUS (Status);
108*29492bb7SDavid van Moolenbroek }
109*29492bb7SDavid van Moolenbroek 
ACPI_EXPORT_SYMBOL(AcpiUpdateAllGpes)110*29492bb7SDavid van Moolenbroek ACPI_EXPORT_SYMBOL (AcpiUpdateAllGpes)
111*29492bb7SDavid van Moolenbroek 
112*29492bb7SDavid van Moolenbroek 
113*29492bb7SDavid van Moolenbroek /*******************************************************************************
114*29492bb7SDavid van Moolenbroek  *
115*29492bb7SDavid van Moolenbroek  * FUNCTION:    AcpiEnableGpe
116*29492bb7SDavid van Moolenbroek  *
117*29492bb7SDavid van Moolenbroek  * PARAMETERS:  GpeDevice           - Parent GPE Device. NULL for GPE0/GPE1
118*29492bb7SDavid van Moolenbroek  *              GpeNumber           - GPE level within the GPE block
119*29492bb7SDavid van Moolenbroek  *
120*29492bb7SDavid van Moolenbroek  * RETURN:      Status
121*29492bb7SDavid van Moolenbroek  *
122*29492bb7SDavid van Moolenbroek  * DESCRIPTION: Add a reference to a GPE. On the first reference, the GPE is
123*29492bb7SDavid van Moolenbroek  *              hardware-enabled.
124*29492bb7SDavid van Moolenbroek  *
125*29492bb7SDavid van Moolenbroek  ******************************************************************************/
126*29492bb7SDavid van Moolenbroek 
127*29492bb7SDavid van Moolenbroek ACPI_STATUS
128*29492bb7SDavid van Moolenbroek AcpiEnableGpe (
129*29492bb7SDavid van Moolenbroek     ACPI_HANDLE             GpeDevice,
130*29492bb7SDavid van Moolenbroek     UINT32                  GpeNumber)
131*29492bb7SDavid van Moolenbroek {
132*29492bb7SDavid van Moolenbroek     ACPI_STATUS             Status = AE_BAD_PARAMETER;
133*29492bb7SDavid van Moolenbroek     ACPI_GPE_EVENT_INFO     *GpeEventInfo;
134*29492bb7SDavid van Moolenbroek     ACPI_CPU_FLAGS          Flags;
135*29492bb7SDavid van Moolenbroek 
136*29492bb7SDavid van Moolenbroek 
137*29492bb7SDavid van Moolenbroek     ACPI_FUNCTION_TRACE (AcpiEnableGpe);
138*29492bb7SDavid van Moolenbroek 
139*29492bb7SDavid van Moolenbroek 
140*29492bb7SDavid van Moolenbroek     Flags = AcpiOsAcquireLock (AcpiGbl_GpeLock);
141*29492bb7SDavid van Moolenbroek 
142*29492bb7SDavid van Moolenbroek     /*
143*29492bb7SDavid van Moolenbroek      * Ensure that we have a valid GPE number and that there is some way
144*29492bb7SDavid van Moolenbroek      * of handling the GPE (handler or a GPE method). In other words, we
145*29492bb7SDavid van Moolenbroek      * won't allow a valid GPE to be enabled if there is no way to handle it.
146*29492bb7SDavid van Moolenbroek      */
147*29492bb7SDavid van Moolenbroek     GpeEventInfo = AcpiEvGetGpeEventInfo (GpeDevice, GpeNumber);
148*29492bb7SDavid van Moolenbroek     if (GpeEventInfo)
149*29492bb7SDavid van Moolenbroek     {
150*29492bb7SDavid van Moolenbroek         if ((GpeEventInfo->Flags & ACPI_GPE_DISPATCH_MASK) !=
151*29492bb7SDavid van Moolenbroek             ACPI_GPE_DISPATCH_NONE)
152*29492bb7SDavid van Moolenbroek         {
153*29492bb7SDavid van Moolenbroek             Status = AcpiEvAddGpeReference (GpeEventInfo);
154*29492bb7SDavid van Moolenbroek         }
155*29492bb7SDavid van Moolenbroek         else
156*29492bb7SDavid van Moolenbroek         {
157*29492bb7SDavid van Moolenbroek             Status = AE_NO_HANDLER;
158*29492bb7SDavid van Moolenbroek         }
159*29492bb7SDavid van Moolenbroek     }
160*29492bb7SDavid van Moolenbroek 
161*29492bb7SDavid van Moolenbroek     AcpiOsReleaseLock (AcpiGbl_GpeLock, Flags);
162*29492bb7SDavid van Moolenbroek     return_ACPI_STATUS (Status);
163*29492bb7SDavid van Moolenbroek }
164*29492bb7SDavid van Moolenbroek 
ACPI_EXPORT_SYMBOL(AcpiEnableGpe)165*29492bb7SDavid van Moolenbroek ACPI_EXPORT_SYMBOL (AcpiEnableGpe)
166*29492bb7SDavid van Moolenbroek 
167*29492bb7SDavid van Moolenbroek 
168*29492bb7SDavid van Moolenbroek /*******************************************************************************
169*29492bb7SDavid van Moolenbroek  *
170*29492bb7SDavid van Moolenbroek  * FUNCTION:    AcpiDisableGpe
171*29492bb7SDavid van Moolenbroek  *
172*29492bb7SDavid van Moolenbroek  * PARAMETERS:  GpeDevice           - Parent GPE Device. NULL for GPE0/GPE1
173*29492bb7SDavid van Moolenbroek  *              GpeNumber           - GPE level within the GPE block
174*29492bb7SDavid van Moolenbroek  *
175*29492bb7SDavid van Moolenbroek  * RETURN:      Status
176*29492bb7SDavid van Moolenbroek  *
177*29492bb7SDavid van Moolenbroek  * DESCRIPTION: Remove a reference to a GPE. When the last reference is
178*29492bb7SDavid van Moolenbroek  *              removed, only then is the GPE disabled (for runtime GPEs), or
179*29492bb7SDavid van Moolenbroek  *              the GPE mask bit disabled (for wake GPEs)
180*29492bb7SDavid van Moolenbroek  *
181*29492bb7SDavid van Moolenbroek  ******************************************************************************/
182*29492bb7SDavid van Moolenbroek 
183*29492bb7SDavid van Moolenbroek ACPI_STATUS
184*29492bb7SDavid van Moolenbroek AcpiDisableGpe (
185*29492bb7SDavid van Moolenbroek     ACPI_HANDLE             GpeDevice,
186*29492bb7SDavid van Moolenbroek     UINT32                  GpeNumber)
187*29492bb7SDavid van Moolenbroek {
188*29492bb7SDavid van Moolenbroek     ACPI_STATUS             Status = AE_BAD_PARAMETER;
189*29492bb7SDavid van Moolenbroek     ACPI_GPE_EVENT_INFO     *GpeEventInfo;
190*29492bb7SDavid van Moolenbroek     ACPI_CPU_FLAGS          Flags;
191*29492bb7SDavid van Moolenbroek 
192*29492bb7SDavid van Moolenbroek 
193*29492bb7SDavid van Moolenbroek     ACPI_FUNCTION_TRACE (AcpiDisableGpe);
194*29492bb7SDavid van Moolenbroek 
195*29492bb7SDavid van Moolenbroek 
196*29492bb7SDavid van Moolenbroek     Flags = AcpiOsAcquireLock (AcpiGbl_GpeLock);
197*29492bb7SDavid van Moolenbroek 
198*29492bb7SDavid van Moolenbroek     /* Ensure that we have a valid GPE number */
199*29492bb7SDavid van Moolenbroek 
200*29492bb7SDavid van Moolenbroek     GpeEventInfo = AcpiEvGetGpeEventInfo (GpeDevice, GpeNumber);
201*29492bb7SDavid van Moolenbroek     if (GpeEventInfo)
202*29492bb7SDavid van Moolenbroek     {
203*29492bb7SDavid van Moolenbroek         Status = AcpiEvRemoveGpeReference (GpeEventInfo);
204*29492bb7SDavid van Moolenbroek     }
205*29492bb7SDavid van Moolenbroek 
206*29492bb7SDavid van Moolenbroek     AcpiOsReleaseLock (AcpiGbl_GpeLock, Flags);
207*29492bb7SDavid van Moolenbroek     return_ACPI_STATUS (Status);
208*29492bb7SDavid van Moolenbroek }
209*29492bb7SDavid van Moolenbroek 
ACPI_EXPORT_SYMBOL(AcpiDisableGpe)210*29492bb7SDavid van Moolenbroek ACPI_EXPORT_SYMBOL (AcpiDisableGpe)
211*29492bb7SDavid van Moolenbroek 
212*29492bb7SDavid van Moolenbroek 
213*29492bb7SDavid van Moolenbroek /*******************************************************************************
214*29492bb7SDavid van Moolenbroek  *
215*29492bb7SDavid van Moolenbroek  * FUNCTION:    AcpiSetGpe
216*29492bb7SDavid van Moolenbroek  *
217*29492bb7SDavid van Moolenbroek  * PARAMETERS:  GpeDevice           - Parent GPE Device. NULL for GPE0/GPE1
218*29492bb7SDavid van Moolenbroek  *              GpeNumber           - GPE level within the GPE block
219*29492bb7SDavid van Moolenbroek  *              Action              - ACPI_GPE_ENABLE or ACPI_GPE_DISABLE
220*29492bb7SDavid van Moolenbroek  *
221*29492bb7SDavid van Moolenbroek  * RETURN:      Status
222*29492bb7SDavid van Moolenbroek  *
223*29492bb7SDavid van Moolenbroek  * DESCRIPTION: Enable or disable an individual GPE. This function bypasses
224*29492bb7SDavid van Moolenbroek  *              the reference count mechanism used in the AcpiEnableGpe and
225*29492bb7SDavid van Moolenbroek  *              AcpiDisableGpe interfaces -- and should be used with care.
226*29492bb7SDavid van Moolenbroek  *
227*29492bb7SDavid van Moolenbroek  * Note: Typically used to disable a runtime GPE for short period of time,
228*29492bb7SDavid van Moolenbroek  * then re-enable it, without disturbing the existing reference counts. This
229*29492bb7SDavid van Moolenbroek  * is useful, for example, in the Embedded Controller (EC) driver.
230*29492bb7SDavid van Moolenbroek  *
231*29492bb7SDavid van Moolenbroek  ******************************************************************************/
232*29492bb7SDavid van Moolenbroek 
233*29492bb7SDavid van Moolenbroek ACPI_STATUS
234*29492bb7SDavid van Moolenbroek AcpiSetGpe (
235*29492bb7SDavid van Moolenbroek     ACPI_HANDLE             GpeDevice,
236*29492bb7SDavid van Moolenbroek     UINT32                  GpeNumber,
237*29492bb7SDavid van Moolenbroek     UINT8                   Action)
238*29492bb7SDavid van Moolenbroek {
239*29492bb7SDavid van Moolenbroek     ACPI_GPE_EVENT_INFO     *GpeEventInfo;
240*29492bb7SDavid van Moolenbroek     ACPI_STATUS             Status;
241*29492bb7SDavid van Moolenbroek     ACPI_CPU_FLAGS          Flags;
242*29492bb7SDavid van Moolenbroek 
243*29492bb7SDavid van Moolenbroek 
244*29492bb7SDavid van Moolenbroek     ACPI_FUNCTION_TRACE (AcpiSetGpe);
245*29492bb7SDavid van Moolenbroek 
246*29492bb7SDavid van Moolenbroek 
247*29492bb7SDavid van Moolenbroek     Flags = AcpiOsAcquireLock (AcpiGbl_GpeLock);
248*29492bb7SDavid van Moolenbroek 
249*29492bb7SDavid van Moolenbroek     /* Ensure that we have a valid GPE number */
250*29492bb7SDavid van Moolenbroek 
251*29492bb7SDavid van Moolenbroek     GpeEventInfo = AcpiEvGetGpeEventInfo (GpeDevice, GpeNumber);
252*29492bb7SDavid van Moolenbroek     if (!GpeEventInfo)
253*29492bb7SDavid van Moolenbroek     {
254*29492bb7SDavid van Moolenbroek         Status = AE_BAD_PARAMETER;
255*29492bb7SDavid van Moolenbroek         goto UnlockAndExit;
256*29492bb7SDavid van Moolenbroek     }
257*29492bb7SDavid van Moolenbroek 
258*29492bb7SDavid van Moolenbroek     /* Perform the action */
259*29492bb7SDavid van Moolenbroek 
260*29492bb7SDavid van Moolenbroek     switch (Action)
261*29492bb7SDavid van Moolenbroek     {
262*29492bb7SDavid van Moolenbroek     case ACPI_GPE_ENABLE:
263*29492bb7SDavid van Moolenbroek 
264*29492bb7SDavid van Moolenbroek         Status = AcpiEvEnableGpe (GpeEventInfo);
265*29492bb7SDavid van Moolenbroek         break;
266*29492bb7SDavid van Moolenbroek 
267*29492bb7SDavid van Moolenbroek     case ACPI_GPE_DISABLE:
268*29492bb7SDavid van Moolenbroek 
269*29492bb7SDavid van Moolenbroek         Status = AcpiHwLowSetGpe (GpeEventInfo, ACPI_GPE_DISABLE);
270*29492bb7SDavid van Moolenbroek         break;
271*29492bb7SDavid van Moolenbroek 
272*29492bb7SDavid van Moolenbroek     default:
273*29492bb7SDavid van Moolenbroek 
274*29492bb7SDavid van Moolenbroek         Status = AE_BAD_PARAMETER;
275*29492bb7SDavid van Moolenbroek         break;
276*29492bb7SDavid van Moolenbroek     }
277*29492bb7SDavid van Moolenbroek 
278*29492bb7SDavid van Moolenbroek UnlockAndExit:
279*29492bb7SDavid van Moolenbroek     AcpiOsReleaseLock (AcpiGbl_GpeLock, Flags);
280*29492bb7SDavid van Moolenbroek     return_ACPI_STATUS (Status);
281*29492bb7SDavid van Moolenbroek }
282*29492bb7SDavid van Moolenbroek 
ACPI_EXPORT_SYMBOL(AcpiSetGpe)283*29492bb7SDavid van Moolenbroek ACPI_EXPORT_SYMBOL (AcpiSetGpe)
284*29492bb7SDavid van Moolenbroek 
285*29492bb7SDavid van Moolenbroek 
286*29492bb7SDavid van Moolenbroek /*******************************************************************************
287*29492bb7SDavid van Moolenbroek  *
288*29492bb7SDavid van Moolenbroek  * FUNCTION:    AcpiMarkGpeForWake
289*29492bb7SDavid van Moolenbroek  *
290*29492bb7SDavid van Moolenbroek  * PARAMETERS:  GpeDevice           - Parent GPE Device. NULL for GPE0/GPE1
291*29492bb7SDavid van Moolenbroek  *              GpeNumber           - GPE level within the GPE block
292*29492bb7SDavid van Moolenbroek  *
293*29492bb7SDavid van Moolenbroek  * RETURN:      Status
294*29492bb7SDavid van Moolenbroek  *
295*29492bb7SDavid van Moolenbroek  * DESCRIPTION: Mark a GPE as having the ability to wake the system. Simply
296*29492bb7SDavid van Moolenbroek  *              sets the ACPI_GPE_CAN_WAKE flag.
297*29492bb7SDavid van Moolenbroek  *
298*29492bb7SDavid van Moolenbroek  * Some potential callers of AcpiSetupGpeForWake may know in advance that
299*29492bb7SDavid van Moolenbroek  * there won't be any notify handlers installed for device wake notifications
300*29492bb7SDavid van Moolenbroek  * from the given GPE (one example is a button GPE in Linux). For these cases,
301*29492bb7SDavid van Moolenbroek  * AcpiMarkGpeForWake should be used instead of AcpiSetupGpeForWake.
302*29492bb7SDavid van Moolenbroek  * This will set the ACPI_GPE_CAN_WAKE flag for the GPE without trying to
303*29492bb7SDavid van Moolenbroek  * setup implicit wake notification for it (since there's no handler method).
304*29492bb7SDavid van Moolenbroek  *
305*29492bb7SDavid van Moolenbroek  ******************************************************************************/
306*29492bb7SDavid van Moolenbroek 
307*29492bb7SDavid van Moolenbroek ACPI_STATUS
308*29492bb7SDavid van Moolenbroek AcpiMarkGpeForWake (
309*29492bb7SDavid van Moolenbroek     ACPI_HANDLE             GpeDevice,
310*29492bb7SDavid van Moolenbroek     UINT32                  GpeNumber)
311*29492bb7SDavid van Moolenbroek {
312*29492bb7SDavid van Moolenbroek     ACPI_GPE_EVENT_INFO     *GpeEventInfo;
313*29492bb7SDavid van Moolenbroek     ACPI_STATUS             Status = AE_BAD_PARAMETER;
314*29492bb7SDavid van Moolenbroek     ACPI_CPU_FLAGS          Flags;
315*29492bb7SDavid van Moolenbroek 
316*29492bb7SDavid van Moolenbroek 
317*29492bb7SDavid van Moolenbroek     ACPI_FUNCTION_TRACE (AcpiMarkGpeForWake);
318*29492bb7SDavid van Moolenbroek 
319*29492bb7SDavid van Moolenbroek 
320*29492bb7SDavid van Moolenbroek     Flags = AcpiOsAcquireLock (AcpiGbl_GpeLock);
321*29492bb7SDavid van Moolenbroek 
322*29492bb7SDavid van Moolenbroek     /* Ensure that we have a valid GPE number */
323*29492bb7SDavid van Moolenbroek 
324*29492bb7SDavid van Moolenbroek     GpeEventInfo = AcpiEvGetGpeEventInfo (GpeDevice, GpeNumber);
325*29492bb7SDavid van Moolenbroek     if (GpeEventInfo)
326*29492bb7SDavid van Moolenbroek     {
327*29492bb7SDavid van Moolenbroek         /* Mark the GPE as a possible wake event */
328*29492bb7SDavid van Moolenbroek 
329*29492bb7SDavid van Moolenbroek         GpeEventInfo->Flags |= ACPI_GPE_CAN_WAKE;
330*29492bb7SDavid van Moolenbroek         Status = AE_OK;
331*29492bb7SDavid van Moolenbroek     }
332*29492bb7SDavid van Moolenbroek 
333*29492bb7SDavid van Moolenbroek     AcpiOsReleaseLock (AcpiGbl_GpeLock, Flags);
334*29492bb7SDavid van Moolenbroek     return_ACPI_STATUS (Status);
335*29492bb7SDavid van Moolenbroek }
336*29492bb7SDavid van Moolenbroek 
ACPI_EXPORT_SYMBOL(AcpiMarkGpeForWake)337*29492bb7SDavid van Moolenbroek ACPI_EXPORT_SYMBOL (AcpiMarkGpeForWake)
338*29492bb7SDavid van Moolenbroek 
339*29492bb7SDavid van Moolenbroek 
340*29492bb7SDavid van Moolenbroek /*******************************************************************************
341*29492bb7SDavid van Moolenbroek  *
342*29492bb7SDavid van Moolenbroek  * FUNCTION:    AcpiSetupGpeForWake
343*29492bb7SDavid van Moolenbroek  *
344*29492bb7SDavid van Moolenbroek  * PARAMETERS:  WakeDevice          - Device associated with the GPE (via _PRW)
345*29492bb7SDavid van Moolenbroek  *              GpeDevice           - Parent GPE Device. NULL for GPE0/GPE1
346*29492bb7SDavid van Moolenbroek  *              GpeNumber           - GPE level within the GPE block
347*29492bb7SDavid van Moolenbroek  *
348*29492bb7SDavid van Moolenbroek  * RETURN:      Status
349*29492bb7SDavid van Moolenbroek  *
350*29492bb7SDavid van Moolenbroek  * DESCRIPTION: Mark a GPE as having the ability to wake the system. This
351*29492bb7SDavid van Moolenbroek  *              interface is intended to be used as the host executes the
352*29492bb7SDavid van Moolenbroek  *              _PRW methods (Power Resources for Wake) in the system tables.
353*29492bb7SDavid van Moolenbroek  *              Each _PRW appears under a Device Object (The WakeDevice), and
354*29492bb7SDavid van Moolenbroek  *              contains the info for the wake GPE associated with the
355*29492bb7SDavid van Moolenbroek  *              WakeDevice.
356*29492bb7SDavid van Moolenbroek  *
357*29492bb7SDavid van Moolenbroek  ******************************************************************************/
358*29492bb7SDavid van Moolenbroek 
359*29492bb7SDavid van Moolenbroek ACPI_STATUS
360*29492bb7SDavid van Moolenbroek AcpiSetupGpeForWake (
361*29492bb7SDavid van Moolenbroek     ACPI_HANDLE             WakeDevice,
362*29492bb7SDavid van Moolenbroek     ACPI_HANDLE             GpeDevice,
363*29492bb7SDavid van Moolenbroek     UINT32                  GpeNumber)
364*29492bb7SDavid van Moolenbroek {
365*29492bb7SDavid van Moolenbroek     ACPI_STATUS             Status;
366*29492bb7SDavid van Moolenbroek     ACPI_GPE_EVENT_INFO     *GpeEventInfo;
367*29492bb7SDavid van Moolenbroek     ACPI_NAMESPACE_NODE     *DeviceNode;
368*29492bb7SDavid van Moolenbroek     ACPI_GPE_NOTIFY_INFO    *Notify;
369*29492bb7SDavid van Moolenbroek     ACPI_GPE_NOTIFY_INFO    *NewNotify;
370*29492bb7SDavid van Moolenbroek     ACPI_CPU_FLAGS          Flags;
371*29492bb7SDavid van Moolenbroek 
372*29492bb7SDavid van Moolenbroek 
373*29492bb7SDavid van Moolenbroek     ACPI_FUNCTION_TRACE (AcpiSetupGpeForWake);
374*29492bb7SDavid van Moolenbroek 
375*29492bb7SDavid van Moolenbroek 
376*29492bb7SDavid van Moolenbroek     /* Parameter Validation */
377*29492bb7SDavid van Moolenbroek 
378*29492bb7SDavid van Moolenbroek     if (!WakeDevice)
379*29492bb7SDavid van Moolenbroek     {
380*29492bb7SDavid van Moolenbroek         /*
381*29492bb7SDavid van Moolenbroek          * By forcing WakeDevice to be valid, we automatically enable the
382*29492bb7SDavid van Moolenbroek          * implicit notify feature on all hosts.
383*29492bb7SDavid van Moolenbroek          */
384*29492bb7SDavid van Moolenbroek         return_ACPI_STATUS (AE_BAD_PARAMETER);
385*29492bb7SDavid van Moolenbroek     }
386*29492bb7SDavid van Moolenbroek 
387*29492bb7SDavid van Moolenbroek     /* Handle root object case */
388*29492bb7SDavid van Moolenbroek 
389*29492bb7SDavid van Moolenbroek     if (WakeDevice == ACPI_ROOT_OBJECT)
390*29492bb7SDavid van Moolenbroek     {
391*29492bb7SDavid van Moolenbroek         DeviceNode = AcpiGbl_RootNode;
392*29492bb7SDavid van Moolenbroek     }
393*29492bb7SDavid van Moolenbroek     else
394*29492bb7SDavid van Moolenbroek     {
395*29492bb7SDavid van Moolenbroek         DeviceNode = ACPI_CAST_PTR (ACPI_NAMESPACE_NODE, WakeDevice);
396*29492bb7SDavid van Moolenbroek     }
397*29492bb7SDavid van Moolenbroek 
398*29492bb7SDavid van Moolenbroek     /* Validate WakeDevice is of type Device */
399*29492bb7SDavid van Moolenbroek 
400*29492bb7SDavid van Moolenbroek     if (DeviceNode->Type != ACPI_TYPE_DEVICE)
401*29492bb7SDavid van Moolenbroek     {
402*29492bb7SDavid van Moolenbroek         return_ACPI_STATUS (AE_BAD_PARAMETER);
403*29492bb7SDavid van Moolenbroek     }
404*29492bb7SDavid van Moolenbroek 
405*29492bb7SDavid van Moolenbroek     /*
406*29492bb7SDavid van Moolenbroek      * Allocate a new notify object up front, in case it is needed.
407*29492bb7SDavid van Moolenbroek      * Memory allocation while holding a spinlock is a big no-no
408*29492bb7SDavid van Moolenbroek      * on some hosts.
409*29492bb7SDavid van Moolenbroek      */
410*29492bb7SDavid van Moolenbroek     NewNotify = ACPI_ALLOCATE_ZEROED (sizeof (ACPI_GPE_NOTIFY_INFO));
411*29492bb7SDavid van Moolenbroek     if (!NewNotify)
412*29492bb7SDavid van Moolenbroek     {
413*29492bb7SDavid van Moolenbroek         return_ACPI_STATUS (AE_NO_MEMORY);
414*29492bb7SDavid van Moolenbroek     }
415*29492bb7SDavid van Moolenbroek 
416*29492bb7SDavid van Moolenbroek     Flags = AcpiOsAcquireLock (AcpiGbl_GpeLock);
417*29492bb7SDavid van Moolenbroek 
418*29492bb7SDavid van Moolenbroek     /* Ensure that we have a valid GPE number */
419*29492bb7SDavid van Moolenbroek 
420*29492bb7SDavid van Moolenbroek     GpeEventInfo = AcpiEvGetGpeEventInfo (GpeDevice, GpeNumber);
421*29492bb7SDavid van Moolenbroek     if (!GpeEventInfo)
422*29492bb7SDavid van Moolenbroek     {
423*29492bb7SDavid van Moolenbroek         Status = AE_BAD_PARAMETER;
424*29492bb7SDavid van Moolenbroek         goto UnlockAndExit;
425*29492bb7SDavid van Moolenbroek     }
426*29492bb7SDavid van Moolenbroek 
427*29492bb7SDavid van Moolenbroek     /*
428*29492bb7SDavid van Moolenbroek      * If there is no method or handler for this GPE, then the
429*29492bb7SDavid van Moolenbroek      * WakeDevice will be notified whenever this GPE fires. This is
430*29492bb7SDavid van Moolenbroek      * known as an "implicit notify". Note: The GPE is assumed to be
431*29492bb7SDavid van Moolenbroek      * level-triggered (for windows compatibility).
432*29492bb7SDavid van Moolenbroek      */
433*29492bb7SDavid van Moolenbroek     if ((GpeEventInfo->Flags & ACPI_GPE_DISPATCH_MASK) ==
434*29492bb7SDavid van Moolenbroek             ACPI_GPE_DISPATCH_NONE)
435*29492bb7SDavid van Moolenbroek     {
436*29492bb7SDavid van Moolenbroek         /*
437*29492bb7SDavid van Moolenbroek          * This is the first device for implicit notify on this GPE.
438*29492bb7SDavid van Moolenbroek          * Just set the flags here, and enter the NOTIFY block below.
439*29492bb7SDavid van Moolenbroek          */
440*29492bb7SDavid van Moolenbroek         GpeEventInfo->Flags =
441*29492bb7SDavid van Moolenbroek             (ACPI_GPE_DISPATCH_NOTIFY | ACPI_GPE_LEVEL_TRIGGERED);
442*29492bb7SDavid van Moolenbroek     }
443*29492bb7SDavid van Moolenbroek 
444*29492bb7SDavid van Moolenbroek     /*
445*29492bb7SDavid van Moolenbroek      * If we already have an implicit notify on this GPE, add
446*29492bb7SDavid van Moolenbroek      * this device to the notify list.
447*29492bb7SDavid van Moolenbroek      */
448*29492bb7SDavid van Moolenbroek     if ((GpeEventInfo->Flags & ACPI_GPE_DISPATCH_MASK) ==
449*29492bb7SDavid van Moolenbroek             ACPI_GPE_DISPATCH_NOTIFY)
450*29492bb7SDavid van Moolenbroek     {
451*29492bb7SDavid van Moolenbroek         /* Ensure that the device is not already in the list */
452*29492bb7SDavid van Moolenbroek 
453*29492bb7SDavid van Moolenbroek         Notify = GpeEventInfo->Dispatch.NotifyList;
454*29492bb7SDavid van Moolenbroek         while (Notify)
455*29492bb7SDavid van Moolenbroek         {
456*29492bb7SDavid van Moolenbroek             if (Notify->DeviceNode == DeviceNode)
457*29492bb7SDavid van Moolenbroek             {
458*29492bb7SDavid van Moolenbroek                 Status = AE_ALREADY_EXISTS;
459*29492bb7SDavid van Moolenbroek                 goto UnlockAndExit;
460*29492bb7SDavid van Moolenbroek             }
461*29492bb7SDavid van Moolenbroek             Notify = Notify->Next;
462*29492bb7SDavid van Moolenbroek         }
463*29492bb7SDavid van Moolenbroek 
464*29492bb7SDavid van Moolenbroek         /* Add this device to the notify list for this GPE */
465*29492bb7SDavid van Moolenbroek 
466*29492bb7SDavid van Moolenbroek         NewNotify->DeviceNode = DeviceNode;
467*29492bb7SDavid van Moolenbroek         NewNotify->Next = GpeEventInfo->Dispatch.NotifyList;
468*29492bb7SDavid van Moolenbroek         GpeEventInfo->Dispatch.NotifyList = NewNotify;
469*29492bb7SDavid van Moolenbroek         NewNotify = NULL;
470*29492bb7SDavid van Moolenbroek     }
471*29492bb7SDavid van Moolenbroek 
472*29492bb7SDavid van Moolenbroek     /* Mark the GPE as a possible wake event */
473*29492bb7SDavid van Moolenbroek 
474*29492bb7SDavid van Moolenbroek     GpeEventInfo->Flags |= ACPI_GPE_CAN_WAKE;
475*29492bb7SDavid van Moolenbroek     Status = AE_OK;
476*29492bb7SDavid van Moolenbroek 
477*29492bb7SDavid van Moolenbroek 
478*29492bb7SDavid van Moolenbroek UnlockAndExit:
479*29492bb7SDavid van Moolenbroek     AcpiOsReleaseLock (AcpiGbl_GpeLock, Flags);
480*29492bb7SDavid van Moolenbroek 
481*29492bb7SDavid van Moolenbroek     /* Delete the notify object if it was not used above */
482*29492bb7SDavid van Moolenbroek 
483*29492bb7SDavid van Moolenbroek     if (NewNotify)
484*29492bb7SDavid van Moolenbroek     {
485*29492bb7SDavid van Moolenbroek         ACPI_FREE (NewNotify);
486*29492bb7SDavid van Moolenbroek     }
487*29492bb7SDavid van Moolenbroek     return_ACPI_STATUS (Status);
488*29492bb7SDavid van Moolenbroek }
489*29492bb7SDavid van Moolenbroek 
ACPI_EXPORT_SYMBOL(AcpiSetupGpeForWake)490*29492bb7SDavid van Moolenbroek ACPI_EXPORT_SYMBOL (AcpiSetupGpeForWake)
491*29492bb7SDavid van Moolenbroek 
492*29492bb7SDavid van Moolenbroek 
493*29492bb7SDavid van Moolenbroek /*******************************************************************************
494*29492bb7SDavid van Moolenbroek  *
495*29492bb7SDavid van Moolenbroek  * FUNCTION:    AcpiSetGpeWakeMask
496*29492bb7SDavid van Moolenbroek  *
497*29492bb7SDavid van Moolenbroek  * PARAMETERS:  GpeDevice           - Parent GPE Device. NULL for GPE0/GPE1
498*29492bb7SDavid van Moolenbroek  *              GpeNumber           - GPE level within the GPE block
499*29492bb7SDavid van Moolenbroek  *              Action              - Enable or Disable
500*29492bb7SDavid van Moolenbroek  *
501*29492bb7SDavid van Moolenbroek  * RETURN:      Status
502*29492bb7SDavid van Moolenbroek  *
503*29492bb7SDavid van Moolenbroek  * DESCRIPTION: Set or clear the GPE's wakeup enable mask bit. The GPE must
504*29492bb7SDavid van Moolenbroek  *              already be marked as a WAKE GPE.
505*29492bb7SDavid van Moolenbroek  *
506*29492bb7SDavid van Moolenbroek  ******************************************************************************/
507*29492bb7SDavid van Moolenbroek 
508*29492bb7SDavid van Moolenbroek ACPI_STATUS
509*29492bb7SDavid van Moolenbroek AcpiSetGpeWakeMask (
510*29492bb7SDavid van Moolenbroek     ACPI_HANDLE             GpeDevice,
511*29492bb7SDavid van Moolenbroek     UINT32                  GpeNumber,
512*29492bb7SDavid van Moolenbroek     UINT8                   Action)
513*29492bb7SDavid van Moolenbroek {
514*29492bb7SDavid van Moolenbroek     ACPI_STATUS             Status = AE_OK;
515*29492bb7SDavid van Moolenbroek     ACPI_GPE_EVENT_INFO     *GpeEventInfo;
516*29492bb7SDavid van Moolenbroek     ACPI_GPE_REGISTER_INFO  *GpeRegisterInfo;
517*29492bb7SDavid van Moolenbroek     ACPI_CPU_FLAGS          Flags;
518*29492bb7SDavid van Moolenbroek     UINT32                  RegisterBit;
519*29492bb7SDavid van Moolenbroek 
520*29492bb7SDavid van Moolenbroek 
521*29492bb7SDavid van Moolenbroek     ACPI_FUNCTION_TRACE (AcpiSetGpeWakeMask);
522*29492bb7SDavid van Moolenbroek 
523*29492bb7SDavid van Moolenbroek 
524*29492bb7SDavid van Moolenbroek     Flags = AcpiOsAcquireLock (AcpiGbl_GpeLock);
525*29492bb7SDavid van Moolenbroek 
526*29492bb7SDavid van Moolenbroek     /*
527*29492bb7SDavid van Moolenbroek      * Ensure that we have a valid GPE number and that this GPE is in
528*29492bb7SDavid van Moolenbroek      * fact a wake GPE
529*29492bb7SDavid van Moolenbroek      */
530*29492bb7SDavid van Moolenbroek     GpeEventInfo = AcpiEvGetGpeEventInfo (GpeDevice, GpeNumber);
531*29492bb7SDavid van Moolenbroek     if (!GpeEventInfo)
532*29492bb7SDavid van Moolenbroek     {
533*29492bb7SDavid van Moolenbroek         Status = AE_BAD_PARAMETER;
534*29492bb7SDavid van Moolenbroek         goto UnlockAndExit;
535*29492bb7SDavid van Moolenbroek     }
536*29492bb7SDavid van Moolenbroek 
537*29492bb7SDavid van Moolenbroek     if (!(GpeEventInfo->Flags & ACPI_GPE_CAN_WAKE))
538*29492bb7SDavid van Moolenbroek     {
539*29492bb7SDavid van Moolenbroek         Status = AE_TYPE;
540*29492bb7SDavid van Moolenbroek         goto UnlockAndExit;
541*29492bb7SDavid van Moolenbroek     }
542*29492bb7SDavid van Moolenbroek 
543*29492bb7SDavid van Moolenbroek     GpeRegisterInfo = GpeEventInfo->RegisterInfo;
544*29492bb7SDavid van Moolenbroek     if (!GpeRegisterInfo)
545*29492bb7SDavid van Moolenbroek     {
546*29492bb7SDavid van Moolenbroek         Status = AE_NOT_EXIST;
547*29492bb7SDavid van Moolenbroek         goto UnlockAndExit;
548*29492bb7SDavid van Moolenbroek     }
549*29492bb7SDavid van Moolenbroek 
550*29492bb7SDavid van Moolenbroek     RegisterBit = AcpiHwGetGpeRegisterBit (GpeEventInfo);
551*29492bb7SDavid van Moolenbroek 
552*29492bb7SDavid van Moolenbroek     /* Perform the action */
553*29492bb7SDavid van Moolenbroek 
554*29492bb7SDavid van Moolenbroek     switch (Action)
555*29492bb7SDavid van Moolenbroek     {
556*29492bb7SDavid van Moolenbroek     case ACPI_GPE_ENABLE:
557*29492bb7SDavid van Moolenbroek 
558*29492bb7SDavid van Moolenbroek         ACPI_SET_BIT (GpeRegisterInfo->EnableForWake, (UINT8) RegisterBit);
559*29492bb7SDavid van Moolenbroek         break;
560*29492bb7SDavid van Moolenbroek 
561*29492bb7SDavid van Moolenbroek     case ACPI_GPE_DISABLE:
562*29492bb7SDavid van Moolenbroek 
563*29492bb7SDavid van Moolenbroek         ACPI_CLEAR_BIT (GpeRegisterInfo->EnableForWake, (UINT8) RegisterBit);
564*29492bb7SDavid van Moolenbroek         break;
565*29492bb7SDavid van Moolenbroek 
566*29492bb7SDavid van Moolenbroek     default:
567*29492bb7SDavid van Moolenbroek 
568*29492bb7SDavid van Moolenbroek         ACPI_ERROR ((AE_INFO, "%u, Invalid action", Action));
569*29492bb7SDavid van Moolenbroek         Status = AE_BAD_PARAMETER;
570*29492bb7SDavid van Moolenbroek         break;
571*29492bb7SDavid van Moolenbroek     }
572*29492bb7SDavid van Moolenbroek 
573*29492bb7SDavid van Moolenbroek UnlockAndExit:
574*29492bb7SDavid van Moolenbroek     AcpiOsReleaseLock (AcpiGbl_GpeLock, Flags);
575*29492bb7SDavid van Moolenbroek     return_ACPI_STATUS (Status);
576*29492bb7SDavid van Moolenbroek }
577*29492bb7SDavid van Moolenbroek 
ACPI_EXPORT_SYMBOL(AcpiSetGpeWakeMask)578*29492bb7SDavid van Moolenbroek ACPI_EXPORT_SYMBOL (AcpiSetGpeWakeMask)
579*29492bb7SDavid van Moolenbroek 
580*29492bb7SDavid van Moolenbroek 
581*29492bb7SDavid van Moolenbroek /*******************************************************************************
582*29492bb7SDavid van Moolenbroek  *
583*29492bb7SDavid van Moolenbroek  * FUNCTION:    AcpiClearGpe
584*29492bb7SDavid van Moolenbroek  *
585*29492bb7SDavid van Moolenbroek  * PARAMETERS:  GpeDevice           - Parent GPE Device. NULL for GPE0/GPE1
586*29492bb7SDavid van Moolenbroek  *              GpeNumber           - GPE level within the GPE block
587*29492bb7SDavid van Moolenbroek  *
588*29492bb7SDavid van Moolenbroek  * RETURN:      Status
589*29492bb7SDavid van Moolenbroek  *
590*29492bb7SDavid van Moolenbroek  * DESCRIPTION: Clear an ACPI event (general purpose)
591*29492bb7SDavid van Moolenbroek  *
592*29492bb7SDavid van Moolenbroek  ******************************************************************************/
593*29492bb7SDavid van Moolenbroek 
594*29492bb7SDavid van Moolenbroek ACPI_STATUS
595*29492bb7SDavid van Moolenbroek AcpiClearGpe (
596*29492bb7SDavid van Moolenbroek     ACPI_HANDLE             GpeDevice,
597*29492bb7SDavid van Moolenbroek     UINT32                  GpeNumber)
598*29492bb7SDavid van Moolenbroek {
599*29492bb7SDavid van Moolenbroek     ACPI_STATUS             Status = AE_OK;
600*29492bb7SDavid van Moolenbroek     ACPI_GPE_EVENT_INFO     *GpeEventInfo;
601*29492bb7SDavid van Moolenbroek     ACPI_CPU_FLAGS          Flags;
602*29492bb7SDavid van Moolenbroek 
603*29492bb7SDavid van Moolenbroek 
604*29492bb7SDavid van Moolenbroek     ACPI_FUNCTION_TRACE (AcpiClearGpe);
605*29492bb7SDavid van Moolenbroek 
606*29492bb7SDavid van Moolenbroek 
607*29492bb7SDavid van Moolenbroek     Flags = AcpiOsAcquireLock (AcpiGbl_GpeLock);
608*29492bb7SDavid van Moolenbroek 
609*29492bb7SDavid van Moolenbroek     /* Ensure that we have a valid GPE number */
610*29492bb7SDavid van Moolenbroek 
611*29492bb7SDavid van Moolenbroek     GpeEventInfo = AcpiEvGetGpeEventInfo (GpeDevice, GpeNumber);
612*29492bb7SDavid van Moolenbroek     if (!GpeEventInfo)
613*29492bb7SDavid van Moolenbroek     {
614*29492bb7SDavid van Moolenbroek         Status = AE_BAD_PARAMETER;
615*29492bb7SDavid van Moolenbroek         goto UnlockAndExit;
616*29492bb7SDavid van Moolenbroek     }
617*29492bb7SDavid van Moolenbroek 
618*29492bb7SDavid van Moolenbroek     Status = AcpiHwClearGpe (GpeEventInfo);
619*29492bb7SDavid van Moolenbroek 
620*29492bb7SDavid van Moolenbroek UnlockAndExit:
621*29492bb7SDavid van Moolenbroek     AcpiOsReleaseLock (AcpiGbl_GpeLock, Flags);
622*29492bb7SDavid van Moolenbroek     return_ACPI_STATUS (Status);
623*29492bb7SDavid van Moolenbroek }
624*29492bb7SDavid van Moolenbroek 
ACPI_EXPORT_SYMBOL(AcpiClearGpe)625*29492bb7SDavid van Moolenbroek ACPI_EXPORT_SYMBOL (AcpiClearGpe)
626*29492bb7SDavid van Moolenbroek 
627*29492bb7SDavid van Moolenbroek 
628*29492bb7SDavid van Moolenbroek /*******************************************************************************
629*29492bb7SDavid van Moolenbroek  *
630*29492bb7SDavid van Moolenbroek  * FUNCTION:    AcpiGetGpeStatus
631*29492bb7SDavid van Moolenbroek  *
632*29492bb7SDavid van Moolenbroek  * PARAMETERS:  GpeDevice           - Parent GPE Device. NULL for GPE0/GPE1
633*29492bb7SDavid van Moolenbroek  *              GpeNumber           - GPE level within the GPE block
634*29492bb7SDavid van Moolenbroek  *              EventStatus         - Where the current status of the event
635*29492bb7SDavid van Moolenbroek  *                                    will be returned
636*29492bb7SDavid van Moolenbroek  *
637*29492bb7SDavid van Moolenbroek  * RETURN:      Status
638*29492bb7SDavid van Moolenbroek  *
639*29492bb7SDavid van Moolenbroek  * DESCRIPTION: Get the current status of a GPE (signalled/not_signalled)
640*29492bb7SDavid van Moolenbroek  *
641*29492bb7SDavid van Moolenbroek  ******************************************************************************/
642*29492bb7SDavid van Moolenbroek 
643*29492bb7SDavid van Moolenbroek ACPI_STATUS
644*29492bb7SDavid van Moolenbroek AcpiGetGpeStatus (
645*29492bb7SDavid van Moolenbroek     ACPI_HANDLE             GpeDevice,
646*29492bb7SDavid van Moolenbroek     UINT32                  GpeNumber,
647*29492bb7SDavid van Moolenbroek     ACPI_EVENT_STATUS       *EventStatus)
648*29492bb7SDavid van Moolenbroek {
649*29492bb7SDavid van Moolenbroek     ACPI_STATUS             Status = AE_OK;
650*29492bb7SDavid van Moolenbroek     ACPI_GPE_EVENT_INFO     *GpeEventInfo;
651*29492bb7SDavid van Moolenbroek     ACPI_CPU_FLAGS          Flags;
652*29492bb7SDavid van Moolenbroek 
653*29492bb7SDavid van Moolenbroek 
654*29492bb7SDavid van Moolenbroek     ACPI_FUNCTION_TRACE (AcpiGetGpeStatus);
655*29492bb7SDavid van Moolenbroek 
656*29492bb7SDavid van Moolenbroek 
657*29492bb7SDavid van Moolenbroek     Flags = AcpiOsAcquireLock (AcpiGbl_GpeLock);
658*29492bb7SDavid van Moolenbroek 
659*29492bb7SDavid van Moolenbroek     /* Ensure that we have a valid GPE number */
660*29492bb7SDavid van Moolenbroek 
661*29492bb7SDavid van Moolenbroek     GpeEventInfo = AcpiEvGetGpeEventInfo (GpeDevice, GpeNumber);
662*29492bb7SDavid van Moolenbroek     if (!GpeEventInfo)
663*29492bb7SDavid van Moolenbroek     {
664*29492bb7SDavid van Moolenbroek         Status = AE_BAD_PARAMETER;
665*29492bb7SDavid van Moolenbroek         goto UnlockAndExit;
666*29492bb7SDavid van Moolenbroek     }
667*29492bb7SDavid van Moolenbroek 
668*29492bb7SDavid van Moolenbroek     /* Obtain status on the requested GPE number */
669*29492bb7SDavid van Moolenbroek 
670*29492bb7SDavid van Moolenbroek     Status = AcpiHwGetGpeStatus (GpeEventInfo, EventStatus);
671*29492bb7SDavid van Moolenbroek 
672*29492bb7SDavid van Moolenbroek UnlockAndExit:
673*29492bb7SDavid van Moolenbroek     AcpiOsReleaseLock (AcpiGbl_GpeLock, Flags);
674*29492bb7SDavid van Moolenbroek     return_ACPI_STATUS (Status);
675*29492bb7SDavid van Moolenbroek }
676*29492bb7SDavid van Moolenbroek 
ACPI_EXPORT_SYMBOL(AcpiGetGpeStatus)677*29492bb7SDavid van Moolenbroek ACPI_EXPORT_SYMBOL (AcpiGetGpeStatus)
678*29492bb7SDavid van Moolenbroek 
679*29492bb7SDavid van Moolenbroek 
680*29492bb7SDavid van Moolenbroek /*******************************************************************************
681*29492bb7SDavid van Moolenbroek  *
682*29492bb7SDavid van Moolenbroek  * FUNCTION:    AcpiFinishGpe
683*29492bb7SDavid van Moolenbroek  *
684*29492bb7SDavid van Moolenbroek  * PARAMETERS:  GpeDevice           - Namespace node for the GPE Block
685*29492bb7SDavid van Moolenbroek  *                                    (NULL for FADT defined GPEs)
686*29492bb7SDavid van Moolenbroek  *              GpeNumber           - GPE level within the GPE block
687*29492bb7SDavid van Moolenbroek  *
688*29492bb7SDavid van Moolenbroek  * RETURN:      Status
689*29492bb7SDavid van Moolenbroek  *
690*29492bb7SDavid van Moolenbroek  * DESCRIPTION: Clear and conditionally reenable a GPE. This completes the GPE
691*29492bb7SDavid van Moolenbroek  *              processing. Intended for use by asynchronous host-installed
692*29492bb7SDavid van Moolenbroek  *              GPE handlers. The GPE is only reenabled if the EnableForRun bit
693*29492bb7SDavid van Moolenbroek  *              is set in the GPE info.
694*29492bb7SDavid van Moolenbroek  *
695*29492bb7SDavid van Moolenbroek  ******************************************************************************/
696*29492bb7SDavid van Moolenbroek 
697*29492bb7SDavid van Moolenbroek ACPI_STATUS
698*29492bb7SDavid van Moolenbroek AcpiFinishGpe (
699*29492bb7SDavid van Moolenbroek     ACPI_HANDLE             GpeDevice,
700*29492bb7SDavid van Moolenbroek     UINT32                  GpeNumber)
701*29492bb7SDavid van Moolenbroek {
702*29492bb7SDavid van Moolenbroek     ACPI_GPE_EVENT_INFO     *GpeEventInfo;
703*29492bb7SDavid van Moolenbroek     ACPI_STATUS             Status;
704*29492bb7SDavid van Moolenbroek     ACPI_CPU_FLAGS          Flags;
705*29492bb7SDavid van Moolenbroek 
706*29492bb7SDavid van Moolenbroek 
707*29492bb7SDavid van Moolenbroek     ACPI_FUNCTION_TRACE (AcpiFinishGpe);
708*29492bb7SDavid van Moolenbroek 
709*29492bb7SDavid van Moolenbroek 
710*29492bb7SDavid van Moolenbroek     Flags = AcpiOsAcquireLock (AcpiGbl_GpeLock);
711*29492bb7SDavid van Moolenbroek 
712*29492bb7SDavid van Moolenbroek     /* Ensure that we have a valid GPE number */
713*29492bb7SDavid van Moolenbroek 
714*29492bb7SDavid van Moolenbroek     GpeEventInfo = AcpiEvGetGpeEventInfo (GpeDevice, GpeNumber);
715*29492bb7SDavid van Moolenbroek     if (!GpeEventInfo)
716*29492bb7SDavid van Moolenbroek     {
717*29492bb7SDavid van Moolenbroek         Status = AE_BAD_PARAMETER;
718*29492bb7SDavid van Moolenbroek         goto UnlockAndExit;
719*29492bb7SDavid van Moolenbroek     }
720*29492bb7SDavid van Moolenbroek 
721*29492bb7SDavid van Moolenbroek     Status = AcpiEvFinishGpe (GpeEventInfo);
722*29492bb7SDavid van Moolenbroek 
723*29492bb7SDavid van Moolenbroek UnlockAndExit:
724*29492bb7SDavid van Moolenbroek     AcpiOsReleaseLock (AcpiGbl_GpeLock, Flags);
725*29492bb7SDavid van Moolenbroek     return_ACPI_STATUS (Status);
726*29492bb7SDavid van Moolenbroek }
727*29492bb7SDavid van Moolenbroek 
ACPI_EXPORT_SYMBOL(AcpiFinishGpe)728*29492bb7SDavid van Moolenbroek ACPI_EXPORT_SYMBOL (AcpiFinishGpe)
729*29492bb7SDavid van Moolenbroek 
730*29492bb7SDavid van Moolenbroek 
731*29492bb7SDavid van Moolenbroek /******************************************************************************
732*29492bb7SDavid van Moolenbroek  *
733*29492bb7SDavid van Moolenbroek  * FUNCTION:    AcpiDisableAllGpes
734*29492bb7SDavid van Moolenbroek  *
735*29492bb7SDavid van Moolenbroek  * PARAMETERS:  None
736*29492bb7SDavid van Moolenbroek  *
737*29492bb7SDavid van Moolenbroek  * RETURN:      Status
738*29492bb7SDavid van Moolenbroek  *
739*29492bb7SDavid van Moolenbroek  * DESCRIPTION: Disable and clear all GPEs in all GPE blocks
740*29492bb7SDavid van Moolenbroek  *
741*29492bb7SDavid van Moolenbroek  ******************************************************************************/
742*29492bb7SDavid van Moolenbroek 
743*29492bb7SDavid van Moolenbroek ACPI_STATUS
744*29492bb7SDavid van Moolenbroek AcpiDisableAllGpes (
745*29492bb7SDavid van Moolenbroek     void)
746*29492bb7SDavid van Moolenbroek {
747*29492bb7SDavid van Moolenbroek     ACPI_STATUS             Status;
748*29492bb7SDavid van Moolenbroek 
749*29492bb7SDavid van Moolenbroek 
750*29492bb7SDavid van Moolenbroek     ACPI_FUNCTION_TRACE (AcpiDisableAllGpes);
751*29492bb7SDavid van Moolenbroek 
752*29492bb7SDavid van Moolenbroek 
753*29492bb7SDavid van Moolenbroek     Status = AcpiUtAcquireMutex (ACPI_MTX_EVENTS);
754*29492bb7SDavid van Moolenbroek     if (ACPI_FAILURE (Status))
755*29492bb7SDavid van Moolenbroek     {
756*29492bb7SDavid van Moolenbroek         return_ACPI_STATUS (Status);
757*29492bb7SDavid van Moolenbroek     }
758*29492bb7SDavid van Moolenbroek 
759*29492bb7SDavid van Moolenbroek     Status = AcpiHwDisableAllGpes ();
760*29492bb7SDavid van Moolenbroek     (void) AcpiUtReleaseMutex (ACPI_MTX_EVENTS);
761*29492bb7SDavid van Moolenbroek 
762*29492bb7SDavid van Moolenbroek     return_ACPI_STATUS (Status);
763*29492bb7SDavid van Moolenbroek }
764*29492bb7SDavid van Moolenbroek 
ACPI_EXPORT_SYMBOL(AcpiDisableAllGpes)765*29492bb7SDavid van Moolenbroek ACPI_EXPORT_SYMBOL (AcpiDisableAllGpes)
766*29492bb7SDavid van Moolenbroek 
767*29492bb7SDavid van Moolenbroek 
768*29492bb7SDavid van Moolenbroek /******************************************************************************
769*29492bb7SDavid van Moolenbroek  *
770*29492bb7SDavid van Moolenbroek  * FUNCTION:    AcpiEnableAllRuntimeGpes
771*29492bb7SDavid van Moolenbroek  *
772*29492bb7SDavid van Moolenbroek  * PARAMETERS:  None
773*29492bb7SDavid van Moolenbroek  *
774*29492bb7SDavid van Moolenbroek  * RETURN:      Status
775*29492bb7SDavid van Moolenbroek  *
776*29492bb7SDavid van Moolenbroek  * DESCRIPTION: Enable all "runtime" GPEs, in all GPE blocks
777*29492bb7SDavid van Moolenbroek  *
778*29492bb7SDavid van Moolenbroek  ******************************************************************************/
779*29492bb7SDavid van Moolenbroek 
780*29492bb7SDavid van Moolenbroek ACPI_STATUS
781*29492bb7SDavid van Moolenbroek AcpiEnableAllRuntimeGpes (
782*29492bb7SDavid van Moolenbroek     void)
783*29492bb7SDavid van Moolenbroek {
784*29492bb7SDavid van Moolenbroek     ACPI_STATUS             Status;
785*29492bb7SDavid van Moolenbroek 
786*29492bb7SDavid van Moolenbroek 
787*29492bb7SDavid van Moolenbroek     ACPI_FUNCTION_TRACE (AcpiEnableAllRuntimeGpes);
788*29492bb7SDavid van Moolenbroek 
789*29492bb7SDavid van Moolenbroek 
790*29492bb7SDavid van Moolenbroek     Status = AcpiUtAcquireMutex (ACPI_MTX_EVENTS);
791*29492bb7SDavid van Moolenbroek     if (ACPI_FAILURE (Status))
792*29492bb7SDavid van Moolenbroek     {
793*29492bb7SDavid van Moolenbroek         return_ACPI_STATUS (Status);
794*29492bb7SDavid van Moolenbroek     }
795*29492bb7SDavid van Moolenbroek 
796*29492bb7SDavid van Moolenbroek     Status = AcpiHwEnableAllRuntimeGpes ();
797*29492bb7SDavid van Moolenbroek     (void) AcpiUtReleaseMutex (ACPI_MTX_EVENTS);
798*29492bb7SDavid van Moolenbroek 
799*29492bb7SDavid van Moolenbroek     return_ACPI_STATUS (Status);
800*29492bb7SDavid van Moolenbroek }
801*29492bb7SDavid van Moolenbroek 
ACPI_EXPORT_SYMBOL(AcpiEnableAllRuntimeGpes)802*29492bb7SDavid van Moolenbroek ACPI_EXPORT_SYMBOL (AcpiEnableAllRuntimeGpes)
803*29492bb7SDavid van Moolenbroek 
804*29492bb7SDavid van Moolenbroek 
805*29492bb7SDavid van Moolenbroek /*******************************************************************************
806*29492bb7SDavid van Moolenbroek  *
807*29492bb7SDavid van Moolenbroek  * FUNCTION:    AcpiInstallGpeBlock
808*29492bb7SDavid van Moolenbroek  *
809*29492bb7SDavid van Moolenbroek  * PARAMETERS:  GpeDevice           - Handle to the parent GPE Block Device
810*29492bb7SDavid van Moolenbroek  *              GpeBlockAddress     - Address and SpaceID
811*29492bb7SDavid van Moolenbroek  *              RegisterCount       - Number of GPE register pairs in the block
812*29492bb7SDavid van Moolenbroek  *              InterruptNumber     - H/W interrupt for the block
813*29492bb7SDavid van Moolenbroek  *
814*29492bb7SDavid van Moolenbroek  * RETURN:      Status
815*29492bb7SDavid van Moolenbroek  *
816*29492bb7SDavid van Moolenbroek  * DESCRIPTION: Create and Install a block of GPE registers. The GPEs are not
817*29492bb7SDavid van Moolenbroek  *              enabled here.
818*29492bb7SDavid van Moolenbroek  *
819*29492bb7SDavid van Moolenbroek  ******************************************************************************/
820*29492bb7SDavid van Moolenbroek 
821*29492bb7SDavid van Moolenbroek ACPI_STATUS
822*29492bb7SDavid van Moolenbroek AcpiInstallGpeBlock (
823*29492bb7SDavid van Moolenbroek     ACPI_HANDLE             GpeDevice,
824*29492bb7SDavid van Moolenbroek     ACPI_GENERIC_ADDRESS    *GpeBlockAddress,
825*29492bb7SDavid van Moolenbroek     UINT32                  RegisterCount,
826*29492bb7SDavid van Moolenbroek     UINT32                  InterruptNumber)
827*29492bb7SDavid van Moolenbroek {
828*29492bb7SDavid van Moolenbroek     ACPI_STATUS             Status;
829*29492bb7SDavid van Moolenbroek     ACPI_OPERAND_OBJECT     *ObjDesc;
830*29492bb7SDavid van Moolenbroek     ACPI_NAMESPACE_NODE     *Node;
831*29492bb7SDavid van Moolenbroek     ACPI_GPE_BLOCK_INFO     *GpeBlock;
832*29492bb7SDavid van Moolenbroek 
833*29492bb7SDavid van Moolenbroek 
834*29492bb7SDavid van Moolenbroek     ACPI_FUNCTION_TRACE (AcpiInstallGpeBlock);
835*29492bb7SDavid van Moolenbroek 
836*29492bb7SDavid van Moolenbroek 
837*29492bb7SDavid van Moolenbroek     if ((!GpeDevice)       ||
838*29492bb7SDavid van Moolenbroek         (!GpeBlockAddress) ||
839*29492bb7SDavid van Moolenbroek         (!RegisterCount))
840*29492bb7SDavid van Moolenbroek     {
841*29492bb7SDavid van Moolenbroek         return_ACPI_STATUS (AE_BAD_PARAMETER);
842*29492bb7SDavid van Moolenbroek     }
843*29492bb7SDavid van Moolenbroek 
844*29492bb7SDavid van Moolenbroek     Status = AcpiUtAcquireMutex (ACPI_MTX_NAMESPACE);
845*29492bb7SDavid van Moolenbroek     if (ACPI_FAILURE (Status))
846*29492bb7SDavid van Moolenbroek     {
847*29492bb7SDavid van Moolenbroek         return_ACPI_STATUS (Status);
848*29492bb7SDavid van Moolenbroek     }
849*29492bb7SDavid van Moolenbroek 
850*29492bb7SDavid van Moolenbroek     Node = AcpiNsValidateHandle (GpeDevice);
851*29492bb7SDavid van Moolenbroek     if (!Node)
852*29492bb7SDavid van Moolenbroek     {
853*29492bb7SDavid van Moolenbroek         Status = AE_BAD_PARAMETER;
854*29492bb7SDavid van Moolenbroek         goto UnlockAndExit;
855*29492bb7SDavid van Moolenbroek     }
856*29492bb7SDavid van Moolenbroek 
857*29492bb7SDavid van Moolenbroek     /* Validate the parent device */
858*29492bb7SDavid van Moolenbroek 
859*29492bb7SDavid van Moolenbroek     if (Node->Type != ACPI_TYPE_DEVICE)
860*29492bb7SDavid van Moolenbroek     {
861*29492bb7SDavid van Moolenbroek         Status = AE_TYPE;
862*29492bb7SDavid van Moolenbroek         goto UnlockAndExit;
863*29492bb7SDavid van Moolenbroek     }
864*29492bb7SDavid van Moolenbroek 
865*29492bb7SDavid van Moolenbroek     if (Node->Object)
866*29492bb7SDavid van Moolenbroek     {
867*29492bb7SDavid van Moolenbroek         Status = AE_ALREADY_EXISTS;
868*29492bb7SDavid van Moolenbroek         goto UnlockAndExit;
869*29492bb7SDavid van Moolenbroek     }
870*29492bb7SDavid van Moolenbroek 
871*29492bb7SDavid van Moolenbroek     /*
872*29492bb7SDavid van Moolenbroek      * For user-installed GPE Block Devices, the GpeBlockBaseNumber
873*29492bb7SDavid van Moolenbroek      * is always zero
874*29492bb7SDavid van Moolenbroek      */
875*29492bb7SDavid van Moolenbroek     Status = AcpiEvCreateGpeBlock (Node, GpeBlockAddress->Address,
876*29492bb7SDavid van Moolenbroek                 GpeBlockAddress->SpaceId, RegisterCount,
877*29492bb7SDavid van Moolenbroek                 0, InterruptNumber, &GpeBlock);
878*29492bb7SDavid van Moolenbroek     if (ACPI_FAILURE (Status))
879*29492bb7SDavid van Moolenbroek     {
880*29492bb7SDavid van Moolenbroek         goto UnlockAndExit;
881*29492bb7SDavid van Moolenbroek     }
882*29492bb7SDavid van Moolenbroek 
883*29492bb7SDavid van Moolenbroek     /* Install block in the DeviceObject attached to the node */
884*29492bb7SDavid van Moolenbroek 
885*29492bb7SDavid van Moolenbroek     ObjDesc = AcpiNsGetAttachedObject (Node);
886*29492bb7SDavid van Moolenbroek     if (!ObjDesc)
887*29492bb7SDavid van Moolenbroek     {
888*29492bb7SDavid van Moolenbroek         /*
889*29492bb7SDavid van Moolenbroek          * No object, create a new one (Device nodes do not always have
890*29492bb7SDavid van Moolenbroek          * an attached object)
891*29492bb7SDavid van Moolenbroek          */
892*29492bb7SDavid van Moolenbroek         ObjDesc = AcpiUtCreateInternalObject (ACPI_TYPE_DEVICE);
893*29492bb7SDavid van Moolenbroek         if (!ObjDesc)
894*29492bb7SDavid van Moolenbroek         {
895*29492bb7SDavid van Moolenbroek             Status = AE_NO_MEMORY;
896*29492bb7SDavid van Moolenbroek             goto UnlockAndExit;
897*29492bb7SDavid van Moolenbroek         }
898*29492bb7SDavid van Moolenbroek 
899*29492bb7SDavid van Moolenbroek         Status = AcpiNsAttachObject (Node, ObjDesc, ACPI_TYPE_DEVICE);
900*29492bb7SDavid van Moolenbroek 
901*29492bb7SDavid van Moolenbroek         /* Remove local reference to the object */
902*29492bb7SDavid van Moolenbroek 
903*29492bb7SDavid van Moolenbroek         AcpiUtRemoveReference (ObjDesc);
904*29492bb7SDavid van Moolenbroek         if (ACPI_FAILURE (Status))
905*29492bb7SDavid van Moolenbroek         {
906*29492bb7SDavid van Moolenbroek             goto UnlockAndExit;
907*29492bb7SDavid van Moolenbroek         }
908*29492bb7SDavid van Moolenbroek     }
909*29492bb7SDavid van Moolenbroek 
910*29492bb7SDavid van Moolenbroek     /* Now install the GPE block in the DeviceObject */
911*29492bb7SDavid van Moolenbroek 
912*29492bb7SDavid van Moolenbroek     ObjDesc->Device.GpeBlock = GpeBlock;
913*29492bb7SDavid van Moolenbroek 
914*29492bb7SDavid van Moolenbroek 
915*29492bb7SDavid van Moolenbroek UnlockAndExit:
916*29492bb7SDavid van Moolenbroek     (void) AcpiUtReleaseMutex (ACPI_MTX_NAMESPACE);
917*29492bb7SDavid van Moolenbroek     return_ACPI_STATUS (Status);
918*29492bb7SDavid van Moolenbroek }
919*29492bb7SDavid van Moolenbroek 
ACPI_EXPORT_SYMBOL(AcpiInstallGpeBlock)920*29492bb7SDavid van Moolenbroek ACPI_EXPORT_SYMBOL (AcpiInstallGpeBlock)
921*29492bb7SDavid van Moolenbroek 
922*29492bb7SDavid van Moolenbroek 
923*29492bb7SDavid van Moolenbroek /*******************************************************************************
924*29492bb7SDavid van Moolenbroek  *
925*29492bb7SDavid van Moolenbroek  * FUNCTION:    AcpiRemoveGpeBlock
926*29492bb7SDavid van Moolenbroek  *
927*29492bb7SDavid van Moolenbroek  * PARAMETERS:  GpeDevice           - Handle to the parent GPE Block Device
928*29492bb7SDavid van Moolenbroek  *
929*29492bb7SDavid van Moolenbroek  * RETURN:      Status
930*29492bb7SDavid van Moolenbroek  *
931*29492bb7SDavid van Moolenbroek  * DESCRIPTION: Remove a previously installed block of GPE registers
932*29492bb7SDavid van Moolenbroek  *
933*29492bb7SDavid van Moolenbroek  ******************************************************************************/
934*29492bb7SDavid van Moolenbroek 
935*29492bb7SDavid van Moolenbroek ACPI_STATUS
936*29492bb7SDavid van Moolenbroek AcpiRemoveGpeBlock (
937*29492bb7SDavid van Moolenbroek     ACPI_HANDLE             GpeDevice)
938*29492bb7SDavid van Moolenbroek {
939*29492bb7SDavid van Moolenbroek     ACPI_OPERAND_OBJECT     *ObjDesc;
940*29492bb7SDavid van Moolenbroek     ACPI_STATUS             Status;
941*29492bb7SDavid van Moolenbroek     ACPI_NAMESPACE_NODE     *Node;
942*29492bb7SDavid van Moolenbroek 
943*29492bb7SDavid van Moolenbroek 
944*29492bb7SDavid van Moolenbroek     ACPI_FUNCTION_TRACE (AcpiRemoveGpeBlock);
945*29492bb7SDavid van Moolenbroek 
946*29492bb7SDavid van Moolenbroek 
947*29492bb7SDavid van Moolenbroek     if (!GpeDevice)
948*29492bb7SDavid van Moolenbroek     {
949*29492bb7SDavid van Moolenbroek         return_ACPI_STATUS (AE_BAD_PARAMETER);
950*29492bb7SDavid van Moolenbroek     }
951*29492bb7SDavid van Moolenbroek 
952*29492bb7SDavid van Moolenbroek     Status = AcpiUtAcquireMutex (ACPI_MTX_NAMESPACE);
953*29492bb7SDavid van Moolenbroek     if (ACPI_FAILURE (Status))
954*29492bb7SDavid van Moolenbroek     {
955*29492bb7SDavid van Moolenbroek         return_ACPI_STATUS (Status);
956*29492bb7SDavid van Moolenbroek     }
957*29492bb7SDavid van Moolenbroek 
958*29492bb7SDavid van Moolenbroek     Node = AcpiNsValidateHandle (GpeDevice);
959*29492bb7SDavid van Moolenbroek     if (!Node)
960*29492bb7SDavid van Moolenbroek     {
961*29492bb7SDavid van Moolenbroek         Status = AE_BAD_PARAMETER;
962*29492bb7SDavid van Moolenbroek         goto UnlockAndExit;
963*29492bb7SDavid van Moolenbroek     }
964*29492bb7SDavid van Moolenbroek 
965*29492bb7SDavid van Moolenbroek     /* Validate the parent device */
966*29492bb7SDavid van Moolenbroek 
967*29492bb7SDavid van Moolenbroek     if (Node->Type != ACPI_TYPE_DEVICE)
968*29492bb7SDavid van Moolenbroek     {
969*29492bb7SDavid van Moolenbroek         Status = AE_TYPE;
970*29492bb7SDavid van Moolenbroek         goto UnlockAndExit;
971*29492bb7SDavid van Moolenbroek     }
972*29492bb7SDavid van Moolenbroek 
973*29492bb7SDavid van Moolenbroek     /* Get the DeviceObject attached to the node */
974*29492bb7SDavid van Moolenbroek 
975*29492bb7SDavid van Moolenbroek     ObjDesc = AcpiNsGetAttachedObject (Node);
976*29492bb7SDavid van Moolenbroek     if (!ObjDesc ||
977*29492bb7SDavid van Moolenbroek         !ObjDesc->Device.GpeBlock)
978*29492bb7SDavid van Moolenbroek     {
979*29492bb7SDavid van Moolenbroek         return_ACPI_STATUS (AE_NULL_OBJECT);
980*29492bb7SDavid van Moolenbroek     }
981*29492bb7SDavid van Moolenbroek 
982*29492bb7SDavid van Moolenbroek     /* Delete the GPE block (but not the DeviceObject) */
983*29492bb7SDavid van Moolenbroek 
984*29492bb7SDavid van Moolenbroek     Status = AcpiEvDeleteGpeBlock (ObjDesc->Device.GpeBlock);
985*29492bb7SDavid van Moolenbroek     if (ACPI_SUCCESS (Status))
986*29492bb7SDavid van Moolenbroek     {
987*29492bb7SDavid van Moolenbroek         ObjDesc->Device.GpeBlock = NULL;
988*29492bb7SDavid van Moolenbroek     }
989*29492bb7SDavid van Moolenbroek 
990*29492bb7SDavid van Moolenbroek UnlockAndExit:
991*29492bb7SDavid van Moolenbroek     (void) AcpiUtReleaseMutex (ACPI_MTX_NAMESPACE);
992*29492bb7SDavid van Moolenbroek     return_ACPI_STATUS (Status);
993*29492bb7SDavid van Moolenbroek }
994*29492bb7SDavid van Moolenbroek 
ACPI_EXPORT_SYMBOL(AcpiRemoveGpeBlock)995*29492bb7SDavid van Moolenbroek ACPI_EXPORT_SYMBOL (AcpiRemoveGpeBlock)
996*29492bb7SDavid van Moolenbroek 
997*29492bb7SDavid van Moolenbroek 
998*29492bb7SDavid van Moolenbroek /*******************************************************************************
999*29492bb7SDavid van Moolenbroek  *
1000*29492bb7SDavid van Moolenbroek  * FUNCTION:    AcpiGetGpeDevice
1001*29492bb7SDavid van Moolenbroek  *
1002*29492bb7SDavid van Moolenbroek  * PARAMETERS:  Index               - System GPE index (0-CurrentGpeCount)
1003*29492bb7SDavid van Moolenbroek  *              GpeDevice           - Where the parent GPE Device is returned
1004*29492bb7SDavid van Moolenbroek  *
1005*29492bb7SDavid van Moolenbroek  * RETURN:      Status
1006*29492bb7SDavid van Moolenbroek  *
1007*29492bb7SDavid van Moolenbroek  * DESCRIPTION: Obtain the GPE device associated with the input index. A NULL
1008*29492bb7SDavid van Moolenbroek  *              gpe device indicates that the gpe number is contained in one of
1009*29492bb7SDavid van Moolenbroek  *              the FADT-defined gpe blocks. Otherwise, the GPE block device.
1010*29492bb7SDavid van Moolenbroek  *
1011*29492bb7SDavid van Moolenbroek  ******************************************************************************/
1012*29492bb7SDavid van Moolenbroek 
1013*29492bb7SDavid van Moolenbroek ACPI_STATUS
1014*29492bb7SDavid van Moolenbroek AcpiGetGpeDevice (
1015*29492bb7SDavid van Moolenbroek     UINT32                  Index,
1016*29492bb7SDavid van Moolenbroek     ACPI_HANDLE             *GpeDevice)
1017*29492bb7SDavid van Moolenbroek {
1018*29492bb7SDavid van Moolenbroek     ACPI_GPE_DEVICE_INFO    Info;
1019*29492bb7SDavid van Moolenbroek     ACPI_STATUS             Status;
1020*29492bb7SDavid van Moolenbroek 
1021*29492bb7SDavid van Moolenbroek 
1022*29492bb7SDavid van Moolenbroek     ACPI_FUNCTION_TRACE (AcpiGetGpeDevice);
1023*29492bb7SDavid van Moolenbroek 
1024*29492bb7SDavid van Moolenbroek 
1025*29492bb7SDavid van Moolenbroek     if (!GpeDevice)
1026*29492bb7SDavid van Moolenbroek     {
1027*29492bb7SDavid van Moolenbroek         return_ACPI_STATUS (AE_BAD_PARAMETER);
1028*29492bb7SDavid van Moolenbroek     }
1029*29492bb7SDavid van Moolenbroek 
1030*29492bb7SDavid van Moolenbroek     if (Index >= AcpiCurrentGpeCount)
1031*29492bb7SDavid van Moolenbroek     {
1032*29492bb7SDavid van Moolenbroek         return_ACPI_STATUS (AE_NOT_EXIST);
1033*29492bb7SDavid van Moolenbroek     }
1034*29492bb7SDavid van Moolenbroek 
1035*29492bb7SDavid van Moolenbroek     /* Setup and walk the GPE list */
1036*29492bb7SDavid van Moolenbroek 
1037*29492bb7SDavid van Moolenbroek     Info.Index = Index;
1038*29492bb7SDavid van Moolenbroek     Info.Status = AE_NOT_EXIST;
1039*29492bb7SDavid van Moolenbroek     Info.GpeDevice = NULL;
1040*29492bb7SDavid van Moolenbroek     Info.NextBlockBaseIndex = 0;
1041*29492bb7SDavid van Moolenbroek 
1042*29492bb7SDavid van Moolenbroek     Status = AcpiEvWalkGpeList (AcpiEvGetGpeDevice, &Info);
1043*29492bb7SDavid van Moolenbroek     if (ACPI_FAILURE (Status))
1044*29492bb7SDavid van Moolenbroek     {
1045*29492bb7SDavid van Moolenbroek         return_ACPI_STATUS (Status);
1046*29492bb7SDavid van Moolenbroek     }
1047*29492bb7SDavid van Moolenbroek 
1048*29492bb7SDavid van Moolenbroek     *GpeDevice = ACPI_CAST_PTR (ACPI_HANDLE, Info.GpeDevice);
1049*29492bb7SDavid van Moolenbroek     return_ACPI_STATUS (Info.Status);
1050*29492bb7SDavid van Moolenbroek }
1051*29492bb7SDavid van Moolenbroek 
1052*29492bb7SDavid van Moolenbroek ACPI_EXPORT_SYMBOL (AcpiGetGpeDevice)
1053*29492bb7SDavid van Moolenbroek 
1054*29492bb7SDavid van Moolenbroek #endif /* !ACPI_REDUCED_HARDWARE */
1055