Blame source/components/events/evxfgpe.c

Packit Service 1fb00e
/******************************************************************************
Packit Service 1fb00e
 *
Packit Service 1fb00e
 * Module Name: evxfgpe - External Interfaces for General Purpose Events (GPEs)
Packit Service 1fb00e
 *
Packit Service 1fb00e
 *****************************************************************************/
Packit Service 1fb00e
Packit Service 1fb00e
/*
Packit Service 1fb00e
 * Copyright (C) 2000 - 2018, Intel Corp.
Packit Service 1fb00e
 * All rights reserved.
Packit Service 1fb00e
 *
Packit Service 1fb00e
 * Redistribution and use in source and binary forms, with or without
Packit Service 1fb00e
 * modification, are permitted provided that the following conditions
Packit Service 1fb00e
 * are met:
Packit Service 1fb00e
 * 1. Redistributions of source code must retain the above copyright
Packit Service 1fb00e
 *    notice, this list of conditions, and the following disclaimer,
Packit Service 1fb00e
 *    without modification.
Packit Service 1fb00e
 * 2. Redistributions in binary form must reproduce at minimum a disclaimer
Packit Service 1fb00e
 *    substantially similar to the "NO WARRANTY" disclaimer below
Packit Service 1fb00e
 *    ("Disclaimer") and any redistribution must be conditioned upon
Packit Service 1fb00e
 *    including a substantially similar Disclaimer requirement for further
Packit Service 1fb00e
 *    binary redistribution.
Packit Service 1fb00e
 * 3. Neither the names of the above-listed copyright holders nor the names
Packit Service 1fb00e
 *    of any contributors may be used to endorse or promote products derived
Packit Service 1fb00e
 *    from this software without specific prior written permission.
Packit Service 1fb00e
 *
Packit Service 1fb00e
 * Alternatively, this software may be distributed under the terms of the
Packit Service 1fb00e
 * GNU General Public License ("GPL") version 2 as published by the Free
Packit Service 1fb00e
 * Software Foundation.
Packit Service 1fb00e
 *
Packit Service 1fb00e
 * NO WARRANTY
Packit Service 1fb00e
 * THIS SOFTWARE IS PROVIDED BY THE COPYRIGHT HOLDERS AND CONTRIBUTORS
Packit Service 1fb00e
 * "AS IS" AND ANY EXPRESS OR IMPLIED WARRANTIES, INCLUDING, BUT NOT
Packit Service 1fb00e
 * LIMITED TO, THE IMPLIED WARRANTIES OF MERCHANTIBILITY AND FITNESS FOR
Packit Service 1fb00e
 * A PARTICULAR PURPOSE ARE DISCLAIMED. IN NO EVENT SHALL THE COPYRIGHT
Packit Service 1fb00e
 * HOLDERS OR CONTRIBUTORS BE LIABLE FOR SPECIAL, EXEMPLARY, OR CONSEQUENTIAL
Packit Service 1fb00e
 * DAMAGES (INCLUDING, BUT NOT LIMITED TO, PROCUREMENT OF SUBSTITUTE GOODS
Packit Service 1fb00e
 * OR SERVICES; LOSS OF USE, DATA, OR PROFITS; OR BUSINESS INTERRUPTION)
Packit Service 1fb00e
 * HOWEVER CAUSED AND ON ANY THEORY OF LIABILITY, WHETHER IN CONTRACT,
Packit Service 1fb00e
 * STRICT LIABILITY, OR TORT (INCLUDING NEGLIGENCE OR OTHERWISE) ARISING
Packit Service 1fb00e
 * IN ANY WAY OUT OF THE USE OF THIS SOFTWARE, EVEN IF ADVISED OF THE
Packit Service 1fb00e
 * POSSIBILITY OF SUCH DAMAGES.
Packit Service 1fb00e
 */
Packit Service 1fb00e
Packit Service 1fb00e
#define EXPORT_ACPI_INTERFACES
Packit Service 1fb00e
Packit Service 1fb00e
#include "acpi.h"
Packit Service 1fb00e
#include "accommon.h"
Packit Service 1fb00e
#include "acevents.h"
Packit Service 1fb00e
#include "acnamesp.h"
Packit Service 1fb00e
Packit Service 1fb00e
#define _COMPONENT          ACPI_EVENTS
Packit Service 1fb00e
        ACPI_MODULE_NAME    ("evxfgpe")
Packit Service 1fb00e
Packit Service 1fb00e
Packit Service 1fb00e
#if (!ACPI_REDUCED_HARDWARE) /* Entire module */
Packit Service 1fb00e
/*******************************************************************************
Packit Service 1fb00e
 *
Packit Service 1fb00e
 * FUNCTION:    AcpiUpdateAllGpes
Packit Service 1fb00e
 *
Packit Service 1fb00e
 * PARAMETERS:  None
Packit Service 1fb00e
 *
Packit Service 1fb00e
 * RETURN:      Status
Packit Service 1fb00e
 *
Packit Service 1fb00e
 * DESCRIPTION: Complete GPE initialization and enable all GPEs that have
Packit Service 1fb00e
 *              associated _Lxx or _Exx methods and are not pointed to by any
Packit Service 1fb00e
 *              device _PRW methods (this indicates that these GPEs are
Packit Service 1fb00e
 *              generally intended for system or device wakeup. Such GPEs
Packit Service 1fb00e
 *              have to be enabled directly when the devices whose _PRW
Packit Service 1fb00e
 *              methods point to them are set up for wakeup signaling.)
Packit Service 1fb00e
 *
Packit Service 1fb00e
 * NOTE: Should be called after any GPEs are added to the system. Primarily,
Packit Service 1fb00e
 * after the system _PRW methods have been run, but also after a GPE Block
Packit Service 1fb00e
 * Device has been added or if any new GPE methods have been added via a
Packit Service 1fb00e
 * dynamic table load.
Packit Service 1fb00e
 *
Packit Service 1fb00e
 ******************************************************************************/
Packit Service 1fb00e
Packit Service 1fb00e
ACPI_STATUS
Packit Service 1fb00e
AcpiUpdateAllGpes (
Packit Service 1fb00e
    void)
Packit Service 1fb00e
{
Packit Service 1fb00e
    ACPI_STATUS             Status;
Packit Service 1fb00e
    BOOLEAN                 IsPollingNeeded = FALSE;
Packit Service 1fb00e
Packit Service 1fb00e
Packit Service 1fb00e
    ACPI_FUNCTION_TRACE (AcpiUpdateAllGpes);
Packit Service 1fb00e
Packit Service 1fb00e
Packit Service 1fb00e
    Status = AcpiUtAcquireMutex (ACPI_MTX_EVENTS);
Packit Service 1fb00e
    if (ACPI_FAILURE (Status))
Packit Service 1fb00e
    {
Packit Service 1fb00e
        return_ACPI_STATUS (Status);
Packit Service 1fb00e
    }
Packit Service 1fb00e
Packit Service 1fb00e
    if (AcpiGbl_AllGpesInitialized)
Packit Service 1fb00e
    {
Packit Service 1fb00e
        goto UnlockAndExit;
Packit Service 1fb00e
    }
Packit Service 1fb00e
Packit Service 1fb00e
    Status = AcpiEvWalkGpeList (AcpiEvInitializeGpeBlock,
Packit Service 1fb00e
        &IsPollingNeeded);
Packit Service 1fb00e
    if (ACPI_SUCCESS (Status))
Packit Service 1fb00e
    {
Packit Service 1fb00e
        AcpiGbl_AllGpesInitialized = TRUE;
Packit Service 1fb00e
    }
Packit Service 1fb00e
Packit Service 1fb00e
UnlockAndExit:
Packit Service 1fb00e
    (void) AcpiUtReleaseMutex (ACPI_MTX_EVENTS);
Packit Service 1fb00e
Packit Service 1fb00e
    if (IsPollingNeeded && AcpiGbl_AllGpesInitialized)
Packit Service 1fb00e
    {
Packit Service 1fb00e
        /* Poll GPEs to handle already triggered events */
Packit Service 1fb00e
Packit Service 1fb00e
        AcpiEvGpeDetect (AcpiGbl_GpeXruptListHead);
Packit Service 1fb00e
    }
Packit Service 1fb00e
    return_ACPI_STATUS (Status);
Packit Service 1fb00e
}
Packit Service 1fb00e
Packit Service 1fb00e
ACPI_EXPORT_SYMBOL (AcpiUpdateAllGpes)
Packit Service 1fb00e
Packit Service 1fb00e
Packit Service 1fb00e
/*******************************************************************************
Packit Service 1fb00e
 *
Packit Service 1fb00e
 * FUNCTION:    AcpiEnableGpe
Packit Service 1fb00e
 *
Packit Service 1fb00e
 * PARAMETERS:  GpeDevice           - Parent GPE Device. NULL for GPE0/GPE1
Packit Service 1fb00e
 *              GpeNumber           - GPE level within the GPE block
Packit Service 1fb00e
 *
Packit Service 1fb00e
 * RETURN:      Status
Packit Service 1fb00e
 *
Packit Service 1fb00e
 * DESCRIPTION: Add a reference to a GPE. On the first reference, the GPE is
Packit Service 1fb00e
 *              hardware-enabled.
Packit Service 1fb00e
 *
Packit Service 1fb00e
 ******************************************************************************/
