2022-06-24 10:24:42 +08:00
// SPDX-License-Identifier: GPL-2.0+
2010-08-04 11:44:14 +05:30
/*
* drivers / char / watchdog / sp805 - wdt . c
*
* Watchdog driver for ARM SP805 watchdog module
*
* Copyright ( C ) 2010 ST Microelectronics
2015-07-17 16:23:50 -07:00
* Viresh Kumar < vireshk @ kernel . org >
2010-08-04 11:44:14 +05:30
*
* This file is licensed under the terms of the GNU General Public
* License version 2 or later . This program is licensed " as is " without any
* warranty of any kind , whether express or implied .
*/
# include <linux/device.h>
# include <linux/resource.h>
# include <linux/amba/bus.h>
# include <linux/bitops.h>
# include <linux/clk.h>
# include <linux/io.h>
# include <linux/ioport.h>
# include <linux/kernel.h>
# include <linux/math64.h>
# include <linux/module.h>
# include <linux/moduleparam.h>
2012-02-24 15:12:37 +05:30
# include <linux/pm.h>
2021-05-17 20:44:56 +03:00
# include <linux/property.h>
2024-02-21 19:56:41 +08:00
# include <linux/reset.h>
2010-08-04 11:44:14 +05:30
# include <linux/slab.h>
# include <linux/spinlock.h>
# include <linux/types.h>
# include <linux/watchdog.h>
/* default timeout in seconds */
# define DEFAULT_TIMEOUT 60
# define MODULE_NAME "sp805-wdt"
/* watchdog register offsets and masks */
# define WDTLOAD 0x000
# define LOAD_MIN 0x00000001
# define LOAD_MAX 0xFFFFFFFF
# define WDTVALUE 0x004
# define WDTCONTROL 0x008
/* control register masks */
# define INT_ENABLE (1 << 0)
# define RESET_ENABLE (1 << 1)
2018-05-28 11:01:35 -07:00
# define ENABLE_MASK (INT_ENABLE | RESET_ENABLE)
2010-08-04 11:44:14 +05:30
# define WDTINTCLR 0x00C
# define WDTRIS 0x010
# define WDTMIS 0x014
# define INT_MASK (1 << 0)
# define WDTLOCK 0xC00
# define UNLOCK 0x1ACCE551
# define LOCK 0x00000001
/**
* struct sp805_wdt : sp805 wdt device structure
2012-03-12 09:52:16 +05:30
* @ wdd : instance of struct watchdog_device
2012-03-12 09:52:13 +05:30
* @ lock : spin lock protecting dev structure and io access
* @ base : base address of wdt
2021-05-17 20:49:12 +03:00
* @ clk : ( optional ) clock structure of wdt
* @ rate : ( optional ) clock rate when provided via properties
2012-03-12 09:52:13 +05:30
* @ adev : amba device structure of wdt
* @ status : current status of wdt
* @ load_val : load value to be set for current timeout
2010-08-04 11:44:14 +05:30
*/
struct sp805_wdt {
2012-03-12 09:52:16 +05:30
struct watchdog_device wdd ;
2010-08-04 11:44:14 +05:30
spinlock_t lock ;
void __iomem * base ;
struct clk * clk ;
2018-07-26 10:28:42 +05:30
u64 rate ;
2010-08-04 11:44:14 +05:30
struct amba_device * adev ;
unsigned int load_val ;
} ;
2012-03-05 16:51:11 +01:00
static bool nowayout = WATCHDOG_NOWAYOUT ;
2012-03-12 09:52:16 +05:30
module_param ( nowayout , bool , 0 ) ;
MODULE_PARM_DESC ( nowayout ,
" Set to 1 to keep watchdog running after device release " ) ;
2010-08-04 11:44:14 +05:30
2018-05-28 11:01:35 -07:00
/* returns true if wdt is running; otherwise returns false */
static bool wdt_is_running ( struct watchdog_device * wdd )
{
struct sp805_wdt * wdt = watchdog_get_drvdata ( wdd ) ;
u32 wdtcontrol = readl_relaxed ( wdt - > base + WDTCONTROL ) ;
return ( wdtcontrol & ENABLE_MASK ) = = ENABLE_MASK ;
}
2022-10-09 16:39:44 +08:00
/* This routine finds load value that will reset system in required timeout */
2012-03-12 09:52:16 +05:30
static int wdt_setload ( struct watchdog_device * wdd , unsigned int timeout )
2010-08-04 11:44:14 +05:30
{
2012-03-12 09:52:16 +05:30
struct sp805_wdt * wdt = watchdog_get_drvdata ( wdd ) ;
2010-08-04 11:44:14 +05:30
u64 load , rate ;
2018-07-26 10:28:42 +05:30
rate = wdt - > rate ;
2010-08-04 11:44:14 +05:30
/*
* sp805 runs counter with given value twice , after the end of first
* counter it gives an interrupt and then starts counter again . If
2011-03-30 22:57:33 -03:00
* interrupt already occurred then it resets the system . This is why
2010-08-04 11:44:14 +05:30
* load is half of what should be required .
*/
load = div_u64 ( rate , 2 ) * timeout - 1 ;
load = ( load > LOAD_MAX ) ? LOAD_MAX : load ;
load = ( load < LOAD_MIN ) ? LOAD_MIN : load ;
spin_lock ( & wdt - > lock ) ;
wdt - > load_val = load ;
/* roundup timeout to closest positive integer value */
2014-05-15 10:01:59 +05:30
wdd - > timeout = div_u64 ( ( load + 1 ) * 2 + ( rate / 2 ) , rate ) ;
2010-08-04 11:44:14 +05:30
spin_unlock ( & wdt - > lock ) ;
2012-03-12 09:52:16 +05:30
return 0 ;
2010-08-04 11:44:14 +05:30
}
/* returns number of seconds left for reset to occur */
2012-03-12 09:52:16 +05:30
static unsigned int wdt_timeleft ( struct watchdog_device * wdd )
2010-08-04 11:44:14 +05:30
{
2012-03-12 09:52:16 +05:30
struct sp805_wdt * wdt = watchdog_get_drvdata ( wdd ) ;
2018-07-26 10:28:42 +05:30
u64 load ;
2010-08-04 11:44:14 +05:30
spin_lock ( & wdt - > lock ) ;
2012-03-12 09:52:14 +05:30
load = readl_relaxed ( wdt - > base + WDTVALUE ) ;
2010-08-04 11:44:14 +05:30
/*If the interrupt is inactive then time left is WDTValue + WDTLoad. */
2012-03-12 09:52:14 +05:30
if ( ! ( readl_relaxed ( wdt - > base + WDTRIS ) & INT_MASK ) )
2010-08-04 11:44:14 +05:30
load + = wdt - > load_val + 1 ;
spin_unlock ( & wdt - > lock ) ;
2018-07-26 10:28:42 +05:30
return div_u64 ( load , wdt - > rate ) ;
2010-08-04 11:44:14 +05:30
}
2018-05-04 15:05:03 +09:00
static int
wdt_restart ( struct watchdog_device * wdd , unsigned long mode , void * cmd )
{
struct sp805_wdt * wdt = watchdog_get_drvdata ( wdd ) ;
2020-03-27 17:24:50 +01:00
writel_relaxed ( UNLOCK , wdt - > base + WDTLOCK ) ;
2018-05-04 15:05:03 +09:00
writel_relaxed ( 0 , wdt - > base + WDTCONTROL ) ;
writel_relaxed ( 0 , wdt - > base + WDTLOAD ) ;
writel_relaxed ( INT_ENABLE | RESET_ENABLE , wdt - > base + WDTCONTROL ) ;
2020-03-27 17:24:50 +01:00
/* Flush posted writes. */
readl_relaxed ( wdt - > base + WDTLOCK ) ;
2018-05-04 15:05:03 +09:00
return 0 ;
}
2012-03-12 09:52:16 +05:30
static int wdt_config ( struct watchdog_device * wdd , bool ping )
2010-08-04 11:44:14 +05:30
{
2012-03-12 09:52:16 +05:30
struct sp805_wdt * wdt = watchdog_get_drvdata ( wdd ) ;
int ret ;
if ( ! ping ) {
2012-03-12 09:52:17 +05:30
2012-08-26 18:01:02 +02:00
ret = clk_prepare_enable ( wdt - > clk ) ;
2012-03-12 09:52:16 +05:30
if ( ret ) {
dev_err ( & wdt - > adev - > dev , " clock enable fail " ) ;
return ret ;
}
}
2010-08-04 11:44:14 +05:30
spin_lock ( & wdt - > lock ) ;
2012-03-12 09:52:14 +05:30
writel_relaxed ( UNLOCK , wdt - > base + WDTLOCK ) ;
writel_relaxed ( wdt - > load_val , wdt - > base + WDTLOAD ) ;
2016-01-19 14:44:49 +05:30
writel_relaxed ( INT_MASK , wdt - > base + WDTINTCLR ) ;
2010-08-04 11:44:14 +05:30
2016-01-19 14:44:49 +05:30
if ( ! ping )
2012-03-12 09:52:16 +05:30
writel_relaxed ( INT_ENABLE | RESET_ENABLE , wdt - > base +
WDTCONTROL ) ;
2010-08-04 11:44:14 +05:30
2012-03-12 09:52:14 +05:30
writel_relaxed ( LOCK , wdt - > base + WDTLOCK ) ;
2010-08-04 11:44:14 +05:30
2011-07-15 11:04:02 -04:00
/* Flush posted writes. */
2012-03-12 09:52:14 +05:30
readl_relaxed ( wdt - > base + WDTLOCK ) ;
2010-08-04 11:44:14 +05:30
spin_unlock ( & wdt - > lock ) ;
2012-03-12 09:52:16 +05:30
return 0 ;
2010-08-04 11:44:14 +05:30
}
2012-03-12 09:52:16 +05:30
static int wdt_ping ( struct watchdog_device * wdd )
2010-08-04 11:44:14 +05:30
{
2012-03-12 09:52:16 +05:30
return wdt_config ( wdd , true ) ;
2010-08-04 11:44:14 +05:30
}
2012-03-12 09:52:16 +05:30
/* enables watchdog timers reset */
static int wdt_enable ( struct watchdog_device * wdd )
2010-08-04 11:44:14 +05:30
{
2012-03-12 09:52:16 +05:30
return wdt_config ( wdd , false ) ;
2010-08-04 11:44:14 +05:30
}
2012-03-12 09:52:16 +05:30
/* disables watchdog timers reset */
static int wdt_disable ( struct watchdog_device * wdd )
2010-08-04 11:44:14 +05:30
{
2012-03-12 09:52:16 +05:30
struct sp805_wdt * wdt = watchdog_get_drvdata ( wdd ) ;
2010-08-04 11:44:14 +05:30
2012-03-12 09:52:16 +05:30
spin_lock ( & wdt - > lock ) ;
2010-08-04 11:44:14 +05:30
2012-03-12 09:52:16 +05:30
writel_relaxed ( UNLOCK , wdt - > base + WDTLOCK ) ;
writel_relaxed ( 0 , wdt - > base + WDTCONTROL ) ;
writel_relaxed ( LOCK , wdt - > base + WDTLOCK ) ;
2010-08-04 11:44:14 +05:30
2012-03-12 09:52:16 +05:30
/* Flush posted writes. */
readl_relaxed ( wdt - > base + WDTLOCK ) ;
spin_unlock ( & wdt - > lock ) ;
2010-08-04 11:44:14 +05:30
2012-08-26 18:01:02 +02:00
clk_disable_unprepare ( wdt - > clk ) ;
2010-08-04 11:44:14 +05:30
return 0 ;
}
2012-03-12 09:52:16 +05:30
static const struct watchdog_info wdt_info = {
. options = WDIOF_MAGICCLOSE | WDIOF_SETTIMEOUT | WDIOF_KEEPALIVEPING ,
. identity = MODULE_NAME ,
2010-08-04 11:44:14 +05:30
} ;
2012-03-12 09:52:16 +05:30
static const struct watchdog_ops wdt_ops = {
. owner = THIS_MODULE ,
. start = wdt_enable ,
. stop = wdt_disable ,
. ping = wdt_ping ,
. set_timeout = wdt_setload ,
. get_timeleft = wdt_timeleft ,
2018-05-04 15:05:03 +09:00
. restart = wdt_restart ,
2010-08-04 11:44:14 +05:30
} ;
2012-11-19 13:21:41 -05:00
static int
2011-02-19 15:55:00 +00:00
sp805_wdt_probe ( struct amba_device * adev , const struct amba_id * id )
2010-08-04 11:44:14 +05:30
{
2012-03-12 09:52:16 +05:30
struct sp805_wdt * wdt ;
2024-02-21 19:56:41 +08:00
struct reset_control * rst ;
2021-05-17 20:44:56 +03:00
u64 rate = 0 ;
2010-08-04 11:44:14 +05:30
int ret = 0 ;
2012-03-12 09:52:15 +05:30
wdt = devm_kzalloc ( & adev - > dev , sizeof ( * wdt ) , GFP_KERNEL ) ;
2010-08-04 11:44:14 +05:30
if ( ! wdt ) {
ret = - ENOMEM ;
2012-03-12 09:52:15 +05:30
goto err ;
}
2014-02-11 21:45:56 +09:00
wdt - > base = devm_ioremap_resource ( & adev - > dev , & adev - > res ) ;
if ( IS_ERR ( wdt - > base ) )
return PTR_ERR ( wdt - > base ) ;
2010-08-04 11:44:14 +05:30
2021-05-17 20:44:56 +03:00
/*
* When driver probe with ACPI device , clock devices
* are not available , so watchdog rate get from
* clock - frequency property given in _DSD object .
*/
device_property_read_u64 ( & adev - > dev , " clock-frequency " , & rate ) ;
wdt - > clk = devm_clk_get_optional ( & adev - > dev , NULL ) ;
if ( IS_ERR ( wdt - > clk ) )
return dev_err_probe ( & adev - > dev , PTR_ERR ( wdt - > clk ) , " Clock not found \n " ) ;
wdt - > rate = clk_get_rate ( wdt - > clk ) ;
if ( ! wdt - > rate )
wdt - > rate = rate ;
if ( ! wdt - > rate ) {
dev_err ( & adev - > dev , " no clock-frequency property \n " ) ;
return - ENODEV ;
2010-08-04 11:44:14 +05:30
}
2024-02-21 19:56:41 +08:00
rst = devm_reset_control_get_optional_exclusive ( & adev - > dev , NULL ) ;
if ( IS_ERR ( rst ) )
return dev_err_probe ( & adev - > dev , PTR_ERR ( rst ) , " Can not get reset \n " ) ;
reset_control_deassert ( rst ) ;
2010-08-04 11:44:14 +05:30
wdt - > adev = adev ;
2012-03-12 09:52:16 +05:30
wdt - > wdd . info = & wdt_info ;
wdt - > wdd . ops = & wdt_ops ;
2015-08-20 14:05:01 +05:30
wdt - > wdd . parent = & adev - > dev ;
2012-03-12 09:52:16 +05:30
2010-08-04 11:44:14 +05:30
spin_lock_init ( & wdt - > lock ) ;
2012-03-12 09:52:16 +05:30
watchdog_set_nowayout ( & wdt - > wdd , nowayout ) ;
watchdog_set_drvdata ( & wdt - > wdd , wdt ) ;
2018-05-04 15:05:03 +09:00
watchdog_set_restart_priority ( & wdt - > wdd , 128 ) ;
2022-04-14 05:42:33 +00:00
watchdog_stop_on_unregister ( & wdt - > wdd ) ;
2018-05-28 11:01:34 -07:00
/*
* If ' timeout - sec ' devicetree property is specified , use that .
* Otherwise , use DEFAULT_TIMEOUT
*/
wdt - > wdd . timeout = DEFAULT_TIMEOUT ;
watchdog_init_timeout ( & wdt - > wdd , 0 , & adev - > dev ) ;
wdt_setload ( & wdt - > wdd , wdt - > wdd . timeout ) ;
2010-08-04 11:44:14 +05:30
2018-05-28 11:01:35 -07:00
/*
* If HW is already running , enable / reset the wdt and set the running
* bit to tell the wdt subsystem
*/
if ( wdt_is_running ( & wdt - > wdd ) ) {
wdt_enable ( & wdt - > wdd ) ;
set_bit ( WDOG_HW_RUNNING , & wdt - > wdd . status ) ;
}
2020-11-27 15:52:17 +08:00
watchdog_stop_on_reboot ( & wdt - > wdd ) ;
2012-03-12 09:52:16 +05:30
ret = watchdog_register_device ( & wdt - > wdd ) ;
2019-05-18 23:27:53 +02:00
if ( ret )
2013-04-29 18:16:49 +09:00
goto err ;
2012-03-12 09:52:16 +05:30
amba_set_drvdata ( adev , wdt ) ;
2010-08-04 11:44:14 +05:30
dev_info ( & adev - > dev , " registration successful \n " ) ;
return 0 ;
err :
dev_err ( & adev - > dev , " Probe Failed!!! \n " ) ;
return ret ;
}
2021-01-26 17:58:34 +01:00
static void sp805_wdt_remove ( struct amba_device * adev )
2010-08-04 11:44:14 +05:30
{
2012-03-12 09:52:16 +05:30
struct sp805_wdt * wdt = amba_get_drvdata ( adev ) ;
watchdog_unregister_device ( & wdt - > wdd ) ;
watchdog_set_drvdata ( & wdt - > wdd , NULL ) ;
2010-08-04 11:44:14 +05:30
}
2012-11-08 10:51:03 +00:00
static int __maybe_unused sp805_wdt_suspend ( struct device * dev )
2012-02-24 15:12:37 +05:30
{
2012-03-12 09:52:16 +05:30
struct sp805_wdt * wdt = dev_get_drvdata ( dev ) ;
if ( watchdog_active ( & wdt - > wdd ) )
return wdt_disable ( & wdt - > wdd ) ;
2012-02-24 15:12:37 +05:30
return 0 ;
}
2012-11-08 10:51:03 +00:00
static int __maybe_unused sp805_wdt_resume ( struct device * dev )
2012-02-24 15:12:37 +05:30
{
2012-03-12 09:52:16 +05:30
struct sp805_wdt * wdt = dev_get_drvdata ( dev ) ;
2012-02-24 15:12:37 +05:30
2012-03-12 09:52:16 +05:30
if ( watchdog_active ( & wdt - > wdd ) )
return wdt_enable ( & wdt - > wdd ) ;
2012-02-24 15:12:37 +05:30
2012-03-12 09:52:16 +05:30
return 0 ;
2012-02-24 15:12:37 +05:30
}
static SIMPLE_DEV_PM_OPS ( sp805_wdt_dev_pm_ops , sp805_wdt_suspend ,
sp805_wdt_resume ) ;
2017-08-23 22:04:08 +05:30
static const struct amba_id sp805_wdt_ids [ ] = {
2010-08-04 11:44:14 +05:30
{
. id = 0x00141805 ,
. mask = 0x00ffffff ,
} ,
2022-06-24 10:24:42 +08:00
{
. id = 0x001bb824 ,
. mask = 0x00ffffff ,
} ,
2010-08-04 11:44:14 +05:30
{ 0 , 0 } ,
} ;
2011-10-05 15:15:23 +01:00
MODULE_DEVICE_TABLE ( amba , sp805_wdt_ids ) ;
2010-08-04 11:44:14 +05:30
static struct amba_driver sp805_wdt_driver = {
. drv = {
. name = MODULE_NAME ,
2012-02-24 15:12:37 +05:30
. pm = & sp805_wdt_dev_pm_ops ,
2010-08-04 11:44:14 +05:30
} ,
. id_table = sp805_wdt_ids ,
. probe = sp805_wdt_probe ,
2012-11-19 13:21:12 -05:00
. remove = sp805_wdt_remove ,
2010-08-04 11:44:14 +05:30
} ;
2012-03-15 10:40:38 +01:00
module_amba_driver ( sp805_wdt_driver ) ;
2010-08-04 11:44:14 +05:30
2015-07-17 16:23:50 -07:00
MODULE_AUTHOR ( " Viresh Kumar <vireshk@kernel.org> " ) ;
2010-08-04 11:44:14 +05:30
MODULE_DESCRIPTION ( " ARM SP805 Watchdog Driver " ) ;
MODULE_LICENSE ( " GPL " ) ;