From e4dad3ea9dc581ff5fa491c2e816483b5e92f264 Mon Sep 17 00:00:00 2001 From: cheloha Date: Sun, 25 Oct 2020 01:55:19 +0000 Subject: setitimer(2): ITIMER_REAL: use kclock timeouts Reimplement the ITIMER_REAL interval timer with a kclock timeout. Couple things of note: - We need to use the high-res nanouptime(9) call, not the low-res getnanouptime(9). - The code is simpler now that we aren't working with ticks. Misc. thoughts: - Still unsure if "kclock" is the right name for these things. - MP-safely cancelling a periodic timeout is very difficult. --- sys/kern/kern_time.c | 20 ++++++-------------- 1 file changed, 6 insertions(+), 14 deletions(-) (limited to 'sys/kern/kern_time.c') diff --git a/sys/kern/kern_time.c b/sys/kern/kern_time.c index 78df8830572..76953229a0e 100644 --- a/sys/kern/kern_time.c +++ b/sys/kern/kern_time.c @@ -1,4 +1,4 @@ -/* $OpenBSD: kern_time.c,v 1.148 2020/10/15 16:31:11 cheloha Exp $ */ +/* $OpenBSD: kern_time.c,v 1.149 2020/10/25 01:55:18 cheloha Exp $ */ /* $NetBSD: kern_time.c,v 1.20 1996/02/18 11:57:06 fvdl Exp $ */ /* @@ -524,7 +524,6 @@ setitimer(int which, const struct itimerval *itv, struct itimerval *olditv) struct timespec now; struct itimerspec *itimer; struct process *pr; - int timo; KASSERT(which >= ITIMER_REAL && which <= ITIMER_PROF); @@ -539,16 +538,15 @@ setitimer(int which, const struct itimerval *itv, struct itimerval *olditv) if (which != ITIMER_REAL) mtx_enter(&itimer_mtx); else - getnanouptime(&now); + nanouptime(&now); if (olditv != NULL) oldits = *itimer; if (itv != NULL) { if (which == ITIMER_REAL) { if (timespecisset(&its.it_value)) { - timo = tstohz(&its.it_value); - timeout_add(&pr->ps_realit_to, timo); timespecadd(&its.it_value, &now, &its.it_value); + timeout_at_ts(&pr->ps_realit_to, &its.it_value); } else timeout_del(&pr->ps_realit_to); } @@ -659,10 +657,9 @@ sys_setitimer(struct proc *p, void *v, register_t *retval) void realitexpire(void *arg) { - struct timespec cts, nts; + struct timespec cts; struct process *pr = arg; struct itimerspec *tp = &pr->ps_timer[ITIMER_REAL]; - int timo; prsignal(pr, SIGALRM); @@ -673,16 +670,11 @@ realitexpire(void *arg) } /* Find the nearest future expiration point and restart the timeout. */ - getnanouptime(&cts); + nanouptime(&cts); while (timespeccmp(&tp->it_value, &cts, <=)) timespecadd(&tp->it_value, &tp->it_interval, &tp->it_value); - nts = tp->it_value; - timespecsub(&nts, &cts, &nts); - timo = tstohz(&nts) - 1; - if (timo <= 0) - timo = 1; if ((pr->ps_flags & PS_EXITING) == 0) - timeout_add(&pr->ps_realit_to, timo); + timeout_at_ts(&pr->ps_realit_to, &tp->it_value); } /* -- cgit v1.2.3