Packit Service 1fb00e
Packit Service 1fb00e
ACPI_STATUS
Packit Service 1fb00e
AcpiEnableGpe (
Packit Service 1fb00e
    ACPI_HANDLE             GpeDevice,
Packit Service 1fb00e
    UINT32                  GpeNumber)
Packit Service 1fb00e
{
Packit Service 1fb00e
    ACPI_STATUS             Status = AE_BAD_PARAMETER;
Packit Service 1fb00e
    ACPI_GPE_EVENT_INFO     *GpeEventInfo;
Packit Service 1fb00e
    ACPI_CPU_FLAGS          Flags;
Packit Service 1fb00e
Packit Service 1fb00e
Packit Service 1fb00e
    ACPI_FUNCTION_TRACE (AcpiEnableGpe);
Packit Service 1fb00e
Packit Service 1fb00e
Packit Service 1fb00e
    Flags = AcpiOsAcquireLock (AcpiGbl_GpeLock);
Packit Service 1fb00e
Packit Service 1fb00e
    /*
Packit Service 1fb00e
     * Ensure that we have a valid GPE number and that there is some way
Packit Service 1fb00e
     * of handling the GPE (handler or a GPE method). In other words, we
Packit Service 1fb00e
     * won't allow a valid GPE to be enabled if there is no way to handle it.
Packit Service 1fb00e
     */
Packit Service 1fb00e
    GpeEventInfo = AcpiEvGetGpeEventInfo (GpeDevice, GpeNumber);
Packit Service 1fb00e
    if (GpeEventInfo)
Packit Service 1fb00e
    {
Packit Service 1fb00e
        if (ACPI_GPE_DISPATCH_TYPE (GpeEventInfo->Flags) !=
Packit Service 1fb00e
            ACPI_GPE_DISPATCH_NONE)
Packit Service 1fb00e
        {
Packit Service 1fb00e
            Status = AcpiEvAddGpeReference (GpeEventInfo);
Packit Service 1fb00e
            if (ACPI_SUCCESS (Status) &&
Packit Service 1fb00e
                ACPI_GPE_IS_POLLING_NEEDED (GpeEventInfo))
Packit Service 1fb00e
            {
Packit Service 1fb00e
                /* Poll edge-triggered GPEs to handle existing events */
Packit Service 1fb00e
Packit Service 1fb00e
                AcpiOsReleaseLock (AcpiGbl_GpeLock, Flags);
Packit Service 1fb00e
                (void) AcpiEvDetectGpe (
Packit Service 1fb00e
                    GpeDevice, GpeEventInfo, GpeNumber);
Packit Service 1fb00e
                Flags = AcpiOsAcquireLock (AcpiGbl_GpeLock);
Packit Service 1fb00e
            }
Packit Service 1fb00e
        }
Packit Service 1fb00e
        else
Packit Service 1fb00e
        {
Packit Service 1fb00e
            Status = AE_NO_HANDLER;
Packit Service 1fb00e
        }
Packit Service 1fb00e
    }
Packit Service 1fb00e
Packit Service 1fb00e
    AcpiOsReleaseLock (AcpiGbl_GpeLock, Flags);
Packit Service 1fb00e
    return_ACPI_STATUS (Status);
Packit Service 1fb00e
}
Packit Service 1fb00e
Packit Service 1fb00e
ACPI_EXPORT_SYMBOL (AcpiEnableGpe)
Packit Service 1fb00e
Packit Service 1fb00e
Packit Service 1fb00e
/*******************************************************************************
Packit Service 1fb00e
 *
Packit Service 1fb00e
 * FUNCTION:    AcpiDisableGpe
Packit Service 1fb00e
 *
Packit Service 1fb00e
 * PARAMETERS:  GpeDevice           - Parent GPE Device. NULL for GPE0/GPE1
Packit Service 1fb00e
 *              GpeNumber           - GPE level within the GPE block
Packit Service 1fb00e
 *
Packit Service 1fb00e
 * RETURN:      Status
Packit Service 1fb00e
 *
Packit Service 1fb00e
 * DESCRIPTION: Remove a reference to a GPE. When the last reference is
Packit Service 1fb00e
 *              removed, only then is the GPE disabled (for runtime GPEs), or
Packit Service 1fb00e
 *              the GPE mask bit disabled (for wake GPEs)
Packit Service 1fb00e
 *
Packit Service 1fb00e
 ******************************************************************************/
Packit Service 1fb00e
Packit Service 1fb00e
ACPI_STATUS
Packit Service 1fb00e
AcpiDisableGpe (
Packit Service 1fb00e
    ACPI_HANDLE             GpeDevice,
Packit Service 1fb00e
    UINT32                  GpeNumber)
Packit Service 1fb00e
{
Packit Service 1fb00e
    ACPI_STATUS             Status = AE_BAD_PARAMETER;
Packit Service 1fb00e
    ACPI_GPE_EVENT_INFO     *GpeEventInfo;
Packit Service 1fb00e
    ACPI_CPU_FLAGS          Flags;
Packit Service 1fb00e
Packit Service 1fb00e
Packit Service 1fb00e
    ACPI_FUNCTION_TRACE (AcpiDisableGpe);
Packit Service 1fb00e
Packit Service 1fb00e
Packit Service 1fb00e
    Flags = AcpiOsAcquireLock (AcpiGbl_GpeLock);
Packit Service 1fb00e
Packit Service 1fb00e
    /* Ensure that we have a valid GPE number */
Packit Service 1fb00e
Packit Service 1fb00e
    GpeEventInfo = AcpiEvGetGpeEventInfo (GpeDevice, GpeNumber);
Packit Service 1fb00e
    if (GpeEventInfo)
Packit Service 1fb00e
    {
Packit Service 1fb00e
        Status = AcpiEvRemoveGpeReference (GpeEventInfo);
Packit Service 1fb00e
    }
Packit Service 1fb00e
Packit Service 1fb00e
    AcpiOsReleaseLock (AcpiGbl_GpeLock, Flags);
Packit Service 1fb00e
    return_ACPI_STATUS (Status);
Packit Service 1fb00e
}
Packit Service 1fb00e
Packit Service 1fb00e
ACPI_EXPORT_SYMBOL (AcpiDisableGpe)
Packit Service 1fb00e
Packit Service 1fb00e
Packit Service 1fb00e
/*******************************************************************************
Packit Service 1fb00e
 *
Packit Service 1fb00e
 * FUNCTION:    AcpiSetGpe
Packit Service 1fb00e
 *
Packit Service 1fb00e
 * PARAMETERS:  GpeDevice           - Parent GPE Device. NULL for GPE0/GPE1
Packit Service 1fb00e
 *              GpeNumber           - GPE level within the GPE block
Packit Service 1fb00e
 *              Action              - ACPI_GPE_ENABLE or ACPI_GPE_DISABLE
Packit Service 1fb00e
 *
Packit Service 1fb00e
 * RETURN:      Status
Packit Service 1fb00e
 *
Packit Service 1fb00e
 * DESCRIPTION: Enable or disable an individual GPE. This function bypasses
Packit Service 1fb00e
 *              the reference count mechanism used in the AcpiEnableGpe(),
Packit Service 1fb00e
 *              AcpiDisableGpe() interfaces.
Packit Service 1fb00e
 *              This API is typically used by the GPE raw handler mode driver
Packit Service 1fb00e
 *              to switch between the polling mode and the interrupt mode after
Packit Service 1fb00e
 *              the driver has enabled the GPE.
Packit Service 1fb00e
 *              The APIs should be invoked in this order:
Packit Service 1fb00e
 *               AcpiEnableGpe()              <- Ensure the reference count > 0
Packit Service 1fb00e
 *               AcpiSetGpe(ACPI_GPE_DISABLE) <- Enter polling mode
Packit Service 1fb00e
 *               AcpiSetGpe(ACPI_GPE_ENABLE)  <- Leave polling mode
Packit Service 1fb00e
 *               AcpiDisableGpe()             <- Decrease the reference count
Packit Service 1fb00e
 *
Packit Service 1fb00e
 * Note: If a GPE is shared by 2 silicon components, then both the drivers
Packit Service 1fb00e
 *       should support GPE polling mode or disabling the GPE for long period
Packit Service 1fb00e
 *       for one driver may break the other. So use it with care since all
Packit Service 1fb00e
 *       firmware _Lxx/_Exx handlers currently rely on the GPE interrupt mode.
Packit Service 1fb00e
 *
Packit Service 1fb00e
 ******************************************************************************/
Packit Service 1fb00e
Packit Service 1fb00e
ACPI_STATUS
Packit Service 1fb00e
AcpiSetGpe (
Packit Service 1fb00e
    ACPI_HANDLE             GpeDevice,
Packit Service 1fb00e
    UINT32                  GpeNumber,
Packit Service 1fb00e
    UINT8                   Action)
