ts72xx_wdt.c 11 KB

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