aboutsummaryrefslogtreecommitdiffstats
diff options
context:
space:
mode:
authorGuillaume Chazarain <guichaz@yahoo.fr>2008-04-19 13:44:57 -0400
committerIngo Molnar <mingo@elte.hu>2008-04-19 13:44:57 -0400
commit15934a37324f32e0fda633dc7984a671ea81cd75 (patch)
tree1f65ac7d910e76b65d0cfcd1c72e156b0a8bd273
parent30914a58af9d21c5f1831adabb5d7a800a378675 (diff)
sched: fix rq->clock overflows detection with CONFIG_NO_HZ
When using CONFIG_NO_HZ, rq->tick_timestamp is not updated every TICK_NSEC. We check that the number of skipped ticks matches the clock jump seen in __update_rq_clock(). Signed-off-by: Guillaume Chazarain <guichaz@yahoo.fr> Signed-off-by: Ingo Molnar <mingo@elte.hu>
-rw-r--r--kernel/sched.c38
1 files changed, 35 insertions, 3 deletions
diff --git a/kernel/sched.c b/kernel/sched.c
index 7fe334ead4f9..d8456a9ac9af 100644
--- a/kernel/sched.c
+++ b/kernel/sched.c
@@ -397,6 +397,7 @@ struct rq {
397 unsigned long cpu_load[CPU_LOAD_IDX_MAX]; 397 unsigned long cpu_load[CPU_LOAD_IDX_MAX];
398 unsigned char idle_at_tick; 398 unsigned char idle_at_tick;
399#ifdef CONFIG_NO_HZ 399#ifdef CONFIG_NO_HZ
400 unsigned long last_tick_seen;
400 unsigned char in_nohz_recently; 401 unsigned char in_nohz_recently;
401#endif 402#endif
402 /* capture load from *all* tasks on this cpu: */ 403 /* capture load from *all* tasks on this cpu: */
@@ -500,6 +501,32 @@ static inline int cpu_of(struct rq *rq)
500#endif 501#endif
501} 502}
502 503
504#ifdef CONFIG_NO_HZ
505static inline bool nohz_on(int cpu)
506{
507 return tick_get_tick_sched(cpu)->nohz_mode != NOHZ_MODE_INACTIVE;
508}
509
510static inline u64 max_skipped_ticks(struct rq *rq)
511{
512 return nohz_on(cpu_of(rq)) ? jiffies - rq->last_tick_seen + 2 : 1;
513}
514
515static inline void update_last_tick_seen(struct rq *rq)
516{
517 rq->last_tick_seen = jiffies;
518}
519#else
520static inline u64 max_skipped_ticks(struct rq *rq)
521{
522 return 1;
523}
524
525static inline void update_last_tick_seen(struct rq *rq)
526{
527}
528#endif
529
503/* 530/*
504 * Update the per-runqueue clock, as finegrained as the platform can give 531 * Update the per-runqueue clock, as finegrained as the platform can give
505 * us, but without assuming monotonicity, etc.: 532 * us, but without assuming monotonicity, etc.:
@@ -524,9 +551,12 @@ static void __update_rq_clock(struct rq *rq)
524 /* 551 /*
525 * Catch too large forward jumps too: 552 * Catch too large forward jumps too:
526 */ 553 */
527 if (unlikely(clock + delta > rq->tick_timestamp + TICK_NSEC)) { 554 u64 max_jump = max_skipped_ticks(rq) * TICK_NSEC;
528 if (clock < rq->tick_timestamp + TICK_NSEC) 555 u64 max_time = rq->tick_timestamp + max_jump;
529 clock = rq->tick_timestamp + TICK_NSEC; 556
557 if (unlikely(clock + delta > max_time)) {
558 if (clock < max_time)
559 clock = max_time;
530 else 560 else
531 clock++; 561 clock++;
532 rq->clock_overflows++; 562 rq->clock_overflows++;
@@ -3812,6 +3842,7 @@ void scheduler_tick(void)
3812 rq->clock_underflows++; 3842 rq->clock_underflows++;
3813 } 3843 }
3814 rq->tick_timestamp = rq->clock; 3844 rq->tick_timestamp = rq->clock;
3845 update_last_tick_seen(rq);
3815 update_cpu_load(rq); 3846 update_cpu_load(rq);
3816 curr->sched_class->task_tick(rq, curr, 0); 3847 curr->sched_class->task_tick(rq, curr, 0);
3817 update_sched_rt_period(rq); 3848 update_sched_rt_period(rq);
@@ -7261,6 +7292,7 @@ void __init sched_init(void)
7261 lockdep_set_class(&rq->lock, &rq->rq_lock_key); 7292 lockdep_set_class(&rq->lock, &rq->rq_lock_key);
7262 rq->nr_running = 0; 7293 rq->nr_running = 0;
7263 rq->clock = 1; 7294 rq->clock = 1;
7295 update_last_tick_seen(rq);
7264 init_cfs_rq(&rq->cfs, rq); 7296 init_cfs_rq(&rq->cfs, rq);
7265 init_rt_rq(&rq->rt, rq); 7297 init_rt_rq(&rq->rt, rq);
7266#ifdef CONFIG_FAIR_GROUP_SCHED 7298#ifdef CONFIG_FAIR_GROUP_SCHED