Packit Service 1fb00e
{
Packit Service 1fb00e
    ACPI_GPE_EVENT_INFO     *GpeEventInfo;
Packit Service 1fb00e
    ACPI_STATUS             Status;
Packit Service 1fb00e
    ACPI_CPU_FLAGS          Flags;
Packit Service 1fb00e
Packit Service 1fb00e
Packit Service 1fb00e
    ACPI_FUNCTION_TRACE (AcpiSetGpe);
Packit Service 1fb00e
Packit Service 1fb00e
Packit Service 1fb00e
    Flags = AcpiOsAcquireLock (AcpiGbl_GpeLock);
Packit Service 1fb00e
Packit Service 1fb00e
    /* Ensure that we have a valid GPE number */
Packit Service 1fb00e
Packit Service 1fb00e
    GpeEventInfo = AcpiEvGetGpeEventInfo (GpeDevice, GpeNumber);
Packit Service 1fb00e
    if (!GpeEventInfo)
Packit Service 1fb00e
    {
Packit Service 1fb00e
        Status = AE_BAD_PARAMETER;
Packit Service 1fb00e
        goto UnlockAndExit;
Packit Service 1fb00e
    }
Packit Service 1fb00e
Packit Service 1fb00e
    /* Perform the action */
Packit Service 1fb00e
Packit Service 1fb00e
    switch (Action)
Packit Service 1fb00e
    {
Packit Service 1fb00e
    case ACPI_GPE_ENABLE:
Packit Service 1fb00e
Packit Service 1fb00e
        Status = AcpiHwLowSetGpe (GpeEventInfo, ACPI_GPE_ENABLE);
Packit Service 1fb00e
        GpeEventInfo->DisableForDispatch = FALSE;
Packit Service 1fb00e
        break;
Packit Service 1fb00e
Packit Service 1fb00e
    case ACPI_GPE_DISABLE:
Packit Service 1fb00e
Packit Service 1fb00e
        Status = AcpiHwLowSetGpe (GpeEventInfo, ACPI_GPE_DISABLE);
Packit Service 1fb00e
        GpeEventInfo->DisableForDispatch = TRUE;
Packit Service 1fb00e
        break;
Packit Service 1fb00e
Packit Service 1fb00e
    default:
Packit Service 1fb00e
Packit Service 1fb00e
        Status = AE_BAD_PARAMETER;
Packit Service 1fb00e
        break;
Packit Service 1fb00e
    }
Packit Service 1fb00e
Packit Service 1fb00e
UnlockAndExit:
Packit Service 1fb00e
    AcpiOsReleaseLock (AcpiGbl_GpeLock, Flags);
Packit Service 1fb00e
    return_ACPI_STATUS (Status);
Packit Service 1fb00e
}
Packit Service 1fb00e
Packit Service 1fb00e
ACPI_EXPORT_SYMBOL (AcpiSetGpe)
Packit Service 1fb00e
Packit Service 1fb00e
Packit Service 1fb00e
/*******************************************************************************
Packit Service 1fb00e
 *
Packit Service 1fb00e
 * FUNCTION:    AcpiMaskGpe
Packit Service 1fb00e
 *
Packit Service 1fb00e
 * PARAMETERS:  GpeDevice           - Parent GPE Device. NULL for GPE0/GPE1
Packit Service 1fb00e
 *              GpeNumber           - GPE level within the GPE block
Packit Service 1fb00e
 *              IsMasked            - Whether the GPE is masked or not
Packit Service 1fb00e
 *
Packit Service 1fb00e
 * RETURN:      Status
Packit Service 1fb00e
 *
Packit Service 1fb00e
 * DESCRIPTION: Unconditionally mask/unmask the an individual GPE, ex., to
Packit Service 1fb00e
 *              prevent a GPE flooding.
Packit Service 1fb00e
 *
Packit Service 1fb00e
 ******************************************************************************/
Packit Service 1fb00e
Packit Service 1fb00e
ACPI_STATUS
Packit Service 1fb00e
AcpiMaskGpe (
Packit Service 1fb00e
    ACPI_HANDLE             GpeDevice,
Packit Service 1fb00e
    UINT32                  GpeNumber,
Packit Service 1fb00e
    BOOLEAN                 IsMasked)
Packit Service 1fb00e
{
Packit Service 1fb00e
    ACPI_GPE_EVENT_INFO     *GpeEventInfo;
Packit Service 1fb00e
    ACPI_STATUS             Status;
Packit Service 1fb00e
    ACPI_CPU_FLAGS          Flags;
Packit Service 1fb00e
Packit Service 1fb00e
Packit Service 1fb00e
    ACPI_FUNCTION_TRACE (AcpiMaskGpe);
Packit Service 1fb00e
Packit Service 1fb00e
Packit Service 1fb00e
    Flags = AcpiOsAcquireLock (AcpiGbl_GpeLock);
Packit Service 1fb00e
Packit Service 1fb00e
    /* Ensure that we have a valid GPE number */
Packit Service 1fb00e
Packit Service 1fb00e
    GpeEventInfo = AcpiEvGetGpeEventInfo (GpeDevice, GpeNumber);
Packit Service 1fb00e
    if (!GpeEventInfo)
Packit Service 1fb00e
    {
Packit Service 1fb00e
        Status = AE_BAD_PARAMETER;
Packit Service 1fb00e
        goto UnlockAndExit;
Packit Service 1fb00e
    }
Packit Service 1fb00e
Packit Service 1fb00e
    Status = AcpiEvMaskGpe (GpeEventInfo, IsMasked);
Packit Service 1fb00e
Packit Service 1fb00e
UnlockAndExit:
Packit Service 1fb00e
    AcpiOsReleaseLock (AcpiGbl_GpeLock, Flags);
Packit Service 1fb00e
    return_ACPI_STATUS (Status);
Packit Service 1fb00e
}
Packit Service 1fb00e
Packit Service 1fb00e
ACPI_EXPORT_SYMBOL (AcpiMaskGpe)
Packit Service 1fb00e
Packit Service 1fb00e
Packit Service 1fb00e
/*******************************************************************************
Packit Service 1fb00e
 *
Packit Service 1fb00e
 * FUNCTION:    AcpiMarkGpeForWake
Packit Service 1fb00e
 *
Packit Service 1fb00e
 * PARAMETERS:  GpeDevice           - Parent GPE Device. NULL for GPE0/GPE1
Packit Service 1fb00e
 *              GpeNumber           - GPE level within the GPE block
Packit Service 1fb00e
 *
Packit Service 1fb00e
 * RETURN:      Status
Packit Service 1fb00e
 *
Packit Service 1fb00e
 * DESCRIPTION: Mark a GPE as having the ability to wake the system. Simply
Packit Service 1fb00e
 *              sets the ACPI_GPE_CAN_WAKE flag.
Packit Service 1fb00e
 *
Packit Service 1fb00e
 * Some potential callers of AcpiSetupGpeForWake may know in advance that
Packit Service 1fb00e
 * there won't be any notify handlers installed for device wake notifications
Packit Service 1fb00e
 * from the given GPE (one example is a button GPE in Linux). For these cases,
Packit Service 1fb00e
 * AcpiMarkGpeForWake should be used instead of AcpiSetupGpeForWake.
Packit Service 1fb00e
 * This will set the ACPI_GPE_CAN_WAKE flag for the GPE without trying to
Packit Service 1fb00e
 * setup implicit wake notification for it (since there's no handler method).
Packit Service 1fb00e
 *
Packit Service 1fb00e
 ******************************************************************************/
Packit Service 1fb00e
Packit Service 1fb00e
ACPI_STATUS
Packit Service 1fb00e
AcpiMarkGpeForWake (
Packit Service 1fb00e
    ACPI_HANDLE             GpeDevice,
Packit Service 1fb00e
    UINT32                  GpeNumber)
Packit Service 1fb00e
{
Packit Service 1fb00e
    ACPI_GPE_EVENT_INFO     *GpeEventInfo;
Packit Service 1fb00e
    ACPI_STATUS             Status = AE_BAD_PARAMETER;
Packit Service 1fb00e
    ACPI_CPU_FLAGS          Flags;
Packit Service 1fb00e
Packit Service 1fb00e
Packit Service 1fb00e
    ACPI_FUNCTION_TRACE (AcpiMarkGpeForWake);
Packit Service 1fb00e
Packit Service 1fb00e
Packit Service 1fb00e
    Flags = AcpiOsAcquireLock (AcpiGbl_GpeLock);
Packit Service 1fb00e
Packit Service 1fb00e
    /* Ensure that we have a valid GPE number */
Packit Service 1fb00e
Packit Service 1fb00e
    GpeEventInfo = AcpiEvGetGpeEventInfo (GpeDevice, GpeNumber);
Packit Service 1fb00e
    if (GpeEventInfo)
Packit Service 1fb00e
    {
Packit Service 1fb00e
        /* Mark the GPE as a possible wake event */
Packit Service 1fb00e
Packit Service 1fb00e
        GpeEventInfo->Flags |= ACPI_GPE_CAN_WAKE;
Packit Service 1fb00e
        Status = AE_OK;
Packit Service 1fb00e
    }
Packit Service 1fb00e
Packit Service 1fb00e
    AcpiOsReleaseLock (AcpiGbl_GpeLock, Flags);
Packit Service 1fb00e
    return_ACPI_STATUS (Status);
Packit Service 1fb00e
}
Packit Service 1fb00e
Packit Service 1fb00e
ACPI_EXPORT_SYMBOL (AcpiMarkGpeForWake)
Packit Service 1fb00e
Packit Service 1fb00e
Packit Service 1fb00e
/*******************************************************************************
Packit Service 1fb00e
 *
Packit Service 1fb00e
 * FUNCTION:    AcpiSetupGpeForWake
Packit Service 1fb00e
 *
Packit Service 1fb00e
 * PARAMETERS:  WakeDevice          - Device associated with the GPE (via _PRW)
Packit Service 1fb00e
 *              GpeDevice           - Parent GPE Device. NULL for GPE0/GPE1
Packit Service 1fb00e
 *              GpeNumber           - GPE level within the GPE block
Packit Service 1fb00e
 *
Packit Service 1fb00e
 * RETURN:      Status
Packit Service 1fb00e
 *
Packit Service 1fb00e
 * DESCRIPTION: Mark a GPE as having the ability to wake the system. This
Packit Service 1fb00e
 *              interface is intended to be used as the host executes the
Packit Service 1fb00e
 *              _PRW methods (Power Resources for Wake) in the system tables.
Packit Service 1fb00e
 *              Each _PRW appears under a Device Object (The WakeDevice), and
Packit Service 1fb00e
 *              contains the info for the wake GPE associated with the
Packit Service 1fb00e
 *              WakeDevice.
Packit Service 1fb00e
 *
Packit Service 1fb00e
 ******************************************************************************/
