2019-06-03 08:44:50 +03:00
// SPDX-License-Identifier: GPL-2.0-only
2013-10-24 23:30:15 +04:00
/*
* CPU kernel entry / exit control
*
* Copyright ( C ) 2013 ARM Ltd .
*/
2015-05-13 16:12:47 +03:00
# include <linux/acpi.h>
2016-08-15 09:45:46 +03:00
# include <linux/cache.h>
2013-10-24 23:30:17 +04:00
# include <linux/errno.h>
# include <linux/of.h>
2013-10-24 23:30:15 +04:00
# include <linux/string.h>
2015-05-13 16:12:47 +03:00
# include <asm/acpi.h>
# include <asm/cpu_ops.h>
# include <asm/smp_plat.h>
2013-10-24 23:30:15 +04:00
extern const struct cpu_operations smp_spin_table_ops ;
2020-03-19 02:01:42 +03:00
# ifdef CONFIG_ARM64_ACPI_PARKING_PROTOCOL
2016-01-26 14:10:38 +03:00
extern const struct cpu_operations acpi_parking_protocol_ops ;
2020-03-19 02:01:42 +03:00
# endif
2013-10-24 23:30:15 +04:00
extern const struct cpu_operations cpu_psci_ops ;
2020-03-19 02:01:44 +03:00
static const struct cpu_operations * cpu_ops [ NR_CPUS ] __ro_after_init ;
2013-10-24 23:30:15 +04:00
2017-11-29 17:03:03 +03:00
static const struct cpu_operations * const dt_supported_cpu_ops [ ] __initconst = {
2013-10-24 23:30:15 +04:00
& smp_spin_table_ops ,
2014-07-17 21:19:18 +04:00
& cpu_psci_ops ,
2013-10-24 23:30:15 +04:00
NULL ,
} ;
2017-11-29 17:03:03 +03:00
static const struct cpu_operations * const acpi_supported_cpu_ops [ ] __initconst = {
2016-01-26 14:10:38 +03:00
# ifdef CONFIG_ARM64_ACPI_PARKING_PROTOCOL
& acpi_parking_protocol_ops ,
# endif
& cpu_psci_ops ,
NULL ,
} ;
2015-05-13 16:12:47 +03:00
static const struct cpu_operations * __init cpu_get_ops ( const char * name )
2013-10-24 23:30:15 +04:00
{
2017-11-29 17:03:03 +03:00
const struct cpu_operations * const * ops ;
2016-01-26 14:10:38 +03:00
ops = acpi_disabled ? dt_supported_cpu_ops : acpi_supported_cpu_ops ;
2013-10-24 23:30:15 +04:00
while ( * ops ) {
if ( ! strcmp ( name , ( * ops ) - > name ) )
return * ops ;
ops + + ;
}
return NULL ;
}
2013-10-24 23:30:17 +04:00
2015-05-13 16:12:47 +03:00
static const char * __init cpu_read_enable_method ( int cpu )
{
const char * enable_method ;
if ( acpi_disabled ) {
struct device_node * dn = of_get_cpu_node ( cpu , NULL ) ;
if ( ! dn ) {
if ( ! cpu )
pr_err ( " Failed to find device node for boot cpu \n " ) ;
return NULL ;
}
enable_method = of_get_property ( dn , " enable-method " , NULL ) ;
if ( ! enable_method ) {
/*
* The boot CPU may not have an enable method ( e . g .
* when spin - table is used for secondaries ) .
* Don ' t warn spuriously .
*/
if ( cpu ! = 0 )
2017-07-19 00:42:42 +03:00
pr_err ( " %pOF: missing enable-method property \n " ,
dn ) ;
2015-05-13 16:12:47 +03:00
}
2019-03-05 14:34:05 +03:00
of_node_put ( dn ) ;
2015-05-13 16:12:47 +03:00
} else {
enable_method = acpi_get_enable_method ( cpu ) ;
2016-01-26 14:10:38 +03:00
if ( ! enable_method ) {
/*
* In ACPI systems the boot CPU does not require
* checking the enable method since for some
* boot protocol ( ie parking protocol ) it need not
* be initialized . Don ' t warn spuriously .
*/
if ( cpu ! = 0 )
pr_err ( " Unsupported ACPI enable-method \n " ) ;
}
2015-05-13 16:12:47 +03:00
}
return enable_method ;
}
2013-10-24 23:30:17 +04:00
/*
2015-05-13 16:12:47 +03:00
* Read a cpu ' s enable method and record it in cpu_ops .
2013-10-24 23:30:17 +04:00
*/
2020-03-19 02:01:43 +03:00
int __init init_cpu_ops ( int cpu )
2013-10-24 23:30:17 +04:00
{
2015-05-13 16:12:47 +03:00
const char * enable_method = cpu_read_enable_method ( cpu ) ;
2015-05-13 16:12:46 +03:00
2015-05-13 16:12:47 +03:00
if ( ! enable_method )
2015-05-13 16:12:46 +03:00
return - ENODEV ;
2013-10-24 23:30:17 +04:00
cpu_ops [ cpu ] = cpu_get_ops ( enable_method ) ;
if ( ! cpu_ops [ cpu ] ) {
2015-05-13 16:12:47 +03:00
pr_warn ( " Unsupported enable-method: %s \n " , enable_method ) ;
2013-10-24 23:30:17 +04:00
return - EOPNOTSUPP ;
}
return 0 ;
}
2020-03-19 02:01:44 +03:00
const struct cpu_operations * get_cpu_ops ( int cpu )
{
return cpu_ops [ cpu ] ;
}