advantechwdt.c
上传用户:jlfgdled
上传日期:2013-04-10
资源大小:33168k
文件大小:6k
源码类别:

Linux/Unix编程

开发平台:

Unix_Linux

  1. /*
  2.  * Advantech Single Board Computer WDT driver for Linux 2.4.x
  3.  *
  4.  * (c) Copyright 2000-2001 Marek Michalkiewicz <marekm@linux.org.pl>
  5.  *
  6.  * Based on acquirewdt.c which is based on wdt.c.
  7.  * Original copyright messages:
  8.  *
  9.  * (c) Copyright 1996 Alan Cox <alan@redhat.com>, All Rights Reserved.
  10.  * http://www.redhat.com
  11.  *
  12.  * This program is free software; you can redistribute it and/or
  13.  * modify it under the terms of the GNU General Public License
  14.  * as published by the Free Software Foundation; either version
  15.  * 2 of the License, or (at your option) any later version.
  16.  *
  17.  * Neither Alan Cox nor CymruNet Ltd. admit liability nor provide
  18.  * warranty for any of this software. This material is provided
  19.  * "AS-IS" and at no charge.
  20.  *
  21.  * (c) Copyright 1995    Alan Cox <alan@redhat.com>
  22.  *
  23.  * 14-Dec-2001 Matt Domsch <Matt_Domsch@dell.com>
  24.  *     Added nowayout module option to override CONFIG_WATCHDOG_NOWAYOUT
  25.  */
  26. #include <linux/config.h>
  27. #include <linux/module.h>
  28. #include <linux/types.h>
  29. #include <linux/errno.h>
  30. #include <linux/kernel.h>
  31. #include <linux/miscdevice.h>
  32. #include <linux/watchdog.h>
  33. #include <linux/ioport.h>
  34. #include <linux/fcntl.h>
  35. #include <asm/io.h>
  36. #include <asm/uaccess.h>
  37. #include <asm/system.h>
  38. #include <linux/notifier.h>
  39. #include <linux/reboot.h>
  40. #include <linux/init.h>
  41. static unsigned long advwdt_is_open;
  42. static char adv_expect_close;
  43. /*
  44.  * You must set these - there is no sane way to probe for this board.
  45.  *
  46.  * To enable or restart, write the timeout value in seconds (1 to 63)
  47.  * to I/O port wdt_start.  To disable, read I/O port wdt_stop.
  48.  * Both are 0x443 for most boards (tested on a PCA-6276VE-00B1), but
  49.  * check your manual (at least the PCA-6159 seems to be different -
  50.  * the manual says wdt_stop is 0x43, not 0x443).
  51.  * (0x43 is also a write-only control register for the 8254 timer!)
  52.  */
  53.  
  54. static int wdt_stop = 0x443;
  55. static int wdt_start = 0x443;
  56. static int wd_margin = 60; /* 60 sec default timeout */
  57. #ifdef CONFIG_WATCHDOG_NOWAYOUT
  58. static int nowayout = 1;
  59. #else
  60. static int nowayout = 0;
  61. #endif
  62. MODULE_PARM(nowayout,"i");
  63. MODULE_PARM_DESC(nowayout, "Watchdog cannot be stopped once started (default=CONFIG_WATCHDOG_NOWAYOUT)");
  64. /*
  65.  * Kernel methods.
  66.  */
  67. #ifndef MODULE
  68. static int __init adv_setup(char *str)
  69. {
  70. int ints[4];
  71. str = get_options(str, ARRAY_SIZE(ints), ints);
  72. if(ints[0] > 0){
  73. wdt_stop = ints[1];
  74. if(ints[0] > 1)
  75. wdt_start = ints[2];
  76. }
  77. return 1;
  78. }
  79. __setup("advwdt=", adv_setup);
  80. #endif /* !MODULE */
  81. MODULE_PARM(wdt_stop, "i");
  82. MODULE_PARM_DESC(wdt_stop, "Advantech WDT 'stop' io port (default 0x443)");
  83. MODULE_PARM(wdt_start, "i");
  84. MODULE_PARM_DESC(wdt_start, "Advantech WDT 'start' io port (default 0x443)");
  85. static void
  86. advwdt_ping(void)
  87. {
  88. /* Write a watchdog value */
  89. outb_p(wd_margin, wdt_start);
  90. }
  91. static void
  92. advwdt_disable(void)
  93. {
  94. inb_p(wdt_stop);
  95. }
  96. static ssize_t
  97. advwdt_write(struct file *file, const char *buf, size_t count, loff_t *ppos)
  98. {
  99. /*  Can't seek (pwrite) on this device  */
  100. if (ppos != &file->f_pos)
  101. return -ESPIPE;
  102. if (count) {
  103. if (!nowayout) {
  104. size_t i;
  105. adv_expect_close = 0;
  106. for (i = 0; i != count; i++) {
  107. char c;
  108. if(get_user(c, buf+i))
  109. return -EFAULT;
  110. if (c == 'V')
  111. adv_expect_close = 42;
  112. }
  113. }
  114. advwdt_ping();
  115. }
  116. return count;
  117. }
  118. static int
  119. advwdt_ioctl(struct inode *inode, struct file *file, unsigned int cmd,
  120.   unsigned long arg)
  121. {
  122. int new_margin;
  123. static struct watchdog_info ident = {
  124. options: WDIOF_KEEPALIVEPING | WDIOF_SETTIMEOUT | WDIOF_MAGICCLOSE,
  125. firmware_version: 0,
  126. identity: "Advantech WDT"
  127. };
  128. switch (cmd) {
  129. case WDIOC_GETSUPPORT:
  130.   if (copy_to_user((struct watchdog_info *)arg, &ident, sizeof(ident)))
  131.     return -EFAULT;
  132.   break;
  133. case WDIOC_GETSTATUS:
  134. case WDIOC_GETBOOTSTATUS:
  135.   return put_user(0, (int *)arg);
  136. case WDIOC_KEEPALIVE:
  137.   advwdt_ping();
  138.   break;
  139. case WDIOC_SETTIMEOUT:
  140.   if (get_user(new_margin, (int *)arg))
  141.   return -EFAULT;
  142.   if ((new_margin < 1) || (new_margin > 63))
  143.   return -EINVAL;
  144.   wd_margin = new_margin;
  145.   advwdt_ping();
  146.   /* Fall */
  147. case WDIOC_GETTIMEOUT:
  148.   return put_user(wd_margin, (int *)arg);
  149. case WDIOC_SETOPTIONS:
  150. {
  151.   int options, retval = -EINVAL;
  152.   if (get_user(options, (int *)arg))
  153.     return -EFAULT;
  154.   if (options & WDIOS_DISABLECARD) {
  155.     advwdt_disable();
  156.     retval = 0;
  157.   }
  158.   if (options & WDIOS_ENABLECARD) {
  159.     advwdt_ping();
  160.     retval = 0;
  161.   }
  162.   return retval;
  163. }
  164. default:
  165.   return -ENOTTY;
  166. }
  167. return 0;
  168. }
  169. static int
  170. advwdt_open(struct inode *inode, struct file *file)
  171. {
  172. if (test_and_set_bit(0, &advwdt_is_open))
  173. return -EBUSY;
  174. /*
  175.  * Activate
  176.  */
  177. advwdt_ping();
  178. return 0;
  179. }
  180. static int
  181. advwdt_close(struct inode *inode, struct file *file)
  182. {
  183. if (adv_expect_close == 42) {
  184. advwdt_disable();
  185. } else {
  186. printk(KERN_CRIT "advancetechwdt: Unexpected close, not stopping watchdog!n");
  187. advwdt_ping();
  188. }
  189. clear_bit(0, &advwdt_is_open);
  190. adv_expect_close = 0;
  191. return 0;
  192. }
  193. /*
  194.  * Notifier for system down
  195.  */
  196. static int
  197. advwdt_notify_sys(struct notifier_block *this, unsigned long code,
  198. void *unused)
  199. {
  200. if (code == SYS_DOWN || code == SYS_HALT) {
  201. /* Turn the WDT off */
  202. advwdt_disable();
  203. }
  204. return NOTIFY_DONE;
  205. }
  206.  
  207. /*
  208.  * Kernel Interfaces
  209.  */
  210.  
  211. static struct file_operations advwdt_fops = {
  212. owner: THIS_MODULE,
  213. llseek: no_llseek,
  214. write: advwdt_write,
  215. ioctl: advwdt_ioctl,
  216. open: advwdt_open,
  217. release: advwdt_close,
  218. };
  219. static struct miscdevice advwdt_miscdev = {
  220. minor: WATCHDOG_MINOR,
  221. name: "watchdog",
  222. fops: &advwdt_fops,
  223. };
  224. /*
  225.  * The WDT needs to learn about soft shutdowns in order to
  226.  * turn the timebomb registers off.
  227.  */
  228.  
  229. static struct notifier_block advwdt_notifier = {
  230. advwdt_notify_sys,
  231. NULL,
  232. 0
  233. };
  234. static int __init
  235. advwdt_init(void)
  236. {
  237. printk(KERN_INFO "WDT driver for Advantech single board computer initialising.n");
  238. misc_register(&advwdt_miscdev);
  239. if(wdt_stop != wdt_start)
  240. request_region(wdt_stop, 1, "Advantech WDT");
  241. request_region(wdt_start, 1, "Advantech WDT");
  242. register_reboot_notifier(&advwdt_notifier);
  243. return 0;
  244. }
  245. static void __exit
  246. advwdt_exit(void)
  247. {
  248. misc_deregister(&advwdt_miscdev);
  249. unregister_reboot_notifier(&advwdt_notifier);
  250. if(wdt_stop != wdt_start)
  251. release_region(wdt_stop,1);
  252. release_region(wdt_start,1);
  253. }
  254. module_init(advwdt_init);
  255. module_exit(advwdt_exit);
  256. MODULE_LICENSE("GPL");
  257. MODULE_AUTHOR("Marek Michalkiewicz <marekm@linux.org.pl>");
  258. MODULE_DESCRIPTION("Advantech Single Board Computer WDT driver");
  259. EXPORT_NO_SYMBOLS;
  260. /* end of advantechwdt.c */