Packit Service 1fb00e
Packit Service 1fb00e
ACPI_STATUS
Packit Service 1fb00e
AcpiSetupGpeForWake (
Packit Service 1fb00e
    ACPI_HANDLE             WakeDevice,
Packit Service 1fb00e
    ACPI_HANDLE             GpeDevice,
Packit Service 1fb00e
    UINT32                  GpeNumber)
Packit Service 1fb00e
{
Packit Service 1fb00e
    ACPI_STATUS             Status;
Packit Service 1fb00e
    ACPI_GPE_EVENT_INFO     *GpeEventInfo;
Packit Service 1fb00e
    ACPI_NAMESPACE_NODE     *DeviceNode;
Packit Service 1fb00e
    ACPI_GPE_NOTIFY_INFO    *Notify;
Packit Service 1fb00e
    ACPI_GPE_NOTIFY_INFO    *NewNotify;
Packit Service 1fb00e
    ACPI_CPU_FLAGS          Flags;
Packit Service 1fb00e
Packit Service 1fb00e
Packit Service 1fb00e
    ACPI_FUNCTION_TRACE (AcpiSetupGpeForWake);
Packit Service 1fb00e
Packit Service 1fb00e
Packit Service 1fb00e
    /* Parameter Validation */
Packit Service 1fb00e
Packit Service 1fb00e
    if (!WakeDevice)
Packit Service 1fb00e
    {
Packit Service 1fb00e
        /*
Packit Service 1fb00e
         * By forcing WakeDevice to be valid, we automatically enable the
Packit Service 1fb00e
         * implicit notify feature on all hosts.
Packit Service 1fb00e
         */
Packit Service 1fb00e
        return_ACPI_STATUS (AE_BAD_PARAMETER);
Packit Service 1fb00e
    }
Packit Service 1fb00e
Packit Service 1fb00e
    /* Handle root object case */
Packit Service 1fb00e
Packit Service 1fb00e
    if (WakeDevice == ACPI_ROOT_OBJECT)
Packit Service 1fb00e
    {
Packit Service 1fb00e
        DeviceNode = AcpiGbl_RootNode;
Packit Service 1fb00e
    }
Packit Service 1fb00e
    else
Packit Service 1fb00e
    {
Packit Service 1fb00e
        DeviceNode = ACPI_CAST_PTR (ACPI_NAMESPACE_NODE, WakeDevice);
Packit Service 1fb00e
    }
Packit Service 1fb00e
Packit Service 1fb00e
    /* Validate WakeDevice is of type Device */
Packit Service 1fb00e
Packit Service 1fb00e
    if (DeviceNode->Type != ACPI_TYPE_DEVICE)
Packit Service 1fb00e
    {
Packit Service 1fb00e
        return_ACPI_STATUS (AE_BAD_PARAMETER);
Packit Service 1fb00e
    }
Packit Service 1fb00e
Packit Service 1fb00e
    /*
Packit Service 1fb00e
     * Allocate a new notify object up front, in case it is needed.
Packit Service 1fb00e
     * Memory allocation while holding a spinlock is a big no-no
Packit Service 1fb00e
     * on some hosts.
Packit Service 1fb00e
     */
Packit Service 1fb00e
    NewNotify = ACPI_ALLOCATE_ZEROED (sizeof (ACPI_GPE_NOTIFY_INFO));
Packit Service 1fb00e
    if (!NewNotify)
Packit Service 1fb00e
    {
Packit Service 1fb00e
        return_ACPI_STATUS (AE_NO_MEMORY);
Packit Service 1fb00e
    }
Packit Service 1fb00e
Packit Service 1fb00e
    Flags = AcpiOsAcquireLock (AcpiGbl_GpeLock);
Packit Service 1fb00e
Packit Service 1fb00e
    /* Ensure that we have a valid GPE number */
Packit Service 1fb00e
Packit Service 1fb00e
    GpeEventInfo = AcpiEvGetGpeEventInfo (GpeDevice, GpeNumber);
Packit Service 1fb00e
    if (!GpeEventInfo)
Packit Service 1fb00e
    {
Packit Service 1fb00e
        Status = AE_BAD_PARAMETER;
Packit Service 1fb00e
        goto UnlockAndExit;
Packit Service 1fb00e
    }
Packit Service 1fb00e
Packit Service 1fb00e
    /*
Packit Service 1fb00e
     * If there is no method or handler for this GPE, then the
Packit Service 1fb00e
     * WakeDevice will be notified whenever this GPE fires. This is
Packit Service 1fb00e
     * known as an "implicit notify". Note: The GPE is assumed to be
Packit Service 1fb00e
     * level-triggered (for windows compatibility).
Packit Service 1fb00e
     */
Packit Service 1fb00e
    if (ACPI_GPE_DISPATCH_TYPE (GpeEventInfo->Flags) ==
Packit Service 1fb00e
        ACPI_GPE_DISPATCH_NONE)
Packit Service 1fb00e
    {
Packit Service 1fb00e
        /*
Packit Service 1fb00e
         * This is the first device for implicit notify on this GPE.
Packit Service 1fb00e
         * Just set the flags here, and enter the NOTIFY block below.
Packit Service 1fb00e
         */
Packit Service 1fb00e
        GpeEventInfo->Flags =
Packit Service 1fb00e
            (ACPI_GPE_DISPATCH_NOTIFY | ACPI_GPE_LEVEL_TRIGGERED);
Packit Service 1fb00e
    }
Packit Service 1fb00e
    else if (GpeEventInfo->Flags & ACPI_GPE_AUTO_ENABLED)
Packit Service 1fb00e
    {
Packit Service 1fb00e
        /*
Packit Service 1fb00e
         * A reference to this GPE has been added during the GPE block
Packit Service 1fb00e
         * initialization, so drop it now to prevent the GPE from being
Packit Service 1fb00e
         * permanently enabled and clear its ACPI_GPE_AUTO_ENABLED flag.
Packit Service 1fb00e
         */
Packit Service 1fb00e
        (void) AcpiEvRemoveGpeReference (GpeEventInfo);
Packit Service 1fb00e
        GpeEventInfo->Flags &= ~~ACPI_GPE_AUTO_ENABLED;
Packit Service 1fb00e
    }
Packit Service 1fb00e
Packit Service 1fb00e
    /*
Packit Service 1fb00e
     * If we already have an implicit notify on this GPE, add
Packit Service 1fb00e
     * this device to the notify list.
Packit Service 1fb00e
     */
Packit Service 1fb00e
    if (ACPI_GPE_DISPATCH_TYPE (GpeEventInfo->Flags) ==
Packit Service 1fb00e
        ACPI_GPE_DISPATCH_NOTIFY)
Packit Service 1fb00e
    {
Packit Service 1fb00e
        /* Ensure that the device is not already in the list */
Packit Service 1fb00e
Packit Service 1fb00e
        Notify = GpeEventInfo->Dispatch.NotifyList;
Packit Service 1fb00e
        while (Notify)
Packit Service 1fb00e
        {
Packit Service 1fb00e
            if (Notify->DeviceNode == DeviceNode)
Packit Service 1fb00e
            {
Packit Service 1fb00e
                Status = AE_ALREADY_EXISTS;
Packit Service 1fb00e
                goto UnlockAndExit;
Packit Service 1fb00e
            }
Packit Service 1fb00e
            Notify = Notify->Next;
Packit Service 1fb00e
        }
Packit Service 1fb00e
Packit Service 1fb00e
        /* Add this device to the notify list for this GPE */
Packit Service 1fb00e
Packit Service 1fb00e
        NewNotify->DeviceNode = DeviceNode;
Packit Service 1fb00e
        NewNotify->Next = GpeEventInfo->Dispatch.NotifyList;
Packit Service 1fb00e
        GpeEventInfo->Dispatch.NotifyList = NewNotify;
Packit Service 1fb00e
        NewNotify = NULL;
Packit Service 1fb00e
    }
Packit Service 1fb00e
Packit Service 1fb00e
    /* Mark the GPE as a possible wake event */
Packit Service 1fb00e
Packit Service 1fb00e
    GpeEventInfo->Flags |= ACPI_GPE_CAN_WAKE;
Packit Service 1fb00e
    Status = AE_OK;
Packit Service 1fb00e
Packit Service 1fb00e
Packit Service 1fb00e
UnlockAndExit:
Packit Service 1fb00e
    AcpiOsReleaseLock (AcpiGbl_GpeLock, Flags);
Packit Service 1fb00e
Packit Service 1fb00e
    /* Delete the notify object if it was not used above */
Packit Service 1fb00e
Packit Service 1fb00e
    if (NewNotify)
Packit Service 1fb00e
    {
Packit Service 1fb00e
        ACPI_FREE (NewNotify);
Packit Service 1fb00e
    }
Packit Service 1fb00e
    return_ACPI_STATUS (Status);
Packit Service 1fb00e
}
Packit Service 1fb00e
Packit Service 1fb00e
ACPI_EXPORT_SYMBOL (AcpiSetupGpeForWake)
Packit Service 1fb00e
Packit Service 1fb00e
Packit Service 1fb00e
/*******************************************************************************
Packit Service 1fb00e
 *
Packit Service 1fb00e
 * FUNCTION:    AcpiSetGpeWakeMask
Packit Service 1fb00e
 *
Packit Service 1fb00e
 * PARAMETERS:  GpeDevice           - Parent GPE Device. NULL for GPE0/GPE1
Packit Service 1fb00e
 *              GpeNumber           - GPE level within the GPE block
Packit Service 1fb00e
 *              Action              - Enable or Disable
Packit Service 1fb00e
 *
Packit Service 1fb00e
 * RETURN:      Status
Packit Service 1fb00e
 *
Packit Service 1fb00e
 * DESCRIPTION: Set or clear the GPE's wakeup enable mask bit. The GPE must
Packit Service 1fb00e
 *              already be marked as a WAKE GPE.
Packit Service 1fb00e
 *
Packit Service 1fb00e
 ******************************************************************************/
