diff options
author | Paul Mackerras <paulus@samba.org> | 2008-05-14 00:30:48 -0400 |
---|---|---|
committer | Paul Mackerras <paulus@samba.org> | 2008-05-16 09:22:28 -0400 |
commit | a560643e21e1ac10f7398b45111aecdd7f47a4a5 (patch) | |
tree | 32cd887f782c35c60cc2a925ab4c49f762851319 | |
parent | fcff474ea5cb17ff015aa40e92ed86fede41f1e2 (diff) |
[POWERPC] Defer processing of interrupts when the CPU wakes from sleep mode
This provides a way to defer processing of an interrupt that wakes the
processor out of sleep mode. On 32-bit platforms that use an
interrupt to wake the processor, we have to have interrupts enabled in
hardware at the point where we go to sleep, otherwise the processor
will never wake up. However, because interrupts are logically
disabled at this point, we don't want to process the interrupt
straight away.
This is handled by setting the _TLF_SLEEPING flag. When we get an
interrupt and _TLF_SLEEPING is set, we firstly clear the MSR_EE
(external interrupt enable) bit in the saved MSR value, and secondly
we then return to the address in the link register, like we do for
_TLF_NAPPING, but without actually handling the interrupt.
Note that this is handled somewhat differently on powerbooks, so this
new code will only be used on non-Apple machines.
Signed-off-by: Paul Mackerras <paulus@samba.org>
-rw-r--r-- | arch/powerpc/kernel/entry_32.S | 8 | ||||
-rw-r--r-- | include/asm-powerpc/thread_info.h | 4 |
2 files changed, 11 insertions, 1 deletions
diff --git a/arch/powerpc/kernel/entry_32.S b/arch/powerpc/kernel/entry_32.S index 3a05e9f93d42..888a364043a8 100644 --- a/arch/powerpc/kernel/entry_32.S +++ b/arch/powerpc/kernel/entry_32.S | |||
@@ -147,6 +147,7 @@ transfer_to_handler: | |||
147 | lwz r12,TI_LOCAL_FLAGS(r9) | 147 | lwz r12,TI_LOCAL_FLAGS(r9) |
148 | mtcrf 0x01,r12 | 148 | mtcrf 0x01,r12 |
149 | bt- 31-TLF_NAPPING,4f | 149 | bt- 31-TLF_NAPPING,4f |
150 | bt- 31-TLF_SLEEPING,7f | ||
150 | #endif /* CONFIG_6xx */ | 151 | #endif /* CONFIG_6xx */ |
151 | .globl transfer_to_handler_cont | 152 | .globl transfer_to_handler_cont |
152 | transfer_to_handler_cont: | 153 | transfer_to_handler_cont: |
@@ -164,6 +165,13 @@ transfer_to_handler_cont: | |||
164 | 4: rlwinm r12,r12,0,~_TLF_NAPPING | 165 | 4: rlwinm r12,r12,0,~_TLF_NAPPING |
165 | stw r12,TI_LOCAL_FLAGS(r9) | 166 | stw r12,TI_LOCAL_FLAGS(r9) |
166 | b power_save_6xx_restore | 167 | b power_save_6xx_restore |
168 | |||
169 | 7: rlwinm r12,r12,0,~_TLF_SLEEPING | ||
170 | stw r12,TI_LOCAL_FLAGS(r9) | ||
171 | lwz r9,_MSR(r11) /* if sleeping, clear MSR.EE */ | ||
172 | rlwinm r9,r9,0,~MSR_EE | ||
173 | lwz r12,_LINK(r11) /* and return to address in LR */ | ||
174 | b fast_exception_return | ||
167 | #endif | 175 | #endif |
168 | 176 | ||
169 | /* | 177 | /* |
diff --git a/include/asm-powerpc/thread_info.h b/include/asm-powerpc/thread_info.h index e079e81051fd..b705c2a7651a 100644 --- a/include/asm-powerpc/thread_info.h +++ b/include/asm-powerpc/thread_info.h | |||
@@ -144,9 +144,11 @@ static inline struct thread_info *current_thread_info(void) | |||
144 | /* Bits in local_flags */ | 144 | /* Bits in local_flags */ |
145 | /* Don't move TLF_NAPPING without adjusting the code in entry_32.S */ | 145 | /* Don't move TLF_NAPPING without adjusting the code in entry_32.S */ |
146 | #define TLF_NAPPING 0 /* idle thread enabled NAP mode */ | 146 | #define TLF_NAPPING 0 /* idle thread enabled NAP mode */ |
147 | #define TLF_RESTORE_SIGMASK 1 /* Restore signal mask in do_signal */ | 147 | #define TLF_SLEEPING 1 /* suspend code enabled SLEEP mode */ |
148 | #define TLF_RESTORE_SIGMASK 2 /* Restore signal mask in do_signal */ | ||
148 | 149 | ||
149 | #define _TLF_NAPPING (1 << TLF_NAPPING) | 150 | #define _TLF_NAPPING (1 << TLF_NAPPING) |
151 | #define _TLF_SLEEPING (1 << TLF_SLEEPING) | ||
150 | #define _TLF_RESTORE_SIGMASK (1 << TLF_RESTORE_SIGMASK) | 152 | #define _TLF_RESTORE_SIGMASK (1 << TLF_RESTORE_SIGMASK) |
151 | 153 | ||
152 | #ifndef __ASSEMBLY__ | 154 | #ifndef __ASSEMBLY__ |