2019-05-27 08:55:06 +02:00
// SPDX-License-Identifier: GPL-2.0-or-later
2008-07-05 10:02:56 +02:00
/*
* Copyright ( C ) 1999 ARM Limited
* Copyright ( C ) 2000 Deep Blue Solutions Ltd
* Copyright 2006 - 2007 Freescale Semiconductor , Inc . All Rights Reserved .
* Copyright 2008 Juergen Beisert , kernel @ pengutronix . de
2009-03-03 02:49:23 +03:00
* Copyright 2009 Ilya Yanok , Emcraft Systems Ltd , yanok @ emcraft . com
2008-07-05 10:02:56 +02:00
*/
# include <linux/kernel.h>
# include <linux/clk.h>
# include <linux/io.h>
2009-03-03 02:49:23 +03:00
# include <linux/err.h>
# include <linux/delay.h>
2013-05-10 10:19:01 +08:00
# include <linux/of.h>
# include <linux/of_address.h>
2008-07-05 10:02:56 +02:00
2012-03-28 18:30:01 +01:00
# include <asm/system_misc.h>
2008-07-05 10:02:56 +02:00
# include <asm/proc-fns.h>
2010-10-27 14:40:55 +02:00
# include <asm/mach-types.h>
2013-07-08 21:45:20 +08:00
# include <asm/hardware/cache-l2x0.h>
2008-07-05 10:02:56 +02:00
2012-09-13 21:01:00 +08:00
# include "common.h"
2012-09-14 14:14:45 +08:00
# include "hardware.h"
2012-09-13 21:01:00 +08:00
2009-06-04 12:19:02 +02:00
static void __iomem * wdog_base ;
2013-05-10 09:13:44 +08:00
static struct clk * wdog_clk ;
2016-06-24 12:49:56 +02:00
static int wcr_enable = ( 1 < < 2 ) ;
2008-07-05 10:02:56 +02:00
/*
* Reset the system . It is called by machine_restart ( ) .
*/
2013-07-08 16:01:40 -07:00
void mxc_restart ( enum reboot_mode mode , const char * cmd )
2008-07-05 10:02:56 +02:00
{
2014-06-13 11:26:13 +04:00
if ( ! wdog_base )
goto reset_fallback ;
2014-06-13 11:26:12 +04:00
if ( ! IS_ERR ( wdog_clk ) )
2013-05-10 09:13:44 +08:00
clk_enable ( wdog_clk ) ;
2008-07-05 10:02:56 +02:00
/* Assert SRS signal */
2016-01-27 17:59:35 +01:00
imx_writew ( wcr_enable , wdog_base ) ;
2013-10-31 10:35:40 +08:00
/*
* Due to imx6q errata ERR004346 ( WDOG : WDOG SRS bit requires to be
* written twice ) , we add another two writes to ensure there must be at
* least two writes happen in the same one 32 kHz clock period . We save
* the target check here , since the writes shouldn ' t be a huge burden
* for other platforms .
*/
2016-01-27 17:59:35 +01:00
imx_writew ( wcr_enable , wdog_base ) ;
imx_writew ( wcr_enable , wdog_base ) ;
2009-03-03 02:49:23 +03:00
/* wait for reset to assert... */
mdelay ( 500 ) ;
2013-05-10 09:13:44 +08:00
pr_err ( " %s: Watchdog reset failed to assert reset \n " , __func__ ) ;
2009-03-03 02:49:23 +03:00
/* delay to allow the serial port to show the message */
mdelay ( 50 ) ;
2014-06-13 11:26:13 +04:00
reset_fallback :
2009-03-03 02:49:23 +03:00
/* we'll take a jump through zero as a poor second */
2011-11-01 13:16:26 +00:00
soft_restart ( 0 ) ;
2008-07-05 10:02:56 +02:00
}
2009-06-04 12:19:02 +02:00
2013-05-10 09:13:44 +08:00
void __init mxc_arch_reset_init ( void __iomem * base )
2009-06-04 12:19:02 +02:00
{
wdog_base = base ;
2013-05-10 09:13:44 +08:00
wdog_clk = clk_get_sys ( " imx2-wdt.0 " , NULL ) ;
2014-06-13 11:26:12 +04:00
if ( IS_ERR ( wdog_clk ) )
2013-05-10 09:13:44 +08:00
pr_warn ( " %s: failed to get wdog clock \n " , __func__ ) ;
2014-06-13 11:26:12 +04:00
else
clk_prepare ( wdog_clk ) ;
2009-06-04 12:19:02 +02:00
}
2013-05-10 10:19:01 +08:00
2016-06-24 12:49:56 +02:00
# ifdef CONFIG_SOC_IMX1
void __init imx1_reset_init ( void __iomem * base )
{
wcr_enable = ( 1 < < 0 ) ;
mxc_arch_reset_init ( base ) ;
}
# endif
2013-07-08 21:45:20 +08:00
# ifdef CONFIG_CACHE_L2X0
2013-07-10 11:45:46 +02:00
void __init imx_init_l2cache ( void )
2013-07-08 21:45:20 +08:00
{
void __iomem * l2x0_base ;
struct device_node * np ;
unsigned int val ;
np = of_find_compatible_node ( NULL , NULL , " arm,pl310-cache " ) ;
if ( ! np )
2016-06-18 18:09:31 -07:00
return ;
2013-07-08 21:45:20 +08:00
l2x0_base = of_iomap ( np , 0 ) ;
2016-06-18 18:09:31 -07:00
if ( ! l2x0_base )
goto put_node ;
2013-07-08 21:45:20 +08:00
2016-06-18 18:09:27 -07:00
if ( ! ( readl_relaxed ( l2x0_base + L2X0_CTRL ) & L2X0_CTRL_EN ) ) {
/* Configure the L2 PREFETCH and POWER registers */
val = readl_relaxed ( l2x0_base + L310_PREFETCH_CTRL ) ;
2016-06-18 18:09:29 -07:00
val | = L310_PREFETCH_CTRL_DBL_LINEFILL |
L310_PREFETCH_CTRL_INSTR_PREFETCH |
2016-06-18 18:09:30 -07:00
L310_PREFETCH_CTRL_DATA_PREFETCH ;
/* Set perfetch offset to improve performance */
val & = ~ L310_PREFETCH_CTRL_OFFSET_MASK ;
val | = 15 ;
2016-06-18 18:09:27 -07:00
writel_relaxed ( val , l2x0_base + L310_PREFETCH_CTRL ) ;
}
2013-07-08 21:45:20 +08:00
iounmap ( l2x0_base ) ;
2016-06-18 18:09:31 -07:00
put_node :
2013-07-08 21:45:20 +08:00
of_node_put ( np ) ;
}
# endif