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/>.
*/
# include <asm/cpu_ops.h>
2013-10-24 23:30:17 +04:00
# include <asm/smp_plat.h>
# include <linux/errno.h>
# include <linux/of.h>
2013-10-24 23:30:15 +04:00
# include <linux/string.h>
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 ,
} ;
2013-10-24 23:30:17 +04: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
/*
* Read a cpu ' s enable method from the device tree and record it in cpu_ops .
*/
int __init cpu_read_ops ( struct device_node * dn , int cpu )
{
const char * 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 ) ;
return - ENOENT ;
}
cpu_ops [ cpu ] = cpu_get_ops ( enable_method ) ;
if ( ! cpu_ops [ cpu ] ) {
2013-10-31 20:37:26 +04:00
pr_warn ( " %s: unsupported enable-method property: %s \n " ,
dn - > full_name , enable_method ) ;
2013-10-24 23:30:17 +04:00
return - EOPNOTSUPP ;
}
return 0 ;
}
void __init cpu_read_bootcpu_ops ( void )
{
2013-10-30 17:47:16 +04:00
struct device_node * dn = of_get_cpu_node ( 0 , NULL ) ;
if ( ! dn ) {
pr_err ( " Failed to find device node for boot cpu \n " ) ;
return ;
2013-10-24 23:30:17 +04:00
}
2013-10-30 17:47:16 +04:00
cpu_read_ops ( dn , 0 ) ;
2013-10-24 23:30:17 +04:00
}