# HG changeset patch # User Wei Huang # Date 1304449177 18000 # Node ID 40c9bab757e1cc3964ae6869022eebba7141d6eb # Parent 83db82b67f65bee91f35e9caaad700a78ac0a3fc FPU: create lazy and non-lazy FPU restore functions Currently Xen relies on #NM (via CR0.TS) to trigger FPU context restore. But not all FPU state is tracked by TS bit. This function creates two FPU restore functions: vcpu_restore_fpu_lazy() and vcpu_restore_fpu_eager(). vcpu_restore_fpu_lazy() is still used when #NM is triggered. vcpu_restore_fpu_eager(), as a comparision, is called for vcpu which is being scheduled in on every context switch. Signed-off-by: Wei Huang diff -r 83db82b67f65 -r 40c9bab757e1 xen/arch/x86/domain.c --- a/xen/arch/x86/domain.c Tue May 03 13:49:27 2011 -0500 +++ b/xen/arch/x86/domain.c Tue May 03 13:59:37 2011 -0500 @@ -1578,6 +1578,7 @@ memcpy(stack_regs, &n->arch.user_regs, CTXT_SWITCH_STACK_BYTES); if ( xsave_enabled(n) && n->arch.xcr0 != get_xcr0() ) set_xcr0(n->arch.xcr0); + vcpu_restore_fpu_eager(n); n->arch.ctxt_switch_to(n); } diff -r 83db82b67f65 -r 40c9bab757e1 xen/arch/x86/hvm/svm/svm.c --- a/xen/arch/x86/hvm/svm/svm.c Tue May 03 13:49:27 2011 -0500 +++ b/xen/arch/x86/hvm/svm/svm.c Tue May 03 13:59:37 2011 -0500 @@ -348,7 +348,7 @@ { struct vmcb_struct *vmcb = v->arch.hvm_svm.vmcb; - vcpu_restore_fpu(v); + vcpu_restore_fpu_lazy(v); vmcb_set_exception_intercepts( vmcb, vmcb_get_exception_intercepts(vmcb) & ~(1U << TRAP_no_device)); } diff -r 83db82b67f65 -r 40c9bab757e1 xen/arch/x86/hvm/vmx/vmx.c --- a/xen/arch/x86/hvm/vmx/vmx.c Tue May 03 13:49:27 2011 -0500 +++ b/xen/arch/x86/hvm/vmx/vmx.c Tue May 03 13:59:37 2011 -0500 @@ -612,7 +612,7 @@ static void vmx_fpu_enter(struct vcpu *v) { - vcpu_restore_fpu(v); + vcpu_restore_fpu_lazy(v); v->arch.hvm_vmx.exception_bitmap &= ~(1u << TRAP_no_device); vmx_update_exception_bitmap(v); v->arch.hvm_vmx.host_cr0 &= ~X86_CR0_TS; diff -r 83db82b67f65 -r 40c9bab757e1 xen/arch/x86/i387.c --- a/xen/arch/x86/i387.c Tue May 03 13:49:27 2011 -0500 +++ b/xen/arch/x86/i387.c Tue May 03 13:59:37 2011 -0500 @@ -160,10 +160,25 @@ /*******************************/ /* VCPU FPU Functions */ /*******************************/ +/* Restore FPU state whenever VCPU is schduled in. */ +void vcpu_restore_fpu_eager(struct vcpu *v) +{ + ASSERT(!is_idle_vcpu(v)); + + /* Avoid recursion */ + clts(); + + /* save the nonlazy extended state which is not tracked by CR0.TS bit */ + if ( xsave_enabled(v) ) + fpu_xrstor(v, XSTATE_NONLAZY); + + stts(); +} + /* * Restore FPU state when #NM is triggered. */ -void vcpu_restore_fpu(struct vcpu *v) +void vcpu_restore_fpu_lazy(struct vcpu *v) { ASSERT(!is_idle_vcpu(v)); @@ -174,7 +189,7 @@ return; if ( xsave_enabled(v) ) - fpu_xrstor(v, XSTATE_ALL); + fpu_xrstor(v, XSTATE_LAZY); else if ( v->fpu_initialised ) { if ( cpu_has_fxsr ) diff -r 83db82b67f65 -r 40c9bab757e1 xen/arch/x86/traps.c --- a/xen/arch/x86/traps.c Tue May 03 13:49:27 2011 -0500 +++ b/xen/arch/x86/traps.c Tue May 03 13:59:37 2011 -0500 @@ -3198,7 +3198,7 @@ BUG_ON(!guest_mode(regs)); - vcpu_restore_fpu(curr); + vcpu_restore_fpu_lazy(curr); if ( curr->arch.pv_vcpu.ctrlreg[0] & X86_CR0_TS ) { diff -r 83db82b67f65 -r 40c9bab757e1 xen/include/asm-x86/i387.h --- a/xen/include/asm-x86/i387.h Tue May 03 13:49:27 2011 -0500 +++ b/xen/include/asm-x86/i387.h Tue May 03 13:59:37 2011 -0500 @@ -14,7 +14,8 @@ #include #include -void vcpu_restore_fpu(struct vcpu *v); +void vcpu_restore_fpu_eager(struct vcpu *v); +void vcpu_restore_fpu_lazy(struct vcpu *v); void vcpu_save_fpu(struct vcpu *v); int vcpu_init_fpu(struct vcpu *v);