Packit Service 1fb00e
Packit Service 1fb00e
ACPI_STATUS
Packit Service 1fb00e
AcpiSetGpeWakeMask (
Packit Service 1fb00e
    ACPI_HANDLE             GpeDevice,
Packit Service 1fb00e
    UINT32                  GpeNumber,
Packit Service 1fb00e
    UINT8                   Action)
Packit Service 1fb00e
{
Packit Service 1fb00e
    ACPI_STATUS             Status = AE_OK;
Packit Service 1fb00e
    ACPI_GPE_EVENT_INFO     *GpeEventInfo;
Packit Service 1fb00e
    ACPI_GPE_REGISTER_INFO  *GpeRegisterInfo;
Packit Service 1fb00e
    ACPI_CPU_FLAGS          Flags;
Packit Service 1fb00e
    UINT32                  RegisterBit;
Packit Service 1fb00e
Packit Service 1fb00e
Packit Service 1fb00e
    ACPI_FUNCTION_TRACE (AcpiSetGpeWakeMask);
Packit Service 1fb00e
Packit Service 1fb00e
Packit Service 1fb00e
    Flags = AcpiOsAcquireLock (AcpiGbl_GpeLock);
Packit Service 1fb00e
Packit Service 1fb00e
    /*
Packit Service 1fb00e
     * Ensure that we have a valid GPE number and that this GPE is in
Packit Service 1fb00e
     * fact a wake GPE
Packit Service 1fb00e
     */
Packit Service 1fb00e
    GpeEventInfo = AcpiEvGetGpeEventInfo (GpeDevice, GpeNumber);
Packit Service 1fb00e
    if (!GpeEventInfo)
Packit Service 1fb00e
    {
Packit Service 1fb00e
        Status = AE_BAD_PARAMETER;
Packit Service 1fb00e
        goto UnlockAndExit;
Packit Service 1fb00e
    }
Packit Service 1fb00e
Packit Service 1fb00e
    if (!(GpeEventInfo->Flags & ACPI_GPE_CAN_WAKE))
Packit Service 1fb00e
    {
Packit Service 1fb00e
        Status = AE_TYPE;
Packit Service 1fb00e
        goto UnlockAndExit;
Packit Service 1fb00e
    }
Packit Service 1fb00e
Packit Service 1fb00e
    GpeRegisterInfo = GpeEventInfo->RegisterInfo;
Packit Service 1fb00e
    if (!GpeRegisterInfo)
Packit Service 1fb00e
    {
Packit Service 1fb00e
        Status = AE_NOT_EXIST;
Packit Service 1fb00e
        goto UnlockAndExit;
Packit Service 1fb00e
    }
Packit Service 1fb00e
Packit Service 1fb00e
    RegisterBit = AcpiHwGetGpeRegisterBit (GpeEventInfo);
Packit Service 1fb00e
Packit Service 1fb00e
    /* Perform the action */
Packit Service 1fb00e
Packit Service 1fb00e
    switch (Action)
Packit Service 1fb00e
    {
Packit Service 1fb00e
    case ACPI_GPE_ENABLE:
Packit Service 1fb00e
Packit Service 1fb00e
        ACPI_SET_BIT (GpeRegisterInfo->EnableForWake, (UINT8) RegisterBit);
Packit Service 1fb00e
        break;
Packit Service 1fb00e
Packit Service 1fb00e
    case ACPI_GPE_DISABLE:
Packit Service 1fb00e
Packit Service 1fb00e
        ACPI_CLEAR_BIT (GpeRegisterInfo->EnableForWake, (UINT8) RegisterBit);
Packit Service 1fb00e
        break;
Packit Service 1fb00e
Packit Service 1fb00e
    default:
Packit Service 1fb00e
Packit Service 1fb00e
        ACPI_ERROR ((AE_INFO, "%u, Invalid action", Action));
Packit Service 1fb00e
        Status = AE_BAD_PARAMETER;
Packit Service 1fb00e
        break;
Packit Service 1fb00e
    }
Packit Service 1fb00e
Packit Service 1fb00e
UnlockAndExit:
Packit Service 1fb00e
    AcpiOsReleaseLock (AcpiGbl_GpeLock, Flags);
Packit Service 1fb00e
    return_ACPI_STATUS (Status);
Packit Service 1fb00e
}
Packit Service 1fb00e
Packit Service 1fb00e
ACPI_EXPORT_SYMBOL (AcpiSetGpeWakeMask)
Packit Service 1fb00e
Packit Service 1fb00e
Packit Service 1fb00e
/*******************************************************************************
Packit Service 1fb00e
 *
Packit Service 1fb00e
 * FUNCTION:    AcpiClearGpe
Packit Service 1fb00e
 *
Packit Service 1fb00e
 * PARAMETERS:  GpeDevice           - Parent GPE Device. NULL for GPE0/GPE1
Packit Service 1fb00e
 *              GpeNumber           - GPE level within the GPE block
Packit Service 1fb00e
 *
Packit Service 1fb00e
 * RETURN:      Status
Packit Service 1fb00e
 *
Packit Service 1fb00e
 * DESCRIPTION: Clear an ACPI event (general purpose)
Packit Service 1fb00e
 *
Packit Service 1fb00e
 ******************************************************************************/
Packit Service 1fb00e
Packit Service 1fb00e
ACPI_STATUS
Packit Service 1fb00e
AcpiClearGpe (
Packit Service 1fb00e
    ACPI_HANDLE             GpeDevice,
Packit Service 1fb00e
    UINT32                  GpeNumber)
Packit Service 1fb00e
{
Packit Service 1fb00e
    ACPI_STATUS             Status = AE_OK;
Packit Service 1fb00e
    ACPI_GPE_EVENT_INFO     *GpeEventInfo;
Packit Service 1fb00e
    ACPI_CPU_FLAGS          Flags;
Packit Service 1fb00e
Packit Service 1fb00e
Packit Service 1fb00e
    ACPI_FUNCTION_TRACE (AcpiClearGpe);
Packit Service 1fb00e
Packit Service 1fb00e
Packit Service 1fb00e
    Flags = AcpiOsAcquireLock (AcpiGbl_GpeLock);
Packit Service 1fb00e
Packit Service 1fb00e
    /* Ensure that we have a valid GPE number */
Packit Service 1fb00e
Packit Service 1fb00e
    GpeEventInfo = AcpiEvGetGpeEventInfo (GpeDevice, GpeNumber);
Packit Service 1fb00e
    if (!GpeEventInfo)
Packit Service 1fb00e
    {
Packit Service 1fb00e
        Status = AE_BAD_PARAMETER;
Packit Service 1fb00e
        goto UnlockAndExit;
Packit Service 1fb00e
    }
Packit Service 1fb00e
Packit Service 1fb00e
    Status = AcpiHwClearGpe (GpeEventInfo);
Packit Service 1fb00e
Packit Service 1fb00e
UnlockAndExit:
Packit Service 1fb00e
    AcpiOsReleaseLock (AcpiGbl_GpeLock, Flags);
Packit Service 1fb00e
    return_ACPI_STATUS (Status);
Packit Service 1fb00e
}
Packit Service 1fb00e
Packit Service 1fb00e
ACPI_EXPORT_SYMBOL (AcpiClearGpe)
Packit Service 1fb00e
Packit Service 1fb00e
Packit Service 1fb00e
/*******************************************************************************
Packit Service 1fb00e
 *
Packit Service 1fb00e
 * FUNCTION:    AcpiGetGpeStatus
Packit Service 1fb00e
 *
Packit Service 1fb00e
 * PARAMETERS:  GpeDevice           - Parent GPE Device. NULL for GPE0/GPE1
Packit Service 1fb00e
 *              GpeNumber           - GPE level within the GPE block
Packit Service 1fb00e
 *              EventStatus         - Where the current status of the event
Packit Service 1fb00e
 *                                    will be returned
Packit Service 1fb00e
 *
Packit Service 1fb00e
 * RETURN:      Status
Packit Service 1fb00e
 *
Packit Service 1fb00e
 * DESCRIPTION: Get the current status of a GPE (signalled/not_signalled)
Packit Service 1fb00e
 *
Packit Service 1fb00e
 ******************************************************************************/
