2011-09-06 14:39:44 +08:00
/*
* Copyright 2011 Freescale Semiconductor , Inc .
* Copyright 2011 Linaro Ltd .
*
* The code contained herein is licensed under the GNU General Public
* License . You may obtain a copy of the GNU General Public License
* Version 2 or later at the following locations :
*
* http : //www.opensource.org/licenses/gpl-license.html
* http : //www.gnu.org/copyleft/gpl.html
*/
# include <linux/init.h>
# include <linux/io.h>
# include <linux/of.h>
# include <linux/of_address.h>
2011-08-09 12:24:07 +01:00
# include <linux/smp.h>
2012-01-20 12:01:12 +01:00
# include <asm/smp_plat.h>
2011-09-06 14:39:44 +08:00
# define SRC_SCR 0x000
# define SRC_GPR1 0x020
2011-12-09 00:51:26 +01:00
# define BP_SRC_SCR_WARM_RESET_ENABLE 0
2011-09-06 14:39:44 +08:00
# define BP_SRC_SCR_CORE1_RST 14
# define BP_SRC_SCR_CORE1_ENABLE 22
static void __iomem * src_base ;
void imx_enable_cpu ( int cpu , bool enable )
{
u32 mask , val ;
2011-08-09 12:24:07 +01:00
cpu = cpu_logical_map ( cpu ) ;
2011-09-06 14:39:44 +08:00
mask = 1 < < ( BP_SRC_SCR_CORE1_ENABLE + cpu - 1 ) ;
val = readl_relaxed ( src_base + SRC_SCR ) ;
val = enable ? val | mask : val & ~ mask ;
writel_relaxed ( val , src_base + SRC_SCR ) ;
}
void imx_set_cpu_jump ( int cpu , void * jump_addr )
{
2011-08-09 12:24:07 +01:00
cpu = cpu_logical_map ( cpu ) ;
2012-01-09 15:41:40 -06:00
writel_relaxed ( virt_to_phys ( jump_addr ) ,
2011-09-06 14:39:44 +08:00
src_base + SRC_GPR1 + cpu * 8 ) ;
}
2013-03-26 16:46:07 +08:00
u32 imx_get_cpu_arg ( int cpu )
{
cpu = cpu_logical_map ( cpu ) ;
return readl_relaxed ( src_base + SRC_GPR1 + cpu * 8 + 4 ) ;
}
void imx_set_cpu_arg ( int cpu , u32 arg )
{
cpu = cpu_logical_map ( cpu ) ;
writel_relaxed ( arg , src_base + SRC_GPR1 + cpu * 8 + 4 ) ;
}
2011-12-09 00:51:26 +01:00
void imx_src_prepare_restart ( void )
{
u32 val ;
/* clear enable bits of secondary cores */
val = readl_relaxed ( src_base + SRC_SCR ) ;
val & = ~ ( 0x7 < < BP_SRC_SCR_CORE1_ENABLE ) ;
writel_relaxed ( val , src_base + SRC_SCR ) ;
/* clear persistent entry register of primary core */
writel_relaxed ( 0 , src_base + SRC_GPR1 ) ;
}
2011-09-06 14:39:44 +08:00
void __init imx_src_init ( void )
{
struct device_node * np ;
2011-12-09 00:51:26 +01:00
u32 val ;
2011-09-06 14:39:44 +08:00
np = of_find_compatible_node ( NULL , NULL , " fsl,imx6q-src " ) ;
src_base = of_iomap ( np , 0 ) ;
WARN_ON ( ! src_base ) ;
2011-12-09 00:51:26 +01:00
/*
* force warm reset sources to generate cold reset
* for a more reliable restart
*/
val = readl_relaxed ( src_base + SRC_SCR ) ;
val & = ~ ( 1 < < BP_SRC_SCR_WARM_RESET_ENABLE ) ;
writel_relaxed ( val , src_base + SRC_SCR ) ;
2011-09-06 14:39:44 +08:00
}