Merge git://git.kernel.org/pub/scm/linux/kernel/git/tglx/linux-2.6-hrt

* git://git.kernel.org/pub/scm/linux/kernel/git/tglx/linux-2.6-hrt:
  clocksource: make clocksource watchdog cycle through online CPUs
  Documentation: move timer related documentation to a single place
  clockevents: optimise tick_nohz_stop_sched_tick() a bit
  locking: remove unused double_spin_lock()
  hrtimers: simplify lockdep handling
  timers: simplify lockdep handling
  posix-timers: fix shadowed variables
  timer_list: add annotations to workqueue.c
  hrtimer: use nanosleep specific restart_block fields
  hrtimer: add nanosleep specific restart_block member
This commit is contained in:
Linus Torvalds 2008-04-18 08:37:41 -07:00
commit 73e3e6481f
14 changed files with 68 additions and 98 deletions

View File

@ -167,10 +167,8 @@ highuid.txt
- notes on the change from 16 bit to 32 bit user/group IDs. - notes on the change from 16 bit to 32 bit user/group IDs.
hpet.txt hpet.txt
- High Precision Event Timer Driver for Linux. - High Precision Event Timer Driver for Linux.
hrtimer/ timers/
- info on the timer_stats debugging facility for timer (ab)use. - info on the timer related topics
hrtimers/
- info on the hrtimers subsystem for high-resolution kernel timers.
hw_random.txt hw_random.txt
- info on Linux support for random number generator in i8xx chipsets. - info on Linux support for random number generator in i8xx chipsets.
hwmon/ hwmon/

View File

@ -173,7 +173,6 @@ struct hrtimer_clock_base {
* struct hrtimer_cpu_base - the per cpu clock bases * struct hrtimer_cpu_base - the per cpu clock bases
* @lock: lock protecting the base and associated clock bases * @lock: lock protecting the base and associated clock bases
* and timers * and timers
* @lock_key: the lock_class_key for use with lockdep
* @clock_base: array of clock bases for this cpu * @clock_base: array of clock bases for this cpu
* @curr_timer: the timer which is executing a callback right now * @curr_timer: the timer which is executing a callback right now
* @expires_next: absolute time of the next event which was scheduled * @expires_next: absolute time of the next event which was scheduled
@ -189,7 +188,6 @@ struct hrtimer_clock_base {
*/ */
struct hrtimer_cpu_base { struct hrtimer_cpu_base {
spinlock_t lock; spinlock_t lock;
struct lock_class_key lock_key;
struct hrtimer_clock_base clock_base[HRTIMER_MAX_CLOCK_BASES]; struct hrtimer_clock_base clock_base[HRTIMER_MAX_CLOCK_BASES];
struct list_head cb_pending; struct list_head cb_pending;
#ifdef CONFIG_HIGH_RES_TIMERS #ifdef CONFIG_HIGH_RES_TIMERS

View File

@ -295,43 +295,6 @@ do { \
1 : ({ local_irq_restore(flags); 0; }); \ 1 : ({ local_irq_restore(flags); 0; }); \
}) })
/*
* Locks two spinlocks l1 and l2.
* l1_first indicates if spinlock l1 should be taken first.
*/
static inline void double_spin_lock(spinlock_t *l1, spinlock_t *l2,
bool l1_first)
__acquires(l1)
__acquires(l2)
{
if (l1_first) {
spin_lock(l1);
spin_lock(l2);
} else {
spin_lock(l2);
spin_lock(l1);
}
}
/*
* Unlocks two spinlocks l1 and l2.
* l1_taken_first indicates if spinlock l1 was taken first and therefore
* should be released after spinlock l2.
*/
static inline void double_spin_unlock(spinlock_t *l1, spinlock_t *l2,
bool l1_taken_first)
__releases(l1)
__releases(l2)
{
if (l1_taken_first) {
spin_unlock(l2);
spin_unlock(l1);
} else {
spin_unlock(l1);
spin_unlock(l2);
}
}
/* /*
* Pull the atomic_t declaration: * Pull the atomic_t declaration:
* (asm-mips/atomic.h needs above definitions) * (asm-mips/atomic.h needs above definitions)

View File

@ -9,6 +9,9 @@
#include <linux/types.h> #include <linux/types.h>
struct timespec;
struct compat_timespec;
/* /*
* System call restart block. * System call restart block.
*/ */
@ -26,6 +29,15 @@ struct restart_block {
u32 bitset; u32 bitset;
u64 time; u64 time;
} futex; } futex;
/* For nanosleep */
struct {
clockid_t index;
struct timespec __user *rmtp;
#ifdef CONFIG_COMPAT
struct compat_timespec __user *compat_rmtp;
#endif
u64 expires;
} nanosleep;
}; };
}; };

