2013-10-24 23:30:15 +04:00
/*
* CPU kernel entry / exit control
*
* Copyright ( C ) 2013 ARM Ltd .
*
* This program is free software ; you can redistribute it and / or modify
* it under the terms of the GNU General Public License version 2 as
* published by the Free Software Foundation .
*
* This program is distributed in the hope that it will be useful ,
* but WITHOUT ANY WARRANTY ; without even the implied warranty of
* MERCHANTABILITY or FITNESS FOR A PARTICULAR PURPOSE . See the
* GNU General Public License for more details .
*
* You should have received a copy of the GNU General Public License
* along with this program . If not , see < http : //www.gnu.org/licenses/>.
*/
2015-05-13 16:12:47 +03:00
# include <linux/acpi.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 ;
extern const struct cpu_operations cpu_psci_ops ;
const struct cpu_operations * cpu_ops [ NR_CPUS ] ;
static const struct cpu_operations * supported_cpu_ops [ ] __initconst = {
# ifdef CONFIG_SMP
& smp_spin_table_ops ,
# endif
2014-07-17 21:19:18 +04:00
& cpu_psci_ops ,
2013-10-24 23:30:15 +04:00
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
{
const struct cpu_operations * * ops = supported_cpu_ops ;
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 )
pr_err ( " %s: missing enable-method property \n " ,
dn - > full_name ) ;
}
} else {
enable_method = acpi_get_enable_method ( cpu ) ;
if ( ! enable_method )
pr_err ( " Unsupported ACPI enable-method \n " ) ;
}
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
*/
2015-05-13 16:12:46 +03:00
int __init cpu_read_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 ;
}