Packit Service 1fb00e
Packit Service 1fb00e
ACPI_STATUS
Packit Service 1fb00e
AcpiGetGpeStatus (
Packit Service 1fb00e
    ACPI_HANDLE             GpeDevice,
Packit Service 1fb00e
    UINT32                  GpeNumber,
Packit Service 1fb00e
    ACPI_EVENT_STATUS       *EventStatus)
Packit Service 1fb00e
{
Packit Service 1fb00e
    ACPI_STATUS             Status = AE_OK;
Packit Service 1fb00e
    ACPI_GPE_EVENT_INFO     *GpeEventInfo;
Packit Service 1fb00e
    ACPI_CPU_FLAGS          Flags;
Packit Service 1fb00e
Packit Service 1fb00e
Packit Service 1fb00e
    ACPI_FUNCTION_TRACE (AcpiGetGpeStatus);
Packit Service 1fb00e
Packit Service 1fb00e
Packit Service 1fb00e
    Flags = AcpiOsAcquireLock (AcpiGbl_GpeLock);
Packit Service 1fb00e
Packit Service 1fb00e
    /* Ensure that we have a valid GPE number */
Packit Service 1fb00e
Packit Service 1fb00e
    GpeEventInfo = AcpiEvGetGpeEventInfo (GpeDevice, GpeNumber);
Packit Service 1fb00e
    if (!GpeEventInfo)
Packit Service 1fb00e
    {
Packit Service 1fb00e
        Status = AE_BAD_PARAMETER;
Packit Service 1fb00e
        goto UnlockAndExit;
Packit Service 1fb00e
    }
Packit Service 1fb00e
Packit Service 1fb00e
    /* Obtain status on the requested GPE number */
Packit Service 1fb00e
Packit Service 1fb00e
    Status = AcpiHwGetGpeStatus (GpeEventInfo, EventStatus);
Packit Service 1fb00e
Packit Service 1fb00e
UnlockAndExit:
Packit Service 1fb00e
    AcpiOsReleaseLock (AcpiGbl_GpeLock, Flags);
Packit Service 1fb00e
    return_ACPI_STATUS (Status);
Packit Service 1fb00e
}
Packit Service 1fb00e
Packit Service 1fb00e
ACPI_EXPORT_SYMBOL (AcpiGetGpeStatus)
Packit Service 1fb00e
Packit Service 1fb00e
Packit Service 1fb00e
/*******************************************************************************
Packit Service 1fb00e
 *
Packit Service 1fb00e
 * FUNCTION:    AcpiFinishGpe
Packit Service 1fb00e
 *
Packit Service 1fb00e
 * PARAMETERS:  GpeDevice           - Namespace node for the GPE Block
Packit Service 1fb00e
 *                                    (NULL for FADT defined GPEs)
Packit Service 1fb00e
 *              GpeNumber           - GPE level within the GPE block
Packit Service 1fb00e
 *
Packit Service 1fb00e
 * RETURN:      Status
Packit Service 1fb00e
 *
Packit Service 1fb00e
 * DESCRIPTION: Clear and conditionally reenable a GPE. This completes the GPE
Packit Service 1fb00e
 *              processing. Intended for use by asynchronous host-installed
Packit Service 1fb00e
 *              GPE handlers. The GPE is only reenabled if the EnableForRun bit
Packit Service 1fb00e
 *              is set in the GPE info.
Packit Service 1fb00e
 *
Packit Service 1fb00e
 ******************************************************************************/
Packit Service 1fb00e
Packit Service 1fb00e
ACPI_STATUS
Packit Service 1fb00e
AcpiFinishGpe (
Packit Service 1fb00e
    ACPI_HANDLE             GpeDevice,
Packit Service 1fb00e
    UINT32                  GpeNumber)
Packit Service 1fb00e
{
Packit Service 1fb00e
    ACPI_GPE_EVENT_INFO     *GpeEventInfo;
Packit Service 1fb00e
    ACPI_STATUS             Status;
Packit Service 1fb00e
    ACPI_CPU_FLAGS          Flags;
Packit Service 1fb00e
Packit Service 1fb00e
Packit Service 1fb00e
    ACPI_FUNCTION_TRACE (AcpiFinishGpe);
Packit Service 1fb00e
Packit Service 1fb00e
Packit Service 1fb00e
    Flags = AcpiOsAcquireLock (AcpiGbl_GpeLock);
Packit Service 1fb00e
Packit Service 1fb00e
    /* Ensure that we have a valid GPE number */
Packit Service 1fb00e
Packit Service 1fb00e
    GpeEventInfo = AcpiEvGetGpeEventInfo (GpeDevice, GpeNumber);
Packit Service 1fb00e
    if (!GpeEventInfo)
Packit Service 1fb00e
    {
Packit Service 1fb00e
        Status = AE_BAD_PARAMETER;
Packit Service 1fb00e
        goto UnlockAndExit;
Packit Service 1fb00e
    }
Packit Service 1fb00e
Packit Service 1fb00e
    Status = AcpiEvFinishGpe (GpeEventInfo);
Packit Service 1fb00e
Packit Service 1fb00e
UnlockAndExit:
Packit Service 1fb00e
    AcpiOsReleaseLock (AcpiGbl_GpeLock, Flags);
Packit Service 1fb00e
    return_ACPI_STATUS (Status);
Packit Service 1fb00e
}
Packit Service 1fb00e
Packit Service 1fb00e
ACPI_EXPORT_SYMBOL (AcpiFinishGpe)
Packit Service 1fb00e
Packit Service 1fb00e
Packit Service 1fb00e
/******************************************************************************
Packit Service 1fb00e
 *
Packit Service 1fb00e
 * FUNCTION:    AcpiDisableAllGpes
Packit Service 1fb00e
 *
Packit Service 1fb00e
 * PARAMETERS:  None
Packit Service 1fb00e
 *
Packit Service 1fb00e
 * RETURN:      Status
Packit Service 1fb00e
 *
Packit Service 1fb00e
 * DESCRIPTION: Disable and clear all GPEs in all GPE blocks
Packit Service 1fb00e
 *
Packit Service 1fb00e
 ******************************************************************************/
Packit Service 1fb00e
Packit Service 1fb00e
ACPI_STATUS
Packit Service 1fb00e
AcpiDisableAllGpes (
Packit Service 1fb00e
    void)
Packit Service 1fb00e
{
Packit Service 1fb00e
    ACPI_STATUS             Status;
Packit Service 1fb00e
Packit Service 1fb00e
Packit Service 1fb00e
    ACPI_FUNCTION_TRACE (AcpiDisableAllGpes);
Packit Service 1fb00e
Packit Service 1fb00e
Packit Service 1fb00e
    Status = AcpiUtAcquireMutex (ACPI_MTX_EVENTS);
Packit Service 1fb00e
    if (ACPI_FAILURE (Status))
Packit Service 1fb00e
    {
Packit Service 1fb00e
        return_ACPI_STATUS (Status);
Packit Service 1fb00e
    }
Packit Service 1fb00e
Packit Service 1fb00e
    Status = AcpiHwDisableAllGpes ();
Packit Service 1fb00e
    (void) AcpiUtReleaseMutex (ACPI_MTX_EVENTS);
Packit Service 1fb00e
Packit Service 1fb00e
    return_ACPI_STATUS (Status);
Packit Service 1fb00e
}
Packit Service 1fb00e
Packit Service 1fb00e
ACPI_EXPORT_SYMBOL (AcpiDisableAllGpes)
Packit Service 1fb00e
Packit Service 1fb00e
Packit Service 1fb00e
/******************************************************************************
Packit Service 1fb00e
 *
Packit Service 1fb00e
 * FUNCTION:    AcpiEnableAllRuntimeGpes
Packit Service 1fb00e
 *
Packit Service 1fb00e
 * PARAMETERS:  None
Packit Service 1fb00e
 *
Packit Service 1fb00e
 * RETURN:      Status
Packit Service 1fb00e
 *
Packit Service 1fb00e
 * DESCRIPTION: Enable all "runtime" GPEs, in all GPE blocks
Packit Service 1fb00e
 *
Packit Service 1fb00e
 ******************************************************************************/