View File

@ -47,15 +47,14 @@ static long compat_nanosleep_restart(struct restart_block *restart)
mm_segment_t oldfs; mm_segment_t oldfs;
long ret; long ret;
rmtp = (struct compat_timespec __user *)(restart->arg1); restart->nanosleep.rmtp = (struct timespec __user *) &rmt;
restart->arg1 = (unsigned long)&rmt;
oldfs = get_fs(); oldfs = get_fs();
set_fs(KERNEL_DS); set_fs(KERNEL_DS);
ret = hrtimer_nanosleep_restart(restart); ret = hrtimer_nanosleep_restart(restart);
set_fs(oldfs); set_fs(oldfs);
if (ret) { if (ret) {
restart->arg1 = (unsigned long)rmtp; rmtp = restart->nanosleep.compat_rmtp;
if (rmtp && put_compat_timespec(&rmt, rmtp)) if (rmtp && put_compat_timespec(&rmt, rmtp))
return -EFAULT; return -EFAULT;
@ -89,7 +88,7 @@ asmlinkage long compat_sys_nanosleep(struct compat_timespec __user *rqtp,
= &current_thread_info()->restart_block; = &current_thread_info()->restart_block;
restart->fn = compat_nanosleep_restart; restart->fn = compat_nanosleep_restart;
restart->arg1 = (unsigned long)rmtp; restart->nanosleep.compat_rmtp = rmtp;
if (rmtp && put_compat_timespec(&rmt, rmtp)) if (rmtp && put_compat_timespec(&rmt, rmtp))
return -EFAULT; return -EFAULT;
@ -607,9 +606,9 @@ static long compat_clock_nanosleep_restart(struct restart_block *restart)
long err; long err;
mm_segment_t oldfs; mm_segment_t oldfs;
struct timespec tu; struct timespec tu;
struct compat_timespec *rmtp = (struct compat_timespec *)(restart->arg1); struct compat_timespec *rmtp = restart->nanosleep.compat_rmtp;
restart->arg1 = (unsigned long) &tu; restart->nanosleep.rmtp = (struct timespec __user *) &tu;
oldfs = get_fs(); oldfs = get_fs();
set_fs(KERNEL_DS); set_fs(KERNEL_DS);
err = clock_nanosleep_restart(restart); err = clock_nanosleep_restart(restart);
@ -621,7 +620,7 @@ static long compat_clock_nanosleep_restart(struct restart_block *restart)
if (err == -ERESTART_RESTARTBLOCK) { if (err == -ERESTART_RESTARTBLOCK) {
restart->fn = compat_clock_nanosleep_restart; restart->fn = compat_clock_nanosleep_restart;
restart->arg1 = (unsigned long) rmtp; restart->nanosleep.compat_rmtp = rmtp;
} }
return err; return err;
} }
@ -652,7 +651,7 @@ long compat_sys_clock_nanosleep(clockid_t which_clock, int flags,
if (err == -ERESTART_RESTARTBLOCK) { if (err == -ERESTART_RESTARTBLOCK) {
restart = &current_thread_info()->restart_block; restart = &current_thread_info()->restart_block;
restart->fn = compat_clock_nanosleep_restart; restart->fn = compat_clock_nanosleep_restart;
restart->arg1 = (unsigned long) rmtp; restart->nanosleep.compat_rmtp = rmtp;
} }
return err; return err;
} }

View File

