2019-05-27 08:55:21 +02:00
// SPDX-License-Identifier: GPL-2.0-only
2015-07-31 15:46:16 +01:00
/*
*
* Copyright ( C ) 2015 ARM Limited
*/
# define pr_fmt(fmt) "psci: " fmt
2016-07-19 18:52:57 +01:00
# include <linux/acpi.h>
2016-01-04 15:46:47 +01:00
# include <linux/arm-smccc.h>
2016-02-01 18:01:30 +01:00
# include <linux/cpuidle.h>
2022-09-26 14:07:58 +03:00
# include <linux/debugfs.h>
2015-07-31 15:46:16 +01:00
# include <linux/errno.h>
# include <linux/linkage.h>
# include <linux/of.h>
# include <linux/pm.h>
# include <linux/printk.h>
# include <linux/psci.h>
# include <linux/reboot.h>
2016-02-01 18:01:30 +01:00
# include <linux/slab.h>
2015-06-18 15:41:34 +01:00
# include <linux/suspend.h>
2015-07-31 15:46:16 +01:00
# include <uapi/linux/psci.h>
2016-02-01 18:01:30 +01:00
# include <asm/cpuidle.h>
2015-07-31 15:46:16 +01:00
# include <asm/cputype.h>
2020-12-09 14:09:24 +08:00
# include <asm/hypervisor.h>
2015-07-31 15:46:16 +01:00
# include <asm/system_misc.h>
# include <asm/smp_plat.h>
2015-06-18 15:41:34 +01:00
# include <asm/suspend.h>
2015-07-31 15:46:16 +01:00
2015-07-31 15:46:17 +01:00
/*
* While a 64 - bit OS can make calls with SMC32 calling conventions , for some
2015-06-18 15:41:33 +01:00
* calls it is necessary to use SMC64 to pass or return 64 - bit values .
* For such calls PSCI_FN_NATIVE ( version , name ) will choose the appropriate
* ( native - width ) function ID .
2015-07-31 15:46:17 +01:00
*/
# ifdef CONFIG_64BIT
2015-06-18 15:41:33 +01:00
# define PSCI_FN_NATIVE(version, name) PSCI_##version##_FN64_##name
2015-07-31 15:46:17 +01:00
# else
2015-06-18 15:41:33 +01:00
# define PSCI_FN_NATIVE(version, name) PSCI_##version##_FN_##name
2015-07-31 15:46:17 +01:00
# endif
2015-07-31 15:46:16 +01:00
/*
* The CPU any Trusted OS is resident on . The trusted OS may reject CPU_OFF
* calls to its resident CPU , so we must avoid issuing those . We never migrate
* a Trusted OS even if it claims to be capable of migration - - doing so will
* require cooperation with a Trusted OS driver .
*/
static int resident_cpu = - 1 ;
2020-05-18 10:12:20 +01:00
struct psci_operations psci_ops ;
static enum arm_smccc_conduit psci_conduit = SMCCC_CONDUIT_NONE ;
2015-07-31 15:46:16 +01:00
bool psci_tos_resident_on ( int cpu )
{
return cpu = = resident_cpu ;
}
typedef unsigned long ( psci_fn ) ( unsigned long , unsigned long ,
unsigned long , unsigned long ) ;
static psci_fn * invoke_psci_fn ;
2020-12-02 18:41:01 +00:00
static struct psci_0_1_function_ids psci_0_1_function_ids ;
2015-07-31 15:46:16 +01:00
2020-12-02 18:41:02 +00:00
struct psci_0_1_function_ids get_psci_0_1_function_ids ( void )
{
return psci_0_1_function_ids ;
}
2015-05-26 16:49:01 +01:00
# define PSCI_0_2_POWER_STATE_MASK \
( PSCI_0_2_POWER_STATE_ID_MASK | \
PSCI_0_2_POWER_STATE_TYPE_MASK | \
PSCI_0_2_POWER_STATE_AFFL_MASK )
2015-05-26 17:10:32 +01:00
# define PSCI_1_0_EXT_POWER_STATE_MASK \
( PSCI_1_0_EXT_POWER_STATE_ID_MASK | \
PSCI_1_0_EXT_POWER_STATE_TYPE_MASK )
static u32 psci_cpu_suspend_feature ;
2019-04-15 12:47:46 +01:00
static bool psci_system_reset2_supported ;
2015-05-26 17:10:32 +01:00
static inline bool psci_has_ext_power_state ( void )
{
return psci_cpu_suspend_feature &
PSCI_1_0_FEATURES_CPU_SUSPEND_PF_MASK ;
}
2019-10-10 12:01:48 +02:00
bool psci_has_osi_support ( void )
2019-04-10 10:20:25 +02:00
{
return psci_cpu_suspend_feature & PSCI_1_0_OS_INITIATED ;
}
2016-04-26 02:13:30 +02:00
static inline bool psci_power_state_loses_context ( u32 state )
2015-05-26 16:49:01 +01:00
{
2015-05-26 17:10:32 +01:00
const u32 mask = psci_has_ext_power_state ( ) ?
PSCI_1_0_EXT_POWER_STATE_TYPE_MASK :
PSCI_0_2_POWER_STATE_TYPE_MASK ;
return state & mask ;
2015-05-26 16:49:01 +01:00
}
2019-08-09 12:03:12 +01:00
bool psci_power_state_is_valid ( u32 state )
2015-05-26 16:49:01 +01:00
{
2015-05-26 17:10:32 +01:00
const u32 valid_mask = psci_has_ext_power_state ( ) ?
PSCI_1_0_EXT_POWER_STATE_MASK :
PSCI_0_2_POWER_STATE_MASK ;
return ! ( state & ~ valid_mask ) ;
2015-05-26 16:49:01 +01:00
}
2016-01-04 15:46:47 +01:00
static unsigned long __invoke_psci_fn_hvc ( unsigned long function_id ,
unsigned long arg0 , unsigned long arg1 ,
unsigned long arg2 )
{
struct arm_smccc_res res ;
arm_smccc_hvc ( function_id , arg0 , arg1 , arg2 , 0 , 0 , 0 , 0 , & res ) ;
return res . a0 ;
}
static unsigned long __invoke_psci_fn_smc ( unsigned long function_id ,
unsigned long arg0 , unsigned long arg1 ,
unsigned long arg2 )
{
struct arm_smccc_res res ;
arm_smccc_smc ( function_id , arg0 , arg1 , arg2 , 0 , 0 , 0 , 0 , & res ) ;
return res . a0 ;
}
2015-07-31 15:46:16 +01:00
static int psci_to_linux_errno ( int errno )
{
switch ( errno ) {
case PSCI_RET_SUCCESS :
return 0 ;
case PSCI_RET_NOT_SUPPORTED :
return - EOPNOTSUPP ;
case PSCI_RET_INVALID_PARAMS :
2015-05-22 14:31:37 +01:00
case PSCI_RET_INVALID_ADDRESS :
2015-07-31 15:46:16 +01:00
return - EINVAL ;
case PSCI_RET_DENIED :
return - EPERM ;
2021-04-29 09:26:29 +08:00
}
2015-07-31 15:46:16 +01:00
return - EINVAL ;
}
2020-12-02 18:40:59 +00:00
static u32 psci_0_1_get_version ( void )
{
return PSCI_VERSION ( 0 , 1 ) ;
}
static u32 psci_0_2_get_version ( void )
2015-07-31 15:46:16 +01:00
{
return invoke_psci_fn ( PSCI_0_2_FN_PSCI_VERSION , 0 , 0 , 0 ) ;
}
2020-09-01 16:28:58 +02:00
int psci_set_osi_mode ( bool enable )
2019-10-10 12:01:48 +02:00
{
2020-09-01 16:28:58 +02:00
unsigned long suspend_mode ;
2019-10-10 12:01:48 +02:00
int err ;
2020-09-01 16:28:58 +02:00
suspend_mode = enable ? PSCI_1_0_SUSPEND_MODE_OSI :
PSCI_1_0_SUSPEND_MODE_PC ;
err = invoke_psci_fn ( PSCI_1_0_FN_SET_SUSPEND_MODE , suspend_mode , 0 , 0 ) ;
2022-09-26 14:02:49 +03:00
if ( err < 0 )
pr_warn ( " failed to set %s mode: %d \n " , enable ? " OSI " : " PC " , err ) ;
2019-10-10 12:01:48 +02:00
return psci_to_linux_errno ( err ) ;
}
2020-12-02 18:41:00 +00:00
static int __psci_cpu_suspend ( u32 fn , u32 state , unsigned long entry_point )
2015-07-31 15:46:16 +01:00
{
int err ;
err = invoke_psci_fn ( fn , state , entry_point , 0 ) ;
return psci_to_linux_errno ( err ) ;
}
2020-12-02 18:41:00 +00:00
static int psci_0_1_cpu_suspend ( u32 state , unsigned long entry_point )
{
2020-12-02 18:41:01 +00:00
return __psci_cpu_suspend ( psci_0_1_function_ids . cpu_suspend ,
2020-12-02 18:41:00 +00:00
state , entry_point ) ;
}
static int psci_0_2_cpu_suspend ( u32 state , unsigned long entry_point )
{
return __psci_cpu_suspend ( PSCI_FN_NATIVE ( 0 _2 , CPU_SUSPEND ) ,
state , entry_point ) ;
}
static int __psci_cpu_off ( u32 fn , u32 state )
2015-07-31 15:46:16 +01:00
{
int err ;
err = invoke_psci_fn ( fn , state , 0 , 0 ) ;
return psci_to_linux_errno ( err ) ;
}
2020-12-02 18:41:00 +00:00
static int psci_0_1_cpu_off ( u32 state )
{
2020-12-02 18:41:01 +00:00
return __psci_cpu_off ( psci_0_1_function_ids . cpu_off , state ) ;
2020-12-02 18:41:00 +00:00
}
static int psci_0_2_cpu_off ( u32 state )
{
return __psci_cpu_off ( PSCI_0_2_FN_CPU_OFF , state ) ;
}
static int __psci_cpu_on ( u32 fn , unsigned long cpuid , unsigned long entry_point )
2015-07-31 15:46:16 +01:00
{
int err ;
err = invoke_psci_fn ( fn , cpuid , entry_point , 0 ) ;
return psci_to_linux_errno ( err ) ;
}
2020-12-02 18:41:00 +00:00
static int psci_0_1_cpu_on ( unsigned long cpuid , unsigned long entry_point )
{
2020-12-02 18:41:01 +00:00
return __psci_cpu_on ( psci_0_1_function_ids . cpu_on , cpuid , entry_point ) ;
2020-12-02 18:41:00 +00:00
}
static int psci_0_2_cpu_on ( unsigned long cpuid , unsigned long entry_point )
{
return __psci_cpu_on ( PSCI_FN_NATIVE ( 0 _2 , CPU_ON ) , cpuid , entry_point ) ;
}
static int __psci_migrate ( u32 fn , unsigned long cpuid )
2015-07-31 15:46:16 +01:00
{
int err ;
err = invoke_psci_fn ( fn , cpuid , 0 , 0 ) ;
return psci_to_linux_errno ( err ) ;
}
2020-12-02 18:41:00 +00:00
static int psci_0_1_migrate ( unsigned long cpuid )
{
2020-12-02 18:41:01 +00:00
return __psci_migrate ( psci_0_1_function_ids . migrate , cpuid ) ;
2020-12-02 18:41:00 +00:00
}
static int psci_0_2_migrate ( unsigned long cpuid )
{
return __psci_migrate ( PSCI_FN_NATIVE ( 0 _2 , MIGRATE ) , cpuid ) ;
}
2015-07-31 15:46:16 +01:00
static int psci_affinity_info ( unsigned long target_affinity ,
unsigned long lowest_affinity_level )
{
2015-06-18 15:41:33 +01:00
return invoke_psci_fn ( PSCI_FN_NATIVE ( 0 _2 , AFFINITY_INFO ) ,
2015-07-31 15:46:17 +01:00
target_affinity , lowest_affinity_level , 0 ) ;
2015-07-31 15:46:16 +01:00
}
static int psci_migrate_info_type ( void )
{
return invoke_psci_fn ( PSCI_0_2_FN_MIGRATE_INFO_TYPE , 0 , 0 , 0 ) ;
}
static unsigned long psci_migrate_info_up_cpu ( void )
{
2015-06-18 15:41:33 +01:00
return invoke_psci_fn ( PSCI_FN_NATIVE ( 0 _2 , MIGRATE_INFO_UP_CPU ) ,
2015-07-31 15:46:17 +01:00
0 , 0 , 0 ) ;
2015-07-31 15:46:16 +01:00
}
2019-08-09 14:22:43 +01:00
static void set_conduit ( enum arm_smccc_conduit conduit )
2018-02-06 17:56:16 +00:00
{
switch ( conduit ) {
2019-08-09 14:22:43 +01:00
case SMCCC_CONDUIT_HVC :
2018-02-06 17:56:16 +00:00
invoke_psci_fn = __invoke_psci_fn_hvc ;
break ;
2019-08-09 14:22:43 +01:00
case SMCCC_CONDUIT_SMC :
2018-02-06 17:56:16 +00:00
invoke_psci_fn = __invoke_psci_fn_smc ;
break ;
default :
WARN ( 1 , " Unexpected PSCI conduit %d \n " , conduit ) ;
}
2020-05-18 10:12:20 +01:00
psci_conduit = conduit ;
2018-02-06 17:56:16 +00:00
}
2022-09-08 14:54:50 -07:00
static int get_set_conduit_method ( const struct device_node * np )
2015-07-31 15:46:16 +01:00
{
const char * method ;
pr_info ( " probing for conduit method from DT. \n " ) ;
if ( of_property_read_string ( np , " method " , & method ) ) {
pr_warn ( " missing \" method \" property \n " ) ;
return - ENXIO ;
}
if ( ! strcmp ( " hvc " , method ) ) {
2019-08-09 14:22:43 +01:00
set_conduit ( SMCCC_CONDUIT_HVC ) ;
2015-07-31 15:46:16 +01:00
} else if ( ! strcmp ( " smc " , method ) ) {
2019-08-09 14:22:43 +01:00
set_conduit ( SMCCC_CONDUIT_SMC ) ;
2015-07-31 15:46:16 +01:00
} else {
pr_warn ( " invalid \" method \" property: %s \n " , method ) ;
return - EINVAL ;
}
return 0 ;
}
2021-06-04 15:07:34 +01:00
static int psci_sys_reset ( struct notifier_block * nb , unsigned long action ,
void * data )
2015-07-31 15:46:16 +01:00
{
2019-04-15 12:47:46 +01:00
if ( ( reboot_mode = = REBOOT_WARM | | reboot_mode = = REBOOT_SOFT ) & &
psci_system_reset2_supported ) {
/*
* reset_type [ 31 ] = 0 ( architectural )
* reset_type [ 30 : 0 ] = 0 ( SYSTEM_WARM_RESET )
* cookie = 0 ( ignored by the implementation )
*/
invoke_psci_fn ( PSCI_FN_NATIVE ( 1 _1 , SYSTEM_RESET2 ) , 0 , 0 , 0 ) ;
} else {
invoke_psci_fn ( PSCI_0_2_FN_SYSTEM_RESET , 0 , 0 , 0 ) ;
}
2021-06-04 15:07:34 +01:00
return NOTIFY_DONE ;
2015-07-31 15:46:16 +01:00
}
2021-06-04 15:07:34 +01:00
static struct notifier_block psci_sys_reset_nb = {
. notifier_call = psci_sys_reset ,
. priority = 129 ,
} ;
2015-07-31 15:46:16 +01:00
static void psci_sys_poweroff ( void )
{
invoke_psci_fn ( PSCI_0_2_FN_SYSTEM_OFF , 0 , 0 , 0 ) ;
}
2022-09-26 14:07:58 +03:00
static int psci_features ( u32 psci_func_id )
2015-05-26 17:06:21 +01:00
{
return invoke_psci_fn ( PSCI_1_0_FN_PSCI_FEATURES ,
psci_func_id , 0 , 0 ) ;
}
2022-09-26 14:07:58 +03:00
# ifdef CONFIG_DEBUG_FS
# define PSCI_ID(ver, _name) \
{ . fn = PSCI_ # # ver # # _FN_ # # _name , . name = # _name , }
# define PSCI_ID_NATIVE(ver, _name) \
{ . fn = PSCI_FN_NATIVE ( ver , _name ) , . name = # _name , }
/* A table of all optional functions */
static const struct {
u32 fn ;
const char * name ;
} psci_fn_ids [ ] = {
PSCI_ID_NATIVE ( 0 _2 , MIGRATE ) ,
PSCI_ID ( 0 _2 , MIGRATE_INFO_TYPE ) ,
PSCI_ID_NATIVE ( 0 _2 , MIGRATE_INFO_UP_CPU ) ,
PSCI_ID ( 1 _0 , CPU_FREEZE ) ,
PSCI_ID_NATIVE ( 1 _0 , CPU_DEFAULT_SUSPEND ) ,
PSCI_ID_NATIVE ( 1 _0 , NODE_HW_STATE ) ,
PSCI_ID_NATIVE ( 1 _0 , SYSTEM_SUSPEND ) ,
PSCI_ID ( 1 _0 , SET_SUSPEND_MODE ) ,
PSCI_ID_NATIVE ( 1 _0 , STAT_RESIDENCY ) ,
PSCI_ID_NATIVE ( 1 _0 , STAT_COUNT ) ,
PSCI_ID_NATIVE ( 1 _1 , SYSTEM_RESET2 ) ,
PSCI_ID ( 1 _1 , MEM_PROTECT ) ,
PSCI_ID_NATIVE ( 1 _1 , MEM_PROTECT_CHECK_RANGE ) ,
} ;
static int psci_debugfs_read ( struct seq_file * s , void * data )
{
int feature , type , i ;
u32 ver ;
ver = psci_ops . get_version ( ) ;
seq_printf ( s , " PSCIv%d.%d \n " ,
PSCI_VERSION_MAJOR ( ver ) ,
PSCI_VERSION_MINOR ( ver ) ) ;
/* PSCI_FEATURES is available only starting from 1.0 */
if ( PSCI_VERSION_MAJOR ( ver ) < 1 )
return 0 ;
feature = psci_features ( ARM_SMCCC_VERSION_FUNC_ID ) ;
if ( feature ! = PSCI_RET_NOT_SUPPORTED ) {
ver = invoke_psci_fn ( ARM_SMCCC_VERSION_FUNC_ID , 0 , 0 , 0 ) ;
seq_printf ( s , " SMC Calling Convention v%d.%d \n " ,
PSCI_VERSION_MAJOR ( ver ) ,
PSCI_VERSION_MINOR ( ver ) ) ;
} else {
seq_puts ( s , " SMC Calling Convention v1.0 is assumed \n " ) ;
}
feature = psci_features ( PSCI_FN_NATIVE ( 0 _2 , CPU_SUSPEND ) ) ;
if ( feature < 0 ) {
seq_printf ( s , " PSCI_FEATURES(CPU_SUSPEND) error (%d) \n " , feature ) ;
} else {
seq_printf ( s , " OSI is %ssupported \n " ,
( feature & BIT ( 0 ) ) ? " " : " not " ) ;
seq_printf ( s , " %s StateID format is used \n " ,
( feature & BIT ( 1 ) ) ? " Extended " : " Original " ) ;
}
type = psci_ops . migrate_info_type ( ) ;
if ( type = = PSCI_0_2_TOS_UP_MIGRATE | |
type = = PSCI_0_2_TOS_UP_NO_MIGRATE ) {
unsigned long cpuid ;
seq_printf ( s , " Trusted OS %smigrate capable \n " ,
type = = PSCI_0_2_TOS_UP_NO_MIGRATE ? " not " : " " ) ;
cpuid = psci_migrate_info_up_cpu ( ) ;
seq_printf ( s , " Trusted OS resident on physical CPU 0x%lx (#%d) \n " ,
cpuid , resident_cpu ) ;
} else if ( type = = PSCI_0_2_TOS_MP ) {
seq_puts ( s , " Trusted OS migration not required \n " ) ;
} else {
if ( type ! = PSCI_RET_NOT_SUPPORTED )
seq_printf ( s , " MIGRATE_INFO_TYPE returned unknown type (%d) \n " , type ) ;
}
for ( i = 0 ; i < ARRAY_SIZE ( psci_fn_ids ) ; i + + ) {
feature = psci_features ( psci_fn_ids [ i ] . fn ) ;
if ( feature = = PSCI_RET_NOT_SUPPORTED )
continue ;
if ( feature < 0 )
seq_printf ( s , " PSCI_FEATURES(%s) error (%d) \n " ,
psci_fn_ids [ i ] . name , feature ) ;
else
seq_printf ( s , " %s is supported \n " , psci_fn_ids [ i ] . name ) ;
}
return 0 ;
}
static int psci_debugfs_open ( struct inode * inode , struct file * f )
{
return single_open ( f , psci_debugfs_read , NULL ) ;
}
static const struct file_operations psci_debugfs_ops = {
. owner = THIS_MODULE ,
. open = psci_debugfs_open ,
. release = single_release ,
. read = seq_read ,
. llseek = seq_lseek
} ;
static int __init psci_debugfs_init ( void )
{
return PTR_ERR_OR_ZERO ( debugfs_create_file ( " psci " , 0444 , NULL , NULL ,
& psci_debugfs_ops ) ) ;
}
late_initcall ( psci_debugfs_init )
# endif
2016-02-01 18:01:30 +01:00
# ifdef CONFIG_CPU_IDLE
2019-08-09 12:03:12 +01:00
static int psci_suspend_finisher ( unsigned long state )
2019-04-10 10:20:23 +02:00
{
2019-08-09 12:03:12 +01:00
u32 power_state = state ;
2022-09-08 14:54:55 -07:00
phys_addr_t pa_cpu_resume = __pa_symbol ( cpu_resume ) ;
2019-04-10 10:20:23 +02:00
2021-04-08 11:28:36 -07:00
return psci_ops . cpu_suspend ( power_state , pa_cpu_resume ) ;
2019-04-10 10:20:23 +02:00
}
2019-08-09 12:03:12 +01:00
int psci_cpu_suspend_enter ( u32 state )
2016-02-01 18:01:30 +01:00
{
int ret ;
2021-06-15 12:12:26 +01:00
if ( ! psci_power_state_loses_context ( state ) ) {
struct arm_cpuidle_irq_context context ;
2023-01-17 11:26:29 +01:00
ct_cpuidle_enter ( ) ;
2021-06-15 12:12:26 +01:00
arm_cpuidle_save_irq_context ( & context ) ;
2019-08-09 12:03:12 +01:00
ret = psci_ops . cpu_suspend ( state , 0 ) ;
2021-06-15 12:12:26 +01:00
arm_cpuidle_restore_irq_context ( & context ) ;
2023-01-17 11:26:29 +01:00
ct_cpuidle_exit ( ) ;
2021-06-15 12:12:26 +01:00
} else {
2023-01-17 11:26:29 +01:00
/*
* ARM64 cpu_suspend ( ) wants to do ct_cpuidle_ * ( ) itself .
*/
if ( ! IS_ENABLED ( CONFIG_ARM64 ) )
ct_cpuidle_enter ( ) ;
2019-08-09 12:03:12 +01:00
ret = cpu_suspend ( state , psci_suspend_finisher ) ;
2023-01-17 11:26:29 +01:00
if ( ! IS_ENABLED ( CONFIG_ARM64 ) )
ct_cpuidle_exit ( ) ;
2021-06-15 12:12:26 +01:00
}
2016-02-01 18:01:30 +01:00
return ret ;
}
# endif
2015-06-18 15:41:34 +01:00
static int psci_system_suspend ( unsigned long unused )
{
2022-09-08 14:54:55 -07:00
phys_addr_t pa_cpu_resume = __pa_symbol ( cpu_resume ) ;
2021-04-08 11:28:36 -07:00
2015-06-18 15:41:34 +01:00
return invoke_psci_fn ( PSCI_FN_NATIVE ( 1 _0 , SYSTEM_SUSPEND ) ,
2021-04-08 11:28:36 -07:00
pa_cpu_resume , 0 , 0 ) ;
2015-06-18 15:41:34 +01:00
}
static int psci_system_suspend_enter ( suspend_state_t state )
{
return cpu_suspend ( 0 , psci_system_suspend ) ;
}
static const struct platform_suspend_ops psci_suspend_ops = {
. valid = suspend_valid_only_mem ,
. enter = psci_system_suspend_enter ,
} ;
2019-04-15 12:47:46 +01:00
static void __init psci_init_system_reset2 ( void )
{
int ret ;
ret = psci_features ( PSCI_FN_NATIVE ( 1 _1 , SYSTEM_RESET2 ) ) ;
if ( ret ! = PSCI_RET_NOT_SUPPORTED )
psci_system_reset2_supported = true ;
}
2015-06-18 15:41:34 +01:00
static void __init psci_init_system_suspend ( void )
{
int ret ;
if ( ! IS_ENABLED ( CONFIG_SUSPEND ) )
return ;
ret = psci_features ( PSCI_FN_NATIVE ( 1 _0 , SYSTEM_SUSPEND ) ) ;
if ( ret ! = PSCI_RET_NOT_SUPPORTED )
suspend_set_ops ( & psci_suspend_ops ) ;
}
2015-05-26 17:10:32 +01:00
static void __init psci_init_cpu_suspend ( void )
{
2020-12-02 18:41:00 +00:00
int feature = psci_features ( PSCI_FN_NATIVE ( 0 _2 , CPU_SUSPEND ) ) ;
2015-05-26 17:10:32 +01:00
if ( feature ! = PSCI_RET_NOT_SUPPORTED )
psci_cpu_suspend_feature = feature ;
}
2015-07-31 15:46:16 +01:00
/*
* Detect the presence of a resident Trusted OS which may cause CPU_OFF to
* return DENIED ( which would be fatal ) .
*/
static void __init psci_init_migrate ( void )
{
unsigned long cpuid ;
int type , cpu = - 1 ;
type = psci_ops . migrate_info_type ( ) ;
if ( type = = PSCI_0_2_TOS_MP ) {
pr_info ( " Trusted OS migration not required \n " ) ;
return ;
}
if ( type = = PSCI_RET_NOT_SUPPORTED ) {
pr_info ( " MIGRATE_INFO_TYPE not supported. \n " ) ;
return ;
}
if ( type ! = PSCI_0_2_TOS_UP_MIGRATE & &
type ! = PSCI_0_2_TOS_UP_NO_MIGRATE ) {
pr_err ( " MIGRATE_INFO_TYPE returned unknown type (%d) \n " , type ) ;
return ;
}
cpuid = psci_migrate_info_up_cpu ( ) ;
if ( cpuid & ~ MPIDR_HWID_BITMASK ) {
pr_warn ( " MIGRATE_INFO_UP_CPU reported invalid physical ID (0x%lx) \n " ,
cpuid ) ;
return ;
}
cpu = get_logical_index ( cpuid ) ;
resident_cpu = cpu > = 0 ? cpu : - 1 ;
pr_info ( " Trusted OS resident on physical CPU 0x%lx \n " , cpuid ) ;
}
2018-02-06 17:56:17 +00:00
static void __init psci_init_smccc ( void )
{
u32 ver = ARM_SMCCC_VERSION_1_0 ;
int feature ;
feature = psci_features ( ARM_SMCCC_VERSION_FUNC_ID ) ;
if ( feature ! = PSCI_RET_NOT_SUPPORTED ) {
u32 ret ;
ret = invoke_psci_fn ( ARM_SMCCC_VERSION_FUNC_ID , 0 , 0 , 0 ) ;
2020-05-18 10:12:19 +01:00
if ( ret > = ARM_SMCCC_VERSION_1_1 ) {
2020-05-18 10:12:20 +01:00
arm_smccc_version_init ( ret , psci_conduit ) ;
2018-02-06 17:56:17 +00:00
ver = ret ;
}
}
/*
* Conveniently , the SMCCC and PSCI versions are encoded the
* same way . No , this isn ' t accidental .
*/
pr_info ( " SMC Calling Convention v%d.%d \n " ,
PSCI_VERSION_MAJOR ( ver ) , PSCI_VERSION_MINOR ( ver ) ) ;
}
2015-07-31 15:46:16 +01:00
static void __init psci_0_2_set_functions ( void )
{
pr_info ( " Using standard PSCI v0.2 function IDs \n " ) ;
2020-12-02 18:41:00 +00:00
psci_ops = ( struct psci_operations ) {
. get_version = psci_0_2_get_version ,
. cpu_suspend = psci_0_2_cpu_suspend ,
. cpu_off = psci_0_2_cpu_off ,
. cpu_on = psci_0_2_cpu_on ,
. migrate = psci_0_2_migrate ,
. affinity_info = psci_affinity_info ,
. migrate_info_type = psci_migrate_info_type ,
} ;
2015-07-31 15:46:16 +01:00
2021-06-04 15:07:34 +01:00
register_restart_handler ( & psci_sys_reset_nb ) ;
2015-07-31 15:46:16 +01:00
pm_power_off = psci_sys_poweroff ;
}
/*
* Probe function for PSCI firmware versions > = 0.2
*/
static int __init psci_probe ( void )
{
2020-12-02 18:40:59 +00:00
u32 ver = psci_0_2_get_version ( ) ;
2015-07-31 15:46:16 +01:00
pr_info ( " PSCIv%d.%d detected in firmware. \n " ,
PSCI_VERSION_MAJOR ( ver ) ,
PSCI_VERSION_MINOR ( ver ) ) ;
if ( PSCI_VERSION_MAJOR ( ver ) = = 0 & & PSCI_VERSION_MINOR ( ver ) < 2 ) {
pr_err ( " Conflicting PSCI version detected. \n " ) ;
return - EINVAL ;
}
psci_0_2_set_functions ( ) ;
psci_init_migrate ( ) ;
2015-10-23 15:46:50 +01:00
if ( PSCI_VERSION_MAJOR ( ver ) > = 1 ) {
2018-02-06 17:56:17 +00:00
psci_init_smccc ( ) ;
2015-10-23 15:46:50 +01:00
psci_init_cpu_suspend ( ) ;
psci_init_system_suspend ( ) ;
2019-04-15 12:47:46 +01:00
psci_init_system_reset2 ( ) ;
2020-12-09 14:09:24 +08:00
kvm_init_hyp_services ( ) ;
2015-10-23 15:46:50 +01:00
}
2015-06-18 15:41:34 +01:00
2015-07-31 15:46:16 +01:00
return 0 ;
}
typedef int ( * psci_initcall_t ) ( const struct device_node * ) ;
/*
* PSCI init function for PSCI versions > = 0.2
*
* Probe based on PSCI PSCI_VERSION function
*/
2022-09-08 14:54:50 -07:00
static int __init psci_0_2_init ( const struct device_node * np )
2015-07-31 15:46:16 +01:00
{
int err ;
err = get_set_conduit_method ( np ) ;
if ( err )
2019-04-10 10:20:24 +02:00
return err ;
2015-07-31 15:46:16 +01:00
/*
* Starting with v0 .2 , the PSCI specification introduced a call
* ( PSCI_VERSION ) that allows probing the firmware version , so
* that PSCI function IDs and version specific initialization
* can be carried out according to the specific version reported
* by firmware
*/
2019-04-10 10:20:24 +02:00
return psci_probe ( ) ;
2015-07-31 15:46:16 +01:00
}
/*
* PSCI < v0 .2 get PSCI Function IDs via DT .
*/
2022-09-08 14:54:50 -07:00
static int __init psci_0_1_init ( const struct device_node * np )
2015-07-31 15:46:16 +01:00
{
u32 id ;
int err ;
err = get_set_conduit_method ( np ) ;
if ( err )
2019-04-10 10:20:24 +02:00
return err ;
2015-07-31 15:46:16 +01:00
pr_info ( " Using PSCI v0.1 Function IDs from DT \n " ) ;
2020-12-02 18:40:59 +00:00
psci_ops . get_version = psci_0_1_get_version ;
2015-07-31 15:46:16 +01:00
if ( ! of_property_read_u32 ( np , " cpu_suspend " , & id ) ) {
2020-12-02 18:41:01 +00:00
psci_0_1_function_ids . cpu_suspend = id ;
2020-12-02 18:41:00 +00:00
psci_ops . cpu_suspend = psci_0_1_cpu_suspend ;
2015-07-31 15:46:16 +01:00
}
if ( ! of_property_read_u32 ( np , " cpu_off " , & id ) ) {
2020-12-02 18:41:01 +00:00
psci_0_1_function_ids . cpu_off = id ;
2020-12-02 18:41:00 +00:00
psci_ops . cpu_off = psci_0_1_cpu_off ;
2015-07-31 15:46:16 +01:00
}
if ( ! of_property_read_u32 ( np , " cpu_on " , & id ) ) {
2020-12-02 18:41:01 +00:00
psci_0_1_function_ids . cpu_on = id ;
2020-12-02 18:41:00 +00:00
psci_ops . cpu_on = psci_0_1_cpu_on ;
2015-07-31 15:46:16 +01:00
}
if ( ! of_property_read_u32 ( np , " migrate " , & id ) ) {
2020-12-02 18:41:01 +00:00
psci_0_1_function_ids . migrate = id ;
2020-12-02 18:41:00 +00:00
psci_ops . migrate = psci_0_1_migrate ;
2015-07-31 15:46:16 +01:00
}
2019-04-10 10:20:24 +02:00
return 0 ;
2015-07-31 15:46:16 +01:00
}
2022-09-08 14:54:50 -07:00
static int __init psci_1_0_init ( const struct device_node * np )
2019-04-10 10:20:25 +02:00
{
int err ;
err = psci_0_2_init ( np ) ;
if ( err )
return err ;
2019-10-10 12:01:48 +02:00
if ( psci_has_osi_support ( ) ) {
2019-04-10 10:20:25 +02:00
pr_info ( " OSI mode supported. \n " ) ;
2019-10-10 12:01:48 +02:00
/* Default to PC mode. */
2020-09-01 16:28:58 +02:00
psci_set_osi_mode ( false ) ;
2019-10-10 12:01:48 +02:00
}
2019-04-10 10:20:25 +02:00
return 0 ;
}
2016-04-20 11:20:27 +01:00
static const struct of_device_id psci_of_match [ ] __initconst = {
2015-07-31 15:46:16 +01:00
{ . compatible = " arm,psci " , . data = psci_0_1_init } ,
{ . compatible = " arm,psci-0.2 " , . data = psci_0_2_init } ,
2019-04-10 10:20:25 +02:00
{ . compatible = " arm,psci-1.0 " , . data = psci_1_0_init } ,
2015-07-31 15:46:16 +01:00
{ } ,
} ;
int __init psci_dt_init ( void )
{
struct device_node * np ;
const struct of_device_id * matched_np ;
psci_initcall_t init_fn ;
2019-04-10 10:20:24 +02:00
int ret ;
2015-07-31 15:46:16 +01:00
np = of_find_matching_node_and_match ( NULL , psci_of_match , & matched_np ) ;
2016-11-08 17:55:47 +00:00
if ( ! np | | ! of_device_is_available ( np ) )
2015-07-31 15:46:16 +01:00
return - ENODEV ;
init_fn = ( psci_initcall_t ) matched_np - > data ;
2019-04-10 10:20:24 +02:00
ret = init_fn ( np ) ;
of_node_put ( np ) ;
return ret ;
2015-07-31 15:46:16 +01:00
}
# ifdef CONFIG_ACPI
/*
* We use PSCI 0.2 + when ACPI is deployed on ARM64 and it ' s
* explicitly clarified in SBBR
*/
int __init psci_acpi_init ( void )
{
if ( ! acpi_psci_present ( ) ) {
pr_info ( " is not implemented in ACPI. \n " ) ;
return - EOPNOTSUPP ;
}
pr_info ( " probing for conduit method from ACPI. \n " ) ;
if ( acpi_psci_use_hvc ( ) )
2019-08-09 14:22:43 +01:00
set_conduit ( SMCCC_CONDUIT_HVC ) ;
2015-07-31 15:46:16 +01:00
else
2019-08-09 14:22:43 +01:00
set_conduit ( SMCCC_CONDUIT_SMC ) ;
2015-07-31 15:46:16 +01:00
return psci_probe ( ) ;
}
# endif