Blame view
drivers/watchdog/shwdt.c
11.8 KB
1da177e4c Linux-2.6.12-rc2 |
1 |
/* |
b1fa888e0 watchdog: shwdt: ... |
2 |
* drivers/watchdog/shwdt.c |
1da177e4c Linux-2.6.12-rc2 |
3 4 5 |
* * Watchdog driver for integrated watchdog in the SuperH processors. * |
b1fa888e0 watchdog: shwdt: ... |
6 |
* Copyright (C) 2001 - 2010 Paul Mundt <lethal@linux-sh.org> |
1da177e4c Linux-2.6.12-rc2 |
7 8 9 10 11 12 13 14 15 16 17 18 19 |
* * This program is free software; you can redistribute it and/or modify it * under the terms of the GNU General Public License as published by the * Free Software Foundation; either version 2 of the License, or (at your * option) any later version. * * 14-Dec-2001 Matt Domsch <Matt_Domsch@dell.com> * Added nowayout module option to override CONFIG_WATCHDOG_NOWAYOUT * * 19-Apr-2002 Rob Radez <rob@osinvestor.com> * Added expect close support, made emulated timeout runtime changeable * general cleanups, add some ioctls */ |
1da177e4c Linux-2.6.12-rc2 |
20 21 |
#include <linux/module.h> #include <linux/moduleparam.h> |
8f5585ec3 watchdog: shwdt: ... |
22 |
#include <linux/platform_device.h> |
1da177e4c Linux-2.6.12-rc2 |
23 24 25 26 27 28 29 30 |
#include <linux/init.h> #include <linux/types.h> #include <linux/miscdevice.h> #include <linux/watchdog.h> #include <linux/reboot.h> #include <linux/notifier.h> #include <linux/ioport.h> #include <linux/fs.h> |
f118420be watchdog: Add a s... |
31 |
#include <linux/mm.h> |
8f5585ec3 watchdog: shwdt: ... |
32 |
#include <linux/slab.h> |
70b814ec1 [WATCHDOG 45/57] ... |
33 34 |
#include <linux/io.h> #include <linux/uaccess.h> |
58cf41984 [WATCHDOG] fix wa... |
35 |
#include <asm/watchdog.h> |
1da177e4c Linux-2.6.12-rc2 |
36 |
|
8f5585ec3 watchdog: shwdt: ... |
37 |
#define DRV_NAME "sh-wdt" |
1da177e4c Linux-2.6.12-rc2 |
38 39 40 41 42 43 44 45 46 47 48 49 50 51 52 |
/* * Default clock division ratio is 5.25 msecs. For an additional table of * values, consult the asm-sh/watchdog.h. Overload this at module load * time. * * In order for this to work reliably we need to have HZ set to 1000 or * something quite higher than 100 (or we need a proper high-res timer * implementation that will deal with this properly), otherwise the 10ms * resolution of a jiffy is enough to trigger the overflow. For things like * the SH-4 and SH-5, this isn't necessarily that big of a problem, though * for the SH-2 and SH-3, this isn't recommended unless the WDT is absolutely * necssary. * * As a result of this timing problem, the only modes that are particularly |
25985edce Fix common misspe... |
53 |
* feasible are the 4096 and the 2048 divisors, which yield 5.25 and 2.62ms |
1da177e4c Linux-2.6.12-rc2 |
54 55 56 |
* overflow periods respectively. * * Also, since we can't really expect userspace to be responsive enough |
ee0fc097e drivers/watchdog/... |
57 |
* before the overflow happens, we maintain two separate timers .. One in |
1da177e4c Linux-2.6.12-rc2 |
58 59 60 61 62 63 64 65 66 |
* the kernel for clearing out WOVF every 2ms or so (again, this depends on * HZ == 1000), and another for monitoring userspace writes to the WDT device. * * As such, we currently use a configurable heartbeat interval which defaults * to 30s. In this case, the userspace daemon is only responsible for periodic * writes to the device before the next heartbeat is scheduled. If the daemon * misses its deadline, the kernel timer will allow the WDT to overflow. */ static int clock_division_ratio = WTCSR_CKS_4096; |
bea190662 watchdog: shwdt: ... |
67 |
#define next_ping_period(cks) (jiffies + msecs_to_jiffies(cks - 4)) |
1da177e4c Linux-2.6.12-rc2 |
68 |
|
58cf41984 [WATCHDOG] fix wa... |
69 |
static const struct watchdog_info sh_wdt_info; |
8f5585ec3 watchdog: shwdt: ... |
70 |
static struct platform_device *sh_wdt_dev; |
70b814ec1 [WATCHDOG 45/57] ... |
71 |
static DEFINE_SPINLOCK(shwdt_lock); |
1da177e4c Linux-2.6.12-rc2 |
72 73 74 |
#define WATCHDOG_HEARTBEAT 30 /* 30 sec default heartbeat */ static int heartbeat = WATCHDOG_HEARTBEAT; /* in seconds */ |
4bfdf3783 [PATCH] consolida... |
75 |
static int nowayout = WATCHDOG_NOWAYOUT; |
8f5585ec3 watchdog: shwdt: ... |
76 77 78 79 80 |
static unsigned long next_heartbeat; struct sh_wdt { void __iomem *base; struct device *dev; |
1da177e4c Linux-2.6.12-rc2 |
81 |
|
8f5585ec3 watchdog: shwdt: ... |
82 83 84 85 86 87 88 |
struct timer_list timer; unsigned long enabled; char expect_close; }; static void sh_wdt_start(struct sh_wdt *wdt) |
1da177e4c Linux-2.6.12-rc2 |
89 |
{ |
70b814ec1 [WATCHDOG 45/57] ... |
90 |
unsigned long flags; |
8f5585ec3 watchdog: shwdt: ... |
91 |
u8 csr; |
70b814ec1 [WATCHDOG 45/57] ... |
92 |
|
58cf41984 [WATCHDOG] fix wa... |
93 |
spin_lock_irqsave(&shwdt_lock, flags); |
1da177e4c Linux-2.6.12-rc2 |
94 95 |
next_heartbeat = jiffies + (heartbeat * HZ); |
8f5585ec3 watchdog: shwdt: ... |
96 |
mod_timer(&wdt->timer, next_ping_period(clock_division_ratio)); |
1da177e4c Linux-2.6.12-rc2 |
97 98 99 100 101 102 103 104 105 106 107 108 109 110 111 112 113 114 115 116 117 |
csr = sh_wdt_read_csr(); csr |= WTCSR_WT | clock_division_ratio; sh_wdt_write_csr(csr); sh_wdt_write_cnt(0); /* * These processors have a bit of an inconsistent initialization * process.. starting with SH-3, RSTS was moved to WTCSR, and the * RSTCSR register was removed. * * On the SH-2 however, in addition with bits being in different * locations, we must deal with RSTCSR outright.. */ csr = sh_wdt_read_csr(); csr |= WTCSR_TME; csr &= ~WTCSR_RSTS; sh_wdt_write_csr(csr); #ifdef CONFIG_CPU_SH2 |
1da177e4c Linux-2.6.12-rc2 |
118 119 120 121 |
csr = sh_wdt_read_rstcsr(); csr &= ~RSTCSR_RSTS; sh_wdt_write_rstcsr(csr); #endif |
58cf41984 [WATCHDOG] fix wa... |
122 |
spin_unlock_irqrestore(&shwdt_lock, flags); |
1da177e4c Linux-2.6.12-rc2 |
123 |
} |
8f5585ec3 watchdog: shwdt: ... |
124 |
static void sh_wdt_stop(struct sh_wdt *wdt) |
1da177e4c Linux-2.6.12-rc2 |
125 |
{ |
70b814ec1 [WATCHDOG 45/57] ... |
126 |
unsigned long flags; |
8f5585ec3 watchdog: shwdt: ... |
127 |
u8 csr; |
70b814ec1 [WATCHDOG 45/57] ... |
128 |
|
58cf41984 [WATCHDOG] fix wa... |
129 |
spin_lock_irqsave(&shwdt_lock, flags); |
1da177e4c Linux-2.6.12-rc2 |
130 |
|
8f5585ec3 watchdog: shwdt: ... |
131 |
del_timer(&wdt->timer); |
1da177e4c Linux-2.6.12-rc2 |
132 133 134 135 |
csr = sh_wdt_read_csr(); csr &= ~WTCSR_TME; sh_wdt_write_csr(csr); |
8f5585ec3 watchdog: shwdt: ... |
136 |
|
58cf41984 [WATCHDOG] fix wa... |
137 |
spin_unlock_irqrestore(&shwdt_lock, flags); |
1da177e4c Linux-2.6.12-rc2 |
138 |
} |
8f5585ec3 watchdog: shwdt: ... |
139 |
static inline void sh_wdt_keepalive(struct sh_wdt *wdt) |
1da177e4c Linux-2.6.12-rc2 |
140 |
{ |
70b814ec1 [WATCHDOG 45/57] ... |
141 |
unsigned long flags; |
58cf41984 [WATCHDOG] fix wa... |
142 |
spin_lock_irqsave(&shwdt_lock, flags); |
1da177e4c Linux-2.6.12-rc2 |
143 |
next_heartbeat = jiffies + (heartbeat * HZ); |
58cf41984 [WATCHDOG] fix wa... |
144 |
spin_unlock_irqrestore(&shwdt_lock, flags); |
1da177e4c Linux-2.6.12-rc2 |
145 |
} |
1da177e4c Linux-2.6.12-rc2 |
146 147 |
static int sh_wdt_set_heartbeat(int t) { |
70b814ec1 [WATCHDOG 45/57] ... |
148 149 150 |
unsigned long flags; if (unlikely(t < 1 || t > 3600)) /* arbitrary upper limit */ |
1da177e4c Linux-2.6.12-rc2 |
151 |
return -EINVAL; |
58cf41984 [WATCHDOG] fix wa... |
152 |
spin_lock_irqsave(&shwdt_lock, flags); |
1da177e4c Linux-2.6.12-rc2 |
153 |
heartbeat = t; |
58cf41984 [WATCHDOG] fix wa... |
154 |
spin_unlock_irqrestore(&shwdt_lock, flags); |
1da177e4c Linux-2.6.12-rc2 |
155 156 |
return 0; } |
1da177e4c Linux-2.6.12-rc2 |
157 158 |
static void sh_wdt_ping(unsigned long data) { |
8f5585ec3 watchdog: shwdt: ... |
159 |
struct sh_wdt *wdt = (struct sh_wdt *)data; |
70b814ec1 [WATCHDOG 45/57] ... |
160 |
unsigned long flags; |
58cf41984 [WATCHDOG] fix wa... |
161 |
spin_lock_irqsave(&shwdt_lock, flags); |
1da177e4c Linux-2.6.12-rc2 |
162 |
if (time_before(jiffies, next_heartbeat)) { |
8f5585ec3 watchdog: shwdt: ... |
163 |
u8 csr; |
1da177e4c Linux-2.6.12-rc2 |
164 165 166 167 168 169 |
csr = sh_wdt_read_csr(); csr &= ~WTCSR_IOVF; sh_wdt_write_csr(csr); sh_wdt_write_cnt(0); |
8f5585ec3 watchdog: shwdt: ... |
170 |
mod_timer(&wdt->timer, next_ping_period(clock_division_ratio)); |
e4c2cfee5 sh: Various cosme... |
171 |
} else |
8f5585ec3 watchdog: shwdt: ... |
172 173 174 |
dev_warn(wdt->dev, "Heartbeat lost! Will not ping " "the watchdog "); |
58cf41984 [WATCHDOG] fix wa... |
175 |
spin_unlock_irqrestore(&shwdt_lock, flags); |
1da177e4c Linux-2.6.12-rc2 |
176 |
} |
1da177e4c Linux-2.6.12-rc2 |
177 178 |
static int sh_wdt_open(struct inode *inode, struct file *file) { |
8f5585ec3 watchdog: shwdt: ... |
179 180 181 |
struct sh_wdt *wdt = platform_get_drvdata(sh_wdt_dev); if (test_and_set_bit(0, &wdt->enabled)) |
1da177e4c Linux-2.6.12-rc2 |
182 183 184 |
return -EBUSY; if (nowayout) __module_get(THIS_MODULE); |
8f5585ec3 watchdog: shwdt: ... |
185 186 187 |
file->private_data = wdt; sh_wdt_start(wdt); |
1da177e4c Linux-2.6.12-rc2 |
188 189 190 |
return nonseekable_open(inode, file); } |
1da177e4c Linux-2.6.12-rc2 |
191 192 |
static int sh_wdt_close(struct inode *inode, struct file *file) { |
8f5585ec3 watchdog: shwdt: ... |
193 194 195 196 |
struct sh_wdt *wdt = file->private_data; if (wdt->expect_close == 42) { sh_wdt_stop(wdt); |
1da177e4c Linux-2.6.12-rc2 |
197 |
} else { |
8f5585ec3 watchdog: shwdt: ... |
198 199 200 201 |
dev_crit(wdt->dev, "Unexpected close, not " "stopping watchdog! "); sh_wdt_keepalive(wdt); |
1da177e4c Linux-2.6.12-rc2 |
202 |
} |
8f5585ec3 watchdog: shwdt: ... |
203 204 |
clear_bit(0, &wdt->enabled); wdt->expect_close = 0; |
1da177e4c Linux-2.6.12-rc2 |
205 206 207 |
return 0; } |
1da177e4c Linux-2.6.12-rc2 |
208 209 210 |
static ssize_t sh_wdt_write(struct file *file, const char *buf, size_t count, loff_t *ppos) { |
8f5585ec3 watchdog: shwdt: ... |
211 |
struct sh_wdt *wdt = file->private_data; |
1da177e4c Linux-2.6.12-rc2 |
212 213 214 |
if (count) { if (!nowayout) { size_t i; |
8f5585ec3 watchdog: shwdt: ... |
215 |
wdt->expect_close = 0; |
1da177e4c Linux-2.6.12-rc2 |
216 217 218 219 220 221 |
for (i = 0; i != count; i++) { char c; if (get_user(c, buf + i)) return -EFAULT; if (c == 'V') |
8f5585ec3 watchdog: shwdt: ... |
222 |
wdt->expect_close = 42; |
1da177e4c Linux-2.6.12-rc2 |
223 224 |
} } |
8f5585ec3 watchdog: shwdt: ... |
225 |
sh_wdt_keepalive(wdt); |
1da177e4c Linux-2.6.12-rc2 |
226 227 228 229 |
} return count; } |
70b814ec1 [WATCHDOG 45/57] ... |
230 231 |
static long sh_wdt_ioctl(struct file *file, unsigned int cmd, unsigned long arg) |
1da177e4c Linux-2.6.12-rc2 |
232 |
{ |
8f5585ec3 watchdog: shwdt: ... |
233 |
struct sh_wdt *wdt = file->private_data; |
1da177e4c Linux-2.6.12-rc2 |
234 235 236 237 |
int new_heartbeat; int options, retval = -EINVAL; switch (cmd) { |
70b814ec1 [WATCHDOG 45/57] ... |
238 239 240 241 242 243 |
case WDIOC_GETSUPPORT: return copy_to_user((struct watchdog_info *)arg, &sh_wdt_info, sizeof(sh_wdt_info)) ? -EFAULT : 0; case WDIOC_GETSTATUS: case WDIOC_GETBOOTSTATUS: return put_user(0, (int *)arg); |
70b814ec1 [WATCHDOG 45/57] ... |
244 245 246 247 248 |
case WDIOC_SETOPTIONS: if (get_user(options, (int *)arg)) return -EFAULT; if (options & WDIOS_DISABLECARD) { |
8f5585ec3 watchdog: shwdt: ... |
249 |
sh_wdt_stop(wdt); |
70b814ec1 [WATCHDOG 45/57] ... |
250 251 252 253 |
retval = 0; } if (options & WDIOS_ENABLECARD) { |
8f5585ec3 watchdog: shwdt: ... |
254 |
sh_wdt_start(wdt); |
70b814ec1 [WATCHDOG 45/57] ... |
255 256 |
retval = 0; } |
1da177e4c Linux-2.6.12-rc2 |
257 |
|
70b814ec1 [WATCHDOG 45/57] ... |
258 |
return retval; |
0c06090c9 [WATCHDOG] Coding... |
259 |
case WDIOC_KEEPALIVE: |
8f5585ec3 watchdog: shwdt: ... |
260 |
sh_wdt_keepalive(wdt); |
0c06090c9 [WATCHDOG] Coding... |
261 262 263 264 265 266 267 |
return 0; case WDIOC_SETTIMEOUT: if (get_user(new_heartbeat, (int *)arg)) return -EFAULT; if (sh_wdt_set_heartbeat(new_heartbeat)) return -EINVAL; |
8f5585ec3 watchdog: shwdt: ... |
268 |
sh_wdt_keepalive(wdt); |
0c06090c9 [WATCHDOG] Coding... |
269 270 271 |
/* Fall */ case WDIOC_GETTIMEOUT: return put_user(heartbeat, (int *)arg); |
70b814ec1 [WATCHDOG 45/57] ... |
272 273 274 |
default: return -ENOTTY; } |
1da177e4c Linux-2.6.12-rc2 |
275 276 |
return 0; } |
1da177e4c Linux-2.6.12-rc2 |
277 278 279 |
static int sh_wdt_notify_sys(struct notifier_block *this, unsigned long code, void *unused) { |
8f5585ec3 watchdog: shwdt: ... |
280 |
struct sh_wdt *wdt = platform_get_drvdata(sh_wdt_dev); |
e4c2cfee5 sh: Various cosme... |
281 |
if (code == SYS_DOWN || code == SYS_HALT) |
8f5585ec3 watchdog: shwdt: ... |
282 |
sh_wdt_stop(wdt); |
1da177e4c Linux-2.6.12-rc2 |
283 284 285 |
return NOTIFY_DONE; } |
62322d255 [PATCH] make more... |
286 |
static const struct file_operations sh_wdt_fops = { |
1da177e4c Linux-2.6.12-rc2 |
287 288 289 |
.owner = THIS_MODULE, .llseek = no_llseek, .write = sh_wdt_write, |
70b814ec1 [WATCHDOG 45/57] ... |
290 |
.unlocked_ioctl = sh_wdt_ioctl, |
1da177e4c Linux-2.6.12-rc2 |
291 292 293 |
.open = sh_wdt_open, .release = sh_wdt_close, }; |
70b814ec1 [WATCHDOG 45/57] ... |
294 |
static const struct watchdog_info sh_wdt_info = { |
e4c2cfee5 sh: Various cosme... |
295 296 |
.options = WDIOF_KEEPALIVEPING | WDIOF_SETTIMEOUT | WDIOF_MAGICCLOSE, |
1da177e4c Linux-2.6.12-rc2 |
297 298 299 300 301 302 303 304 305 306 307 308 309 |
.firmware_version = 1, .identity = "SH WDT", }; static struct notifier_block sh_wdt_notifier = { .notifier_call = sh_wdt_notify_sys, }; static struct miscdevice sh_wdt_miscdev = { .minor = WATCHDOG_MINOR, .name = "watchdog", .fops = &sh_wdt_fops, }; |
8f5585ec3 watchdog: shwdt: ... |
310 |
static int __devinit sh_wdt_probe(struct platform_device *pdev) |
1da177e4c Linux-2.6.12-rc2 |
311 |
{ |
8f5585ec3 watchdog: shwdt: ... |
312 313 |
struct sh_wdt *wdt; struct resource *res; |
1da177e4c Linux-2.6.12-rc2 |
314 |
int rc; |
8f5585ec3 watchdog: shwdt: ... |
315 316 317 318 319 320 321 322 323 324 325 326 327 328 329 330 331 332 333 |
/* * As this driver only covers the global watchdog case, reject * any attempts to register per-CPU watchdogs. */ if (pdev->id != -1) return -EINVAL; res = platform_get_resource(pdev, IORESOURCE_MEM, 0); if (unlikely(!res)) return -EINVAL; if (!devm_request_mem_region(&pdev->dev, res->start, resource_size(res), DRV_NAME)) return -EBUSY; wdt = devm_kzalloc(&pdev->dev, sizeof(struct sh_wdt), GFP_KERNEL); if (unlikely(!wdt)) { rc = -ENOMEM; goto out_release; |
1da177e4c Linux-2.6.12-rc2 |
334 |
} |
8f5585ec3 watchdog: shwdt: ... |
335 336 337 338 339 340 |
wdt->dev = &pdev->dev; wdt->base = devm_ioremap(&pdev->dev, res->start, resource_size(res)); if (unlikely(!wdt->base)) { rc = -ENXIO; goto out_err; |
1da177e4c Linux-2.6.12-rc2 |
341 |
} |
1da177e4c Linux-2.6.12-rc2 |
342 |
rc = register_reboot_notifier(&sh_wdt_notifier); |
e4c2cfee5 sh: Various cosme... |
343 |
if (unlikely(rc)) { |
8f5585ec3 watchdog: shwdt: ... |
344 |
dev_err(&pdev->dev, |
70b814ec1 [WATCHDOG 45/57] ... |
345 346 |
"Can't register reboot notifier (err=%d) ", rc); |
8f5585ec3 watchdog: shwdt: ... |
347 |
goto out_unmap; |
1da177e4c Linux-2.6.12-rc2 |
348 |
} |
8f5585ec3 watchdog: shwdt: ... |
349 |
sh_wdt_miscdev.parent = wdt->dev; |
1da177e4c Linux-2.6.12-rc2 |
350 |
rc = misc_register(&sh_wdt_miscdev); |
e4c2cfee5 sh: Various cosme... |
351 |
if (unlikely(rc)) { |
8f5585ec3 watchdog: shwdt: ... |
352 |
dev_err(&pdev->dev, |
70b814ec1 [WATCHDOG 45/57] ... |
353 354 355 |
"Can't register miscdev on minor=%d (err=%d) ", sh_wdt_miscdev.minor, rc); |
8f5585ec3 watchdog: shwdt: ... |
356 |
goto out_unreg; |
1da177e4c Linux-2.6.12-rc2 |
357 |
} |
8f5585ec3 watchdog: shwdt: ... |
358 359 360 361 362 363 364 365 366 367 |
init_timer(&wdt->timer); wdt->timer.function = sh_wdt_ping; wdt->timer.data = (unsigned long)wdt; wdt->timer.expires = next_ping_period(clock_division_ratio); platform_set_drvdata(pdev, wdt); sh_wdt_dev = pdev; dev_info(&pdev->dev, "initialized. "); |
1da177e4c Linux-2.6.12-rc2 |
368 369 |
return 0; |
8f5585ec3 watchdog: shwdt: ... |
370 371 372 373 374 375 376 377 378 379 380 |
out_unreg: unregister_reboot_notifier(&sh_wdt_notifier); out_unmap: devm_iounmap(&pdev->dev, wdt->base); out_err: devm_kfree(&pdev->dev, wdt); out_release: devm_release_mem_region(&pdev->dev, res->start, resource_size(res)); return rc; |
1da177e4c Linux-2.6.12-rc2 |
381 |
} |
8f5585ec3 watchdog: shwdt: ... |
382 |
static int __devexit sh_wdt_remove(struct platform_device *pdev) |
1da177e4c Linux-2.6.12-rc2 |
383 |
{ |
8f5585ec3 watchdog: shwdt: ... |
384 385 386 387 |
struct sh_wdt *wdt = platform_get_drvdata(pdev); struct resource *res = platform_get_resource(pdev, IORESOURCE_MEM, 0); platform_set_drvdata(pdev, NULL); |
1da177e4c Linux-2.6.12-rc2 |
388 |
misc_deregister(&sh_wdt_miscdev); |
8f5585ec3 watchdog: shwdt: ... |
389 390 |
sh_wdt_dev = NULL; |
1da177e4c Linux-2.6.12-rc2 |
391 |
unregister_reboot_notifier(&sh_wdt_notifier); |
8f5585ec3 watchdog: shwdt: ... |
392 393 394 395 396 397 398 399 400 401 402 403 404 405 406 407 408 409 410 411 412 413 414 415 416 417 418 419 420 421 422 423 424 425 426 427 428 429 430 431 432 433 434 435 |
devm_release_mem_region(&pdev->dev, res->start, resource_size(res)); devm_iounmap(&pdev->dev, wdt->base); devm_kfree(&pdev->dev, wdt); return 0; } static struct platform_driver sh_wdt_driver = { .driver = { .name = DRV_NAME, .owner = THIS_MODULE, }, .probe = sh_wdt_probe, .remove = __devexit_p(sh_wdt_remove), }; static int __init sh_wdt_init(void) { int rc; if (unlikely(clock_division_ratio < 0x5 || clock_division_ratio > 0x7)) { clock_division_ratio = WTCSR_CKS_4096; pr_info("%s: divisor must be 0x5<=x<=0x7, using %d ", DRV_NAME, clock_division_ratio); } rc = sh_wdt_set_heartbeat(heartbeat); if (unlikely(rc)) { heartbeat = WATCHDOG_HEARTBEAT; pr_info("%s: heartbeat value must be 1<=x<=3600, using %d ", DRV_NAME, heartbeat); } pr_info("%s: configured with heartbeat=%d sec (nowayout=%d) ", DRV_NAME, heartbeat, nowayout); return platform_driver_register(&sh_wdt_driver); |
1da177e4c Linux-2.6.12-rc2 |
436 |
} |
8f5585ec3 watchdog: shwdt: ... |
437 438 439 440 441 442 |
static void __exit sh_wdt_exit(void) { platform_driver_unregister(&sh_wdt_driver); } module_init(sh_wdt_init); module_exit(sh_wdt_exit); |
1da177e4c Linux-2.6.12-rc2 |
443 444 445 |
MODULE_AUTHOR("Paul Mundt <lethal@linux-sh.org>"); MODULE_DESCRIPTION("SuperH watchdog driver"); MODULE_LICENSE("GPL"); |
8f5585ec3 watchdog: shwdt: ... |
446 |
MODULE_ALIAS("platform:" DRV_NAME); |
1da177e4c Linux-2.6.12-rc2 |
447 448 449 |
MODULE_ALIAS_MISCDEV(WATCHDOG_MINOR); module_param(clock_division_ratio, int, 0); |
a77dba7e4 [WATCHDOG] Some m... |
450 451 |
MODULE_PARM_DESC(clock_division_ratio, "Clock division ratio. Valid ranges are from 0x5 (1.31ms) " |
76550d329 watchdog: fix sev... |
452 |
"to 0x7 (5.25ms). (default=" __MODULE_STRING(WTCSR_CKS_4096) ")"); |
1da177e4c Linux-2.6.12-rc2 |
453 454 |
module_param(heartbeat, int, 0); |
70b814ec1 [WATCHDOG 45/57] ... |
455 456 457 |
MODULE_PARM_DESC(heartbeat, "Watchdog heartbeat in seconds. (1 <= heartbeat <= 3600, default=" __MODULE_STRING(WATCHDOG_HEARTBEAT) ")"); |
1da177e4c Linux-2.6.12-rc2 |
458 459 |
module_param(nowayout, int, 0); |
70b814ec1 [WATCHDOG 45/57] ... |
460 461 462 |
MODULE_PARM_DESC(nowayout, "Watchdog cannot be stopped once started (default=" __MODULE_STRING(WATCHDOG_NOWAYOUT) ")"); |