#endif
 extern void save_fpregs_to_fpstate(struct fpu *fpu);
 
-/* Returns 0 or the negated trap number, which results in -EFAULT for #PF */
+/*
+ * Returns 0 on success or the trap number when the operation raises an
+ * exception.
+ */
 #define user_insn(insn, output, input...)                              \
 ({                                                                     \
        int err;                                                        \
        asm volatile(ASM_STAC "\n"                                      \
                     "1: " #insn "\n"                                   \
                     "2: " ASM_CLAC "\n"                                \
-                    ".section .fixup,\"ax\"\n"                         \
-                    "3:  negl %%eax\n"                                 \
-                    "    jmp  2b\n"                                    \
-                    ".previous\n"                                      \
-                    _ASM_EXTABLE_TYPE(1b, 3b, EX_TYPE_FAULT_MCE_SAFE)  \
+                    _ASM_EXTABLE_TYPE(1b, 2b, EX_TYPE_FAULT_MCE_SAFE)  \
                     : [err] "=a" (err), output                         \
                     : "0"(0), input);                                  \
        err;                                                            \
 #define XRSTORS                ".byte " REX_PREFIX "0x0f,0xc7,0x1f"
 
 /*
- * After this @err contains 0 on success or the negated trap number when
- * the operation raises an exception. For faults this results in -EFAULT.
+ * After this @err contains 0 on success or the trap number when the
+ * operation raises an exception.
  */
 #define XSTATE_OP(op, st, lmask, hmask, err)                           \
        asm volatile("1:" op "\n\t"                                     \
                     "xor %[err], %[err]\n"                             \
                     "2:\n\t"                                           \
-                    ".pushsection .fixup,\"ax\"\n\t"                   \
-                    "3: negl %%eax\n\t"                                \
-                    "jmp 2b\n\t"                                       \
-                    ".popsection\n\t"                                  \
-                    _ASM_EXTABLE_TYPE(1b, 3b, EX_TYPE_FAULT_MCE_SAFE)  \
+                    _ASM_EXTABLE_TYPE(1b, 2b, EX_TYPE_FAULT_MCE_SAFE)  \
                     : [err] "=a" (err)                                 \
                     : "D" (st), "m" (*st), "a" (lmask), "d" (hmask)    \
                     : "memory")
 
 #include <asm/fpu/xstate.h>
 
 #include <asm/sigframe.h>
+#include <asm/trapnr.h>
 #include <asm/trace/fpu.h>
 
 static struct _fpx_sw_bytes fx_sw_reserved __ro_after_init;
                fpregs_unlock();
 
                /* Try to handle #PF, but anything else is fatal. */
-               if (ret != -EFAULT)
+               if (ret != X86_TRAP_PF)
                        return -EINVAL;
 
                ret = fault_in_pages_readable(buf, size);
                u64 mask = user_xfeatures | xfeatures_mask_supervisor();
 
                fpu->state.xsave.header.xfeatures &= mask;
-               ret = os_xrstor_safe(&fpu->state.xsave, xfeatures_mask_all);
+               ret = os_xrstor_safe(&fpu->state.xsave, xfeatures_mask_all) ? -EINVAL : 0;
        } else {
                ret = fxrstor_safe(&fpu->state.fxsave);
        }