@ -1354,13 +1354,13 @@ long __sched hrtimer_nanosleep_restart(struct restart_block *restart)
struct hrtimer_sleeper t; struct hrtimer_sleeper t;
struct timespec __user *rmtp; struct timespec __user *rmtp;
hrtimer_init(&t.timer, restart->arg0, HRTIMER_MODE_ABS); hrtimer_init(&t.timer, restart->nanosleep.index, HRTIMER_MODE_ABS);
t.timer.expires.tv64 = ((u64)restart->arg3 << 32) | (u64) restart->arg2; t.timer.expires.tv64 = restart->nanosleep.expires;
if (do_nanosleep(&t, HRTIMER_MODE_ABS)) if (do_nanosleep(&t, HRTIMER_MODE_ABS))
return 0; return 0;
rmtp = (struct timespec __user *)restart->arg1; rmtp = restart->nanosleep.rmtp;
if (rmtp) { if (rmtp) {
int ret = update_rmtp(&t.timer, rmtp); int ret = update_rmtp(&t.timer, rmtp);
if (ret <= 0) if (ret <= 0)
@ -1394,10 +1394,9 @@ long hrtimer_nanosleep(struct timespec *rqtp, struct timespec __user *rmtp,
restart = &current_thread_info()->restart_block; restart = &current_thread_info()->restart_block;
restart->fn = hrtimer_nanosleep_restart; restart->fn = hrtimer_nanosleep_restart;
restart->arg0 = (unsigned long) t.timer.base->index; restart->nanosleep.index = t.timer.base->index;
restart->arg1 = (unsigned long) rmtp; restart->nanosleep.rmtp = rmtp;
restart->arg2 = t.timer.expires.tv64 & 0xFFFFFFFF; restart->nanosleep.expires = t.timer.expires.tv64;
restart->arg3 = t.timer.expires.tv64 >> 32;
return -ERESTART_RESTARTBLOCK; return -ERESTART_RESTARTBLOCK;
} }
@ -1425,7 +1424,6 @@ static void __cpuinit init_hrtimers_cpu(int cpu)
int i; int i;
spin_lock_init(&cpu_base->lock); spin_lock_init(&cpu_base->lock);
lockdep_set_class(&cpu_base->lock, &cpu_base->lock_key);
for (i = 0; i < HRTIMER_MAX_CLOCK_BASES; i++) for (i = 0; i < HRTIMER_MAX_CLOCK_BASES; i++)
cpu_base->clock_base[i].cpu_base = cpu_base; cpu_base->clock_base[i].cpu_base = cpu_base;
@ -1466,16 +1464,16 @@ static void migrate_hrtimers(int cpu)
tick_cancel_sched_timer(cpu); tick_cancel_sched_timer(cpu);
local_irq_disable(); local_irq_disable();
double_spin_lock(&new_base->lock, &old_base->lock, spin_lock(&new_base->lock);
smp_processor_id() < cpu); spin_lock_nested(&old_base->lock, SINGLE_DEPTH_NESTING);
for (i = 0; i < HRTIMER_MAX_CLOCK_BASES; i++) { for (i = 0; i < HRTIMER_MAX_CLOCK_BASES; i++) {
migrate_hrtimer_list(&old_base->clock_base[i], migrate_hrtimer_list(&old_base->clock_base[i],
&new_base->clock_base[i]); &new_base->clock_base[i]);
} }
double_spin_unlock(&new_base->lock, &old_base->lock, spin_unlock(&old_base->lock);
smp_processor_id() < cpu); spin_unlock(&new_base->lock);
local_irq_enable(); local_irq_enable();
put_cpu_var(hrtimer_bases); put_cpu_var(hrtimer_bases);
} }

View File

