2007-06-12 18:09:50 +04:00
/*
* drivers / char / watchdog / davinci_wdt . c
*
* Watchdog driver for DaVinci DM644x / DM646x processors
*
* Copyright ( C ) 2006 Texas Instruments .
*
* 2007 ( c ) MontaVista Software , Inc . This file is licensed under
* the terms of the GNU General Public License version 2. This program
* is licensed " as is " without any warranty of any kind , whether express
* or implied .
*/
# include <linux/module.h>
# include <linux/moduleparam.h>
# include <linux/types.h>
# include <linux/kernel.h>
# include <linux/fs.h>
# include <linux/miscdevice.h>
# include <linux/watchdog.h>
# include <linux/init.h>
# include <linux/bitops.h>
# include <linux/platform_device.h>
# include <linux/spinlock.h>
2008-05-19 14:05:30 +01:00
# include <linux/uaccess.h>
# include <linux/io.h>
2009-01-29 14:14:30 -08:00
# include <linux/device.h>
2009-02-10 20:30:37 -08:00
# include <linux/clk.h>
2007-06-12 18:09:50 +04:00
# define MODULE_NAME "DAVINCI-WDT: "
# define DEFAULT_HEARTBEAT 60
# define MAX_HEARTBEAT 600 /* really the max margin is 264/27MHz*/
/* Timer register set definition */
# define PID12 (0x0)
# define EMUMGT (0x4)
# define TIM12 (0x10)
# define TIM34 (0x14)
# define PRD12 (0x18)
# define PRD34 (0x1C)
# define TCR (0x20)
# define TGCR (0x24)
# define WDTCR (0x28)
/* TCR bit definitions */
# define ENAMODE12_DISABLED (0 << 6)
# define ENAMODE12_ONESHOT (1 << 6)
# define ENAMODE12_PERIODIC (2 << 6)
/* TGCR bit definitions */
# define TIM12RS_UNRESET (1 << 0)
# define TIM34RS_UNRESET (1 << 1)
# define TIMMODE_64BIT_WDOG (2 << 2)
/* WDTCR bit definitions */
# define WDEN (1 << 14)
# define WDFLAG (1 << 15)
# define WDKEY_SEQ0 (0xa5c6 << 16)
# define WDKEY_SEQ1 (0xda7e << 16)
static int heartbeat = DEFAULT_HEARTBEAT ;
2007-11-01 16:27:08 -07:00
static DEFINE_SPINLOCK ( io_lock ) ;
2007-06-12 18:09:50 +04:00
static unsigned long wdt_status ;
# define WDT_IN_USE 0
# define WDT_OK_TO_CLOSE 1
# define WDT_REGION_INITED 2
# define WDT_DEVICE_INITED 3
static struct resource * wdt_mem ;
static void __iomem * wdt_base ;
2009-02-10 20:30:37 -08:00
struct clk * wdt_clk ;
2007-06-12 18:09:50 +04:00
static void wdt_service ( void )
{
spin_lock ( & io_lock ) ;
/* put watchdog in service state */
2009-01-29 14:14:30 -08:00
iowrite32 ( WDKEY_SEQ0 , wdt_base + WDTCR ) ;
2007-06-12 18:09:50 +04:00
/* put watchdog in active state */
2009-01-29 14:14:30 -08:00
iowrite32 ( WDKEY_SEQ1 , wdt_base + WDTCR ) ;
2007-06-12 18:09:50 +04:00
spin_unlock ( & io_lock ) ;
}
static void wdt_enable ( void )
{
u32 tgcr ;
u32 timer_margin ;
2009-02-10 20:30:37 -08:00
unsigned long wdt_freq ;
wdt_freq = clk_get_rate ( wdt_clk ) ;
2007-06-12 18:09:50 +04:00
spin_lock ( & io_lock ) ;
/* disable, internal clock source */
2009-01-29 14:14:30 -08:00
iowrite32 ( 0 , wdt_base + TCR ) ;
2007-06-12 18:09:50 +04:00
/* reset timer, set mode to 64-bit watchdog, and unreset */
2009-01-29 14:14:30 -08:00
iowrite32 ( 0 , wdt_base + TGCR ) ;
2007-06-12 18:09:50 +04:00
tgcr = TIMMODE_64BIT_WDOG | TIM12RS_UNRESET | TIM34RS_UNRESET ;
2009-01-29 14:14:30 -08:00
iowrite32 ( tgcr , wdt_base + TGCR ) ;
2007-06-12 18:09:50 +04:00
/* clear counter regs */
2009-01-29 14:14:30 -08:00
iowrite32 ( 0 , wdt_base + TIM12 ) ;
iowrite32 ( 0 , wdt_base + TIM34 ) ;
2007-06-12 18:09:50 +04:00
/* set timeout period */
2009-02-10 20:30:37 -08:00
timer_margin = ( ( ( u64 ) heartbeat * wdt_freq ) & 0xffffffff ) ;
2009-01-29 14:14:30 -08:00
iowrite32 ( timer_margin , wdt_base + PRD12 ) ;
2009-02-10 20:30:37 -08:00
timer_margin = ( ( ( u64 ) heartbeat * wdt_freq ) > > 32 ) ;
2009-01-29 14:14:30 -08:00
iowrite32 ( timer_margin , wdt_base + PRD34 ) ;
2007-06-12 18:09:50 +04:00
/* enable run continuously */
2009-01-29 14:14:30 -08:00
iowrite32 ( ENAMODE12_PERIODIC , wdt_base + TCR ) ;
2007-06-12 18:09:50 +04:00
/* Once the WDT is in pre-active state write to
* TIM12 , TIM34 , PRD12 , PRD34 , TCR , TGCR , WDTCR are
* write protected ( except for the WDKEY field )
*/
/* put watchdog in pre-active state */
2009-01-29 14:14:30 -08:00
iowrite32 ( WDKEY_SEQ0 | WDEN , wdt_base + WDTCR ) ;
2007-06-12 18:09:50 +04:00
/* put watchdog in active state */
2009-01-29 14:14:30 -08:00
iowrite32 ( WDKEY_SEQ1 | WDEN , wdt_base + WDTCR ) ;
2007-06-12 18:09:50 +04:00
spin_unlock ( & io_lock ) ;
}
static int davinci_wdt_open ( struct inode * inode , struct file * file )
{
if ( test_and_set_bit ( WDT_IN_USE , & wdt_status ) )
return - EBUSY ;
wdt_enable ( ) ;
return nonseekable_open ( inode , file ) ;
}
static ssize_t
davinci_wdt_write ( struct file * file , const char * data , size_t len ,
loff_t * ppos )
{
if ( len )
wdt_service ( ) ;
return len ;
}
static struct watchdog_info ident = {
2007-07-20 21:47:55 +00:00
. options = WDIOF_KEEPALIVEPING ,
2007-06-12 18:09:50 +04:00
. identity = " DaVinci Watchdog " ,
} ;
2008-05-19 14:05:30 +01:00
static long davinci_wdt_ioctl ( struct file * file ,
unsigned int cmd , unsigned long arg )
2007-06-12 18:09:50 +04:00
{
int ret = - ENOTTY ;
switch ( cmd ) {
case WDIOC_GETSUPPORT :
ret = copy_to_user ( ( struct watchdog_info * ) arg , & ident ,
sizeof ( ident ) ) ? - EFAULT : 0 ;
break ;
case WDIOC_GETSTATUS :
2007-07-20 21:47:55 +00:00
case WDIOC_GETBOOTSTATUS :
2007-06-12 18:09:50 +04:00
ret = put_user ( 0 , ( int * ) arg ) ;
break ;
case WDIOC_KEEPALIVE :
wdt_service ( ) ;
ret = 0 ;
break ;
2008-07-18 11:41:17 +00:00
case WDIOC_GETTIMEOUT :
ret = put_user ( heartbeat , ( int * ) arg ) ;
break ;
2007-06-12 18:09:50 +04:00
}
return ret ;
}
static int davinci_wdt_release ( struct inode * inode , struct file * file )
{
wdt_service ( ) ;
clear_bit ( WDT_IN_USE , & wdt_status ) ;
return 0 ;
}
static const struct file_operations davinci_wdt_fops = {
. owner = THIS_MODULE ,
. llseek = no_llseek ,
. write = davinci_wdt_write ,
2008-05-19 14:05:30 +01:00
. unlocked_ioctl = davinci_wdt_ioctl ,
2007-06-12 18:09:50 +04:00
. open = davinci_wdt_open ,
. release = davinci_wdt_release ,
} ;
static struct miscdevice davinci_wdt_miscdev = {
. minor = WATCHDOG_MINOR ,
. name = " watchdog " ,
. fops = & davinci_wdt_fops ,
} ;
2009-04-14 20:30:55 +00:00
static int __devinit davinci_wdt_probe ( struct platform_device * pdev )
2007-06-12 18:09:50 +04:00
{
int ret = 0 , size ;
struct resource * res ;
2009-01-29 14:14:30 -08:00
struct device * dev = & pdev - > dev ;
2007-06-12 18:09:50 +04:00
2009-02-10 20:30:37 -08:00
wdt_clk = clk_get ( dev , NULL ) ;
if ( WARN_ON ( IS_ERR ( wdt_clk ) ) )
return PTR_ERR ( wdt_clk ) ;
clk_enable ( wdt_clk ) ;
2007-06-12 18:09:50 +04:00
if ( heartbeat < 1 | | heartbeat > MAX_HEARTBEAT )
heartbeat = DEFAULT_HEARTBEAT ;
2009-01-29 14:14:30 -08:00
dev_info ( dev , " heartbeat %d sec \n " , heartbeat ) ;
2007-06-12 18:09:50 +04:00
res = platform_get_resource ( pdev , IORESOURCE_MEM , 0 ) ;
if ( res = = NULL ) {
2009-01-29 14:14:30 -08:00
dev_err ( dev , " failed to get memory region resource \n " ) ;
2007-06-12 18:09:50 +04:00
return - ENOENT ;
}
2009-12-04 12:24:04 -05:00
size = resource_size ( res ) ;
2007-06-12 18:09:50 +04:00
wdt_mem = request_mem_region ( res - > start , size , pdev - > name ) ;
if ( wdt_mem = = NULL ) {
2009-01-29 14:14:30 -08:00
dev_err ( dev , " failed to get memory region \n " ) ;
2007-06-12 18:09:50 +04:00
return - ENOENT ;
}
2009-01-29 14:14:30 -08:00
wdt_base = ioremap ( res - > start , size ) ;
if ( ! wdt_base ) {
dev_err ( dev , " failed to map memory region \n " ) ;
return - ENOMEM ;
}
2007-06-12 18:09:50 +04:00
ret = misc_register ( & davinci_wdt_miscdev ) ;
if ( ret < 0 ) {
2009-01-29 14:14:30 -08:00
dev_err ( dev , " cannot register misc device \n " ) ;
2007-06-12 18:09:50 +04:00
release_resource ( wdt_mem ) ;
kfree ( wdt_mem ) ;
} else {
set_bit ( WDT_DEVICE_INITED , & wdt_status ) ;
}
2009-01-29 14:14:30 -08:00
iounmap ( wdt_base ) ;
2007-06-12 18:09:50 +04:00
return ret ;
}
2009-04-14 20:30:55 +00:00
static int __devexit davinci_wdt_remove ( struct platform_device * pdev )
2007-06-12 18:09:50 +04:00
{
misc_deregister ( & davinci_wdt_miscdev ) ;
if ( wdt_mem ) {
release_resource ( wdt_mem ) ;
kfree ( wdt_mem ) ;
wdt_mem = NULL ;
}
2009-02-10 20:30:37 -08:00
clk_disable ( wdt_clk ) ;
clk_put ( wdt_clk ) ;
2007-06-12 18:09:50 +04:00
return 0 ;
}
static struct platform_driver platform_wdt_driver = {
. driver = {
. name = " watchdog " ,
2008-04-10 21:29:23 -07:00
. owner = THIS_MODULE ,
2007-06-12 18:09:50 +04:00
} ,
. probe = davinci_wdt_probe ,
2009-04-14 20:30:55 +00:00
. remove = __devexit_p ( davinci_wdt_remove ) ,
2007-06-12 18:09:50 +04:00
} ;
static int __init davinci_wdt_init ( void )
{
return platform_driver_register ( & platform_wdt_driver ) ;
}
static void __exit davinci_wdt_exit ( void )
{
2007-10-17 15:42:22 +02:00
platform_driver_unregister ( & platform_wdt_driver ) ;
2007-06-12 18:09:50 +04:00
}
module_init ( davinci_wdt_init ) ;
module_exit ( davinci_wdt_exit ) ;
MODULE_AUTHOR ( " Texas Instruments " ) ;
MODULE_DESCRIPTION ( " DaVinci Watchdog Driver " ) ;
module_param ( heartbeat , int , 0 ) ;
MODULE_PARM_DESC ( heartbeat ,
" Watchdog heartbeat period in seconds from 1 to "
__MODULE_STRING ( MAX_HEARTBEAT ) " , default "
__MODULE_STRING ( DEFAULT_HEARTBEAT ) ) ;
MODULE_LICENSE ( " GPL " ) ;
MODULE_ALIAS_MISCDEV ( WATCHDOG_MINOR ) ;
2008-04-10 21:29:23 -07:00
MODULE_ALIAS ( " platform:watchdog " ) ;