Skip to content

Commit 175cc3a

Browse files
Frederic WeisbeckerKAGA-KOKO
authored andcommitted
posix-cpu-timers: Force next_expiration recalc after timer deletion
A timer deletion only dequeues the timer but it doesn't shutdown the related costly process wide cputimer counter and the tick dependency. The following code snippet keeps this overhead around for one week after the timer deletion: void trigger_process_counter(void) { timer_t id; struct itimerspec val = { }; val.it_value.tv_sec = 604800; timer_create(CLOCK_PROCESS_CPUTIME_ID, NULL, &id); timer_settime(id, 0, &val, NULL); timer_delete(id); } Make sure the next target's tick recalculates the nearest expiration and clears the process wide counter and tick dependency if necessary. Signed-off-by: Frederic Weisbecker <[email protected]> Signed-off-by: Thomas Gleixner <[email protected]> Acked-by: Peter Zijlstra (Intel) <[email protected]> Link: https://lore.kernel.org/r/[email protected]
1 parent a5dec9f commit 175cc3a

File tree

2 files changed

+35
-2
lines changed

2 files changed

+35
-2
lines changed

include/linux/posix-timers.h

Lines changed: 3 additions & 1 deletion
Original file line numberDiff line numberDiff line change
@@ -82,12 +82,14 @@ static inline bool cpu_timer_enqueue(struct timerqueue_head *head,
8282
return timerqueue_add(head, &ctmr->node);
8383
}
8484

85-
static inline void cpu_timer_dequeue(struct cpu_timer *ctmr)
85+
static inline bool cpu_timer_dequeue(struct cpu_timer *ctmr)
8686
{
8787
if (ctmr->head) {
8888
timerqueue_del(ctmr->head, &ctmr->node);
8989
ctmr->head = NULL;
90+
return true;
9091
}
92+
return false;
9193
}
9294

9395
static inline u64 cpu_timer_getexpires(struct cpu_timer *ctmr)

kernel/time/posix-cpu-timers.c

Lines changed: 32 additions & 1 deletion
Original file line numberDiff line numberDiff line change
@@ -407,6 +407,37 @@ static int posix_cpu_timer_create(struct k_itimer *new_timer)
407407
return 0;
408408
}
409409

410+
/*
411+
* Dequeue the timer and reset the base if it was its earliest expiration.
412+
* It makes sure the next tick recalculates the base next expiration so we
413+
* don't keep the costly process wide cputime counter around for a random
414+
* amount of time, along with the tick dependency.
415+
*
416+
* If another timer gets queued between this and the next tick, its
417+
* expiration will update the base next event if necessary on the next
418+
* tick.
419+
*/
420+
static void disarm_timer(struct k_itimer *timer, struct task_struct *p)
421+
{
422+
struct cpu_timer *ctmr = &timer->it.cpu;
423+
struct posix_cputimer_base *base;
424+
int clkidx;
425+
426+
if (!cpu_timer_dequeue(ctmr))
427+
return;
428+
429+
clkidx = CPUCLOCK_WHICH(timer->it_clock);
430+
431+
if (CPUCLOCK_PERTHREAD(timer->it_clock))
432+
base = p->posix_cputimers.bases + clkidx;
433+
else
434+
base = p->signal->posix_cputimers.bases + clkidx;
435+
436+
if (cpu_timer_getexpires(ctmr) == base->nextevt)
437+
base->nextevt = 0;
438+
}
439+
440+
410441
/*
411442
* Clean up a CPU-clock timer that is about to be destroyed.
412443
* This is called from timer deletion with the timer already locked.
@@ -441,7 +472,7 @@ static int posix_cpu_timer_del(struct k_itimer *timer)
441472
if (timer->it.cpu.firing)
442473
ret = TIMER_RETRY;
443474
else
444-
cpu_timer_dequeue(ctmr);
475+
disarm_timer(timer, p);
445476

446477
unlock_task_sighand(p, &flags);
447478
}

0 commit comments

Comments
 (0)