aboutsummaryrefslogtreecommitdiffstats
path: root/include/asm-sh/fpu.h
diff options
context:
space:
mode:
authorPaul Mundt <lethal@linux-sh.org>2008-03-26 06:02:47 -0400
committerPaul Mundt <lethal@linux-sh.org>2008-03-26 06:02:47 -0400
commit9bbafce2eec190ef7e44b0eb1095ba17ce6ad3af (patch)
tree8deae79efc43d39763bdba8ca9b670e6a1782859 /include/asm-sh/fpu.h
parent05dda977f2574c3341abef9b74c27d2b362e1e3a (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.h32
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
29extern int do_fpu_inst(unsigned short, struct pt_regs *); 28extern int do_fpu_inst(unsigned short, struct pt_regs *);
30 29
31#define unlazy_fpu(tsk, regs) do { \ 30static 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); \ 38static 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