2012-02-14 15:00:53 +08:00
/******************************************************************************
*
* Name : hwxfsleep . c - ACPI Hardware Sleep / Wake External Interfaces
*
* * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * */
/*
2016-01-15 08:17:03 +08:00
* Copyright ( C ) 2000 - 2016 , Intel Corp .
2012-02-14 15:00:53 +08:00
* All rights reserved .
*
* Redistribution and use in source and binary forms , with or without
* modification , are permitted provided that the following conditions
* are met :
* 1. Redistributions of source code must retain the above copyright
* notice , this list of conditions , and the following disclaimer ,
* without modification .
* 2. Redistributions in binary form must reproduce at minimum a disclaimer
* substantially similar to the " NO WARRANTY " disclaimer below
* ( " Disclaimer " ) and any redistribution must be conditioned upon
* including a substantially similar Disclaimer requirement for further
* binary redistribution .
* 3. Neither the names of the above - listed copyright holders nor the names
* of any contributors may be used to endorse or promote products derived
* from this software without specific prior written permission .
*
* Alternatively , this software may be distributed under the terms of the
* GNU General Public License ( " GPL " ) version 2 as published by the Free
* Software Foundation .
*
* NO WARRANTY
* THIS SOFTWARE IS PROVIDED BY THE COPYRIGHT HOLDERS AND CONTRIBUTORS
* " AS IS " AND ANY EXPRESS OR IMPLIED WARRANTIES , INCLUDING , BUT NOT
* LIMITED TO , THE IMPLIED WARRANTIES OF MERCHANTIBILITY AND FITNESS FOR
* A PARTICULAR PURPOSE ARE DISCLAIMED . IN NO EVENT SHALL THE COPYRIGHT
* HOLDERS OR CONTRIBUTORS BE LIABLE FOR SPECIAL , EXEMPLARY , OR CONSEQUENTIAL
* DAMAGES ( INCLUDING , BUT NOT LIMITED TO , PROCUREMENT OF SUBSTITUTE GOODS
* OR SERVICES ; LOSS OF USE , DATA , OR PROFITS ; OR BUSINESS INTERRUPTION )
* HOWEVER CAUSED AND ON ANY THEORY OF LIABILITY , WHETHER IN CONTRACT ,
* STRICT LIABILITY , OR TORT ( INCLUDING NEGLIGENCE OR OTHERWISE ) ARISING
* IN ANY WAY OUT OF THE USE OF THIS SOFTWARE , EVEN IF ADVISED OF THE
* POSSIBILITY OF SUCH DAMAGES .
*/
2013-10-29 09:29:51 +08:00
# define EXPORT_ACPI_INTERFACES
2012-02-14 15:00:53 +08:00
# include <acpi/acpi.h>
# include "accommon.h"
# define _COMPONENT ACPI_HARDWARE
ACPI_MODULE_NAME ( " hwxfsleep " )
2012-02-14 18:57:13 +08:00
/* Local prototypes */
2015-07-01 14:43:18 +08:00
# if (!ACPI_REDUCED_HARDWARE)
static acpi_status
2016-01-04 22:05:20 +01:00
acpi_hw_set_firmware_waking_vector ( struct acpi_table_facs * facs ,
acpi_physical_address physical_address ,
acpi_physical_address physical_address64 ) ;
2015-07-01 14:43:18 +08:00
# endif
2012-10-31 02:25:45 +00:00
static acpi_status acpi_hw_sleep_dispatch ( u8 sleep_state , u32 function_id ) ;
2012-02-14 18:57:13 +08:00
/*
* Dispatch table used to efficiently branch to the various sleep
* functions .
*/
2012-03-21 10:01:49 +08:00
# define ACPI_SLEEP_FUNCTION_ID 0
# define ACPI_WAKE_PREP_FUNCTION_ID 1
# define ACPI_WAKE_FUNCTION_ID 2
2012-02-14 18:57:13 +08:00
/* Legacy functions are optional, based upon ACPI_REDUCED_HARDWARE */
static struct acpi_sleep_functions acpi_sleep_dispatch [ ] = {
{ ACPI_HW_OPTIONAL_FUNCTION ( acpi_hw_legacy_sleep ) ,
acpi_hw_extended_sleep } ,
{ ACPI_HW_OPTIONAL_FUNCTION ( acpi_hw_legacy_wake_prep ) ,
acpi_hw_extended_wake_prep } ,
{ ACPI_HW_OPTIONAL_FUNCTION ( acpi_hw_legacy_wake ) , acpi_hw_extended_wake }
} ;
/*
* These functions are removed for the ACPI_REDUCED_HARDWARE case :
* acpi_set_firmware_waking_vector
* acpi_enter_sleep_state_s4bios
*/
2012-02-14 18:14:27 +08:00
# if (!ACPI_REDUCED_HARDWARE)
2012-02-14 15:00:53 +08:00
/*******************************************************************************
*
2016-01-04 22:05:20 +01:00
* FUNCTION : acpi_hw_set_firmware_waking_vector
2012-02-14 15:00:53 +08:00
*
2015-07-01 14:43:18 +08:00
* PARAMETERS : facs - Pointer to FACS table
* physical_address - 32 - bit physical address of ACPI real mode
2016-01-04 22:05:20 +01:00
* entry point
2015-07-01 14:43:04 +08:00
* physical_address64 - 64 - bit physical address of ACPI protected
2016-01-04 22:05:20 +01:00
* mode entry point
2012-02-14 15:00:53 +08:00
*
* RETURN : Status
*
2015-07-01 14:43:04 +08:00
* DESCRIPTION : Sets the firmware_waking_vector fields of the FACS
2012-02-14 15:00:53 +08:00
*
* * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * */
2012-02-14 18:57:13 +08:00
2015-07-01 14:43:18 +08:00
static acpi_status
2016-01-04 22:05:20 +01:00
acpi_hw_set_firmware_waking_vector ( struct acpi_table_facs * facs ,
acpi_physical_address physical_address ,
acpi_physical_address physical_address64 )
2012-02-14 15:00:53 +08:00
{
2016-01-04 22:05:20 +01:00
ACPI_FUNCTION_TRACE ( acpi_hw_set_firmware_waking_vector ) ;
2012-02-14 15:00:53 +08:00
/*
* According to the ACPI specification 2.0 c and later , the 64 - bit
* waking vector should be cleared and the 32 - bit waking vector should
* be used , unless we want the wake - up code to be called by the BIOS in
* Protected Mode . Some systems ( for example HP dv5 - 1004 nr ) are known
* to fail to resume if the 64 - bit vector is used .
*/
/* Set the 32-bit vector */
2015-07-01 14:43:18 +08:00
facs - > firmware_waking_vector = ( u32 ) physical_address ;
2012-02-14 15:00:53 +08:00
2015-07-01 14:43:18 +08:00
if ( facs - > length > 32 ) {
if ( facs - > version > = 1 ) {
2012-02-14 15:00:53 +08:00
2015-07-01 14:43:04 +08:00
/* Set the 64-bit vector */
2015-07-01 14:43:18 +08:00
facs - > xfirmware_waking_vector = physical_address64 ;
2015-07-01 14:43:04 +08:00
} else {
/* Clear the 64-bit vector if it exists */
2015-07-01 14:43:18 +08:00
facs - > xfirmware_waking_vector = 0 ;
2015-07-01 14:43:04 +08:00
}
2012-02-14 15:00:53 +08:00
}
return_ACPI_STATUS ( AE_OK ) ;
}
2015-07-01 14:43:18 +08:00
/*******************************************************************************
*
2016-01-04 22:05:20 +01:00
* FUNCTION : acpi_set_firmware_waking_vector
2015-07-01 14:43:18 +08:00
*
* PARAMETERS : physical_address - 32 - bit physical address of ACPI real mode
2016-01-04 22:05:20 +01:00
* entry point
2015-07-01 14:43:18 +08:00
* physical_address64 - 64 - bit physical address of ACPI protected
2016-01-04 22:05:20 +01:00
* mode entry point
2015-07-01 14:43:18 +08:00
*
* RETURN : Status
*
* DESCRIPTION : Sets the firmware_waking_vector fields of the FACS
*
* * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * */
acpi_status
2016-01-04 22:05:20 +01:00
acpi_set_firmware_waking_vector ( acpi_physical_address physical_address ,
acpi_physical_address physical_address64 )
2015-07-01 14:43:18 +08:00
{
2016-01-04 22:05:20 +01:00
ACPI_FUNCTION_TRACE ( acpi_set_firmware_waking_vector ) ;
2015-07-01 14:43:18 +08:00
2015-08-25 10:29:08 +08:00
if ( acpi_gbl_FACS ) {
2016-01-04 22:05:20 +01:00
( void ) acpi_hw_set_firmware_waking_vector ( acpi_gbl_FACS ,
physical_address ,
physical_address64 ) ;
2015-07-01 14:43:18 +08:00
}
return_ACPI_STATUS ( AE_OK ) ;
}
2012-02-14 15:00:53 +08:00
ACPI_EXPORT_SYMBOL ( acpi_set_firmware_waking_vector )
/*******************************************************************************
*
* FUNCTION : acpi_enter_sleep_state_s4bios
*
* PARAMETERS : None
*
* RETURN : Status
*
* DESCRIPTION : Perform a S4 bios request .
* THIS FUNCTION MUST BE CALLED WITH INTERRUPTS DISABLED
*
* * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * */
2013-10-31 09:31:18 +08:00
acpi_status acpi_enter_sleep_state_s4bios ( void )
2012-02-14 15:00:53 +08:00
{
u32 in_value ;
acpi_status status ;
ACPI_FUNCTION_TRACE ( acpi_enter_sleep_state_s4bios ) ;
/* Clear the wake status bit (PM1) */
status =
acpi_write_bit_register ( ACPI_BITREG_WAKE_STATUS , ACPI_CLEAR_STATUS ) ;
if ( ACPI_FAILURE ( status ) ) {
return_ACPI_STATUS ( status ) ;
}
status = acpi_hw_clear_acpi_status ( ) ;
if ( ACPI_FAILURE ( status ) ) {
return_ACPI_STATUS ( status ) ;
}
/*
* 1 ) Disable / Clear all GPEs
* 2 ) Enable all wakeup GPEs
*/
status = acpi_hw_disable_all_gpes ( ) ;
if ( ACPI_FAILURE ( status ) ) {
return_ACPI_STATUS ( status ) ;
}
acpi_gbl_system_awake_and_running = FALSE ;
status = acpi_hw_enable_all_wakeup_gpes ( ) ;
if ( ACPI_FAILURE ( status ) ) {
return_ACPI_STATUS ( status ) ;
}
ACPI_FLUSH_CPU_CACHE ( ) ;
status = acpi_hw_write_port ( acpi_gbl_FADT . smi_command ,
2012-07-12 09:40:10 +08:00
( u32 ) acpi_gbl_FADT . s4_bios_request , 8 ) ;
2012-02-14 15:00:53 +08:00
do {
2012-12-31 00:05:46 +00:00
acpi_os_stall ( ACPI_USEC_PER_MSEC ) ;
2012-02-14 15:00:53 +08:00
status =
acpi_read_bit_register ( ACPI_BITREG_WAKE_STATUS , & in_value ) ;
if ( ACPI_FAILURE ( status ) ) {
return_ACPI_STATUS ( status ) ;
}
2015-12-29 13:54:36 +08:00
2012-02-14 15:00:53 +08:00
} while ( ! in_value ) ;
return_ACPI_STATUS ( AE_OK ) ;
}
ACPI_EXPORT_SYMBOL ( acpi_enter_sleep_state_s4bios )
2012-02-14 18:14:27 +08:00
# endif /* !ACPI_REDUCED_HARDWARE */
2012-02-14 18:57:13 +08:00
/*******************************************************************************
*
* FUNCTION : acpi_hw_sleep_dispatch
*
* PARAMETERS : sleep_state - Which sleep state to enter / exit
* function_id - Sleep , wake_prep , or Wake
*
* RETURN : Status from the invoked sleep handling function .
*
* DESCRIPTION : Dispatch a sleep / wake request to the appropriate handling
* function .
*
* * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * */
2012-10-31 02:25:45 +00:00
static acpi_status acpi_hw_sleep_dispatch ( u8 sleep_state , u32 function_id )
2012-02-14 18:57:13 +08:00
{
acpi_status status ;
struct acpi_sleep_functions * sleep_functions =
& acpi_sleep_dispatch [ function_id ] ;
# if (!ACPI_REDUCED_HARDWARE)
/*
* If the Hardware Reduced flag is set ( from the FADT ) , we must
2013-06-08 00:59:18 +00:00
* use the extended sleep registers ( FADT ) . Note : As per the ACPI
* specification , these extended registers are to be used for HW - reduced
* platforms only . They are not general - purpose replacements for the
* legacy PM register sleep support .
2012-02-14 18:57:13 +08:00
*/
2013-06-08 00:59:18 +00:00
if ( acpi_gbl_reduced_hardware ) {
2012-07-26 20:08:54 -04:00
status = sleep_functions - > extended_function ( sleep_state ) ;
2012-02-14 18:57:13 +08:00
} else {
/* Legacy sleep */
2012-07-26 20:08:54 -04:00
status = sleep_functions - > legacy_function ( sleep_state ) ;
2012-02-14 18:57:13 +08:00
}
return ( status ) ;
# else
/*
* For the case where reduced - hardware - only code is being generated ,
* we know that only the extended sleep registers are available
*/
2012-07-26 20:08:54 -04:00
status = sleep_functions - > extended_function ( sleep_state ) ;
2012-02-14 18:57:13 +08:00
return ( status ) ;
# endif /* !ACPI_REDUCED_HARDWARE */
}
2012-02-14 15:00:53 +08:00
/*******************************************************************************
*
* FUNCTION : acpi_enter_sleep_state_prep
*
* PARAMETERS : sleep_state - Which sleep state to enter
*
* RETURN : Status
*
2012-02-14 18:57:13 +08:00
* DESCRIPTION : Prepare to enter a system sleep state .
2012-02-14 15:00:53 +08:00
* This function must execute with interrupts enabled .
* We break sleeping into 2 stages so that OSPM can handle
* various OS - specific tasks between the two steps .
*
* * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * */
2012-02-14 18:57:13 +08:00
2012-02-14 15:00:53 +08:00
acpi_status acpi_enter_sleep_state_prep ( u8 sleep_state )
{
acpi_status status ;
struct acpi_object_list arg_list ;
union acpi_object arg ;
u32 sst_value ;
ACPI_FUNCTION_TRACE ( acpi_enter_sleep_state_prep ) ;
status = acpi_get_sleep_type_data ( sleep_state ,
& acpi_gbl_sleep_type_a ,
& acpi_gbl_sleep_type_b ) ;
if ( ACPI_FAILURE ( status ) ) {
return_ACPI_STATUS ( status ) ;
}
/* Execute the _PTS method (Prepare To Sleep) */
arg_list . count = 1 ;
arg_list . pointer = & arg ;
arg . type = ACPI_TYPE_INTEGER ;
arg . integer . value = sleep_state ;
2012-03-21 09:42:45 +08:00
status =
acpi_evaluate_object ( NULL , METHOD_PATHNAME__PTS , & arg_list , NULL ) ;
2012-02-14 15:00:53 +08:00
if ( ACPI_FAILURE ( status ) & & status ! = AE_NOT_FOUND ) {
return_ACPI_STATUS ( status ) ;
}
/* Setup the argument to the _SST method (System STatus) */
switch ( sleep_state ) {
case ACPI_STATE_S0 :
2013-06-08 00:58:14 +00:00
2012-02-14 15:00:53 +08:00
sst_value = ACPI_SST_WORKING ;
break ;
case ACPI_STATE_S1 :
case ACPI_STATE_S2 :
case ACPI_STATE_S3 :
2013-06-08 00:58:14 +00:00
2012-02-14 15:00:53 +08:00
sst_value = ACPI_SST_SLEEPING ;
break ;
case ACPI_STATE_S4 :
2013-06-08 00:58:14 +00:00
2012-02-14 15:00:53 +08:00
sst_value = ACPI_SST_SLEEP_CONTEXT ;
break ;
default :
2013-06-08 00:58:14 +00:00
2012-02-14 15:00:53 +08:00
sst_value = ACPI_SST_INDICATOR_OFF ; /* Default is off */
break ;
}
/*
* Set the system indicators to show the desired sleep state .
* _SST is an optional method ( return no error if not found )
*/
2012-03-21 09:42:45 +08:00
acpi_hw_execute_sleep_method ( METHOD_PATHNAME__SST , sst_value ) ;
2012-02-14 15:00:53 +08:00
return_ACPI_STATUS ( AE_OK ) ;
}
ACPI_EXPORT_SYMBOL ( acpi_enter_sleep_state_prep )
/*******************************************************************************
*
* FUNCTION : acpi_enter_sleep_state
*
* PARAMETERS : sleep_state - Which sleep state to enter
*
* RETURN : Status
*
2012-12-19 05:36:49 +00:00
* DESCRIPTION : Enter a system sleep state
2012-02-14 15:00:53 +08:00
* THIS FUNCTION MUST BE CALLED WITH INTERRUPTS DISABLED
*
* * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * */
2013-10-31 09:31:18 +08:00
acpi_status acpi_enter_sleep_state ( u8 sleep_state )
2012-02-14 15:00:53 +08:00
{
acpi_status status ;
ACPI_FUNCTION_TRACE ( acpi_enter_sleep_state ) ;
if ( ( acpi_gbl_sleep_type_a > ACPI_SLEEP_TYPE_MAX ) | |
( acpi_gbl_sleep_type_b > ACPI_SLEEP_TYPE_MAX ) ) {
ACPI_ERROR ( ( AE_INFO , " Sleep values out of range: A=0x%X B=0x%X " ,
acpi_gbl_sleep_type_a , acpi_gbl_sleep_type_b ) ) ;
return_ACPI_STATUS ( AE_AML_OPERAND_VALUE ) ;
}
2012-02-14 18:14:27 +08:00
2012-10-31 02:25:45 +00:00
status = acpi_hw_sleep_dispatch ( sleep_state , ACPI_SLEEP_FUNCTION_ID ) ;
2012-02-14 15:00:53 +08:00
return_ACPI_STATUS ( status ) ;
}
ACPI_EXPORT_SYMBOL ( acpi_enter_sleep_state )
/*******************************************************************************
*
* FUNCTION : acpi_leave_sleep_state_prep
*
* PARAMETERS : sleep_state - Which sleep state we are exiting
*
* RETURN : Status
*
* DESCRIPTION : Perform the first state of OS - independent ACPI cleanup after a
2012-12-19 05:36:49 +00:00
* sleep . Called with interrupts DISABLED .
* We break wake / resume into 2 stages so that OSPM can handle
* various OS - specific tasks between the two steps .
2012-02-14 15:00:53 +08:00
*
* * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * */
2012-07-26 20:08:54 -04:00
acpi_status acpi_leave_sleep_state_prep ( u8 sleep_state )
2012-02-14 15:00:53 +08:00
{
acpi_status status ;
2012-02-14 18:57:13 +08:00
ACPI_FUNCTION_TRACE ( acpi_leave_sleep_state_prep ) ;
2012-02-14 15:00:53 +08:00
2012-03-21 10:01:49 +08:00
status =
2012-10-31 02:25:45 +00:00
acpi_hw_sleep_dispatch ( sleep_state , ACPI_WAKE_PREP_FUNCTION_ID ) ;
2012-02-14 15:00:53 +08:00
return_ACPI_STATUS ( status ) ;
}
ACPI_EXPORT_SYMBOL ( acpi_leave_sleep_state_prep )
/*******************************************************************************
*
* FUNCTION : acpi_leave_sleep_state
*
2012-02-14 18:57:13 +08:00
* PARAMETERS : sleep_state - Which sleep state we are exiting
2012-02-14 15:00:53 +08:00
*
* RETURN : Status
*
* DESCRIPTION : Perform OS - independent ACPI cleanup after a sleep
* Called with interrupts ENABLED .
*
* * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * */
acpi_status acpi_leave_sleep_state ( u8 sleep_state )
{
acpi_status status ;
ACPI_FUNCTION_TRACE ( acpi_leave_sleep_state ) ;
2012-07-26 20:08:54 -04:00
status = acpi_hw_sleep_dispatch ( sleep_state , ACPI_WAKE_FUNCTION_ID ) ;
2012-02-14 15:00:53 +08:00
return_ACPI_STATUS ( status ) ;
}
ACPI_EXPORT_SYMBOL ( acpi_leave_sleep_state )