aboutsummaryrefslogtreecommitdiffstats
path: root/include/asm-x86/i387.h
diff options
context:
space:
mode:
Diffstat (limited to 'include/asm-x86/i387.h')
-rw-r--r--include/asm-x86/i387.h64
1 files changed, 59 insertions, 5 deletions
diff --git a/include/asm-x86/i387.h b/include/asm-x86/i387.h
index 6a6647896670..a6d256f4ac81 100644
--- a/include/asm-x86/i387.h
+++ b/include/asm-x86/i387.h
@@ -36,6 +36,8 @@ extern int save_i387_ia32(struct _fpstate_ia32 __user *buf);
36extern int restore_i387_ia32(struct _fpstate_ia32 __user *buf); 36extern int restore_i387_ia32(struct _fpstate_ia32 __user *buf);
37#endif 37#endif
38 38
39#define X87_FSW_ES (1 << 7) /* Exception Summary */
40
39#ifdef CONFIG_X86_64 41#ifdef CONFIG_X86_64
40 42
41/* Ignore delayed exceptions from user space */ 43/* Ignore delayed exceptions from user space */
@@ -46,7 +48,7 @@ static inline void tolerant_fwait(void)
46 _ASM_EXTABLE(1b, 2b)); 48 _ASM_EXTABLE(1b, 2b));
47} 49}
48 50
49static inline int restore_fpu_checking(struct i387_fxsave_struct *fx) 51static inline int fxrstor_checking(struct i387_fxsave_struct *fx)
50{ 52{
51 int err; 53 int err;
52 54
@@ -66,15 +68,31 @@ static inline int restore_fpu_checking(struct i387_fxsave_struct *fx)
66 return err; 68 return err;
67} 69}
68 70
69#define X87_FSW_ES (1 << 7) /* Exception Summary */ 71static inline int restore_fpu_checking(struct task_struct *tsk)
72{
73 if (task_thread_info(tsk)->status & TS_XSAVE)
74 return xrstor_checking(&tsk->thread.xstate->xsave);
75 else
76 return fxrstor_checking(&tsk->thread.xstate->fxsave);
77}
70 78
71/* AMD CPUs don't save/restore FDP/FIP/FOP unless an exception 79/* AMD CPUs don't save/restore FDP/FIP/FOP unless an exception
72 is pending. Clear the x87 state here by setting it to fixed 80 is pending. Clear the x87 state here by setting it to fixed
73 values. The kernel data segment can be sometimes 0 and sometimes 81 values. The kernel data segment can be sometimes 0 and sometimes
74 new user value. Both should be ok. 82 new user value. Both should be ok.
75 Use the PDA as safe address because it should be already in L1. */ 83 Use the PDA as safe address because it should be already in L1. */
76static inline void clear_fpu_state(struct i387_fxsave_struct *fx) 84static inline void clear_fpu_state(struct task_struct *tsk)
77{ 85{
86 struct xsave_struct *xstate = &tsk->thread.xstate->xsave;
87 struct i387_fxsave_struct *fx = &tsk->thread.xstate->fxsave;
88
89 /*
90 * xsave header may indicate the init state of the FP.
91 */
92 if ((task_thread_info(tsk)->status & TS_XSAVE) &&
93 !(xstate->xsave_hdr.xstate_bv & XSTATE_FP))
94 return;
95
78 if (unlikely(fx->swd & X87_FSW_ES)) 96 if (unlikely(fx->swd & X87_FSW_ES))
79 asm volatile("fnclex"); 97 asm volatile("fnclex");
80 alternative_input(ASM_NOP8 ASM_NOP2, 98 alternative_input(ASM_NOP8 ASM_NOP2,
@@ -107,7 +125,7 @@ static inline int save_i387_checking(struct i387_fxsave_struct __user *fx)
107 return err; 125 return err;
108} 126}
109 127
110static inline void __save_init_fpu(struct task_struct *tsk) 128static inline void fxsave(struct task_struct *tsk)
111{ 129{
112 /* Using "rex64; fxsave %0" is broken because, if the memory operand 130 /* Using "rex64; fxsave %0" is broken because, if the memory operand
113 uses any extended registers for addressing, a second REX prefix 131 uses any extended registers for addressing, a second REX prefix
@@ -132,7 +150,16 @@ static inline void __save_init_fpu(struct task_struct *tsk)
132 : "=m" (tsk->thread.xstate->fxsave) 150 : "=m" (tsk->thread.xstate->fxsave)
133 : "cdaSDb" (&tsk->thread.xstate->fxsave)); 151 : "cdaSDb" (&tsk->thread.xstate->fxsave));
134#endif 152#endif
135 clear_fpu_state(&tsk->thread.xstate->fxsave); 153}
154
155static inline void __save_init_fpu(struct task_struct *tsk)
156{
157 if (task_thread_info(tsk)->status & TS_XSAVE)
158 xsave(tsk);
159 else
160 fxsave(tsk);
161
162 clear_fpu_state(tsk);
136 task_thread_info(tsk)->status &= ~TS_USEDFPU; 163 task_thread_info(tsk)->status &= ~TS_USEDFPU;
137} 164}
138 165
@@ -147,6 +174,10 @@ static inline void tolerant_fwait(void)
147 174
148static inline void restore_fpu(struct task_struct *tsk) 175static inline void restore_fpu(struct task_struct *tsk)
149{ 176{
177 if (task_thread_info(tsk)->status & TS_XSAVE) {
178 xrstor_checking(&tsk->thread.xstate->xsave);
179 return;
180 }
150 /* 181 /*
151 * The "nop" is needed to make the instructions the same 182 * The "nop" is needed to make the instructions the same
152 * length. 183 * length.
@@ -172,6 +203,27 @@ static inline void restore_fpu(struct task_struct *tsk)
172 */ 203 */
173static inline void __save_init_fpu(struct task_struct *tsk) 204static inline void __save_init_fpu(struct task_struct *tsk)
174{ 205{
206 if (task_thread_info(tsk)->status & TS_XSAVE) {
207 struct xsave_struct *xstate = &tsk->thread.xstate->xsave;
208 struct i387_fxsave_struct *fx = &tsk->thread.xstate->fxsave;
209
210 xsave(tsk);
211
212 /*
213 * xsave header may indicate the init state of the FP.
214 */
215 if (!(xstate->xsave_hdr.xstate_bv & XSTATE_FP))
216 goto end;
217
218 if (unlikely(fx->swd & X87_FSW_ES))
219 asm volatile("fnclex");
220
221 /*
222 * we can do a simple return here or be paranoid :)
223 */
224 goto clear_state;
225 }
226
175 /* Use more nops than strictly needed in case the compiler 227 /* Use more nops than strictly needed in case the compiler
176 varies code */ 228 varies code */
177 alternative_input( 229 alternative_input(
@@ -181,6 +233,7 @@ static inline void __save_init_fpu(struct task_struct *tsk)
181 X86_FEATURE_FXSR, 233 X86_FEATURE_FXSR,
182 [fx] "m" (tsk->thread.xstate->fxsave), 234 [fx] "m" (tsk->thread.xstate->fxsave),
183 [fsw] "m" (tsk->thread.xstate->fxsave.swd) : "memory"); 235 [fsw] "m" (tsk->thread.xstate->fxsave.swd) : "memory");
236clear_state:
184 /* AMD K7/K8 CPUs don't save/restore FDP/FIP/FOP unless an exception 237 /* AMD K7/K8 CPUs don't save/restore FDP/FIP/FOP unless an exception
185 is pending. Clear the x87 state here by setting it to fixed 238 is pending. Clear the x87 state here by setting it to fixed
186 values. safe_address is a random variable that should be in L1 */ 239 values. safe_address is a random variable that should be in L1 */
@@ -190,6 +243,7 @@ static inline void __save_init_fpu(struct task_struct *tsk)
190 "fildl %[addr]", /* set F?P to defined value */ 243 "fildl %[addr]", /* set F?P to defined value */
191 X86_FEATURE_FXSAVE_LEAK, 244 X86_FEATURE_FXSAVE_LEAK,
192 [addr] "m" (safe_address)); 245 [addr] "m" (safe_address));
246end:
193 task_thread_info(tsk)->status &= ~TS_USEDFPU; 247 task_thread_info(tsk)->status &= ~TS_USEDFPU;
194} 248}
195 249