Skip to content

Commit

Permalink
- Add the possibility to reuse the already last used timeout when pat…
Browse files Browse the repository at this point in the history
…ting

  the watchdog, via the watchdog(9) interface.
  For that, the WD_LASTVAL bitwise operation is used. It is mutually
  exclusive with any explicit timout passing to the watchdogs.
  The last timeout can be returned via the wdog_kern_last_timeout()
  KPI.
- Add the possibility to pat the watchdogs installed via the watchdog(9)
  interface from the kernel.
  In order to do that the new KPI wdog_kern_pat() is offered and it does
  accept normalized nanoseconds or WD_LASTVAL.
- Avoid to pass WD_ACTIVE down in the watchdog handlers. All the control
  bit processing should over to the upper layer functions and not passed
  down to the handlers at all.

These changes are intended to be used in order to fix up the watchdog
tripping in situation when the userland is busted, but protection is still
wanted (examples: shutdown syncing / disk dumping).

Sponsored by:	Sandvine Incorporated
Reviewed by:	emaste, des, cognet
MFC after:	2 weeks
  • Loading branch information
Attilio Rao authored and Attilio Rao committed Apr 27, 2011
1 parent 0e24a63 commit 7126ba4
Show file tree
Hide file tree
Showing 3 changed files with 60 additions and 21 deletions.
8 changes: 0 additions & 8 deletions sys/ddb/db_command.c
Original file line number Diff line number Diff line change
Expand Up @@ -724,14 +724,6 @@ db_watchdog(dummy1, dummy2, dummy3, dummy4)
} else if ((tout & WD_INTERVAL) == WD_TO_NEVER) {
db_error("Out of range watchdog interval\n");
return;
} else {

/*
* XXX: Right now we only support WD_ACTIVE, in the future we
* may be possibly needing a more convoluted function for
* dealing with different cases.
*/
tout |= WD_ACTIVE;
}
EVENTHANDLER_INVOKE(watchdog_list, tout, &i);
}
Expand Down
64 changes: 51 additions & 13 deletions sys/dev/watchdog/watchdog.c
Original file line number Diff line number Diff line change
Expand Up @@ -40,35 +40,73 @@ __FBSDID("$FreeBSD$");
#include <machine/bus.h>

static struct cdev *wd_dev;
static volatile u_int wd_last_u;

static int
kern_do_pat(u_int utim)
{
int error;

if ((utim & WD_LASTVAL) != 0 && (utim & WD_INTERVAL) > 0)
return (EINVAL);

if ((utim & WD_LASTVAL) != 0) {
MPASS((wd_last_u & ~WD_INTERVAL) == 0);
utim &= ~WD_LASTVAL;
utim |= wd_last_u;
} else
wd_last_u = (utim & WD_INTERVAL);
if ((utim & WD_INTERVAL) == WD_TO_NEVER) {
utim = 0;

/* Assume all is well; watchdog signals failure. */
error = 0;
} else {
/* Assume no watchdog available; watchdog flags success */
error = EOPNOTSUPP;
}
EVENTHANDLER_INVOKE(watchdog_list, utim, &error);
return (error);
}

static int
wd_ioctl(struct cdev *dev __unused, u_long cmd, caddr_t data,
int flags __unused, struct thread *td)
{
int error;
u_int u;

if (cmd != WDIOCPATPAT)
return (ENOIOCTL);
u = *(u_int *)data;
if (u & ~(WD_ACTIVE | WD_PASSIVE | WD_INTERVAL))
if (u & ~(WD_ACTIVE | WD_PASSIVE | WD_LASTVAL | WD_INTERVAL))
return (EINVAL);
if ((u & (WD_ACTIVE | WD_PASSIVE)) == (WD_ACTIVE | WD_PASSIVE))
return (EINVAL);
if ((u & (WD_ACTIVE | WD_PASSIVE)) == 0 && (u & WD_INTERVAL) > 0)
if ((u & (WD_ACTIVE | WD_PASSIVE)) == 0 && ((u & WD_INTERVAL) > 0 ||
(u & WD_LASTVAL) != 0))
return (EINVAL);
if (u & WD_PASSIVE)
return (ENOSYS); /* XXX Not implemented yet */
if ((u & WD_INTERVAL) == WD_TO_NEVER) {
u = 0;
/* Assume all is well; watchdog signals failure. */
error = 0;
} else {
/* Assume no watchdog available; watchdog flags success */
error = EOPNOTSUPP;
}
EVENTHANDLER_INVOKE(watchdog_list, u, &error);
return (error);
u &= ~(WD_ACTIVE | WD_PASSIVE);

return (kern_do_pat(u));
}

u_int
wdog_kern_last_timeout(void)
{

return (wd_last_u);
}

int
wdog_kern_pat(u_int utim)
{

if (utim & ~(WD_LASTVAL | WD_INTERVAL))
return (EINVAL);

return (kern_do_pat(utim));
}

static struct cdevsw wd_cdevsw = {
Expand Down
9 changes: 9 additions & 0 deletions sys/sys/watchdog.h
Original file line number Diff line number Diff line change
Expand Up @@ -50,6 +50,12 @@
* right to the kernel.
*/

#define WD_LASTVAL 0x0200000
/*
* Use the already last used timeout value.
* The kernel will use as timeout the last valid timeout provided.
*/

#define WD_INTERVAL 0x00000ff
/*
* Mask for duration bits.
Expand Down Expand Up @@ -78,6 +84,9 @@
typedef void (*watchdog_fn)(void *, u_int, int *);

EVENTHANDLER_DECLARE(watchdog_list, watchdog_fn);

u_int wdog_kern_last_timeout(void);
int wdog_kern_pat(u_int utim);
#endif

#endif /* _SYS_WATCHDOG_H */

0 comments on commit 7126ba4

Please sign in to comment.