/* $NetBSD: swwdog.c,v 1.12 2010/11/11 21:55:04 pooka Exp $ */ /* * Copyright (c) 2004, 2005 Steven M. Bellovin * All rights reserved. * * Redistribution and use in source and binary forms, with or without * modification, are permitted provided that the following conditions * are met: * 1. Redistributions of source code must retain the above copyright * notice, this list of conditions and the following disclaimer. * 2. Redistributions in binary form must reproduce the above copyright * notice, this list of conditions and the following disclaimer in the * documentation and/or other materials provided with the distribution. * 3. All advertising materials mentioning features or use of this software * must display the following acknowledgement: * This product includes software developed by Steven M. Bellovin * 4. The name of the author contributors may not be used to endorse or * promote products derived from this software without specific prior * written permission. * * THIS SOFTWARE IS PROVIDED BY THE AUTHOR AND CONTRIBUTORS * ``AS IS'' AND ANY EXPRESS OR IMPLIED WARRANTIES, INCLUDING, BUT NOT LIMITED * TO, THE IMPLIED WARRANTIES OF MERCHANTABILITY AND FITNESS FOR A PARTICULAR * PURPOSE ARE DISCLAIMED. IN NO EVENT SHALL THE AUTHOR OR CONTRIBUTORS * BE LIABLE FOR ANY DIRECT, INDIRECT, INCIDENTAL, SPECIAL, EXEMPLARY, OR * CONSEQUENTIAL DAMAGES (INCLUDING, BUT NOT LIMITED TO, PROCUREMENT OF * SUBSTITUTE GOODS OR SERVICES; LOSS OF USE, DATA, OR PROFITS; OR BUSINESS * INTERRUPTION) HOWEVER CAUSED AND ON ANY THEORY OF LIABILITY, WHETHER IN * CONTRACT, STRICT LIABILITY, OR TORT (INCLUDING NEGLIGENCE OR OTHERWISE) * ARISING IN ANY WAY OUT OF THE USE OF THIS SOFTWARE, EVEN IF ADVISED OF THE * POSSIBILITY OF SUCH DAMAGE. */ #include __KERNEL_RCSID(0, "$NetBSD: swwdog.c,v 1.12 2010/11/11 21:55:04 pooka Exp $"); /* * * Software watchdog timer * */ #include #include #include #include #include #include #include #include #include #include #include #include "ioconf.h" struct swwdog_softc { device_t sc_dev; struct sysmon_wdog sc_smw; struct callout sc_c; int sc_wdog_armed; }; void swwdogattach(int); static int swwdog_match(device_t, cfdata_t, void *); static void swwdog_attach(device_t, device_t, void *); static int swwdog_detach(device_t, int); static bool swwdog_suspend(device_t, const pmf_qual_t *); static int swwdog_setmode(struct sysmon_wdog *); static int swwdog_tickle(struct sysmon_wdog *); static int swwdog_arm(struct swwdog_softc *); static int swwdog_disarm(struct swwdog_softc *); static void swwdog_panic(void *); bool swwdog_reboot = false; /* set for panic instead of reboot */ #define SWDOG_DEFAULT 60 /* 60-second default period */ CFATTACH_DECL_NEW(swwdog, sizeof(struct swwdog_softc), swwdog_match, swwdog_attach, swwdog_detach, NULL); static void swwdog_sysctl_setup(void); static struct sysctllog *swwdog_sysctllog; static void doreboot(struct work *wrkwrkwrk, void *p) { cpu_reboot(0, NULL); } static struct workqueue *wq; void swwdogattach(int n __unused) { int err; static struct cfdata cf; if (workqueue_create(&wq, "swwreboot", doreboot, NULL, PRI_NONE, IPL_NONE, 0) != 0) { aprint_error("failed to create swwdog reboot wq"); } err = config_cfattach_attach(swwdog_cd.cd_name, &swwdog_ca); if (err) { aprint_error("%s: couldn't register cfattach: %d\n", swwdog_cd.cd_name, err); config_cfdriver_detach(&swwdog_cd); workqueue_destroy(wq); return; } cf.cf_name = swwdog_cd.cd_name; cf.cf_atname = swwdog_cd.cd_name; cf.cf_unit = 0; cf.cf_fstate = FSTATE_STAR; (void)config_attach_pseudo(&cf); return; } static int swwdog_match(device_t parent, cfdata_t data, void *aux) { return 1; } static void swwdog_attach(device_t parent, device_t self, void *aux) { struct swwdog_softc *sc = device_private(self); sc->sc_dev = self; sc->sc_smw.smw_name = device_xname(self); sc->sc_smw.smw_cookie = sc; sc->sc_smw.smw_setmode = swwdog_setmode; sc->sc_smw.smw_tickle = swwdog_tickle; sc->sc_smw.smw_period = SWDOG_DEFAULT; callout_init(&sc->sc_c, 0); callout_setfunc(&sc->sc_c, swwdog_panic, sc); if (sysmon_wdog_register(&sc->sc_smw) == 0) aprint_normal_dev(self, "software watchdog initialized\n"); else aprint_error_dev(self, "unable to register software " "watchdog with sysmon\n"); if (!pmf_device_register(self, swwdog_suspend, NULL)) aprint_error_dev(self, "couldn't establish power handler\n"); swwdog_sysctl_setup(); } static int swwdog_detach(device_t self, int flags) { struct swwdog_softc *sc = device_private(self); swwdog_disarm(sc); callout_destroy(&sc->sc_c); sysctl_teardown(&swwdog_sysctllog); workqueue_destroy(wq); return 1; } static bool swwdog_suspend(device_t dev, const pmf_qual_t *qual) { struct swwdog_softc *sc = device_private(dev); /* Don't allow suspend if watchdog is armed */ if ((sc->sc_smw.smw_mode & WDOG_MODE_MASK) != WDOG_MODE_DISARMED) return false; return true; } static int swwdog_setmode(struct sysmon_wdog *smw) { struct swwdog_softc *sc = smw->smw_cookie; int error = 0; if ((smw->smw_mode & WDOG_MODE_MASK) == WDOG_MODE_DISARMED) { error = swwdog_disarm(sc); } else { if (smw->smw_period == 0) return EINVAL; else if (smw->smw_period == WDOG_PERIOD_DEFAULT) sc->sc_smw.smw_period = SWDOG_DEFAULT; else sc->sc_smw.smw_period = smw->smw_period; error = swwdog_arm(sc); } return error; } static int swwdog_tickle(struct sysmon_wdog *smw) { return swwdog_arm(smw->smw_cookie); } static int swwdog_arm(struct swwdog_softc *sc) { callout_schedule(&sc->sc_c, sc->sc_smw.smw_period * hz); return 0; } static int swwdog_disarm(struct swwdog_softc *sc) { callout_stop(&sc->sc_c); return 0; } static void swwdog_panic(void *vsc) { struct swwdog_softc *sc = vsc; static struct work wk; /* we'll need it max once */ bool do_panic; do_panic = !swwdog_reboot; swwdog_reboot = false; callout_schedule(&sc->sc_c, 60 * hz); /* deliberate double-panic */ printf("%s: %d second timer expired\n", device_xname(sc->sc_dev), sc->sc_smw.smw_period); if (do_panic) panic("watchdog timer expired"); else workqueue_enqueue(wq, &wk, NULL); } static void swwdog_sysctl_setup(void) { const struct sysctlnode *me; KASSERT(swwdog_sysctllog == NULL); sysctl_createv(&swwdog_sysctllog, 0, NULL, &me, CTLFLAG_READWRITE, CTLTYPE_NODE, "swwdog", NULL, NULL, 0, NULL, 0, CTL_HW, CTL_CREATE, CTL_EOL); sysctl_createv(&swwdog_sysctllog, 0, NULL, NULL, CTLFLAG_READWRITE, CTLTYPE_BOOL, "reboot", "reboot if timer expires", NULL, 0, &swwdog_reboot, sizeof(bool), CTL_HW, me->sysctl_num, CTL_CREATE, CTL_EOL); }