diff options
author | Nick Piggin <nickpiggin@yahoo.com.au> | 2005-11-09 00:39:01 -0500 |
---|---|---|
committer | Linus Torvalds <torvalds@g5.osdl.org> | 2005-11-09 10:56:33 -0500 |
commit | 5bfb5d690f36d316a5f3b4f7775fda996faa6b12 (patch) | |
tree | ea53f15293d1ddb49c316eb65df85e939a4f6e5e /arch/s390 | |
parent | ede3d0fba99520f268067917b50858d788bc41da (diff) |
[PATCH] sched: disable preempt in idle tasks
Run idle threads with preempt disabled.
Also corrected a bugs in arm26's cpu_idle (make it actually call schedule()).
How did it ever work before?
Might fix the CPU hotplugging hang which Nigel Cunningham noted.
We think the bug hits if the idle thread is preempted after checking
need_resched() and before going to sleep, then the CPU offlined.
After calling stop_machine_run, the CPU eventually returns from preemption and
into the idle thread and goes to sleep. The CPU will continue executing
previous idle and have no chance to call play_dead.
By disabling preemption until we are ready to explicitly schedule, this bug is
fixed and the idle threads generally become more robust.
From: alexs <ashepard@u.washington.edu>
PPC build fix
From: Yoichi Yuasa <yuasa@hh.iij4u.or.jp>
MIPS build fix
Signed-off-by: Nick Piggin <npiggin@suse.de>
Signed-off-by: Yoichi Yuasa <yuasa@hh.iij4u.or.jp>
Signed-off-by: Andrew Morton <akpm@osdl.org>
Signed-off-by: Linus Torvalds <torvalds@osdl.org>
Diffstat (limited to 'arch/s390')
-rw-r--r-- | arch/s390/kernel/process.c | 11 | ||||
-rw-r--r-- | arch/s390/kernel/smp.c | 1 |
2 files changed, 9 insertions, 3 deletions
diff --git a/arch/s390/kernel/process.c b/arch/s390/kernel/process.c index 9f3dff6c0b72..66ca5757e368 100644 --- a/arch/s390/kernel/process.c +++ b/arch/s390/kernel/process.c | |||
@@ -102,7 +102,6 @@ void default_idle(void) | |||
102 | local_irq_disable(); | 102 | local_irq_disable(); |
103 | if (need_resched()) { | 103 | if (need_resched()) { |
104 | local_irq_enable(); | 104 | local_irq_enable(); |
105 | schedule(); | ||
106 | return; | 105 | return; |
107 | } | 106 | } |
108 | 107 | ||
@@ -139,8 +138,14 @@ void default_idle(void) | |||
139 | 138 | ||
140 | void cpu_idle(void) | 139 | void cpu_idle(void) |
141 | { | 140 | { |
142 | for (;;) | 141 | for (;;) { |
143 | default_idle(); | 142 | while (!need_resched()) |
143 | default_idle(); | ||
144 | |||
145 | preempt_enable_no_resched(); | ||
146 | schedule(); | ||
147 | preempt_disable(); | ||
148 | } | ||
144 | } | 149 | } |
145 | 150 | ||
146 | void show_regs(struct pt_regs *regs) | 151 | void show_regs(struct pt_regs *regs) |
diff --git a/arch/s390/kernel/smp.c b/arch/s390/kernel/smp.c index e13c87b446b2..5856b3fda6bf 100644 --- a/arch/s390/kernel/smp.c +++ b/arch/s390/kernel/smp.c | |||
@@ -533,6 +533,7 @@ int __devinit start_secondary(void *cpuvoid) | |||
533 | { | 533 | { |
534 | /* Setup the cpu */ | 534 | /* Setup the cpu */ |
535 | cpu_init(); | 535 | cpu_init(); |
536 | preempt_disable(); | ||
536 | /* init per CPU timer */ | 537 | /* init per CPU timer */ |
537 | init_cpu_timer(); | 538 | init_cpu_timer(); |
538 | #ifdef CONFIG_VIRT_TIMER | 539 | #ifdef CONFIG_VIRT_TIMER |