diff options
author | Paul Mundt <lethal@linux-sh.org> | 2008-03-26 06:02:47 -0400 |
---|---|---|
committer | Paul Mundt <lethal@linux-sh.org> | 2008-03-26 06:02:47 -0400 |
commit | 9bbafce2eec190ef7e44b0eb1095ba17ce6ad3af (patch) | |
tree | 8deae79efc43d39763bdba8ca9b670e6a1782859 /include/asm-sh/fpu.h | |
parent | 05dda977f2574c3341abef9b74c27d2b362e1e3a (diff) |
sh: Fix occasional FPU register corruption under preempt.
Presently with preempt enabled there's the possibility to be preempted
after the TIF_USEDFPU test and the register save, leading to bogus
state post-__switch_to(). Use an explicit preempt_disable()/enable()
pair around unlazy_fpu()/clear_fpu() to avoid this. Follows the x86
change.
Reported-by: Takuo Koguchi <takuo.koguchi.sw@hitachi.com>
Signed-off-by: Paul Mundt <lethal@linux-sh.org>
Diffstat (limited to 'include/asm-sh/fpu.h')
-rw-r--r-- | include/asm-sh/fpu.h | 32 |
1 files changed, 18 insertions, 14 deletions
diff --git a/include/asm-sh/fpu.h b/include/asm-sh/fpu.h index f8429880a270..f89abf5920d8 100644 --- a/include/asm-sh/fpu.h +++ b/include/asm-sh/fpu.h | |||
@@ -1,9 +1,8 @@ | |||
1 | #ifndef __ASM_SH_FPU_H | 1 | #ifndef __ASM_SH_FPU_H |
2 | #define __ASM_SH_FPU_H | 2 | #define __ASM_SH_FPU_H |
3 | 3 | ||
4 | #define SR_FD 0x00008000 | ||
5 | |||
6 | #ifndef __ASSEMBLY__ | 4 | #ifndef __ASSEMBLY__ |
5 | #include <linux/preempt.h> | ||
7 | #include <asm/ptrace.h> | 6 | #include <asm/ptrace.h> |
8 | 7 | ||
9 | #ifdef CONFIG_SH_FPU | 8 | #ifdef CONFIG_SH_FPU |
@@ -28,18 +27,23 @@ extern void save_fpu(struct task_struct *__tsk, struct pt_regs *regs); | |||
28 | 27 | ||
29 | extern int do_fpu_inst(unsigned short, struct pt_regs *); | 28 | extern int do_fpu_inst(unsigned short, struct pt_regs *); |
30 | 29 | ||
31 | #define unlazy_fpu(tsk, regs) do { \ | 30 | static inline void unlazy_fpu(struct task_struct *tsk, struct pt_regs *regs) |
32 | if (test_tsk_thread_flag(tsk, TIF_USEDFPU)) { \ | 31 | { |
33 | save_fpu(tsk, regs); \ | 32 | preempt_disable(); |
34 | } \ | 33 | if (test_tsk_thread_flag(tsk, TIF_USEDFPU)) |
35 | } while (0) | 34 | save_fpu(tsk, regs); |
36 | 35 | preempt_enable(); | |
37 | #define clear_fpu(tsk, regs) do { \ | 36 | } |
38 | if (test_tsk_thread_flag(tsk, TIF_USEDFPU)) { \ | 37 | |
39 | clear_tsk_thread_flag(tsk, TIF_USEDFPU); \ | 38 | static inline void clear_fpu(struct task_struct *tsk, struct pt_regs *regs) |
40 | release_fpu(regs); \ | 39 | { |
41 | } \ | 40 | preempt_disable(); |
42 | } while (0) | 41 | if (test_tsk_thread_flag(tsk, TIF_USEDFPU)) { |
42 | clear_tsk_thread_flag(tsk, TIF_USEDFPU); | ||
43 | release_fpu(regs); | ||
44 | } | ||
45 | preempt_enable(); | ||
46 | } | ||
43 | 47 | ||
44 | #endif /* __ASSEMBLY__ */ | 48 | #endif /* __ASSEMBLY__ */ |
45 | 49 | ||