ts72xx_wdt.c 11 KB

123456789101112131415161718192021222324252627282930313233343536373839404142434445464748495051525354555657585960616263646566676869707172737475767778798081828384858687888990919293949596979899100101102103104105106107108109110111112113114115116117118119120121122123124125126127128129130131132133134135136137138139140141142143144145146147148149150151152153154155156157158159160161162163164165166167168169170171172173174175176177178179180181182183184185186187188189190191192193194195196197198199200201202203204205206207208209210211212213214215216217218219220221222223224225226227228229230231232233234235236237238239240241242243244245246247248249250251252253254255256257258259260261262263264265266267268269270271272273274275276277278279280281282283284285286287288289290291292293294295296297298299300301302303304305306307308309310311312313314315316317318319320321322323324325326327328329330331332333334335336337338339340341342343344345346347348349350351352353354355356357358359360361362363364365366367368369370371372373374375376377378379380381382383384385386387388389390391392393394395396397398399400401402403404405406407408409410411412413414415416417418419420421422423424425426427428429430431432433434435436437438439440441442443444445446447448449450451452453454455456457458459460461462463464465466467468469470471472473474475476477478479480481482483484485486487488489490491492493494495496497498499500501502503504505506507508509510511512513514515516517518519520
  1. /*
  2. * Watchdog driver for Technologic Systems TS-72xx based SBCs
  3. * (TS-7200, TS-7250 and TS-7260). These boards have external
  4. * glue logic CPLD chip, which includes programmable watchdog
  5. * timer.
  6. *
  7. * Copyright (c) 2009 Mika Westerberg <mika.westerberg@iki.fi>
  8. *
  9. * This driver is based on ep93xx_wdt and wm831x_wdt drivers.
  10. *
  11. * This file is licensed under the terms of the GNU General Public
  12. * License version 2. This program is licensed "as is" without any
  13. * warranty of any kind, whether express or implied.
  14. */
  15. #include <linux/fs.h>
  16. #include <linux/io.h>
  17. #include <linux/module.h>
  18. #include <linux/moduleparam.h>
  19. #include <linux/miscdevice.h>
  20. #include <linux/mutex.h>
  21. #include <linux/platform_device.h>
  22. #include <linux/watchdog.h>
  23. #include <linux/uaccess.h>
  24. #define TS72XX_WDT_FEED_VAL 0x05
  25. #define TS72XX_WDT_DEFAULT_TIMEOUT 8
  26. static int timeout = TS72XX_WDT_DEFAULT_TIMEOUT;
  27. module_param(timeout, int, 0);
  28. MODULE_PARM_DESC(timeout, "Watchdog timeout in seconds. "
  29. "(1 <= timeout <= 8, default="
  30. __MODULE_STRING(TS72XX_WDT_DEFAULT_TIMEOUT)
  31. ")");
  32. static int nowayout = WATCHDOG_NOWAYOUT;
  33. module_param(nowayout, int, 0);
  34. MODULE_PARM_DESC(nowayout, "Disable watchdog shutdown on close");
  35. /**
  36. * struct ts72xx_wdt - watchdog control structure
  37. * @lock: lock that protects this structure
  38. * @regval: watchdog timeout value suitable for control register
  39. * @flags: flags controlling watchdog device state
  40. * @control_reg: watchdog control register
  41. * @feed_reg: watchdog feed register
  42. * @pdev: back pointer to platform dev
  43. */
  44. struct ts72xx_wdt {
  45. struct mutex lock;
  46. int regval;
  47. #define TS72XX_WDT_BUSY_FLAG 1
  48. #define TS72XX_WDT_EXPECT_CLOSE_FLAG 2
  49. int flags;
  50. void __iomem *control_reg;
  51. void __iomem *feed_reg;
  52. struct platform_device *pdev;
  53. };
  54. struct platform_device *ts72xx_wdt_pdev;
  55. /*
  56. * TS-72xx Watchdog supports following timeouts (value written
  57. * to control register):
  58. * value description
  59. * -------------------------
  60. * 0x00 watchdog disabled
  61. * 0x01 250ms
  62. * 0x02 500ms
  63. * 0x03 1s
  64. * 0x04 reserved
  65. * 0x05 2s
  66. * 0x06 4s
  67. * 0x07 8s
  68. *
  69. * Timeouts below 1s are not very usable so we don't
  70. * allow them at all.
  71. *
  72. * We provide two functions that convert between these:
  73. * timeout_to_regval() and regval_to_timeout().
  74. */
  75. static const struct {
  76. int timeout;
  77. int regval;
  78. } ts72xx_wdt_map[] = {
  79. { 1, 3 },
  80. { 2, 5 },
  81. { 4, 6 },
  82. { 8, 7 },
  83. };
  84. /**
  85. * timeout_to_regval() - converts given timeout to control register value
  86. * @new_timeout: timeout in seconds to be converted
  87. *
  88. * Function converts given @new_timeout into valid value that can
  89. * be programmed into watchdog control register. When conversion is
  90. * not possible, function returns %-EINVAL.
  91. */
  92. static int timeout_to_regval(int new_timeout)
  93. {
  94. int i;
  95. /* first limit it to 1 - 8 seconds */
  96. new_timeout = clamp_val(new_timeout, 1, 8);
  97. for (i = 0; i < ARRAY_SIZE(ts72xx_wdt_map); i++) {
  98. if (ts72xx_wdt_map[i].timeout >= new_timeout)
  99. return ts72xx_wdt_map[i].regval;
  100. }
  101. return -EINVAL;
  102. }
  103. /**
  104. * regval_to_timeout() - converts control register value to timeout
  105. * @regval: control register value to be converted
  106. *
  107. * Function converts given @regval to timeout in seconds (1, 2, 4 or 8).
  108. * If @regval cannot be converted, function returns %-EINVAL.
  109. */
  110. static int regval_to_timeout(int regval)
  111. {
  112. int i;
  113. for (i = 0; i < ARRAY_SIZE(ts72xx_wdt_map); i++) {
  114. if (ts72xx_wdt_map[i].regval == regval)
  115. return ts72xx_wdt_map[i].timeout;
  116. }
  117. return -EINVAL;
  118. }
  119. /**
  120. * ts72xx_wdt_kick() - kick the watchdog
  121. * @wdt: watchdog to be kicked
  122. *
  123. * Called with @wdt->lock held.
  124. */
  125. static inline void ts72xx_wdt_kick(struct ts72xx_wdt *wdt)
  126. {
  127. __raw_writeb(TS72XX_WDT_FEED_VAL, wdt->feed_reg);
  128. }
  129. /**
  130. * ts72xx_wdt_start() - starts the watchdog timer
  131. * @wdt: watchdog to be started
  132. *
  133. * This function programs timeout to watchdog timer
  134. * and starts it.
  135. *
  136. * Called with @wdt->lock held.
  137. */
  138. static void ts72xx_wdt_start(struct ts72xx_wdt *wdt)
  139. {
  140. /*
  141. * To program the wdt, it first must be "fed" and
  142. * only after that (within 30 usecs) the configuration
  143. * can be changed.
  144. */
  145. ts72xx_wdt_kick(wdt);
  146. __raw_writeb((u8)wdt->regval, wdt->control_reg);
  147. }
  148. /**
  149. * ts72xx_wdt_stop() - stops the watchdog timer
  150. * @wdt: watchdog to be stopped
  151. *
  152. * Called with @wdt->lock held.
  153. */
  154. static void ts72xx_wdt_stop(struct ts72xx_wdt *wdt)
  155. {
  156. ts72xx_wdt_kick(wdt);
  157. __raw_writeb(0, wdt->control_reg);
  158. }
  159. static int ts72xx_wdt_open(struct inode *inode, struct file *file)
  160. {
  161. struct ts72xx_wdt *wdt = platform_get_drvdata(ts72xx_wdt_pdev);
  162. int regval;
  163. /*
  164. * Try to convert default timeout to valid register
  165. * value first.
  166. */
  167. regval = timeout_to_regval(timeout);
  168. if (regval < 0) {
  169. dev_err(&wdt->pdev->dev,
  170. "failed to convert timeout (%d) to register value\n",
  171. timeout);
  172. return -EINVAL;
  173. }
  174. if (mutex_lock_interruptible(&wdt->lock))
  175. return -ERESTARTSYS;
  176. if ((wdt->flags & TS72XX_WDT_BUSY_FLAG) != 0) {
  177. mutex_unlock(&wdt->lock);
  178. return -EBUSY;
  179. }
  180. wdt->flags = TS72XX_WDT_BUSY_FLAG;
  181. wdt->regval = regval;
  182. file->private_data = wdt;
  183. ts72xx_wdt_start(wdt);
  184. mutex_unlock(&wdt->lock);
  185. return nonseekable_open(inode, file);
  186. }
  187. static int ts72xx_wdt_release(struct inode *inode, struct file *file)
  188. {
  189. struct ts72xx_wdt *wdt = file->private_data;
  190. if (mutex_lock_interruptible(&wdt->lock))
  191. return -ERESTARTSYS;
  192. if ((wdt->flags & TS72XX_WDT_EXPECT_CLOSE_FLAG) != 0) {
  193. ts72xx_wdt_stop(wdt);
  194. } else {
  195. dev_warn(&wdt->pdev->dev,
  196. "TS-72XX WDT device closed unexpectly. "
  197. "Watchdog timer will not stop!\n");
  198. /*
  199. * Kick it one more time, to give userland some time
  200. * to recover (for example, respawning the kicker
  201. * daemon).
  202. */
  203. ts72xx_wdt_kick(wdt);
  204. }
  205. wdt->flags = 0;
  206. mutex_unlock(&wdt->lock);
  207. return 0;
  208. }
  209. static ssize_t ts72xx_wdt_write(struct file *file,
  210. const char __user *data,
  211. size_t len,
  212. loff_t *ppos)
  213. {
  214. struct ts72xx_wdt *wdt = file->private_data;
  215. if (!len)
  216. return 0;
  217. if (mutex_lock_interruptible(&wdt->lock))
  218. return -ERESTARTSYS;
  219. ts72xx_wdt_kick(wdt);
  220. /*
  221. * Support for magic character closing. User process
  222. * writes 'V' into the device, just before it is closed.
  223. * This means that we know that the wdt timer can be
  224. * stopped after user closes the device.
  225. */
  226. if (!nowayout) {
  227. int i;
  228. for (i = 0; i < len; i++) {
  229. char c;
  230. /* In case it was set long ago */
  231. wdt->flags &= ~TS72XX_WDT_EXPECT_CLOSE_FLAG;
  232. if (get_user(c, data + i)) {
  233. mutex_unlock(&wdt->lock);
  234. return -EFAULT;
  235. }
  236. if (c == 'V') {
  237. wdt->flags |= TS72XX_WDT_EXPECT_CLOSE_FLAG;
  238. break;
  239. }
  240. }
  241. }
  242. mutex_unlock(&wdt->lock);
  243. return len;
  244. }
  245. static const struct watchdog_info winfo = {
  246. .options = WDIOF_KEEPALIVEPING | WDIOF_SETTIMEOUT |
  247. WDIOF_MAGICCLOSE,
  248. .firmware_version = 1,
  249. .identity = "TS-72XX WDT",
  250. };
  251. static long ts72xx_wdt_ioctl(struct file *file, unsigned int cmd,
  252. unsigned long arg)
  253. {
  254. struct ts72xx_wdt *wdt = file->private_data;
  255. void __user *argp = (void __user *)arg;
  256. int __user *p = (int __user *)argp;
  257. int error = 0;
  258. if (mutex_lock_interruptible(&wdt->lock))
  259. return -ERESTARTSYS;
  260. switch (cmd) {
  261. case WDIOC_GETSUPPORT:
  262. error = copy_to_user(argp, &winfo, sizeof(winfo));
  263. break;
  264. case WDIOC_GETSTATUS:
  265. case WDIOC_GETBOOTSTATUS:
  266. return put_user(0, p);
  267. case WDIOC_KEEPALIVE:
  268. ts72xx_wdt_kick(wdt);
  269. break;
  270. case WDIOC_SETOPTIONS: {
  271. int options;
  272. if (get_user(options, p)) {
  273. error = -EFAULT;
  274. break;
  275. }
  276. error = -EINVAL;
  277. if ((options & WDIOS_DISABLECARD) != 0) {
  278. ts72xx_wdt_stop(wdt);
  279. error = 0;
  280. }
  281. if ((options & WDIOS_ENABLECARD) != 0) {
  282. ts72xx_wdt_start(wdt);
  283. error = 0;
  284. }
  285. break;
  286. }
  287. case WDIOC_SETTIMEOUT: {
  288. int new_timeout;
  289. if (get_user(new_timeout, p)) {
  290. error = -EFAULT;
  291. } else {
  292. int regval;
  293. regval = timeout_to_regval(new_timeout);
  294. if (regval < 0) {
  295. error = -EINVAL;
  296. } else {
  297. ts72xx_wdt_stop(wdt);
  298. wdt->regval = regval;
  299. ts72xx_wdt_start(wdt);
  300. }
  301. }
  302. if (error)
  303. break;
  304. /*FALLTHROUGH*/
  305. }
  306. case WDIOC_GETTIMEOUT:
  307. if (put_user(regval_to_timeout(wdt->regval), p))
  308. error = -EFAULT;
  309. break;
  310. default:
  311. error = -ENOTTY;
  312. break;
  313. }
  314. mutex_unlock(&wdt->lock);
  315. return error;
  316. }
  317. static const struct file_operations ts72xx_wdt_fops = {
  318. .owner = THIS_MODULE,
  319. .llseek = no_llseek,
  320. .open = ts72xx_wdt_open,
  321. .release = ts72xx_wdt_release,
  322. .write = ts72xx_wdt_write,
  323. .unlocked_ioctl = ts72xx_wdt_ioctl,
  324. };
  325. static struct miscdevice ts72xx_wdt_miscdev = {
  326. .minor = WATCHDOG_MINOR,
  327. .name = "watchdog",
  328. .fops = &ts72xx_wdt_fops,
  329. };
  330. static __devinit int ts72xx_wdt_probe(struct platform_device *pdev)
  331. {
  332. struct ts72xx_wdt *wdt;
  333. struct resource *r1, *r2;
  334. int error = 0;
  335. wdt = kzalloc(sizeof(struct ts72xx_wdt), GFP_KERNEL);
  336. if (!wdt) {
  337. dev_err(&pdev->dev, "failed to allocate memory\n");
  338. return -ENOMEM;
  339. }
  340. r1 = platform_get_resource(pdev, IORESOURCE_MEM, 0);
  341. if (!r1) {
  342. dev_err(&pdev->dev, "failed to get memory resource\n");
  343. error = -ENODEV;
  344. goto fail;
  345. }
  346. r1 = request_mem_region(r1->start, resource_size(r1), pdev->name);
  347. if (!r1) {
  348. dev_err(&pdev->dev, "cannot request memory region\n");
  349. error = -EBUSY;
  350. goto fail;
  351. }
  352. wdt->control_reg = ioremap(r1->start, resource_size(r1));
  353. if (!wdt->control_reg) {
  354. dev_err(&pdev->dev, "failed to map memory\n");
  355. error = -ENODEV;
  356. goto fail_free_control;
  357. }
  358. r2 = platform_get_resource(pdev, IORESOURCE_MEM, 1);
  359. if (!r2) {
  360. dev_err(&pdev->dev, "failed to get memory resource\n");
  361. error = -ENODEV;
  362. goto fail_unmap_control;
  363. }
  364. r2 = request_mem_region(r2->start, resource_size(r2), pdev->name);
  365. if (!r2) {
  366. dev_err(&pdev->dev, "cannot request memory region\n");
  367. error = -EBUSY;
  368. goto fail_unmap_control;
  369. }
  370. wdt->feed_reg = ioremap(r2->start, resource_size(r2));
  371. if (!wdt->feed_reg) {
  372. dev_err(&pdev->dev, "failed to map memory\n");
  373. error = -ENODEV;
  374. goto fail_free_feed;
  375. }
  376. platform_set_drvdata(pdev, wdt);
  377. ts72xx_wdt_pdev = pdev;
  378. wdt->pdev = pdev;
  379. mutex_init(&wdt->lock);
  380. error = misc_register(&ts72xx_wdt_miscdev);
  381. if (error) {
  382. dev_err(&pdev->dev, "failed to register miscdev\n");
  383. goto fail_unmap_feed;
  384. }
  385. dev_info(&pdev->dev, "TS-72xx Watchdog driver\n");
  386. return 0;
  387. fail_unmap_feed:
  388. platform_set_drvdata(pdev, NULL);
  389. iounmap(wdt->feed_reg);
  390. fail_free_feed:
  391. release_mem_region(r2->start, resource_size(r2));
  392. fail_unmap_control:
  393. iounmap(wdt->control_reg);
  394. fail_free_control:
  395. release_mem_region(r1->start, resource_size(r1));
  396. fail:
  397. kfree(wdt);
  398. return error;
  399. }
  400. static __devexit int ts72xx_wdt_remove(struct platform_device *pdev)
  401. {
  402. struct ts72xx_wdt *wdt = platform_get_drvdata(pdev);
  403. struct resource *res;
  404. int error;
  405. error = misc_deregister(&ts72xx_wdt_miscdev);
  406. platform_set_drvdata(pdev, NULL);
  407. iounmap(wdt->feed_reg);
  408. res = platform_get_resource(pdev, IORESOURCE_MEM, 1);
  409. release_mem_region(res->start, resource_size(res));
  410. iounmap(wdt->control_reg);
  411. res = platform_get_resource(pdev, IORESOURCE_MEM, 0);
  412. release_mem_region(res->start, resource_size(res));
  413. kfree(wdt);
  414. return error;
  415. }
  416. static struct platform_driver ts72xx_wdt_driver = {
  417. .probe = ts72xx_wdt_probe,
  418. .remove = __devexit_p(ts72xx_wdt_remove),
  419. .driver = {
  420. .name = "ts72xx-wdt",
  421. .owner = THIS_MODULE,
  422. },
  423. };
  424. static __init int ts72xx_wdt_init(void)
  425. {
  426. return platform_driver_register(&ts72xx_wdt_driver);
  427. }
  428. module_init(ts72xx_wdt_init);
  429. static __exit void ts72xx_wdt_exit(void)
  430. {
  431. platform_driver_unregister(&ts72xx_wdt_driver);
  432. }
  433. module_exit(ts72xx_wdt_exit);
  434. MODULE_AUTHOR("Mika Westerberg <mika.westerberg@iki.fi>");
  435. MODULE_DESCRIPTION("TS-72xx SBC Watchdog");
  436. MODULE_LICENSE("GPL");
  437. MODULE_ALIAS("platform:ts72xx-wdt");