Packit Service 1fb00e
Packit Service 1fb00e
ACPI_STATUS
Packit Service 1fb00e
AcpiEnableAllRuntimeGpes (
Packit Service 1fb00e
    void)
Packit Service 1fb00e
{
Packit Service 1fb00e
    ACPI_STATUS             Status;
Packit Service 1fb00e
Packit Service 1fb00e
Packit Service 1fb00e
    ACPI_FUNCTION_TRACE (AcpiEnableAllRuntimeGpes);
Packit Service 1fb00e
Packit Service 1fb00e
Packit Service 1fb00e
    Status = AcpiUtAcquireMutex (ACPI_MTX_EVENTS);
Packit Service 1fb00e
    if (ACPI_FAILURE (Status))
Packit Service 1fb00e
    {
Packit Service 1fb00e
        return_ACPI_STATUS (Status);
Packit Service 1fb00e
    }
Packit Service 1fb00e
Packit Service 1fb00e
    Status = AcpiHwEnableAllRuntimeGpes ();
Packit Service 1fb00e
    (void) AcpiUtReleaseMutex (ACPI_MTX_EVENTS);
Packit Service 1fb00e
Packit Service 1fb00e
    return_ACPI_STATUS (Status);
Packit Service 1fb00e
}
Packit Service 1fb00e
Packit Service 1fb00e
ACPI_EXPORT_SYMBOL (AcpiEnableAllRuntimeGpes)
Packit Service 1fb00e
Packit Service 1fb00e
Packit Service 1fb00e
/******************************************************************************
Packit Service 1fb00e
 *
Packit Service 1fb00e
 * FUNCTION:    AcpiEnableAllWakeupGpes
Packit Service 1fb00e
 *
Packit Service 1fb00e
 * PARAMETERS:  None
Packit Service 1fb00e
 *
Packit Service 1fb00e
 * RETURN:      Status
Packit Service 1fb00e
 *
Packit Service 1fb00e
 * DESCRIPTION: Enable all "wakeup" GPEs and disable all of the other GPEs, in
Packit Service 1fb00e
 *              all GPE blocks.
Packit Service 1fb00e
 *
Packit Service 1fb00e
 ******************************************************************************/
Packit Service 1fb00e
Packit Service 1fb00e
ACPI_STATUS
Packit Service 1fb00e
AcpiEnableAllWakeupGpes (
Packit Service 1fb00e
    void)
Packit Service 1fb00e
{
Packit Service 1fb00e
    ACPI_STATUS             Status;
Packit Service 1fb00e
Packit Service 1fb00e
Packit Service 1fb00e
    ACPI_FUNCTION_TRACE (AcpiEnableAllWakeupGpes);
Packit Service 1fb00e
Packit Service 1fb00e
Packit Service 1fb00e
    Status = AcpiUtAcquireMutex (ACPI_MTX_EVENTS);
Packit Service 1fb00e
    if (ACPI_FAILURE (Status))
Packit Service 1fb00e
    {
Packit Service 1fb00e
        return_ACPI_STATUS (Status);
Packit Service 1fb00e
    }
Packit Service 1fb00e
Packit Service 1fb00e
    Status = AcpiHwEnableAllWakeupGpes ();
Packit Service 1fb00e
    (void) AcpiUtReleaseMutex (ACPI_MTX_EVENTS);
Packit Service 1fb00e
Packit Service 1fb00e
    return_ACPI_STATUS (Status);
Packit Service 1fb00e
}
Packit Service 1fb00e
Packit Service 1fb00e
ACPI_EXPORT_SYMBOL (AcpiEnableAllWakeupGpes)
Packit Service 1fb00e
Packit Service 1fb00e
Packit Service 1fb00e
/*******************************************************************************
Packit Service 1fb00e
 *
Packit Service 1fb00e
 * FUNCTION:    AcpiInstallGpeBlock
Packit Service 1fb00e
 *
Packit Service 1fb00e
 * PARAMETERS:  GpeDevice           - Handle to the parent GPE Block Device
Packit Service 1fb00e
 *              GpeBlockAddress     - Address and SpaceID
Packit Service 1fb00e
 *              RegisterCount       - Number of GPE register pairs in the block
Packit Service 1fb00e
 *              InterruptNumber     - H/W interrupt for the block
Packit Service 1fb00e
 *
Packit Service 1fb00e
 * RETURN:      Status
Packit Service 1fb00e
 *
Packit Service 1fb00e
 * DESCRIPTION: Create and Install a block of GPE registers. The GPEs are not
Packit Service 1fb00e
 *              enabled here.
Packit Service 1fb00e
 *
Packit Service 1fb00e
 ******************************************************************************/
Packit Service 1fb00e
Packit Service 1fb00e
ACPI_STATUS
Packit Service 1fb00e
AcpiInstallGpeBlock (
Packit Service 1fb00e
    ACPI_HANDLE             GpeDevice,
Packit Service 1fb00e
    ACPI_GENERIC_ADDRESS    *GpeBlockAddress,
Packit Service 1fb00e
    UINT32                  RegisterCount,
Packit Service 1fb00e
    UINT32                  InterruptNumber)
Packit Service 1fb00e
{
Packit Service 1fb00e
    ACPI_STATUS             Status;
Packit Service 1fb00e
    ACPI_OPERAND_OBJECT     *ObjDesc;
Packit Service 1fb00e
    ACPI_NAMESPACE_NODE     *Node;
Packit Service 1fb00e
    ACPI_GPE_BLOCK_INFO     *GpeBlock;
Packit Service 1fb00e
Packit Service 1fb00e
Packit Service 1fb00e
    ACPI_FUNCTION_TRACE (AcpiInstallGpeBlock);
Packit Service 1fb00e
Packit Service 1fb00e
Packit Service 1fb00e
    if ((!GpeDevice)       ||
Packit Service 1fb00e
        (!GpeBlockAddress) ||
Packit Service 1fb00e
        (!RegisterCount))
Packit Service 1fb00e
    {
Packit Service 1fb00e
        return_ACPI_STATUS (AE_BAD_PARAMETER);
Packit Service 1fb00e
    }
Packit Service 1fb00e
Packit Service 1fb00e
    Status = AcpiUtAcquireMutex (ACPI_MTX_NAMESPACE);
Packit Service 1fb00e
    if (ACPI_FAILURE (Status))
Packit Service 1fb00e
    {
Packit Service 1fb00e
        return_ACPI_STATUS (Status);
Packit Service 1fb00e
    }
Packit Service 1fb00e
Packit Service 1fb00e
    Node = AcpiNsValidateHandle (GpeDevice);
Packit Service 1fb00e
    if (!Node)
Packit Service 1fb00e
    {
Packit Service 1fb00e
        Status = AE_BAD_PARAMETER;
Packit Service 1fb00e
        goto UnlockAndExit;
Packit Service 1fb00e
    }
Packit Service 1fb00e
Packit Service 1fb00e
    /* Validate the parent device */
Packit Service 1fb00e
Packit Service 1fb00e
    if (Node->Type != ACPI_TYPE_DEVICE)
Packit Service 1fb00e
    {
Packit Service 1fb00e
        Status = AE_TYPE;
Packit Service 1fb00e
        goto UnlockAndExit;
Packit Service 1fb00e
    }
Packit Service 1fb00e
Packit Service 1fb00e
    if (Node->Object)
Packit Service 1fb00e
    {
Packit Service 1fb00e
        Status = AE_ALREADY_EXISTS;
Packit Service 1fb00e
        goto UnlockAndExit;
Packit Service 1fb00e
    }
Packit Service 1fb00e
Packit Service 1fb00e
    /*
Packit Service 1fb00e
     * For user-installed GPE Block Devices, the GpeBlockBaseNumber
Packit Service 1fb00e
     * is always zero
Packit Service 1fb00e
     */
Packit Service 1fb00e
    Status = AcpiEvCreateGpeBlock (Node, GpeBlockAddress->Address,
Packit Service 1fb00e
        GpeBlockAddress->SpaceId, RegisterCount,
Packit Service 1fb00e
        0, InterruptNumber, &GpeBlock);
Packit Service 1fb00e
    if (ACPI_FAILURE (Status))
Packit Service 1fb00e
    {
Packit Service 1fb00e
        goto UnlockAndExit;
Packit Service 1fb00e
    }
Packit Service 1fb00e
Packit Service 1fb00e
    /* Install block in the DeviceObject attached to the node */
Packit Service 1fb00e
Packit Service 1fb00e
    ObjDesc = AcpiNsGetAttachedObject (Node);
Packit Service 1fb00e
    if (!ObjDesc)
Packit Service 1fb00e
    {
Packit Service 1fb00e
        /*
Packit Service 1fb00e
         * No object, create a new one (Device nodes do not always have
Packit Service 1fb00e
         * an attached object)
Packit Service 1fb00e
         */
Packit Service 1fb00e
        ObjDesc = AcpiUtCreateInternalObject (ACPI_TYPE_DEVICE);
Packit Service 1fb00e
        if (!ObjDesc)
Packit Service 1fb00e
        {
Packit Service 1fb00e
            Status = AE_NO_MEMORY;
Packit Service 1fb00e
            goto UnlockAndExit;
Packit Service 1fb00e
        }
Packit Service 1fb00e
Packit Service 1fb00e
        Status = AcpiNsAttachObject (Node, ObjDesc, ACPI_TYPE_DEVICE);
Packit Service 1fb00e
Packit Service 1fb00e
        /* Remove local reference to the object */
Packit Service 1fb00e
Packit Service 1fb00e
        AcpiUtRemoveReference (ObjDesc);
Packit Service 1fb00e
        if (ACPI_FAILURE (Status))
Packit Service 1fb00e
        {
Packit Service 1fb00e
            goto UnlockAndExit;
Packit Service 1fb00e
        }
Packit Service 1fb00e
    }
Packit Service 1fb00e
Packit Service 1fb00e
    /* Now install the GPE block in the DeviceObject */
Packit Service 1fb00e
Packit Service 1fb00e
    ObjDesc->Device.GpeBlock = GpeBlock;
Packit Service 1fb00e
Packit Service 1fb00e
Packit Service 1fb00e
UnlockAndExit:
Packit Service 1fb00e
    (void) AcpiUtReleaseMutex (ACPI_MTX_NAMESPACE);
Packit Service 1fb00e
    return_ACPI_STATUS (Status);
Packit Service 1fb00e
}
Packit Service 1fb00e
Packit Service 1fb00e
ACPI_EXPORT_SYMBOL (AcpiInstallGpeBlock)
Packit Service 1fb00e
Packit Service 1fb00e
Packit Service 1fb00e
/*******************************************************************************
Packit Service 1fb00e
 *
Packit Service 1fb00e
 * FUNCTION:    AcpiRemoveGpeBlock
Packit Service 1fb00e
 *
Packit Service 1fb00e
 * PARAMETERS:  GpeDevice           - Handle to the parent GPE Block Device
Packit Service 1fb00e
 *
Packit Service 1fb00e
 * RETURN:      Status
Packit Service 1fb00e
 *
Packit Service 1fb00e
 * DESCRIPTION: Remove a previously installed block of GPE registers
Packit Service 1fb00e
 *
Packit Service 1fb00e
 ******************************************************************************/
