1 /* SPDX-License-Identifier: GPL-2.0 */
2 #ifndef __X86_KERNEL_FPU_LEGACY_H
3 #define __X86_KERNEL_FPU_LEGACY_H
5 #include <asm/fpu/types.h>
7 extern unsigned int mxcsr_feature_mask;
9 static inline void ldmxcsr(u32 mxcsr)
11 asm volatile("ldmxcsr %0" :: "m" (mxcsr));
15 * Returns 0 on success or the trap number when the operation raises an
18 #define user_insn(insn, output, input...) \
24 asm volatile(ASM_STAC "\n" \
27 _ASM_EXTABLE_TYPE(1b, 2b, EX_TYPE_FAULT_MCE_SAFE) \
28 : [err] "=a" (err), output \
33 #define kernel_insn_err(insn, output, input...) \
36 asm volatile("1:" #insn "\n\t" \
38 _ASM_EXTABLE_TYPE_REG(1b, 2b, EX_TYPE_EFAULT_REG, %[err]) \
39 : [err] "=r" (err), output \
44 #define kernel_insn(insn, output, input...) \
45 asm volatile("1:" #insn "\n\t" \
47 _ASM_EXTABLE_TYPE(1b, 2b, EX_TYPE_FPU_RESTORE) \
50 static inline int fnsave_to_user_sigframe(struct fregs_state __user *fx)
52 return user_insn(fnsave %[fx]; fwait, [fx] "=m" (*fx), "m" (*fx));
55 static inline int fxsave_to_user_sigframe(struct fxregs_state __user *fx)
57 if (IS_ENABLED(CONFIG_X86_32))
58 return user_insn(fxsave %[fx], [fx] "=m" (*fx), "m" (*fx));
60 return user_insn(fxsaveq %[fx], [fx] "=m" (*fx), "m" (*fx));
64 static inline void fxrstor(struct fxregs_state *fx)
66 if (IS_ENABLED(CONFIG_X86_32))
67 kernel_insn(fxrstor %[fx], "=m" (*fx), [fx] "m" (*fx));
69 kernel_insn(fxrstorq %[fx], "=m" (*fx), [fx] "m" (*fx));
72 static inline int fxrstor_safe(struct fxregs_state *fx)
74 if (IS_ENABLED(CONFIG_X86_32))
75 return kernel_insn_err(fxrstor %[fx], "=m" (*fx), [fx] "m" (*fx));
77 return kernel_insn_err(fxrstorq %[fx], "=m" (*fx), [fx] "m" (*fx));
80 static inline int fxrstor_from_user_sigframe(struct fxregs_state __user *fx)
82 if (IS_ENABLED(CONFIG_X86_32))
83 return user_insn(fxrstor %[fx], "=m" (*fx), [fx] "m" (*fx));
85 return user_insn(fxrstorq %[fx], "=m" (*fx), [fx] "m" (*fx));
88 static inline void frstor(struct fregs_state *fx)
90 kernel_insn(frstor %[fx], "=m" (*fx), [fx] "m" (*fx));
93 static inline int frstor_safe(struct fregs_state *fx)
95 return kernel_insn_err(frstor %[fx], "=m" (*fx), [fx] "m" (*fx));
98 static inline int frstor_from_user_sigframe(struct fregs_state __user *fx)
100 return user_insn(frstor %[fx], "=m" (*fx), [fx] "m" (*fx));
103 static inline void fxsave(struct fxregs_state *fx)
105 if (IS_ENABLED(CONFIG_X86_32))
106 asm volatile( "fxsave %[fx]" : [fx] "=m" (*fx));
108 asm volatile("fxsaveq %[fx]" : [fx] "=m" (*fx));