diff options
Diffstat (limited to 'arch/x86/kernel/xsave.c')
| -rw-r--r-- | arch/x86/kernel/xsave.c | 170 |
1 files changed, 152 insertions, 18 deletions
diff --git a/arch/x86/kernel/xsave.c b/arch/x86/kernel/xsave.c index a4ae302f03aa..9c253bd65e24 100644 --- a/arch/x86/kernel/xsave.c +++ b/arch/x86/kernel/xsave.c | |||
| @@ -16,11 +16,88 @@ | |||
| 16 | */ | 16 | */ |
| 17 | u64 pcntxt_mask; | 17 | u64 pcntxt_mask; |
| 18 | 18 | ||
| 19 | /* | ||
| 20 | * Represents init state for the supported extended state. | ||
| 21 | */ | ||
| 22 | static struct xsave_struct *init_xstate_buf; | ||
| 23 | |||
| 19 | struct _fpx_sw_bytes fx_sw_reserved; | 24 | struct _fpx_sw_bytes fx_sw_reserved; |
| 20 | #ifdef CONFIG_IA32_EMULATION | 25 | #ifdef CONFIG_IA32_EMULATION |
| 21 | struct _fpx_sw_bytes fx_sw_reserved_ia32; | 26 | struct _fpx_sw_bytes fx_sw_reserved_ia32; |
| 22 | #endif | 27 | #endif |
| 23 | 28 | ||
| 29 | static unsigned int *xstate_offsets, *xstate_sizes, xstate_features; | ||
| 30 | |||
| 31 | /* | ||
| 32 | * If a processor implementation discern that a processor state component is | ||
| 33 | * in its initialized state it may modify the corresponding bit in the | ||
| 34 | * xsave_hdr.xstate_bv as '0', with out modifying the corresponding memory | ||
| 35 | * layout in the case of xsaveopt. While presenting the xstate information to | ||
| 36 | * the user, we always ensure that the memory layout of a feature will be in | ||
| 37 | * the init state if the corresponding header bit is zero. This is to ensure | ||
| 38 | * that the user doesn't see some stale state in the memory layout during | ||
| 39 | * signal handling, debugging etc. | ||
| 40 | */ | ||
| 41 | void __sanitize_i387_state(struct task_struct *tsk) | ||
| 42 | { | ||
| 43 | u64 xstate_bv; | ||
| 44 | int feature_bit = 0x2; | ||
| 45 | struct i387_fxsave_struct *fx = &tsk->thread.fpu.state->fxsave; | ||
| 46 | |||
| 47 | if (!fx) | ||
| 48 | return; | ||
| 49 | |||
| 50 | BUG_ON(task_thread_info(tsk)->status & TS_USEDFPU); | ||
| 51 | |||
| 52 | xstate_bv = tsk->thread.fpu.state->xsave.xsave_hdr.xstate_bv; | ||
| 53 | |||
| 54 | /* | ||
| 55 | * None of the feature bits are in init state. So nothing else | ||
| 56 | * to do for us, as the memory layout is upto date. | ||
| 57 | */ | ||
| 58 | if ((xstate_bv & pcntxt_mask) == pcntxt_mask) | ||
| 59 | return; | ||
| 60 | |||
| 61 | /* | ||
| 62 | * FP is in init state | ||
| 63 | */ | ||
| 64 | if (!(xstate_bv & XSTATE_FP)) { | ||
| 65 | fx->cwd = 0x37f; | ||
| 66 | fx->swd = 0; | ||
| 67 | fx->twd = 0; | ||
| 68 | fx->fop = 0; | ||
| 69 | fx->rip = 0; | ||
| 70 | fx->rdp = 0; | ||
| 71 | memset(&fx->st_space[0], 0, 128); | ||
| 72 | } | ||
| 73 | |||
| 74 | /* | ||
| 75 | * SSE is in init state | ||
| 76 | */ | ||
| 77 | if (!(xstate_bv & XSTATE_SSE)) | ||
| 78 | memset(&fx->xmm_space[0], 0, 256); | ||
| 79 | |||
| 80 | xstate_bv = (pcntxt_mask & ~xstate_bv) >> 2; | ||
| 81 | |||
| 82 | /* | ||
| 83 | * Update all the other memory layouts for which the corresponding | ||
| 84 | * header bit is in the init state. | ||
| 85 | */ | ||
| 86 | while (xstate_bv) { | ||
| 87 | if (xstate_bv & 0x1) { | ||
| 88 | int offset = xstate_offsets[feature_bit]; | ||
| 89 | int size = xstate_sizes[feature_bit]; | ||
| 90 | |||
| 91 | memcpy(((void *) fx) + offset, | ||
| 92 | ((void *) init_xstate_buf) + offset, | ||
| 93 | size); | ||
| 94 | } | ||
| 95 | |||
| 96 | xstate_bv >>= 1; | ||
| 97 | feature_bit++; | ||
| 98 | } | ||
| 99 | } | ||
| 100 | |||
| 24 | /* | 101 | /* |
| 25 | * Check for the presence of extended state information in the | 102 | * Check for the presence of extended state information in the |
| 26 | * user fpstate pointer in the sigcontext. | 103 | * user fpstate pointer in the sigcontext. |
| @@ -102,6 +179,7 @@ int save_i387_xstate(void __user *buf) | |||
| 102 | task_thread_info(tsk)->status &= ~TS_USEDFPU; | 179 | task_thread_info(tsk)->status &= ~TS_USEDFPU; |
| 103 | stts(); | 180 | stts(); |
| 104 | } else { | 181 | } else { |
| 182 | sanitize_i387_state(tsk); | ||
| 105 | if (__copy_to_user(buf, &tsk->thread.fpu.state->fxsave, | 183 | if (__copy_to_user(buf, &tsk->thread.fpu.state->fxsave, |
| 106 | xstate_size)) | 184 | xstate_size)) |
| 107 | return -1; | 185 | return -1; |
| @@ -267,11 +345,6 @@ static void prepare_fx_sw_frame(void) | |||
| 267 | #endif | 345 | #endif |
| 268 | } | 346 | } |
| 269 | 347 | ||
| 270 | /* | ||
| 271 | * Represents init state for the supported extended state. | ||
| 272 | */ | ||
| 273 | struct xsave_struct *init_xstate_buf; | ||
| 274 | |||
| 275 | #ifdef CONFIG_X86_64 | 348 | #ifdef CONFIG_X86_64 |
| 276 | unsigned int sig_xstate_size = sizeof(struct _fpstate); | 349 | unsigned int sig_xstate_size = sizeof(struct _fpstate); |
| 277 | #endif | 350 | #endif |
| @@ -279,37 +352,77 @@ unsigned int sig_xstate_size = sizeof(struct _fpstate); | |||
| 279 | /* | 352 | /* |
| 280 | * Enable the extended processor state save/restore feature | 353 | * Enable the extended processor state save/restore feature |
| 281 | */ | 354 | */ |
| 282 | void __cpuinit xsave_init(void) | 355 | static inline void xstate_enable(void) |
| 283 | { | 356 | { |
| 284 | if (!cpu_has_xsave) | ||
| 285 | return; | ||
| 286 | |||
| 287 | set_in_cr4(X86_CR4_OSXSAVE); | 357 | set_in_cr4(X86_CR4_OSXSAVE); |
| 288 | |||
| 289 | /* | ||
| 290 | * Enable all the features that the HW is capable of | ||
| 291 | * and the Linux kernel is aware of. | ||
| 292 | */ | ||
| 293 | xsetbv(XCR_XFEATURE_ENABLED_MASK, pcntxt_mask); | 358 | xsetbv(XCR_XFEATURE_ENABLED_MASK, pcntxt_mask); |
| 294 | } | 359 | } |
| 295 | 360 | ||
| 296 | /* | 361 | /* |
| 362 | * Record the offsets and sizes of different state managed by the xsave | ||
| 363 | * memory layout. | ||
| 364 | */ | ||
| 365 | static void __init setup_xstate_features(void) | ||
| 366 | { | ||
| 367 | int eax, ebx, ecx, edx, leaf = 0x2; | ||
| 368 | |||
| 369 | xstate_features = fls64(pcntxt_mask); | ||
| 370 | xstate_offsets = alloc_bootmem(xstate_features * sizeof(int)); | ||
| 371 | xstate_sizes = alloc_bootmem(xstate_features * sizeof(int)); | ||
| 372 | |||
| 373 | do { | ||
| 374 | cpuid_count(XSTATE_CPUID, leaf, &eax, &ebx, &ecx, &edx); | ||
| 375 | |||
| 376 | if (eax == 0) | ||
| 377 | break; | ||
| 378 | |||
| 379 | xstate_offsets[leaf] = ebx; | ||
| 380 | xstate_sizes[leaf] = eax; | ||
| 381 | |||
| 382 | leaf++; | ||
| 383 | } while (1); | ||
| 384 | } | ||
| 385 | |||
| 386 | /* | ||
| 297 | * setup the xstate image representing the init state | 387 | * setup the xstate image representing the init state |
| 298 | */ | 388 | */ |
| 299 | static void __init setup_xstate_init(void) | 389 | static void __init setup_xstate_init(void) |
| 300 | { | 390 | { |
| 391 | setup_xstate_features(); | ||
| 392 | |||
| 393 | /* | ||
| 394 | * Setup init_xstate_buf to represent the init state of | ||
| 395 | * all the features managed by the xsave | ||
| 396 | */ | ||
| 301 | init_xstate_buf = alloc_bootmem(xstate_size); | 397 | init_xstate_buf = alloc_bootmem(xstate_size); |
| 302 | init_xstate_buf->i387.mxcsr = MXCSR_DEFAULT; | 398 | init_xstate_buf->i387.mxcsr = MXCSR_DEFAULT; |
| 399 | |||
| 400 | clts(); | ||
| 401 | /* | ||
| 402 | * Init all the features state with header_bv being 0x0 | ||
| 403 | */ | ||
| 404 | xrstor_state(init_xstate_buf, -1); | ||
| 405 | /* | ||
| 406 | * Dump the init state again. This is to identify the init state | ||
| 407 | * of any feature which is not represented by all zero's. | ||
| 408 | */ | ||
| 409 | xsave_state(init_xstate_buf, -1); | ||
| 410 | stts(); | ||
| 303 | } | 411 | } |
| 304 | 412 | ||
| 305 | /* | 413 | /* |
| 306 | * Enable and initialize the xsave feature. | 414 | * Enable and initialize the xsave feature. |
| 307 | */ | 415 | */ |
| 308 | void __ref xsave_cntxt_init(void) | 416 | static void __init xstate_enable_boot_cpu(void) |
| 309 | { | 417 | { |
| 310 | unsigned int eax, ebx, ecx, edx; | 418 | unsigned int eax, ebx, ecx, edx; |
| 311 | 419 | ||
| 312 | cpuid_count(0xd, 0, &eax, &ebx, &ecx, &edx); | 420 | if (boot_cpu_data.cpuid_level < XSTATE_CPUID) { |
| 421 | WARN(1, KERN_ERR "XSTATE_CPUID missing\n"); | ||
| 422 | return; | ||
| 423 | } | ||
| 424 | |||
| 425 | cpuid_count(XSTATE_CPUID, 0, &eax, &ebx, &ecx, &edx); | ||
| 313 | pcntxt_mask = eax + ((u64)edx << 32); | 426 | pcntxt_mask = eax + ((u64)edx << 32); |
| 314 | 427 | ||
| 315 | if ((pcntxt_mask & XSTATE_FPSSE) != XSTATE_FPSSE) { | 428 | if ((pcntxt_mask & XSTATE_FPSSE) != XSTATE_FPSSE) { |
| @@ -322,12 +435,13 @@ void __ref xsave_cntxt_init(void) | |||
| 322 | * Support only the state known to OS. | 435 | * Support only the state known to OS. |
| 323 | */ | 436 | */ |
| 324 | pcntxt_mask = pcntxt_mask & XCNTXT_MASK; | 437 | pcntxt_mask = pcntxt_mask & XCNTXT_MASK; |
| 325 | xsave_init(); | 438 | |
| 439 | xstate_enable(); | ||
| 326 | 440 | ||
| 327 | /* | 441 | /* |
| 328 | * Recompute the context size for enabled features | 442 | * Recompute the context size for enabled features |
| 329 | */ | 443 | */ |
| 330 | cpuid_count(0xd, 0, &eax, &ebx, &ecx, &edx); | 444 | cpuid_count(XSTATE_CPUID, 0, &eax, &ebx, &ecx, &edx); |
| 331 | xstate_size = ebx; | 445 | xstate_size = ebx; |
| 332 | 446 | ||
| 333 | update_regset_xstate_info(xstate_size, pcntxt_mask); | 447 | update_regset_xstate_info(xstate_size, pcntxt_mask); |
| @@ -339,3 +453,23 @@ void __ref xsave_cntxt_init(void) | |||
| 339 | "cntxt size 0x%x\n", | 453 | "cntxt size 0x%x\n", |
| 340 | pcntxt_mask, xstate_size); | 454 | pcntxt_mask, xstate_size); |
| 341 | } | 455 | } |
| 456 | |||
| 457 | /* | ||
| 458 | * For the very first instance, this calls xstate_enable_boot_cpu(); | ||
| 459 | * for all subsequent instances, this calls xstate_enable(). | ||
| 460 | * | ||
| 461 | * This is somewhat obfuscated due to the lack of powerful enough | ||
| 462 | * overrides for the section checks. | ||
| 463 | */ | ||
| 464 | void __cpuinit xsave_init(void) | ||
| 465 | { | ||
| 466 | static __refdata void (*next_func)(void) = xstate_enable_boot_cpu; | ||
| 467 | void (*this_func)(void); | ||
| 468 | |||
| 469 | if (!cpu_has_xsave) | ||
| 470 | return; | ||
| 471 | |||
| 472 | this_func = next_func; | ||
| 473 | next_func = xstate_enable; | ||
| 474 | this_func(); | ||
| 475 | } | ||
