2  * xsave/xrstor support.
 
   4  * Author: Suresh Siddha <suresh.b.siddha@intel.com>
 
   6 #include <linux/bootmem.h>
 
   7 #include <linux/compat.h>
 
   9 #ifdef CONFIG_IA32_EMULATION
 
  10 #include <asm/sigcontext32.h>
 
  15  * Supported feature mask by the CPU and the kernel.
 
  19 struct _fpx_sw_bytes fx_sw_reserved;
 
  20 #ifdef CONFIG_IA32_EMULATION
 
  21 struct _fpx_sw_bytes fx_sw_reserved_ia32;
 
  25  * Check for the presence of extended state information in the
 
  26  * user fpstate pointer in the sigcontext.
 
  28 int check_for_xstate(struct i387_fxsave_struct __user *buf,
 
  30                      struct _fpx_sw_bytes *fx_sw_user)
 
  32         int min_xstate_size = sizeof(struct i387_fxsave_struct) +
 
  33                               sizeof(struct xsave_hdr_struct);
 
  37         err = __copy_from_user(fx_sw_user, &buf->sw_reserved[0],
 
  38                                sizeof(struct _fpx_sw_bytes));
 
  44          * First Magic check failed.
 
  46         if (fx_sw_user->magic1 != FP_XSTATE_MAGIC1)
 
  50          * Check for error scenarios.
 
  52         if (fx_sw_user->xstate_size < min_xstate_size ||
 
  53             fx_sw_user->xstate_size > xstate_size ||
 
  54             fx_sw_user->xstate_size > fx_sw_user->extended_size)
 
  57         err = __get_user(magic2, (__u32 *) (((void *)fpstate) +
 
  58                                             fx_sw_user->extended_size -
 
  59                                             FP_XSTATE_MAGIC2_SIZE));
 
  61          * Check for the presence of second magic word at the end of memory
 
  62          * layout. This detects the case where the user just copied the legacy
 
  63          * fpstate layout with out copying the extended state information
 
  64          * in the memory layout.
 
  66         if (err || magic2 != FP_XSTATE_MAGIC2)
 
  74  * Signal frame handlers.
 
  77 int save_i387_xstate(void __user *buf)
 
  79         struct task_struct *tsk = current;
 
  82         if (!access_ok(VERIFY_WRITE, buf, sig_xstate_size))
 
  85         BUG_ON(sig_xstate_size < xstate_size);
 
  87         if ((unsigned long)buf % 64)
 
  88                 printk("save_i387_xstate: bad fpstate %p\n", buf);
 
  92         clear_used_math(); /* trigger finit */
 
  93         if (task_thread_info(tsk)->status & TS_USEDFPU) {
 
  95                  * Start with clearing the user buffer. This will present a
 
  96                  * clean context for the bytes not touched by the fxsave/xsave.
 
  98                 err = __clear_user(buf, sig_xstate_size);
 
 102                 if (task_thread_info(tsk)->status & TS_XSAVE)
 
 103                         err = xsave_user(buf);
 
 105                         err = fxsave_user(buf);
 
 109                 task_thread_info(tsk)->status &= ~TS_USEDFPU;
 
 112                 if (__copy_to_user(buf, &tsk->thread.xstate->fxsave,
 
 117         if (task_thread_info(tsk)->status & TS_XSAVE) {
 
 118                 struct _fpstate __user *fx = buf;
 
 119                 struct _xstate __user *x = buf;
 
 122                 err = __copy_to_user(&fx->sw_reserved, &fx_sw_reserved,
 
 123                                      sizeof(struct _fpx_sw_bytes));
 
 125                 err |= __put_user(FP_XSTATE_MAGIC2,
 
 126                                   (__u32 __user *) (buf + sig_xstate_size
 
 127                                                     - FP_XSTATE_MAGIC2_SIZE));
 
 130                  * Read the xstate_bv which we copied (directly from the cpu or
 
 131                  * from the state in task struct) to the user buffers and
 
 132                  * set the FP/SSE bits.
 
 134                 err |= __get_user(xstate_bv, &x->xstate_hdr.xstate_bv);
 
 137                  * For legacy compatible, we always set FP/SSE bits in the bit
 
 138                  * vector while saving the state to the user context. This will
 
 139                  * enable us capturing any changes(during sigreturn) to
 
 140                  * the FP/SSE bits by the legacy applications which don't touch
 
 141                  * xstate_bv in the xsave header.
 
 143                  * xsave aware apps can change the xstate_bv in the xsave
 
 144                  * header as well as change any contents in the memory layout.
 
 145                  * xrestore as part of sigreturn will capture all the changes.
 
 147                 xstate_bv |= XSTATE_FPSSE;
 
 149                 err |= __put_user(xstate_bv, &x->xstate_hdr.xstate_bv);
 
 159  * Restore the extended state if present. Otherwise, restore the FP/SSE
 
 162 static int restore_user_xstate(void __user *buf)
 
 164         struct _fpx_sw_bytes fx_sw_user;
 
 168         if (((unsigned long)buf % 64) ||
 
 169              check_for_xstate(buf, buf, &fx_sw_user))
 
 172         mask = fx_sw_user.xstate_bv;
 
 175          * restore the state passed by the user.
 
 177         err = xrestore_user(buf, mask);
 
 182          * init the state skipped by the user.
 
 184         mask = pcntxt_mask & ~mask;
 
 186         xrstor_state(init_xstate_buf, mask);
 
 192          * couldn't find the extended state information in the
 
 193          * memory layout. Restore just the FP/SSE and init all
 
 194          * the other extended state.
 
 196         xrstor_state(init_xstate_buf, pcntxt_mask & ~XSTATE_FPSSE);
 
 197         return fxrstor_checking((__force struct i387_fxsave_struct *)buf);
 
 201  * This restores directly out of user space. Exceptions are handled.
 
 203 int restore_i387_xstate(void __user *buf)
 
 205         struct task_struct *tsk = current;
 
 213                 if (!access_ok(VERIFY_READ, buf, sig_xstate_size))
 
 222         if (!(task_thread_info(current)->status & TS_USEDFPU)) {
 
 224                 task_thread_info(current)->status |= TS_USEDFPU;
 
 226         if (task_thread_info(tsk)->status & TS_XSAVE)
 
 227                 err = restore_user_xstate(buf);
 
 229                 err = fxrstor_checking((__force struct i387_fxsave_struct *)
 
 233                  * Encountered an error while doing the restore from the
 
 234                  * user buffer, clear the fpu state.
 
 245  * Prepare the SW reserved portion of the fxsave memory layout, indicating
 
 246  * the presence of the extended state information in the memory layout
 
 247  * pointed by the fpstate pointer in the sigcontext.
 
 248  * This will be saved when ever the FP and extended state context is
 
 249  * saved on the user stack during the signal handler delivery to the user.
 
 251 static void prepare_fx_sw_frame(void)
 
 253         int size_extended = (xstate_size - sizeof(struct i387_fxsave_struct)) +
 
 254                              FP_XSTATE_MAGIC2_SIZE;
 
 256         sig_xstate_size = sizeof(struct _fpstate) + size_extended;
 
 258 #ifdef CONFIG_IA32_EMULATION
 
 259         sig_xstate_ia32_size = sizeof(struct _fpstate_ia32) + size_extended;
 
 262         memset(&fx_sw_reserved, 0, sizeof(fx_sw_reserved));
 
 264         fx_sw_reserved.magic1 = FP_XSTATE_MAGIC1;
 
 265         fx_sw_reserved.extended_size = sig_xstate_size;
 
 266         fx_sw_reserved.xstate_bv = pcntxt_mask;
 
 267         fx_sw_reserved.xstate_size = xstate_size;
 
 268 #ifdef CONFIG_IA32_EMULATION
 
 269         memcpy(&fx_sw_reserved_ia32, &fx_sw_reserved,
 
 270                sizeof(struct _fpx_sw_bytes));
 
 271         fx_sw_reserved_ia32.extended_size = sig_xstate_ia32_size;
 
 276  * Represents init state for the supported extended state.
 
 278 struct xsave_struct *init_xstate_buf;
 
 281 unsigned int sig_xstate_size = sizeof(struct _fpstate);
 
 285  * Enable the extended processor state save/restore feature
 
 287 void __cpuinit xsave_init(void)
 
 292         set_in_cr4(X86_CR4_OSXSAVE);
 
 295          * Enable all the features that the HW is capable of
 
 296          * and the Linux kernel is aware of.
 
 298         xsetbv(XCR_XFEATURE_ENABLED_MASK, pcntxt_mask);
 
 302  * setup the xstate image representing the init state
 
 304 static void __init setup_xstate_init(void)
 
 306         init_xstate_buf = alloc_bootmem(xstate_size);
 
 307         init_xstate_buf->i387.mxcsr = MXCSR_DEFAULT;
 
 311  * Enable and initialize the xsave feature.
 
 313 void __ref xsave_cntxt_init(void)
 
 315         unsigned int eax, ebx, ecx, edx;
 
 317         cpuid_count(0xd, 0, &eax, &ebx, &ecx, &edx);
 
 318         pcntxt_mask = eax + ((u64)edx << 32);
 
 320         if ((pcntxt_mask & XSTATE_FPSSE) != XSTATE_FPSSE) {
 
 321                 printk(KERN_ERR "FP/SSE not shown under xsave features 0x%llx\n",
 
 327          * Support only the state known to OS.
 
 329         pcntxt_mask = pcntxt_mask & XCNTXT_MASK;
 
 333          * Recompute the context size for enabled features
 
 335         cpuid_count(0xd, 0, &eax, &ebx, &ecx, &edx);
 
 338         prepare_fx_sw_frame();
 
 342         printk(KERN_INFO "xsave/xrstor: enabled xstate_bv 0x%llx, "
 
 344                pcntxt_mask, xstate_size);