1 /* Watchdog timer for the Geode GX/LX with the CS5535/CS5536 companion chip
 
   3  * Copyright (C) 2006-2007, Advanced Micro Devices, Inc.
 
   5  * This program is free software; you can redistribute it and/or
 
   6  * modify it under the terms of the GNU General Public License
 
   7  * as published by the Free Software Foundation; either version
 
   8  * 2 of the License, or (at your option) any later version.
 
  12 #include <linux/module.h>
 
  13 #include <linux/moduleparam.h>
 
  14 #include <linux/types.h>
 
  15 #include <linux/miscdevice.h>
 
  16 #include <linux/watchdog.h>
 
  18 #include <linux/platform_device.h>
 
  19 #include <linux/reboot.h>
 
  21 #include <asm/uaccess.h>
 
  22 #include <asm/geode.h>
 
  24 #define GEODEWDT_HZ 500
 
  25 #define GEODEWDT_SCALE 6
 
  26 #define GEODEWDT_MAX_SECONDS 131
 
  28 #define WDT_FLAGS_OPEN 1
 
  29 #define WDT_FLAGS_ORPHAN 2
 
  31 #define DRV_NAME "geodewdt"
 
  32 #define WATCHDOG_NAME "Geode GX/LX WDT"
 
  33 #define WATCHDOG_TIMEOUT 60
 
  35 static int timeout = WATCHDOG_TIMEOUT;
 
  36 module_param(timeout, int, 0);
 
  37 MODULE_PARM_DESC(timeout, "Watchdog timeout in seconds. 1<= timeout <=131, default=" __MODULE_STRING(WATCHDOG_TIMEOUT) ".");
 
  39 static int nowayout = WATCHDOG_NOWAYOUT;
 
  40 module_param(nowayout, int, 0);
 
  41 MODULE_PARM_DESC(nowayout, "Watchdog cannot be stopped once started (default=" __MODULE_STRING(WATCHDOG_NOWAYOUT) ")");
 
  43 static struct platform_device *geodewdt_platform_device;
 
  44 static unsigned long wdt_flags;
 
  46 static int safe_close;
 
  48 static void geodewdt_ping(void)
 
  50         /* Stop the counter */
 
  51         geode_mfgpt_write(wdt_timer, MFGPT_REG_SETUP, 0);
 
  53         /* Reset the counter */
 
  54         geode_mfgpt_write(wdt_timer, MFGPT_REG_COUNTER, 0);
 
  56         /* Enable the counter */
 
  57         geode_mfgpt_write(wdt_timer, MFGPT_REG_SETUP, MFGPT_SETUP_CNTEN);
 
  60 static void geodewdt_disable(void)
 
  62         geode_mfgpt_write(wdt_timer, MFGPT_REG_SETUP, 0);
 
  63         geode_mfgpt_write(wdt_timer, MFGPT_REG_COUNTER, 0);
 
  66 static int geodewdt_set_heartbeat(int val)
 
  68         if (val < 1 || val > GEODEWDT_MAX_SECONDS)
 
  71         geode_mfgpt_write(wdt_timer, MFGPT_REG_SETUP, 0);
 
  72         geode_mfgpt_write(wdt_timer, MFGPT_REG_CMP2, val * GEODEWDT_HZ);
 
  73         geode_mfgpt_write(wdt_timer, MFGPT_REG_COUNTER, 0);
 
  74         geode_mfgpt_write(wdt_timer, MFGPT_REG_SETUP, MFGPT_SETUP_CNTEN);
 
  81 geodewdt_open(struct inode *inode, struct file *file)
 
  83         if (test_and_set_bit(WDT_FLAGS_OPEN, &wdt_flags))
 
  86         if (!test_and_clear_bit(WDT_FLAGS_ORPHAN, &wdt_flags))
 
  87                 __module_get(THIS_MODULE);
 
  90         return nonseekable_open(inode, file);
 
  94 geodewdt_release(struct inode *inode, struct file *file)
 
  98                 module_put(THIS_MODULE);
 
 101                 printk(KERN_CRIT "Unexpected close - watchdog is not stopping.\n");
 
 104                 set_bit(WDT_FLAGS_ORPHAN, &wdt_flags);
 
 107         clear_bit(WDT_FLAGS_OPEN, &wdt_flags);
 
 113 geodewdt_write(struct file *file, const char __user *data, size_t len,
 
 121                         for (i = 0; i != len; i++) {
 
 124                                 if (get_user(c, data + i))
 
 138 geodewdt_ioctl(struct inode *inode, struct file *file, unsigned int cmd,
 
 141         void __user *argp = (void __user *)arg;
 
 142         int __user *p = argp;
 
 145         static struct watchdog_info ident = {
 
 146                 .options = WDIOF_SETTIMEOUT | WDIOF_KEEPALIVEPING
 
 148                 .firmware_version =     1,
 
 149                 .identity =             WATCHDOG_NAME,
 
 153         case WDIOC_GETSUPPORT:
 
 154                 return copy_to_user(argp, &ident,
 
 155                                     sizeof(ident)) ? -EFAULT : 0;
 
 158         case WDIOC_GETSTATUS:
 
 159         case WDIOC_GETBOOTSTATUS:
 
 160                 return put_user(0, p);
 
 162         case WDIOC_KEEPALIVE:
 
 166         case WDIOC_SETTIMEOUT:
 
 167                 if (get_user(interval, p))
 
 170                 if (geodewdt_set_heartbeat(interval))
 
 175         case WDIOC_GETTIMEOUT:
 
 176                 return put_user(timeout, p);
 
 178         case WDIOC_SETOPTIONS:
 
 180                 int options, ret = -EINVAL;
 
 182                 if (get_user(options, p))
 
 185                 if (options & WDIOS_DISABLECARD) {
 
 190                 if (options & WDIOS_ENABLECARD) {
 
 204 static const struct file_operations geodewdt_fops = {
 
 205         .owner          = THIS_MODULE,
 
 207         .write          = geodewdt_write,
 
 208         .ioctl          = geodewdt_ioctl,
 
 209         .open           = geodewdt_open,
 
 210         .release        = geodewdt_release,
 
 213 static struct miscdevice geodewdt_miscdev = {
 
 214         .minor = WATCHDOG_MINOR,
 
 216         .fops = &geodewdt_fops
 
 220 geodewdt_probe(struct platform_device *dev)
 
 224         timer = geode_mfgpt_alloc_timer(MFGPT_TIMER_ANY, MFGPT_DOMAIN_WORKING);
 
 227                 printk(KERN_ERR "geodewdt:  No timers were available\n");
 
 233         /* Set up the timer */
 
 235         geode_mfgpt_write(wdt_timer, MFGPT_REG_SETUP,
 
 236                           GEODEWDT_SCALE | (3 << 8));
 
 238         /* Set up comparator 2 to reset when the event fires */
 
 239         geode_mfgpt_toggle_event(wdt_timer, MFGPT_CMP2, MFGPT_EVENT_RESET, 1);
 
 241         /* Set up the initial timeout */
 
 243         geode_mfgpt_write(wdt_timer, MFGPT_REG_CMP2,
 
 244                 timeout * GEODEWDT_HZ);
 
 246         ret = misc_register(&geodewdt_miscdev);
 
 252 geodewdt_remove(struct platform_device *dev)
 
 254         misc_deregister(&geodewdt_miscdev);
 
 259 geodewdt_shutdown(struct platform_device *dev)
 
 264 static struct platform_driver geodewdt_driver = {
 
 265         .probe          = geodewdt_probe,
 
 266         .remove         = __devexit_p(geodewdt_remove),
 
 267         .shutdown       = geodewdt_shutdown,
 
 269                 .owner  = THIS_MODULE,
 
 279         ret = platform_driver_register(&geodewdt_driver);
 
 283         geodewdt_platform_device = platform_device_register_simple(DRV_NAME, -1, NULL, 0);
 
 284         if (IS_ERR(geodewdt_platform_device)) {
 
 285                 ret = PTR_ERR(geodewdt_platform_device);
 
 291         platform_driver_unregister(&geodewdt_driver);
 
 298         platform_device_unregister(geodewdt_platform_device);
 
 299         platform_driver_unregister(&geodewdt_driver);
 
 302 module_init(geodewdt_init);
 
 303 module_exit(geodewdt_exit);
 
 305 MODULE_AUTHOR("Advanced Micro Devices, Inc");
 
 306 MODULE_DESCRIPTION("Geode GX/LX Watchdog Driver");
 
 307 MODULE_LICENSE("GPL");
 
 308 MODULE_ALIAS_MISCDEV(WATCHDOG_MINOR);