@ -1087,45 +1087,45 @@ static void check_process_timers(struct task_struct *tsk,
maxfire = 20; maxfire = 20;
prof_expires = cputime_zero; prof_expires = cputime_zero;
while (!list_empty(timers)) { while (!list_empty(timers)) {
struct cpu_timer_list *t = list_first_entry(timers, struct cpu_timer_list *tl = list_first_entry(timers,
struct cpu_timer_list, struct cpu_timer_list,
entry); entry);
if (!--maxfire || cputime_lt(ptime, t->expires.cpu)) { if (!--maxfire || cputime_lt(ptime, tl->expires.cpu)) {
prof_expires = t->expires.cpu; prof_expires = tl->expires.cpu;
break; break;
} }
t->firing = 1; tl->firing = 1;
list_move_tail(&t->entry, firing); list_move_tail(&tl->entry, firing);
} }
++timers; ++timers;
maxfire = 20; maxfire = 20;
virt_expires = cputime_zero; virt_expires = cputime_zero;
while (!list_empty(timers)) { while (!list_empty(timers)) {
struct cpu_timer_list *t = list_first_entry(timers, struct cpu_timer_list *tl = list_first_entry(timers,
struct cpu_timer_list, struct cpu_timer_list,
entry); entry);
if (!--maxfire || cputime_lt(utime, t->expires.cpu)) { if (!--maxfire || cputime_lt(utime, tl->expires.cpu)) {
virt_expires = t->expires.cpu; virt_expires = tl->expires.cpu;
break; break;
} }
t->firing = 1; tl->firing = 1;
list_move_tail(&t->entry, firing); list_move_tail(&tl->entry, firing);
} }
++timers; ++timers;
maxfire = 20; maxfire = 20;
sched_expires = 0; sched_expires = 0;
while (!list_empty(timers)) { while (!list_empty(timers)) {
struct cpu_timer_list *t = list_first_entry(timers, struct cpu_timer_list *tl = list_first_entry(timers,
struct cpu_timer_list, struct cpu_timer_list,
entry); entry);
if (!--maxfire || sum_sched_runtime < t->expires.sched) { if (!--maxfire || sum_sched_runtime < tl->expires.sched) {
sched_expires = t->expires.sched; sched_expires = tl->expires.sched;
break; break;
} }
t->firing = 1; tl->firing = 1;
list_move_tail(&t->entry, firing); list_move_tail(&tl->entry, firing);
} }
/* /*

View File

@ -141,8 +141,16 @@ static void clocksource_watchdog(unsigned long data)
} }
if (!list_empty(&watchdog_list)) { if (!list_empty(&watchdog_list)) {
__mod_timer(&watchdog_timer, /*
watchdog_timer.expires + WATCHDOG_INTERVAL); * Cycle through CPUs to check if the CPUs stay
* synchronized to each other.
*/
int next_cpu = next_cpu(raw_smp_processor_id(), cpu_online_map);
if (next_cpu >= NR_CPUS)
next_cpu = first_cpu(cpu_online_map);
watchdog_timer.expires += WATCHDOG_INTERVAL;
add_timer_on(&watchdog_timer, next_cpu);
} }
spin_unlock(&watchdog_lock); spin_unlock(&watchdog_lock);
} }
@ -164,7 +172,8 @@ static void clocksource_check_watchdog(struct clocksource *cs)
if (!started && watchdog) { if (!started && watchdog) {
watchdog_last = watchdog->read(); watchdog_last = watchdog->read();
watchdog_timer.expires = jiffies + WATCHDOG_INTERVAL; watchdog_timer.expires = jiffies + WATCHDOG_INTERVAL;
add_timer(&watchdog_timer); add_timer_on(&watchdog_timer,
first_cpu(cpu_online_map));
} }
} else { } else {
if (cs->flags & CLOCK_SOURCE_IS_CONTINUOUS) if (cs->flags & CLOCK_SOURCE_IS_CONTINUOUS)
@ -185,7 +194,8 @@ static void clocksource_check_watchdog(struct clocksource *cs)
watchdog_last = watchdog->read(); watchdog_last = watchdog->read();
watchdog_timer.expires = watchdog_timer.expires =
jiffies + WATCHDOG_INTERVAL; jiffies + WATCHDOG_INTERVAL;
add_timer(&watchdog_timer); add_timer_on(&watchdog_timer,
first_cpu(cpu_online_map));
} }
} }
} }

View File