Packit Service 1fb00e
Packit Service 1fb00e
ACPI_STATUS
Packit Service 1fb00e
AcpiRemoveGpeBlock (
Packit Service 1fb00e
    ACPI_HANDLE             GpeDevice)
Packit Service 1fb00e
{
Packit Service 1fb00e
    ACPI_OPERAND_OBJECT     *ObjDesc;
Packit Service 1fb00e
    ACPI_STATUS             Status;
Packit Service 1fb00e
    ACPI_NAMESPACE_NODE     *Node;
Packit Service 1fb00e
Packit Service 1fb00e
Packit Service 1fb00e
    ACPI_FUNCTION_TRACE (AcpiRemoveGpeBlock);
Packit Service 1fb00e
Packit Service 1fb00e
Packit Service 1fb00e
    if (!GpeDevice)
Packit Service 1fb00e
    {
Packit Service 1fb00e
        return_ACPI_STATUS (AE_BAD_PARAMETER);
Packit Service 1fb00e
    }
Packit Service 1fb00e
Packit Service 1fb00e
    Status = AcpiUtAcquireMutex (ACPI_MTX_NAMESPACE);
Packit Service 1fb00e
    if (ACPI_FAILURE (Status))
Packit Service 1fb00e
    {
Packit Service 1fb00e
        return_ACPI_STATUS (Status);
Packit Service 1fb00e
    }
Packit Service 1fb00e
Packit Service 1fb00e
    Node = AcpiNsValidateHandle (GpeDevice);
Packit Service 1fb00e
    if (!Node)
Packit Service 1fb00e
    {
Packit Service 1fb00e
        Status = AE_BAD_PARAMETER;
Packit Service 1fb00e
        goto UnlockAndExit;
Packit Service 1fb00e
    }
Packit Service 1fb00e
Packit Service 1fb00e
    /* Validate the parent device */
Packit Service 1fb00e
Packit Service 1fb00e
    if (Node->Type != ACPI_TYPE_DEVICE)
Packit Service 1fb00e
    {
Packit Service 1fb00e
        Status = AE_TYPE;
Packit Service 1fb00e
        goto UnlockAndExit;
Packit Service 1fb00e
    }
Packit Service 1fb00e
Packit Service 1fb00e
    /* Get the DeviceObject attached to the node */
Packit Service 1fb00e
Packit Service 1fb00e
    ObjDesc = AcpiNsGetAttachedObject (Node);
Packit Service 1fb00e
    if (!ObjDesc ||
Packit Service 1fb00e
        !ObjDesc->Device.GpeBlock)
Packit Service 1fb00e
    {
Packit Service 1fb00e
        return_ACPI_STATUS (AE_NULL_OBJECT);
Packit Service 1fb00e
    }
Packit Service 1fb00e
Packit Service 1fb00e
    /* Delete the GPE block (but not the DeviceObject) */
Packit Service 1fb00e
Packit Service 1fb00e
    Status = AcpiEvDeleteGpeBlock (ObjDesc->Device.GpeBlock);
Packit Service 1fb00e
    if (ACPI_SUCCESS (Status))
Packit Service 1fb00e
    {
Packit Service 1fb00e
        ObjDesc->Device.GpeBlock = NULL;
Packit Service 1fb00e
    }
Packit Service 1fb00e
Packit Service 1fb00e
UnlockAndExit:
Packit Service 1fb00e
    (void) AcpiUtReleaseMutex (ACPI_MTX_NAMESPACE);
Packit Service 1fb00e
    return_ACPI_STATUS (Status);
Packit Service 1fb00e
}
Packit Service 1fb00e
Packit Service 1fb00e
ACPI_EXPORT_SYMBOL (AcpiRemoveGpeBlock)
Packit Service 1fb00e
Packit Service 1fb00e
Packit Service 1fb00e
/*******************************************************************************
Packit Service 1fb00e
 *
Packit Service 1fb00e
 * FUNCTION:    AcpiGetGpeDevice
Packit Service 1fb00e
 *
Packit Service 1fb00e
 * PARAMETERS:  Index               - System GPE index (0-CurrentGpeCount)
Packit Service 1fb00e
 *              GpeDevice           - Where the parent GPE Device is returned
Packit Service 1fb00e
 *
Packit Service 1fb00e
 * RETURN:      Status
Packit Service 1fb00e
 *
Packit Service 1fb00e
 * DESCRIPTION: Obtain the GPE device associated with the input index. A NULL
Packit Service 1fb00e
 *              gpe device indicates that the gpe number is contained in one of
Packit Service 1fb00e
 *              the FADT-defined gpe blocks. Otherwise, the GPE block device.
Packit Service 1fb00e
 *
Packit Service 1fb00e
 ******************************************************************************/
Packit Service 1fb00e
Packit Service 1fb00e
ACPI_STATUS
Packit Service 1fb00e
AcpiGetGpeDevice (
Packit Service 1fb00e
    UINT32                  Index,
Packit Service 1fb00e
    ACPI_HANDLE             *GpeDevice)
Packit Service 1fb00e
{
Packit Service 1fb00e
    ACPI_GPE_DEVICE_INFO    Info;
Packit Service 1fb00e
    ACPI_STATUS             Status;
Packit Service 1fb00e
Packit Service 1fb00e
Packit Service 1fb00e
    ACPI_FUNCTION_TRACE (AcpiGetGpeDevice);
Packit Service 1fb00e
Packit Service 1fb00e
Packit Service 1fb00e
    if (!GpeDevice)
Packit Service 1fb00e
    {
Packit Service 1fb00e
        return_ACPI_STATUS (AE_BAD_PARAMETER);
Packit Service 1fb00e
    }
Packit Service 1fb00e
Packit Service 1fb00e
    if (Index >= AcpiCurrentGpeCount)
Packit Service 1fb00e
    {
Packit Service 1fb00e
        return_ACPI_STATUS (AE_NOT_EXIST);
Packit Service 1fb00e
    }
Packit Service 1fb00e
Packit Service 1fb00e
    /* Setup and walk the GPE list */
Packit Service 1fb00e
Packit Service 1fb00e
    Info.Index = Index;
Packit Service 1fb00e
    Info.Status = AE_NOT_EXIST;
Packit Service 1fb00e
    Info.GpeDevice = NULL;
Packit Service 1fb00e
    Info.NextBlockBaseIndex = 0;
Packit Service 1fb00e
Packit Service 1fb00e
    Status = AcpiEvWalkGpeList (AcpiEvGetGpeDevice, &Info);
Packit Service 1fb00e
    if (ACPI_FAILURE (Status))
Packit Service 1fb00e
    {
Packit Service 1fb00e
        return_ACPI_STATUS (Status);
Packit Service 1fb00e
    }
Packit Service 1fb00e
Packit Service 1fb00e
    *GpeDevice = ACPI_CAST_PTR (ACPI_HANDLE, Info.GpeDevice);
Packit Service 1fb00e
    return_ACPI_STATUS (Info.Status);
Packit Service 1fb00e
}
Packit Service 1fb00e
Packit Service 1fb00e
ACPI_EXPORT_SYMBOL (AcpiGetGpeDevice)
Packit Service 1fb00e
Packit Service 1fb00e
#endif /* !ACPI_REDUCED_HARDWARE */