summaryrefslogtreecommitdiff
diff options
context:
space:
mode:
authorArtur Grabowski <art@cvs.openbsd.org>2004-07-20 20:20:53 +0000
committerArtur Grabowski <art@cvs.openbsd.org>2004-07-20 20:20:53 +0000
commit8075c18474252d8f65d5a42d733a8b26745b7d12 (patch)
tree98e3d0a229e94188afad660aa3299e7294bbe1d5
parentefbc8ef7df9425ed2171c341c62979657387bfa5 (diff)
The timeout wheels are manipulated by hardclock that's not protected with
biglock. We need to protect them with a mutex.
-rw-r--r--sys/kern/kern_timeout.c49
1 files changed, 19 insertions, 30 deletions
diff --git a/sys/kern/kern_timeout.c b/sys/kern/kern_timeout.c
index 7f996fd25e4..921f4be9cff 100644
--- a/sys/kern/kern_timeout.c
+++ b/sys/kern/kern_timeout.c
@@ -1,4 +1,4 @@
-/* $OpenBSD: kern_timeout.c,v 1.18 2003/06/03 12:05:25 art Exp $ */
+/* $OpenBSD: kern_timeout.c,v 1.19 2004/07/20 20:20:52 art Exp $ */
/*
* Copyright (c) 2001 Thomas Nordin <nordin@openbsd.org>
* Copyright (c) 2000-2001 Artur Grabowski <art@openbsd.org>
@@ -29,6 +29,7 @@
#include <sys/systm.h>
#include <sys/lock.h>
#include <sys/timeout.h>
+#include <sys/mutex.h>
#ifdef DDB
#include <machine/db_machdep.h>
@@ -69,15 +70,12 @@ struct circq timeout_todo; /* Worklist */
&timeout_wheel[MASKWHEEL((wheel), (time)) + (wheel)*WHEELSIZE])
/*
- * All wheels are locked with the same lock (which must also block out all
- * interrupts).
+ * All wheels are locked with the same mutex.
+ *
+ * We need locking since the timeouts are manipulated from hardclock that's
+ * not behind the big lock.
*/
-struct simplelock _timeout_lock;
-
-#define timeout_wheel_lock(s) \
- do { *(s) = splhigh(); simple_lock(&_timeout_lock); } while (0)
-#define timeout_wheel_unlock(s) \
- do { simple_unlock(&_timeout_lock); splx(s); } while (0)
+struct mutex timeout_mutex = MUTEX_INITIALIZER(IPL_HIGH);
/*
* Circular queue definitions.
@@ -137,7 +135,6 @@ timeout_startup(void)
CIRCQ_INIT(&timeout_todo);
for (b = 0; b < BUCKETS; b++)
CIRCQ_INIT(&timeout_wheel[b]);
- simple_lock_init(&_timeout_lock);
}
void
@@ -152,7 +149,6 @@ timeout_set(struct timeout *new, void (*fn)(void *), void *arg)
void
timeout_add(struct timeout *new, int to_ticks)
{
- int s;
int old_time;
#ifdef DIAGNOSTIC
@@ -162,7 +158,7 @@ timeout_add(struct timeout *new, int to_ticks)
panic("timeout_add: to_ticks < 0");
#endif
- timeout_wheel_lock(&s);
+ mtx_enter(&timeout_mutex);
/* Initialize the time here, it won't change. */
old_time = new->to_time;
new->to_time = to_ticks + ticks;
@@ -182,33 +178,29 @@ timeout_add(struct timeout *new, int to_ticks)
new->to_flags |= TIMEOUT_ONQUEUE;
CIRCQ_INSERT(&new->to_list, &timeout_todo);
}
-
- timeout_wheel_unlock(s);
+ mtx_leave(&timeout_mutex);
}
void
timeout_del(struct timeout *to)
{
- int s;
-
- timeout_wheel_lock(&s);
+ mtx_enter(&timeout_mutex);
if (to->to_flags & TIMEOUT_ONQUEUE) {
CIRCQ_REMOVE(&to->to_list);
to->to_flags &= ~TIMEOUT_ONQUEUE;
}
to->to_flags &= ~TIMEOUT_TRIGGERED;
- timeout_wheel_unlock(s);
+ mtx_leave(&timeout_mutex);
}
/*
* This is called from hardclock() once every tick.
* We return !0 if we need to schedule a softclock.
- *
- * We don't need locking in here.
*/
int
timeout_hardclock_update(void)
{
+ mtx_enter(&timeout_mutex);
MOVEBUCKET(0, ticks);
if (MASKWHEEL(0, ticks) == 0) {
MOVEBUCKET(1, ticks);
@@ -218,6 +210,7 @@ timeout_hardclock_update(void)
MOVEBUCKET(3, ticks);
}
}
+ mtx_leave(&timeout_mutex);
return (!CIRCQ_EMPTY(&timeout_todo));
}
@@ -225,11 +218,10 @@ void
softclock(void)
{
struct timeout *to;
- int s;
void (*fn)(void *);
void *arg;
- timeout_wheel_lock(&s);
+ mtx_enter(&timeout_mutex);
while (!CIRCQ_EMPTY(&timeout_todo)) {
to = (struct timeout *)CIRCQ_FIRST(&timeout_todo); /* XXX */
@@ -251,12 +243,12 @@ softclock(void)
fn = to->to_func;
arg = to->to_arg;
- timeout_wheel_unlock(s);
+ mtx_leave(&timeout_mutex);
fn(arg);
- timeout_wheel_lock(&s);
+ mtx_enter(&timeout_mutex);
}
}
- timeout_wheel_unlock(s);
+ mtx_leave(&timeout_mutex);
}
#ifdef DDB
@@ -283,18 +275,15 @@ db_show_callout_bucket(struct circq *bucket)
void
db_show_callout(db_expr_t addr, int haddr, db_expr_t count, char *modif)
{
- int s;
int b;
db_printf("ticks now: %d\n", ticks);
db_printf(" ticks wheel arg func\n");
- timeout_wheel_lock(&s);
-
+ mtx_enter(&timeout_mutex);
db_show_callout_bucket(&timeout_todo);
for (b = 0; b < BUCKETS; b++)
db_show_callout_bucket(&timeout_wheel[b]);
-
- timeout_wheel_unlock(s);
+ mtx_leave(&timeout_mutex);
}
#endif