2 * Copyright (c) 2009 The DragonFly Project. All rights reserved.
4 * This code is derived from software contributed to The DragonFly Project
5 * by Alex Hornung <ahornung@gmail.com>
7 * Redistribution and use in source and binary forms, with or without
8 * modification, are permitted provided that the following conditions
11 * 1. Redistributions of source code must retain the above copyright
12 * notice, this list of conditions and the following disclaimer.
13 * 2. Redistributions in binary form must reproduce the above copyright
14 * notice, this list of conditions and the following disclaimer in
15 * the documentation and/or other materials provided with the
17 * 3. Neither the name of The DragonFly Project nor the names of its
18 * contributors may be used to endorse or promote products derived
19 * from this software without specific, prior written permission.
21 * THIS SOFTWARE IS PROVIDED BY THE COPYRIGHT HOLDERS AND CONTRIBUTORS
22 * ``AS IS'' AND ANY EXPRESS OR IMPLIED WARRANTIES, INCLUDING, BUT NOT
23 * LIMITED TO, THE IMPLIED WARRANTIES OF MERCHANTABILITY AND FITNESS
24 * FOR A PARTICULAR PURPOSE ARE DISCLAIMED. IN NO EVENT SHALL THE
25 * COPYRIGHT HOLDERS OR CONTRIBUTORS BE LIABLE FOR ANY DIRECT, INDIRECT,
26 * INCIDENTAL, SPECIAL, EXEMPLARY OR CONSEQUENTIAL DAMAGES (INCLUDING,
27 * BUT NOT LIMITED TO, PROCUREMENT OF SUBSTITUTE GOODS OR SERVICES;
28 * LOSS OF USE, DATA, OR PROFITS; OR BUSINESS INTERRUPTION) HOWEVER CAUSED
29 * AND ON ANY THEORY OF LIABILITY, WHETHER IN CONTRACT, STRICT LIABILITY,
30 * OR TORT (INCLUDING NEGLIGENCE OR OTHERWISE) ARISING IN ANY WAY OUT
31 * OF THE USE OF THIS SOFTWARE, EVEN IF ADVISED OF THE POSSIBILITY OF
35 #include <sys/param.h>
36 #include <sys/kernel.h>
37 #include <sys/systm.h>
38 #include <sys/malloc.h>
39 #include <sys/sysproto.h>
40 #include <sys/spinlock2.h>
42 #include <sys/device.h>
43 #include <sys/filedesc.h>
44 #include <sys/sysctl.h>
45 #include <sys/unistd.h>
46 #include <sys/event.h>
47 #include <sys/queue.h>
49 #include <machine/limits.h>
51 static LIST_HEAD(, watchdog
) wdoglist
= LIST_HEAD_INITIALIZER(&wdoglist
);
52 static struct spinlock wdogmtx
;
53 static struct callout wdog_callout
;
55 static int wdog_auto_enable
= 1;
56 static int wdog_auto_period
= WDOG_DEFAULT_PERIOD
;
58 static void wdog_reset_all(void *unused
);
61 wdog_register(struct watchdog
*wd
)
64 wd
->period
= WDOG_DEFAULT_PERIOD
;
65 LIST_INSERT_HEAD(&wdoglist
, wd
, link
);
66 spin_unlock(&wdogmtx
);
70 kprintf("wdog: Watchdog %s registered, max period = %ds , period = %ds\n",
71 wd
->name
, wd
->period_max
, wd
->period
);
75 wdog_unregister(struct watchdog
*wd
)
78 LIST_REMOVE(wd
, link
);
79 spin_unlock(&wdogmtx
);
81 kprintf("wdog: Watchdog %s unregistered\n", wd
->name
);
85 wdog_reset(struct watchdog
*wd
)
87 return (wd
->period
= wd
->wdog_fn(wd
->arg
, wd
->period
));
91 wdog_reset_all(void *unused
)
94 int period
, min_period
= INT_MAX
;
97 if (LIST_EMPTY(&wdoglist
))
99 LIST_FOREACH(wd
, &wdoglist
, link
) {
100 period
= wdog_reset(wd
);
101 if (period
< min_period
)
104 if (wdog_auto_enable
)
105 callout_reset(&wdog_callout
, min_period
* hz
/ 2, wdog_reset_all
, NULL
);
107 wdog_auto_period
= min_period
;
110 spin_unlock(&wdogmtx
);
114 wdog_set_period(int period
)
119 LIST_FOREACH(wd
, &wdoglist
, link
) {
120 /* XXX: check for period_max */
123 spin_unlock(&wdogmtx
);
128 wdog_sysctl_auto(SYSCTL_HANDLER_ARGS
)
132 error
= sysctl_handle_int(oidp
, &wdog_auto_enable
, 1, req
);
133 if (error
|| req
->newptr
== NULL
)
136 /* has changed, do something */
137 callout_stop(&wdog_callout
);
138 if (wdog_auto_enable
) {
139 wdog_reset_all(NULL
);
142 kprintf("wdog: In-kernel automatic watchdog reset %s\n",
143 (wdog_auto_enable
)?"enabled":"disabled");
149 wdog_sysctl_period(SYSCTL_HANDLER_ARGS
)
153 error
= sysctl_handle_int(oidp
, &wdog_auto_period
, WDOG_DEFAULT_PERIOD
, req
);
154 if (error
|| req
->newptr
== NULL
)
157 /* has changed, do something */
158 callout_stop(&wdog_callout
);
159 wdog_set_period(wdog_auto_period
);
160 wdog_reset_all(NULL
);
162 if (wdog_auto_period
!= 0)
163 kprintf("wdog: Watchdog period set to %ds\n", wdog_auto_period
);
165 kprintf("wdog: Disabled watchdog(s)\n");
173 callout_stop(&wdog_callout
);
175 wdog_reset_all(NULL
);
178 static SYSCTL_NODE(_kern
, OID_AUTO
, watchdog
, CTLFLAG_RW
, 0, "watchdog");
179 SYSCTL_PROC(_kern_watchdog
, OID_AUTO
, auto, CTLTYPE_INT
| CTLFLAG_RW
,
180 NULL
, 0, wdog_sysctl_auto
, "I", "auto in-kernel watchdog reset "
181 "(0 = disabled, 1 = enabled)");
182 SYSCTL_PROC(_kern_watchdog
, OID_AUTO
, period
, CTLTYPE_INT
| CTLFLAG_RW
,
183 NULL
, 0, wdog_sysctl_period
, "I", "watchdog period "
184 "(value in seconds)");
188 wdog_ioctl(struct dev_ioctl_args
*ap
)
190 if (wdog_auto_enable
)
193 if (ap
->a_cmd
== WDIOCRESET
) {
194 wdog_reset_all(NULL
);
202 static struct dev_ops wdog_ops
= {
204 .d_ioctl
= wdog_ioctl
,
210 spin_init(&wdogmtx
, "wdog");
211 make_dev(&wdog_ops
, 0,
212 UID_ROOT
, GID_WHEEL
, 0600, "wdog");
213 callout_init_mp(&wdog_callout
);
215 kprintf("wdog: In-kernel automatic watchdog reset %s\n",
216 (wdog_auto_enable
)?"enabled":"disabled");
222 callout_stop(&wdog_callout
);
223 callout_deactivate(&wdog_callout
);
224 dev_ops_remove_all(&wdog_ops
);
225 spin_uninit(&wdogmtx
);
228 SYSINIT(wdog_register
, SI_SUB_PRE_DRIVERS
, SI_ORDER_ANY
, wdog_init
, NULL
);
229 SYSUNINIT(wdog_register
, SI_SUB_PRE_DRIVERS
, SI_ORDER_ANY
, wdog_uninit
, NULL
);