diff options
| -rw-r--r-- | drivers/char/watchdog/Kconfig | 9 | ||||
| -rw-r--r-- | drivers/char/watchdog/Makefile | 1 | ||||
| -rw-r--r-- | drivers/char/watchdog/mpcore_wdt.c | 434 |
3 files changed, 444 insertions, 0 deletions
diff --git a/drivers/char/watchdog/Kconfig b/drivers/char/watchdog/Kconfig index c3898afce3ae..fa789ea36bbe 100644 --- a/drivers/char/watchdog/Kconfig +++ b/drivers/char/watchdog/Kconfig | |||
| @@ -139,6 +139,15 @@ config SA1100_WATCHDOG | |||
| 139 | To compile this driver as a module, choose M here: the | 139 | To compile this driver as a module, choose M here: the |
| 140 | module will be called sa1100_wdt. | 140 | module will be called sa1100_wdt. |
| 141 | 141 | ||
| 142 | config MPCORE_WATCHDOG | ||
| 143 | tristate "MPcore watchdog" | ||
| 144 | depends on WATCHDOG && ARM_MPCORE_PLATFORM && LOCAL_TIMERS | ||
| 145 | help | ||
| 146 | Watchdog timer embedded into the MPcore system. | ||
| 147 | |||
| 148 | To compile this driver as a module, choose M here: the | ||
| 149 | module will be called mpcore_wdt. | ||
| 150 | |||
| 142 | # X86 (i386 + ia64 + x86_64) Architecture | 151 | # X86 (i386 + ia64 + x86_64) Architecture |
| 143 | 152 | ||
| 144 | config ACQUIRE_WDT | 153 | config ACQUIRE_WDT |
diff --git a/drivers/char/watchdog/Makefile b/drivers/char/watchdog/Makefile index cfeac6f10137..bc6f5fe88c8c 100644 --- a/drivers/char/watchdog/Makefile +++ b/drivers/char/watchdog/Makefile | |||
| @@ -29,6 +29,7 @@ obj-$(CONFIG_IXP2000_WATCHDOG) += ixp2000_wdt.o | |||
| 29 | obj-$(CONFIG_IXP4XX_WATCHDOG) += ixp4xx_wdt.o | 29 | obj-$(CONFIG_IXP4XX_WATCHDOG) += ixp4xx_wdt.o |
| 30 | obj-$(CONFIG_S3C2410_WATCHDOG) += s3c2410_wdt.o | 30 | obj-$(CONFIG_S3C2410_WATCHDOG) += s3c2410_wdt.o |
| 31 | obj-$(CONFIG_SA1100_WATCHDOG) += sa1100_wdt.o | 31 | obj-$(CONFIG_SA1100_WATCHDOG) += sa1100_wdt.o |
| 32 | obj-$(CONFIG_MPCORE_WATCHDOG) += mpcore_wdt.o | ||
| 32 | 33 | ||
| 33 | # X86 (i386 + ia64 + x86_64) Architecture | 34 | # X86 (i386 + ia64 + x86_64) Architecture |
| 34 | obj-$(CONFIG_ACQUIRE_WDT) += acquirewdt.o | 35 | obj-$(CONFIG_ACQUIRE_WDT) += acquirewdt.o |
diff --git a/drivers/char/watchdog/mpcore_wdt.c b/drivers/char/watchdog/mpcore_wdt.c new file mode 100644 index 000000000000..c694eee1fb24 --- /dev/null +++ b/drivers/char/watchdog/mpcore_wdt.c | |||
| @@ -0,0 +1,434 @@ | |||
| 1 | /* | ||
| 2 | * Watchdog driver for the mpcore watchdog timer | ||
| 3 | * | ||
| 4 | * (c) Copyright 2004 ARM Limited | ||
| 5 | * | ||
| 6 | * Based on the SoftDog driver: | ||
| 7 | * (c) Copyright 1996 Alan Cox <alan@redhat.com>, All Rights Reserved. | ||
| 8 | * http://www.redhat.com | ||
| 9 | * | ||
| 10 | * This program is free software; you can redistribute it and/or | ||
| 11 | * modify it under the terms of the GNU General Public License | ||
| 12 | * as published by the Free Software Foundation; either version | ||
| 13 | * 2 of the License, or (at your option) any later version. | ||
| 14 | * | ||
| 15 | * Neither Alan Cox nor CymruNet Ltd. admit liability nor provide | ||
| 16 | * warranty for any of this software. This material is provided | ||
| 17 | * "AS-IS" and at no charge. | ||
| 18 | * | ||
| 19 | * (c) Copyright 1995 Alan Cox <alan@lxorguk.ukuu.org.uk> | ||
| 20 | * | ||
| 21 | */ | ||
| 22 | #include <linux/module.h> | ||
| 23 | #include <linux/moduleparam.h> | ||
| 24 | #include <linux/config.h> | ||
| 25 | #include <linux/types.h> | ||
| 26 | #include <linux/miscdevice.h> | ||
| 27 | #include <linux/watchdog.h> | ||
| 28 | #include <linux/fs.h> | ||
| 29 | #include <linux/reboot.h> | ||
| 30 | #include <linux/init.h> | ||
| 31 | #include <linux/interrupt.h> | ||
| 32 | #include <linux/device.h> | ||
| 33 | #include <asm/uaccess.h> | ||
| 34 | |||
| 35 | struct mpcore_wdt { | ||
| 36 | unsigned long timer_alive; | ||
| 37 | struct device *dev; | ||
| 38 | void __iomem *base; | ||
| 39 | int irq; | ||
| 40 | unsigned int perturb; | ||
| 41 | char expect_close; | ||
| 42 | }; | ||
| 43 | |||
| 44 | static struct platform_device *mpcore_wdt_dev; | ||
| 45 | |||
| 46 | extern unsigned int mpcore_timer_rate; | ||
| 47 | |||
| 48 | #define TIMER_MARGIN 60 | ||
| 49 | static int mpcore_margin = TIMER_MARGIN; | ||
| 50 | module_param(mpcore_margin, int, 0); | ||
| 51 | MODULE_PARM_DESC(mpcore_margin, "MPcore timer margin in seconds. (0<mpcore_margin<65536, default=" __MODULE_STRING(TIMER_MARGIN) ")"); | ||
| 52 | |||
| 53 | static int nowayout = WATCHDOG_NOWAYOUT; | ||
| 54 | module_param(nowayout, int, 0); | ||
| 55 | MODULE_PARM_DESC(nowayout, "Watchdog cannot be stopped once started (default=" __MODULE_STRING(WATCHDOG_NOWAYOUT) ")"); | ||
| 56 | |||
| 57 | #define ONLY_TESTING 0 | ||
| 58 | static int mpcore_noboot = ONLY_TESTING; | ||
| 59 | module_param(mpcore_noboot, int, 0); | ||
| 60 | MODULE_PARM_DESC(mpcore_noboot, "MPcore watchdog action, set to 1 to ignore reboots, 0 to reboot (default=" __MODULE_STRING(ONLY_TESTING) ")"); | ||
| 61 | |||
| 62 | /* | ||
| 63 | * This is the interrupt handler. Note that we only use this | ||
| 64 | * in testing mode, so don't actually do a reboot here. | ||
| 65 | */ | ||
| 66 | static irqreturn_t mpcore_wdt_fire(int irq, void *arg, struct pt_regs *regs) | ||
| 67 | { | ||
| 68 | struct mpcore_wdt *wdt = arg; | ||
| 69 | |||
| 70 | /* Check it really was our interrupt */ | ||
| 71 | if (readl(wdt->base + TWD_WDOG_INTSTAT)) { | ||
| 72 | dev_printk(KERN_CRIT, wdt->dev, "Triggered - Reboot ignored.\n"); | ||
| 73 | |||
| 74 | /* Clear the interrupt on the watchdog */ | ||
| 75 | writel(1, wdt->base + TWD_WDOG_INTSTAT); | ||
| 76 | |||
| 77 | return IRQ_HANDLED; | ||
| 78 | } | ||
| 79 | |||
| 80 | return IRQ_NONE; | ||
| 81 | } | ||
| 82 | |||
| 83 | /* | ||
| 84 | * mpcore_wdt_keepalive - reload the timer | ||
| 85 | * | ||
| 86 | * Note that the spec says a DIFFERENT value must be written to the reload | ||
| 87 | * register each time. The "perturb" variable deals with this by adding 1 | ||
| 88 | * to the count every other time the function is called. | ||
| 89 | */ | ||
| 90 | static void mpcore_wdt_keepalive(struct mpcore_wdt *wdt) | ||
| 91 | { | ||
| 92 | unsigned int count; | ||
| 93 | |||
| 94 | /* Assume prescale is set to 256 */ | ||
| 95 | count = (mpcore_timer_rate / 256) * mpcore_margin; | ||
| 96 | |||
| 97 | /* Reload the counter */ | ||
| 98 | writel(count + wdt->perturb, wdt->base + TWD_WDOG_LOAD); | ||
| 99 | |||
| 100 | wdt->perturb = wdt->perturb ? 0 : 1; | ||
| 101 | } | ||
| 102 | |||
| 103 | static void mpcore_wdt_stop(struct mpcore_wdt *wdt) | ||
| 104 | { | ||
| 105 | writel(0x12345678, wdt->base + TWD_WDOG_DISABLE); | ||
| 106 | writel(0x87654321, wdt->base + TWD_WDOG_DISABLE); | ||
| 107 | writel(0x0, wdt->base + TWD_WDOG_CONTROL); | ||
| 108 | } | ||
| 109 | |||
| 110 | static void mpcore_wdt_start(struct mpcore_wdt *wdt) | ||
| 111 | { | ||
| 112 | dev_printk(KERN_INFO, wdt->dev, "enabling watchdog.\n"); | ||
| 113 | |||
| 114 | /* This loads the count register but does NOT start the count yet */ | ||
| 115 | mpcore_wdt_keepalive(wdt); | ||
| 116 | |||
| 117 | if (mpcore_noboot) { | ||
| 118 | /* Enable watchdog - prescale=256, watchdog mode=0, enable=1 */ | ||
| 119 | writel(0x0000FF01, wdt->base + TWD_WDOG_CONTROL); | ||
| 120 | } else { | ||
| 121 | /* Enable watchdog - prescale=256, watchdog mode=1, enable=1 */ | ||
| 122 | writel(0x0000FF09, wdt->base + TWD_WDOG_CONTROL); | ||
| 123 | } | ||
| 124 | } | ||
| 125 | |||
| 126 | static int mpcore_wdt_set_heartbeat(int t) | ||
| 127 | { | ||
| 128 | if (t < 0x0001 || t > 0xFFFF) | ||
| 129 | return -EINVAL; | ||
| 130 | |||
| 131 | mpcore_margin = t; | ||
| 132 | return 0; | ||
| 133 | } | ||
| 134 | |||
| 135 | /* | ||
| 136 | * /dev/watchdog handling | ||
| 137 | */ | ||
| 138 | static int mpcore_wdt_open(struct inode *inode, struct file *file) | ||
| 139 | { | ||
| 140 | struct mpcore_wdt *wdt = dev_get_drvdata(&mpcore_wdt_dev->dev); | ||
| 141 | |||
| 142 | if (test_and_set_bit(0, &wdt->timer_alive)) | ||
| 143 | return -EBUSY; | ||
| 144 | |||
| 145 | if (nowayout) | ||
| 146 | __module_get(THIS_MODULE); | ||
| 147 | |||
| 148 | file->private_data = wdt; | ||
| 149 | |||
| 150 | /* | ||
| 151 | * Activate timer | ||
| 152 | */ | ||
| 153 | mpcore_wdt_start(wdt); | ||
| 154 | |||
| 155 | return nonseekable_open(inode, file); | ||
| 156 | } | ||
| 157 | |||
| 158 | static int mpcore_wdt_release(struct inode *inode, struct file *file) | ||
| 159 | { | ||
| 160 | struct mpcore_wdt *wdt = file->private_data; | ||
| 161 | |||
| 162 | /* | ||
| 163 | * Shut off the timer. | ||
| 164 | * Lock it in if it's a module and we set nowayout | ||
| 165 | */ | ||
| 166 | if (wdt->expect_close == 42) { | ||
| 167 | mpcore_wdt_stop(wdt); | ||
| 168 | } else { | ||
| 169 | dev_printk(KERN_CRIT, wdt->dev, "unexpected close, not stopping watchdog!\n"); | ||
| 170 | mpcore_wdt_keepalive(wdt); | ||
| 171 | } | ||
| 172 | clear_bit(0, &wdt->timer_alive); | ||
| 173 | wdt->expect_close = 0; | ||
| 174 | return 0; | ||
| 175 | } | ||
| 176 | |||
| 177 | static ssize_t mpcore_wdt_write(struct file *file, const char *data, size_t len, loff_t *ppos) | ||
| 178 | { | ||
| 179 | struct mpcore_wdt *wdt = file->private_data; | ||
| 180 | |||
| 181 | /* Can't seek (pwrite) on this device */ | ||
| 182 | if (ppos != &file->f_pos) | ||
| 183 | return -ESPIPE; | ||
| 184 | |||
| 185 | /* | ||
| 186 | * Refresh the timer. | ||
| 187 | */ | ||
| 188 | if (len) { | ||
| 189 | if (!nowayout) { | ||
| 190 | size_t i; | ||
| 191 | |||
| 192 | /* In case it was set long ago */ | ||
| 193 | wdt->expect_close = 0; | ||
| 194 | |||
| 195 | for (i = 0; i != len; i++) { | ||
| 196 | char c; | ||
| 197 | |||
| 198 | if (get_user(c, data + i)) | ||
| 199 | return -EFAULT; | ||
| 200 | if (c == 'V') | ||
| 201 | wdt->expect_close = 42; | ||
| 202 | } | ||
| 203 | } | ||
| 204 | mpcore_wdt_keepalive(wdt); | ||
| 205 | } | ||
| 206 | return len; | ||
| 207 | } | ||
| 208 | |||
| 209 | static struct watchdog_info ident = { | ||
| 210 | .options = WDIOF_SETTIMEOUT | | ||
| 211 | WDIOF_KEEPALIVEPING | | ||
| 212 | WDIOF_MAGICCLOSE, | ||
| 213 | .identity = "MPcore Watchdog", | ||
| 214 | }; | ||
| 215 | |||
| 216 | static int mpcore_wdt_ioctl(struct inode *inode, struct file *file, | ||
| 217 | unsigned int cmd, unsigned long arg) | ||
| 218 | { | ||
| 219 | struct mpcore_wdt *wdt = file->private_data; | ||
| 220 | int ret; | ||
| 221 | union { | ||
| 222 | struct watchdog_info ident; | ||
| 223 | int i; | ||
| 224 | } uarg; | ||
| 225 | |||
| 226 | if (_IOC_DIR(cmd) && _IOC_SIZE(cmd) > sizeof(uarg)) | ||
| 227 | return -ENOIOCTLCMD; | ||
| 228 | |||
| 229 | if (_IOC_DIR(cmd) & _IOC_WRITE) { | ||
| 230 | ret = copy_from_user(&uarg, (void __user *)arg, _IOC_SIZE(cmd)); | ||
| 231 | if (ret) | ||
| 232 | return -EFAULT; | ||
| 233 | } | ||
| 234 | |||
| 235 | switch (cmd) { | ||
| 236 | case WDIOC_GETSUPPORT: | ||
| 237 | uarg.ident = ident; | ||
| 238 | ret = 0; | ||
| 239 | break; | ||
| 240 | |||
| 241 | case WDIOC_SETOPTIONS: | ||
| 242 | ret = -EINVAL; | ||
| 243 | if (uarg.i & WDIOS_DISABLECARD) { | ||
| 244 | mpcore_wdt_stop(wdt); | ||
| 245 | ret = 0; | ||
| 246 | } | ||
| 247 | if (uarg.i & WDIOS_ENABLECARD) { | ||
| 248 | mpcore_wdt_start(wdt); | ||
| 249 | ret = 0; | ||
| 250 | } | ||
| 251 | break; | ||
| 252 | |||
| 253 | case WDIOC_GETSTATUS: | ||
| 254 | case WDIOC_GETBOOTSTATUS: | ||
| 255 | uarg.i = 0; | ||
| 256 | ret = 0; | ||
| 257 | break; | ||
| 258 | |||
| 259 | case WDIOC_KEEPALIVE: | ||
| 260 | mpcore_wdt_keepalive(wdt); | ||
| 261 | ret = 0; | ||
| 262 | break; | ||
| 263 | |||
| 264 | case WDIOC_SETTIMEOUT: | ||
| 265 | ret = mpcore_wdt_set_heartbeat(uarg.i); | ||
| 266 | if (ret) | ||
| 267 | break; | ||
| 268 | |||
| 269 | mpcore_wdt_keepalive(wdt); | ||
| 270 | /* Fall */ | ||
| 271 | case WDIOC_GETTIMEOUT: | ||
| 272 | uarg.i = mpcore_margin; | ||
| 273 | ret = 0; | ||
| 274 | break; | ||
| 275 | |||
| 276 | default: | ||
| 277 | return -ENOIOCTLCMD; | ||
| 278 | } | ||
| 279 | |||
| 280 | if (ret == 0 && _IOC_DIR(cmd) & _IOC_READ) { | ||
| 281 | ret = copy_to_user((void __user *)arg, &uarg, _IOC_SIZE(cmd)); | ||
| 282 | if (ret) | ||
| 283 | ret = -EFAULT; | ||
| 284 | } | ||
| 285 | return ret; | ||
| 286 | } | ||
| 287 | |||
| 288 | /* | ||
| 289 | * System shutdown handler. Turn off the watchdog if we're | ||
| 290 | * restarting or halting the system. | ||
| 291 | */ | ||
| 292 | static void mpcore_wdt_shutdown(struct device *_dev) | ||
| 293 | { | ||
| 294 | struct mpcore_wdt *wdt = dev_get_drvdata(_dev); | ||
| 295 | |||
| 296 | if (system_state == SYSTEM_RESTART || system_state == SYSTEM_HALT) | ||
| 297 | mpcore_wdt_stop(wdt); | ||
| 298 | } | ||
| 299 | |||
| 300 | /* | ||
| 301 | * Kernel Interfaces | ||
| 302 | */ | ||
| 303 | static struct file_operations mpcore_wdt_fops = { | ||
| 304 | .owner = THIS_MODULE, | ||
| 305 | .llseek = no_llseek, | ||
| 306 | .write = mpcore_wdt_write, | ||
| 307 | .ioctl = mpcore_wdt_ioctl, | ||
| 308 | .open = mpcore_wdt_open, | ||
| 309 | .release = mpcore_wdt_release, | ||
| 310 | }; | ||
| 311 | |||
| 312 | static struct miscdevice mpcore_wdt_miscdev = { | ||
| 313 | .minor = WATCHDOG_MINOR, | ||
| 314 | .name = "watchdog", | ||
| 315 | .fops = &mpcore_wdt_fops, | ||
| 316 | }; | ||
| 317 | |||
| 318 | static int __devinit mpcore_wdt_probe(struct device *_dev) | ||
| 319 | { | ||
| 320 | struct platform_device *dev = to_platform_device(_dev); | ||
| 321 | struct mpcore_wdt *wdt; | ||
| 322 | struct resource *res; | ||
| 323 | int ret; | ||
| 324 | |||
| 325 | /* We only accept one device, and it must have an id of -1 */ | ||
| 326 | if (dev->id != -1) | ||
| 327 | return -ENODEV; | ||
| 328 | |||
| 329 | res = platform_get_resource(dev, IORESOURCE_MEM, 0); | ||
| 330 | if (!res) { | ||
| 331 | ret = -ENODEV; | ||
| 332 | goto err_out; | ||
| 333 | } | ||
| 334 | |||
| 335 | wdt = kmalloc(sizeof(struct mpcore_wdt), GFP_KERNEL); | ||
| 336 | if (!wdt) { | ||
| 337 | ret = -ENOMEM; | ||
| 338 | goto err_out; | ||
| 339 | } | ||
| 340 | memset(wdt, 0, sizeof(struct mpcore_wdt)); | ||
| 341 | |||
| 342 | wdt->dev = &dev->dev; | ||
| 343 | wdt->irq = platform_get_irq(dev, 0); | ||
| 344 | wdt->base = ioremap(res->start, res->end - res->start + 1); | ||
| 345 | if (!wdt->base) { | ||
| 346 | ret = -ENOMEM; | ||
| 347 | goto err_free; | ||
| 348 | } | ||
| 349 | |||
| 350 | mpcore_wdt_miscdev.dev = &dev->dev; | ||
| 351 | ret = misc_register(&mpcore_wdt_miscdev); | ||
| 352 | if (ret) { | ||
| 353 | dev_printk(KERN_ERR, _dev, "cannot register miscdev on minor=%d (err=%d)\n", | ||
| 354 | WATCHDOG_MINOR, ret); | ||
| 355 | goto err_misc; | ||
| 356 | } | ||
| 357 | |||
| 358 | ret = request_irq(wdt->irq, mpcore_wdt_fire, SA_INTERRUPT, "mpcore_wdt", wdt); | ||
| 359 | if (ret) { | ||
| 360 | dev_printk(KERN_ERR, _dev, "cannot register IRQ%d for watchdog\n", wdt->irq); | ||
| 361 | goto err_irq; | ||
| 362 | } | ||
| 363 | |||
| 364 | mpcore_wdt_stop(wdt); | ||
| 365 | dev_set_drvdata(&dev->dev, wdt); | ||
| 366 | mpcore_wdt_dev = dev; | ||
| 367 | |||
| 368 | return 0; | ||
| 369 | |||
| 370 | err_irq: | ||
| 371 | misc_deregister(&mpcore_wdt_miscdev); | ||
| 372 | err_misc: | ||
| 373 | iounmap(wdt->base); | ||
| 374 | err_free: | ||
| 375 | kfree(wdt); | ||
| 376 | err_out: | ||
| 377 | return ret; | ||
| 378 | } | ||
| 379 | |||
| 380 | static int __devexit mpcore_wdt_remove(struct device *dev) | ||
| 381 | { | ||
| 382 | struct mpcore_wdt *wdt = dev_get_drvdata(dev); | ||
| 383 | |||
| 384 | dev_set_drvdata(dev, NULL); | ||
| 385 | |||
| 386 | misc_deregister(&mpcore_wdt_miscdev); | ||
| 387 | |||
| 388 | mpcore_wdt_dev = NULL; | ||
| 389 | |||
| 390 | free_irq(wdt->irq, wdt); | ||
| 391 | iounmap(wdt->base); | ||
| 392 | kfree(wdt); | ||
| 393 | return 0; | ||
| 394 | } | ||
| 395 | |||
| 396 | static struct device_driver mpcore_wdt_driver = { | ||
| 397 | .name = "mpcore_wdt", | ||
| 398 | .bus = &platform_bus_type, | ||
| 399 | .probe = mpcore_wdt_probe, | ||
| 400 | .remove = __devexit_p(mpcore_wdt_remove), | ||
| 401 | .shutdown = mpcore_wdt_shutdown, | ||
| 402 | }; | ||
| 403 | |||
| 404 | static char banner[] __initdata = KERN_INFO "MPcore Watchdog Timer: 0.1. mpcore_noboot=%d mpcore_margin=%d sec (nowayout= %d)\n"; | ||
| 405 | |||
| 406 | static int __init mpcore_wdt_init(void) | ||
| 407 | { | ||
| 408 | /* | ||
| 409 | * Check that the margin value is within it's range; | ||
| 410 | * if not reset to the default | ||
| 411 | */ | ||
| 412 | if (mpcore_wdt_set_heartbeat(mpcore_margin)) { | ||
| 413 | mpcore_wdt_set_heartbeat(TIMER_MARGIN); | ||
| 414 | printk(KERN_INFO "mpcore_margin value must be 0<mpcore_margin<65536, using %d\n", | ||
| 415 | TIMER_MARGIN); | ||
| 416 | } | ||
| 417 | |||
| 418 | printk(banner, mpcore_noboot, mpcore_margin, nowayout); | ||
| 419 | |||
| 420 | return driver_register(&mpcore_wdt_driver); | ||
| 421 | } | ||
| 422 | |||
| 423 | static void __exit mpcore_wdt_exit(void) | ||
| 424 | { | ||
| 425 | driver_unregister(&mpcore_wdt_driver); | ||
| 426 | } | ||
| 427 | |||
| 428 | module_init(mpcore_wdt_init); | ||
| 429 | module_exit(mpcore_wdt_exit); | ||
| 430 | |||
| 431 | MODULE_AUTHOR("ARM Limited"); | ||
| 432 | MODULE_DESCRIPTION("MPcore Watchdog Device Driver"); | ||
| 433 | MODULE_LICENSE("GPL"); | ||
| 434 | MODULE_ALIAS_MISCDEV(WATCHDOG_MINOR); | ||
