2 * Copyright (c) 2000-2010 Apple Inc. All rights reserved.
4 * @APPLE_OSREFERENCE_LICENSE_HEADER_START@
6 * This file contains Original Code and/or Modifications of Original Code
7 * as defined in and that are subject to the Apple Public Source License
8 * Version 2.0 (the 'License'). You may not use this file except in
9 * compliance with the License. The rights granted to you under the License
10 * may not be used to create, or enable the creation or redistribution of,
11 * unlawful or unlicensed copies of an Apple operating system, or to
12 * circumvent, violate, or enable the circumvention or violation of, any
13 * terms of an Apple operating system software license agreement.
15 * Please obtain a copy of the License at
16 * http://www.opensource.apple.com/apsl/ and read it before using this file.
18 * The Original Code and all software distributed under the License are
19 * distributed on an 'AS IS' basis, WITHOUT WARRANTY OF ANY KIND, EITHER
20 * EXPRESS OR IMPLIED, AND APPLE HEREBY DISCLAIMS ALL SUCH WARRANTIES,
21 * INCLUDING WITHOUT LIMITATION, ANY WARRANTIES OF MERCHANTABILITY,
22 * FITNESS FOR A PARTICULAR PURPOSE, QUIET ENJOYMENT OR NON-INFRINGEMENT.
23 * Please see the License for the specific language governing rights and
24 * limitations under the License.
26 * @APPLE_OSREFERENCE_LICENSE_HEADER_END@
32 * Mach Operating System
33 * Copyright (c) 1992-1990 Carnegie Mellon University
34 * All Rights Reserved.
36 * Permission to use, copy, modify and distribute this software and its
37 * documentation is hereby granted, provided that both the copyright
38 * notice and this permission notice appear in all copies of the
39 * software, derivative works or modified versions, and any portions
40 * thereof, and that both notices appear in supporting documentation.
42 * CARNEGIE MELLON ALLOWS FREE USE OF THIS SOFTWARE IN ITS "AS IS"
43 * CONDITION. CARNEGIE MELLON DISCLAIMS ANY LIABILITY OF ANY KIND FOR
44 * ANY DAMAGES WHATSOEVER RESULTING FROM THE USE OF THIS SOFTWARE.
46 * Carnegie Mellon requests users of this software to return to
48 * Software Distribution Coordinator or Software.Distribution@CS.CMU.EDU
49 * School of Computer Science
50 * Carnegie Mellon University
51 * Pittsburgh PA 15213-3890
53 * any improvements or extensions that they make and grant Carnegie Mellon
54 * the rights to redistribute these changes.
59 #include <platforms.h>
61 #include <mach/exception_types.h>
62 #include <mach/i386/thread_status.h>
63 #include <mach/i386/fp_reg.h>
64 #include <mach/branch_predicates.h>
66 #include <kern/mach_param.h>
67 #include <kern/processor.h>
68 #include <kern/thread.h>
69 #include <kern/zalloc.h>
70 #include <kern/misc_protos.h>
72 #include <kern/assert.h>
74 #include <libkern/OSAtomic.h>
76 #include <architecture/i386/pio.h>
77 #include <i386/cpuid.h>
79 #include <i386/proc_reg.h>
80 #include <i386/misc_protos.h>
81 #include <i386/thread.h>
82 #include <i386/trap.h>
84 int fp_kind
= FP_NO
; /* not inited */
85 zone_t ifps_zone
; /* zone for FPU save area */
87 #define ALIGNED(addr,size) (((uintptr_t)(addr)&((size)-1))==0)
91 extern void fpinit(void);
97 static void configure_mxcsr_capability_mask(struct x86_avx_thread_state
*fps
);
99 struct x86_avx_thread_state initial_fp_state
__attribute((aligned(64)));
102 /* Global MXCSR capability bitmask */
103 static unsigned int mxcsr_capability_mask
;
106 __asm__ volatile("fninit")
108 #define fnstcw(control) \
109 __asm__("fnstcw %0" : "=m" (*(unsigned short *)(control)))
111 #define fldcw(control) \
112 __asm__ volatile("fldcw %0" : : "m" (*(unsigned short *) &(control)) )
115 __asm__ volatile("fnclex")
117 #define fnsave(state) \
118 __asm__ volatile("fnsave %0" : "=m" (*state))
120 #define frstor(state) \
121 __asm__ volatile("frstor %0" : : "m" (state))
126 #define fxrstor(addr) __asm__ __volatile__("fxrstor %0" : : "m" (*(addr)))
127 #define fxsave(addr) __asm__ __volatile__("fxsave %0" : "=m" (*(addr)))
129 static uint32_t fp_register_state_size
= 0;
130 static uint32_t fpu_YMM_present
= FALSE
;
131 static uint32_t cpuid_reevaluated
= 0;
133 static void fpu_store_registers(void *, boolean_t
);
134 static void fpu_load_registers(void *);
136 extern void xsave64o(void);
137 extern void xrstor64o(void);
139 #define XMASK ((uint32_t) (XFEM_X87 | XFEM_SSE | XFEM_YMM))
141 /* DRK: TODO replace opcodes with mnemonics when assembler support available */
143 static inline void xsetbv(uint32_t mask_hi
, uint32_t mask_lo
) {
144 __asm__
__volatile__(".short 0x010F\n\t.byte 0xD1" :: "a"(mask_lo
), "d"(mask_hi
), "c" (XCR0
));
147 static inline void xsave(void *a
) {
148 /* MOD 0x4, operand ECX 0x1 */
149 __asm__
__volatile__(".short 0xAE0F\n\t.byte 0x21" :: "a"(XMASK
), "d"(0), "c" (a
));
152 static inline void xrstor(void *a
) {
153 /* MOD 0x5, operand ECX 0x1 */
154 __asm__
__volatile__(".short 0xAE0F\n\t.byte 0x29" :: "a"(XMASK
), "d"(0), "c" (a
));
157 static inline void xsave64(void *a
) {
158 /* Out of line call that executes in 64-bit mode on K32 */
159 __asm__
__volatile__("call _xsave64o" :: "a"(XMASK
), "d"(0), "c" (a
));
162 static inline void xrstor64(void *a
) {
163 /* Out of line call that executes in 64-bit mode on K32 */
164 __asm__
__volatile__("call _xrstor64o" :: "a"(XMASK
), "d"(0), "c" (a
));
167 static inline unsigned short
170 unsigned short status
;
171 __asm__
volatile("fnstsw %0" : "=ma" (status
));
176 * Configure the initial FPU state presented to new threads.
177 * Determine the MXCSR capability mask, which allows us to mask off any
178 * potentially unsafe "reserved" bits before restoring the FPU context.
179 * *Not* per-cpu, assumes symmetry.
183 configure_mxcsr_capability_mask(struct x86_avx_thread_state
*fps
)
185 /* XSAVE requires a 64 byte aligned store */
186 assert(ALIGNED(fps
, 64));
187 /* Clear, to prepare for the diagnostic FXSAVE */
188 bzero(fps
, sizeof(*fps
));
191 fpu_store_registers(fps
, FALSE
);
193 mxcsr_capability_mask
= fps
->fx_MXCSR_MASK
;
195 /* Set default mask value if necessary */
196 if (mxcsr_capability_mask
== 0)
197 mxcsr_capability_mask
= 0xffbf;
199 /* Clear vector register store */
200 bzero(&fps
->fx_XMM_reg
[0][0], sizeof(fps
->fx_XMM_reg
));
201 bzero(&fps
->x_YMMH_reg
[0][0], sizeof(fps
->x_YMMH_reg
));
203 fps
->fp_valid
= TRUE
;
204 fps
->fp_save_layout
= fpu_YMM_present
? XSAVE32
: FXSAVE32
;
205 fpu_load_registers(fps
);
207 /* Poison values to trap unsafe usage */
208 fps
->fp_valid
= 0xFFFFFFFF;
209 fps
->fp_save_layout
= FP_UNUSED
;
211 /* Re-enable FPU/SSE DNA exceptions */
217 * Look for FPU and initialize it.
218 * Called on each CPU.
224 unsigned short status
;
225 unsigned short control
;
228 * Check for FPU by initializing it,
229 * then trying to read the correct bit patterns from
230 * the control and status registers.
232 set_cr0((get_cr0() & ~(CR0_EM
|CR0_TS
)) | CR0_NE
); /* allow use of FPU */
238 assert(((status
& 0xff) == 0) && ((control
& 0x103f) == 0x3f));
240 /* Advertise SSE support */
241 if (cpuid_features() & CPUID_FEATURE_FXSR
) {
243 set_cr4(get_cr4() | CR4_OSFXS
);
244 /* And allow SIMD exceptions if present */
245 if (cpuid_features() & CPUID_FEATURE_SSE
) {
246 set_cr4(get_cr4() | CR4_OSXMM
);
248 fp_register_state_size
= sizeof(struct x86_fx_thread_state
);
251 panic("fpu is not FP_FXSR");
253 /* Configure the XSAVE context mechanism if the processor supports
256 if (cpuid_features() & CPUID_FEATURE_XSAVE
) {
257 cpuid_xsave_leaf_t
*xsp
= &cpuid_info()->cpuid_xsave_leaf
;
258 if (xsp
->extended_state
[0] & (uint32_t)XFEM_YMM
) {
259 assert(xsp
->extended_state
[0] & (uint32_t) XFEM_SSE
);
260 /* XSAVE container size for all features */
261 assert(xsp
->extended_state
[2] == sizeof(struct x86_avx_thread_state
));
262 fp_register_state_size
= sizeof(struct x86_avx_thread_state
);
263 fpu_YMM_present
= TRUE
;
264 set_cr4(get_cr4() | CR4_OSXSAVE
);
266 /* Re-evaluate CPUID, once, to reflect OSXSAVE */
267 if (OSCompareAndSwap(0, 1, &cpuid_reevaluated
))
269 /* DRK: consider verifying AVX offset with cpuid(d, ECX:2) */
273 fpu_YMM_present
= FALSE
;
278 * Trap wait instructions. Turn off FPU for now.
280 set_cr0(get_cr0() | CR0_TS
| CR0_MP
);
284 * Allocate and initialize FP state for current thread.
290 void *ifps
= zalloc(ifps_zone
);
293 if (!(ALIGNED(ifps
,64))) {
294 panic("fp_state_alloc: %p, %u, %p, %u", ifps
, (unsigned) ifps_zone
->elem_size
, (void *) ifps_zone
->free_elements
, (unsigned) ifps_zone
->alloc_size
);
301 fp_state_free(void *ifps
)
303 zfree(ifps_zone
, ifps
);
312 static void fpu_load_registers(void *fstate
) {
313 struct x86_fx_thread_state
*ifps
= fstate
;
314 fp_save_layout_t layout
= ifps
->fp_save_layout
;
316 assert(layout
== FXSAVE32
|| layout
== FXSAVE64
|| layout
== XSAVE32
|| layout
== XSAVE64
);
317 assert(ALIGNED(ifps
, 64));
318 assert(ml_get_interrupts_enabled() == FALSE
);
321 if (layout
== XSAVE32
|| layout
== XSAVE64
) {
322 struct x86_avx_thread_state
*iavx
= fstate
;
324 /* Verify reserved bits in the XSAVE header*/
325 if (iavx
->_xh
.xsbv
& ~7)
326 panic("iavx->_xh.xsbv: 0x%llx", iavx
->_xh
.xsbv
);
327 for (i
= 0; i
< sizeof(iavx
->_xh
.xhrsvd
); i
++)
328 if (iavx
->_xh
.xhrsvd
[i
])
329 panic("Reserved bit set");
331 if (fpu_YMM_present
) {
332 if (layout
!= XSAVE32
&& layout
!= XSAVE64
)
333 panic("Inappropriate layout: %u\n", layout
);
337 #if defined(__i386__)
338 if (layout
== FXSAVE32
) {
339 /* Restore the compatibility/legacy mode XMM+x87 state */
342 else if (layout
== FXSAVE64
) {
345 else if (layout
== XSAVE32
) {
348 else if (layout
== XSAVE64
) {
351 #elif defined(__x86_64__)
352 if ((layout
== XSAVE64
) || (layout
== XSAVE32
))
359 static void fpu_store_registers(void *fstate
, boolean_t is64
) {
360 struct x86_fx_thread_state
*ifps
= fstate
;
361 assert(ALIGNED(ifps
, 64));
362 #if defined(__i386__)
364 if (fpu_YMM_present
) {
366 ifps
->fp_save_layout
= XSAVE32
;
369 /* save the compatibility/legacy mode XMM+x87 state */
371 ifps
->fp_save_layout
= FXSAVE32
;
375 if (fpu_YMM_present
) {
377 ifps
->fp_save_layout
= XSAVE64
;
381 ifps
->fp_save_layout
= FXSAVE64
;
384 #elif defined(__x86_64__)
385 if (fpu_YMM_present
) {
387 ifps
->fp_save_layout
= is64
? XSAVE64
: XSAVE32
;
391 ifps
->fp_save_layout
= is64
? FXSAVE64
: FXSAVE32
;
397 * Initialize FP handling.
401 fpu_module_init(void)
403 if ((fp_register_state_size
!= sizeof(struct x86_fx_thread_state
)) &&
404 (fp_register_state_size
!= sizeof(struct x86_avx_thread_state
)))
405 panic("fpu_module_init: incorrect savearea size %u\n", fp_register_state_size
);
407 assert(fpu_YMM_present
!= 0xFFFFFFFF);
409 /* We explicitly choose an allocation size of 64
410 * to eliminate waste for the 832 byte sized
411 * AVX XSAVE register save area.
413 ifps_zone
= zinit(fp_register_state_size
,
414 thread_max
* fp_register_state_size
,
415 64 * fp_register_state_size
,
418 /* To maintain the required alignment, disable
419 * zone debugging for this zone as that appends
420 * 16 bytes to each element.
422 zone_change(ifps_zone
, Z_ALIGNMENT_REQUIRED
, TRUE
);
423 /* Determine MXCSR reserved bits and configure initial FPU state*/
424 configure_mxcsr_capability_mask(&initial_fp_state
);
428 * Save thread`s FPU context.
431 fpu_save_context(thread_t thread
)
433 struct x86_fx_thread_state
*ifps
;
435 assert(ml_get_interrupts_enabled() == FALSE
);
436 ifps
= (thread
)->machine
.ifps
;
438 if (ifps
&& ((ifps
->fp_valid
!= FALSE
) && (ifps
->fp_valid
!= TRUE
))) {
439 panic("ifps->fp_valid: %u\n", ifps
->fp_valid
);
442 if (ifps
!= 0 && (ifps
->fp_valid
== FALSE
)) {
443 /* Clear CR0.TS in preparation for the FP context save. In
444 * theory, this shouldn't be necessary since a live FPU should
445 * indicate that TS is clear. However, various routines
446 * (such as sendsig & sigreturn) manipulate TS directly.
449 /* registers are in FPU - save to memory */
450 fpu_store_registers(ifps
, (thread_is_64bit(thread
) && is_saved_state64(thread
->machine
.iss
)));
451 ifps
->fp_valid
= TRUE
;
458 * Free a FPU save area.
459 * Called only when thread terminating - no locking necessary.
468 * Set the floating-point state for a thread based
469 * on the FXSave formatted data. This is basically
470 * the same as fpu_set_state except it uses the
471 * expanded data structure.
472 * If the thread is not the current thread, it is
473 * not running (held). Locking needed against
474 * concurrent fpu_set_state or fpu_get_state.
479 thread_state_t tstate
,
482 struct x86_fx_thread_state
*ifps
;
483 struct x86_fx_thread_state
*new_ifps
;
484 x86_float_state64_t
*state
;
486 size_t state_size
= sizeof(struct x86_fx_thread_state
);
488 if (fp_kind
== FP_NO
)
491 if ((f
== x86_AVX_STATE32
|| f
== x86_AVX_STATE64
) &&
492 !ml_fpu_avx_enabled())
495 state
= (x86_float_state64_t
*)tstate
;
497 assert(thr_act
!= THREAD_NULL
);
498 pcb
= THREAD_TO_PCB(thr_act
);
502 * new FPU state is 'invalid'.
503 * Deallocate the fp state if it exists.
505 simple_lock(&pcb
->lock
);
510 simple_unlock(&pcb
->lock
);
516 * Valid state. Allocate the fp state if there is none.
520 simple_lock(&pcb
->lock
);
525 simple_unlock(&pcb
->lock
);
526 new_ifps
= fp_state_alloc();
534 * now copy over the new data.
536 old_valid
= ifps
->fp_valid
;
539 if ((old_valid
== FALSE
) && (thr_act
!= current_thread())) {
540 panic("fpu_set_fxstate inconsistency, thread: %p not stopped", thr_act
);
544 * Clear any reserved bits in the MXCSR to prevent a GPF
545 * when issuing an FXRSTOR.
548 state
->fpu_mxcsr
&= mxcsr_capability_mask
;
550 bcopy((char *)&state
->fpu_fcw
, (char *)ifps
, state_size
);
552 if (fpu_YMM_present
) {
553 struct x86_avx_thread_state
*iavx
= (void *) ifps
;
554 uint32_t fpu_nyreg
= 0;
556 if (f
== x86_AVX_STATE32
)
558 else if (f
== x86_AVX_STATE64
)
562 x86_avx_state64_t
*ystate
= (x86_avx_state64_t
*) state
;
563 bcopy(&ystate
->__fpu_ymmh0
, &iavx
->x_YMMH_reg
[0][0], fpu_nyreg
* sizeof(_STRUCT_XMM_REG
));
566 iavx
->fp_save_layout
= thread_is_64bit(thr_act
) ? XSAVE64
: XSAVE32
;
567 /* Sanitize XSAVE header */
568 bzero(&iavx
->_xh
.xhrsvd
[0], sizeof(iavx
->_xh
.xhrsvd
));
570 iavx
->_xh
.xsbv
= (XFEM_YMM
| XFEM_SSE
| XFEM_X87
);
572 iavx
->_xh
.xsbv
= (XFEM_SSE
| XFEM_X87
);
575 ifps
->fp_save_layout
= thread_is_64bit(thr_act
) ? FXSAVE64
: FXSAVE32
;
576 ifps
->fp_valid
= old_valid
;
578 if (old_valid
== FALSE
) {
579 boolean_t istate
= ml_set_interrupts_enabled(FALSE
);
580 ifps
->fp_valid
= TRUE
;
582 ml_set_interrupts_enabled(istate
);
585 simple_unlock(&pcb
->lock
);
588 fp_state_free(new_ifps
);
594 * Get the floating-point state for a thread.
595 * If the thread is not the current thread, it is
596 * not running (held). Locking needed against
597 * concurrent fpu_set_state or fpu_get_state.
602 thread_state_t tstate
,
605 struct x86_fx_thread_state
*ifps
;
606 x86_float_state64_t
*state
;
607 kern_return_t ret
= KERN_FAILURE
;
609 size_t state_size
= sizeof(struct x86_fx_thread_state
);
611 if (fp_kind
== FP_NO
)
614 if ((f
== x86_AVX_STATE32
|| f
== x86_AVX_STATE64
) &&
615 !ml_fpu_avx_enabled())
618 state
= (x86_float_state64_t
*)tstate
;
620 assert(thr_act
!= THREAD_NULL
);
621 pcb
= THREAD_TO_PCB(thr_act
);
623 simple_lock(&pcb
->lock
);
628 * No valid floating-point state.
631 bcopy((char *)&initial_fp_state
, (char *)&state
->fpu_fcw
,
634 simple_unlock(&pcb
->lock
);
639 * Make sure we`ve got the latest fp state info
640 * If the live fpu state belongs to our target
642 if (thr_act
== current_thread()) {
645 intr
= ml_set_interrupts_enabled(FALSE
);
651 (void)ml_set_interrupts_enabled(intr
);
653 if (ifps
->fp_valid
) {
654 bcopy((char *)ifps
, (char *)&state
->fpu_fcw
, state_size
);
655 if (fpu_YMM_present
) {
656 struct x86_avx_thread_state
*iavx
= (void *) ifps
;
657 uint32_t fpu_nyreg
= 0;
659 if (f
== x86_AVX_STATE32
)
661 else if (f
== x86_AVX_STATE64
)
665 x86_avx_state64_t
*ystate
= (x86_avx_state64_t
*) state
;
666 bcopy(&iavx
->x_YMMH_reg
[0][0], &ystate
->__fpu_ymmh0
, fpu_nyreg
* sizeof(_STRUCT_XMM_REG
));
672 simple_unlock(&pcb
->lock
);
680 * the child thread is 'stopped' with the thread
681 * mutex held and is currently not known by anyone
682 * so no way for fpu state to get manipulated by an
683 * outside agency -> no need for pcb lock
691 struct x86_fx_thread_state
*new_ifps
= NULL
;
695 ppcb
= THREAD_TO_PCB(parent
);
697 if (ppcb
->ifps
== NULL
)
700 if (child
->machine
.ifps
)
701 panic("fpu_dup_fxstate: child's ifps non-null");
703 new_ifps
= fp_state_alloc();
705 simple_lock(&ppcb
->lock
);
707 if (ppcb
->ifps
!= NULL
) {
708 struct x86_fx_thread_state
*ifps
= ppcb
->ifps
;
710 * Make sure we`ve got the latest fp state info
712 intr
= ml_set_interrupts_enabled(FALSE
);
713 assert(current_thread() == parent
);
718 (void)ml_set_interrupts_enabled(intr
);
720 if (ifps
->fp_valid
) {
721 child
->machine
.ifps
= new_ifps
;
722 assert((fp_register_state_size
== sizeof(struct x86_fx_thread_state
)) ||
723 (fp_register_state_size
== sizeof(struct x86_avx_thread_state
)));
724 bcopy((char *)(ppcb
->ifps
),
725 (char *)(child
->machine
.ifps
), fp_register_state_size
);
727 /* Mark the new fp saved state as non-live. */
728 /* Temporarily disabled: radar 4647827
729 * new_ifps->fp_valid = TRUE;
733 * Clear any reserved bits in the MXCSR to prevent a GPF
734 * when issuing an FXRSTOR.
736 new_ifps
->fx_MXCSR
&= mxcsr_capability_mask
;
740 simple_unlock(&ppcb
->lock
);
742 if (new_ifps
!= NULL
)
743 fp_state_free(new_ifps
);
755 unsigned short control
;
760 control
&= ~(FPC_PC
|FPC_RC
); /* Clear precision & rounding control */
761 control
|= (FPC_PC_64
| /* Set precision */
762 FPC_RC_RN
| /* round-to-nearest */
763 FPC_ZE
| /* Suppress zero-divide */
764 FPC_OE
| /* and overflow */
765 FPC_UE
| /* underflow */
766 FPC_IE
| /* Allow NaNQs and +-INF */
767 FPC_DE
| /* Allow denorms as operands */
768 FPC_PE
); /* No trap for precision loss */
771 /* Initialize SSE/SSE2 */
772 __builtin_ia32_ldmxcsr(0x1f80);
776 * Coprocessor not present.
785 struct x86_fx_thread_state
*ifps
= 0;
787 thr_act
= current_thread();
788 pcb
= THREAD_TO_PCB(thr_act
);
790 assert(fp_register_state_size
!= 0);
792 if (pcb
->ifps
== 0 && !get_interrupt_level()) {
793 ifps
= fp_state_alloc();
794 bcopy((char *)&initial_fp_state
, (char *)ifps
,
795 fp_register_state_size
);
796 if (!thread_is_64bit(thr_act
)) {
797 ifps
->fp_save_layout
= fpu_YMM_present
? XSAVE32
: FXSAVE32
;
800 ifps
->fp_save_layout
= fpu_YMM_present
? XSAVE64
: FXSAVE64
;
801 ifps
->fp_valid
= TRUE
;
803 intr
= ml_set_interrupts_enabled(FALSE
);
805 clear_ts(); /* Enable FPU use */
807 if (__improbable(get_interrupt_level())) {
809 * Save current coprocessor context if valid
810 * Initialize coprocessor live context
815 if (pcb
->ifps
== 0) {
820 * Load this thread`s state into coprocessor live context.
824 (void)ml_set_interrupts_enabled(intr
);
831 * FPU overran end of segment.
832 * Re-initialize FPU. Floating point state is not valid.
838 thread_t thr_act
= current_thread();
840 struct x86_fx_thread_state
*ifps
;
843 intr
= ml_set_interrupts_enabled(FALSE
);
845 if (get_interrupt_level())
846 panic("FPU segment overrun exception at interrupt context\n");
847 if (current_task() == kernel_task
)
848 panic("FPU segment overrun exception in kernel thread context\n");
851 * This is a non-recoverable error.
852 * Invalidate the thread`s FPU state.
854 pcb
= THREAD_TO_PCB(thr_act
);
855 simple_lock(&pcb
->lock
);
858 simple_unlock(&pcb
->lock
);
861 * Re-initialize the FPU.
867 * And disable access.
871 (void)ml_set_interrupts_enabled(intr
);
874 zfree(ifps_zone
, ifps
);
879 i386_exception(EXC_BAD_ACCESS
, VM_PROT_READ
|VM_PROT_EXECUTE
, 0);
884 * FPU error. Called by AST.
890 thread_t thr_act
= current_thread();
891 struct x86_fx_thread_state
*ifps
= thr_act
->machine
.ifps
;
894 intr
= ml_set_interrupts_enabled(FALSE
);
896 if (get_interrupt_level())
897 panic("FPU error exception at interrupt context\n");
898 if (current_task() == kernel_task
)
899 panic("FPU error exception in kernel thread context\n");
902 * Save the FPU state and turn off the FPU.
906 (void)ml_set_interrupts_enabled(intr
);
909 * Raise FPU exception.
910 * Locking not needed on pcb->ifps,
911 * since thread is running.
913 i386_exception(EXC_ARITHMETIC
,
923 * Locking not needed:
924 * . if called from fpu_get_state, pcb already locked.
925 * . if called from fpnoextflt or fp_intr, we are single-cpu
926 * . otherwise, thread is running.
927 * N.B.: Must be called with interrupts disabled
934 pcb_t pcb
= THREAD_TO_PCB(thr_act
);
935 struct x86_fx_thread_state
*ifps
= pcb
->ifps
;
938 if (ifps
!= 0 && !ifps
->fp_valid
) {
939 assert((get_cr0() & CR0_TS
) == 0);
940 /* registers are in FPU */
941 ifps
->fp_valid
= TRUE
;
942 fpu_store_registers(ifps
, thread_is_64bit(thr_act
));
947 * Restore FPU state from PCB.
949 * Locking not needed; always called on the current thread.
956 pcb_t pcb
= THREAD_TO_PCB(thr_act
);
957 struct x86_fx_thread_state
*ifps
= pcb
->ifps
;
960 assert(ifps
->fp_valid
== FALSE
|| ifps
->fp_valid
== TRUE
);
962 if (ifps
->fp_valid
== FALSE
) {
965 fpu_load_registers(ifps
);
967 ifps
->fp_valid
= FALSE
; /* in FPU */
971 * SSE arithmetic exception handling code.
972 * Basically the same as the x87 exception handler with a different subtype
978 thread_t thr_act
= current_thread();
979 struct x86_fx_thread_state
*ifps
= thr_act
->machine
.ifps
;
982 intr
= ml_set_interrupts_enabled(FALSE
);
984 if (get_interrupt_level())
985 panic("SSE exception at interrupt context\n");
986 if (current_task() == kernel_task
)
987 panic("SSE exception in kernel thread context\n");
990 * Save the FPU state and turn off the FPU.
994 (void)ml_set_interrupts_enabled(intr
);
996 * Raise FPU exception.
997 * Locking not needed on pcb->ifps,
998 * since thread is running.
1000 assert(ifps
->fp_save_layout
== FXSAVE32
|| ifps
->fp_save_layout
== FXSAVE64
);
1001 i386_exception(EXC_ARITHMETIC
,
1008 fp_setvalid(boolean_t value
) {
1009 thread_t thr_act
= current_thread();
1010 struct x86_fx_thread_state
*ifps
= thr_act
->machine
.ifps
;
1013 ifps
->fp_valid
= value
;
1015 if (value
== TRUE
) {
1016 boolean_t istate
= ml_set_interrupts_enabled(FALSE
);
1018 ml_set_interrupts_enabled(istate
);
1024 ml_fpu_avx_enabled(void) {
1025 return (fpu_YMM_present
== TRUE
);