@ -158,9 +158,8 @@ void tick_nohz_stop_idle(int cpu)
} }
} }
static ktime_t tick_nohz_start_idle(int cpu) static ktime_t tick_nohz_start_idle(struct tick_sched *ts)
{ {
struct tick_sched *ts = &per_cpu(tick_cpu_sched, cpu);
ktime_t now, delta; ktime_t now, delta;
now = ktime_get(); now = ktime_get();
@ -201,8 +200,8 @@ void tick_nohz_stop_sched_tick(void)
local_irq_save(flags); local_irq_save(flags);
cpu = smp_processor_id(); cpu = smp_processor_id();
now = tick_nohz_start_idle(cpu);
ts = &per_cpu(tick_cpu_sched, cpu); ts = &per_cpu(tick_cpu_sched, cpu);
now = tick_nohz_start_idle(ts);
/* /*
* If this cpu is offline and it is the one which updates * If this cpu is offline and it is the one which updates
@ -222,7 +221,6 @@ void tick_nohz_stop_sched_tick(void)
if (need_resched()) if (need_resched())
goto end; goto end;
cpu = smp_processor_id();
if (unlikely(local_softirq_pending())) { if (unlikely(local_softirq_pending())) {
static int ratelimit; static int ratelimit;

View File

@ -1228,13 +1228,6 @@ asmlinkage long sys_sysinfo(struct sysinfo __user *info)
return 0; return 0;
} }
/*
* lockdep: we want to track each per-CPU base as a separate lock-class,
* but timer-bases are kmalloc()-ed, so we need to attach separate
* keys to them:
*/
static struct lock_class_key base_lock_keys[NR_CPUS];
static int __cpuinit init_timers_cpu(int cpu) static int __cpuinit init_timers_cpu(int cpu)
{ {
int j; int j;
@ -1277,7 +1270,6 @@ static int __cpuinit init_timers_cpu(int cpu)
} }
spin_lock_init(&base->lock); spin_lock_init(&base->lock);
lockdep_set_class(&base->lock, base_lock_keys + cpu);
for (j = 0; j < TVN_SIZE; j++) { for (j = 0; j < TVN_SIZE; j++) {
INIT_LIST_HEAD(base->tv5.vec + j); INIT_LIST_HEAD(base->tv5.vec + j);
@ -1316,8 +1308,8 @@ static void __cpuinit migrate_timers(int cpu)
new_base = get_cpu_var(tvec_bases); new_base = get_cpu_var(tvec_bases);
local_irq_disable(); local_irq_disable();
double_spin_lock(&new_base->lock, &old_base->lock, spin_lock(&new_base->lock);
smp_processor_id() < cpu); spin_lock_nested(&old_base->lock, SINGLE_DEPTH_NESTING);
BUG_ON(old_base->running_timer); BUG_ON(old_base->running_timer);
@ -1330,8 +1322,8 @@ static void __cpuinit migrate_timers(int cpu)
migrate_timer_list(new_base, old_base->tv5.vec + i); migrate_timer_list(new_base, old_base->tv5.vec + i);
} }
double_spin_unlock(&new_base->lock, &old_base->lock, spin_unlock(&old_base->lock);
smp_processor_id() < cpu); spin_unlock(&new_base->lock);
local_irq_enable(); local_irq_enable();
put_cpu_var(tvec_bases); put_cpu_var(tvec_bases);
} }

View File

@ -219,6 +219,7 @@ int queue_delayed_work_on(int cpu, struct workqueue_struct *wq,
struct timer_list *timer = &dwork->timer; struct timer_list *timer = &dwork->timer;
struct work_struct *work = &dwork->work; struct work_struct *work = &dwork->work;
timer_stats_timer_set_start_info(&dwork->timer);
if (!test_and_set_bit(WORK_STRUCT_PENDING, work_data_bits(work))) { if (!test_and_set_bit(WORK_STRUCT_PENDING, work_data_bits(work))) {
BUG_ON(timer_pending(timer)); BUG_ON(timer_pending(timer));
BUG_ON(!list_empty(&work->entry)); BUG_ON(!list_empty(&work->entry));
@ -580,6 +581,7 @@ EXPORT_SYMBOL(schedule_delayed_work);
int schedule_delayed_work_on(int cpu, int schedule_delayed_work_on(int cpu,
struct delayed_work *dwork, unsigned long delay) struct delayed_work *dwork, unsigned long delay)
{ {
timer_stats_timer_set_start_info(&dwork->timer);
return queue_delayed_work_on(cpu, keventd_wq, dwork, delay); return queue_delayed_work_on(cpu, keventd_wq, dwork, delay);
} }
EXPORT_SYMBOL(schedule_delayed_work_on); EXPORT_SYMBOL(schedule_delayed_work_on);