[Date Prev][Date Next][Thread Prev][Thread Next][Date Index][Thread Index] [Xen-changelog] [xen master] x86emul: correct handling of FPU insns faulting on memory write
commit a8fcd80c8f02a02e5c39cd9f76b2988b270d45e7 Author: Jan Beulich <jbeulich@xxxxxxxx> AuthorDate: Tue Mar 21 15:12:59 2017 +0100 Commit: Jan Beulich <jbeulich@xxxxxxxx> CommitDate: Tue Mar 21 15:12:59 2017 +0100 x86emul: correct handling of FPU insns faulting on memory write When an FPU instruction with a memory destination fails during the memory write, it should not affect FPU register state. Due to the way we emulate FPU (and SIMD) instructions, we can only guarantee this by - backing out changes to the FPU register state in such a case or - doing a descriptor read and/or page walk up front, perhaps with the stubs accessing the actual memory location then. The latter would require a significant change in how the emulator does its guest memory accessing, so for now the former variant is being chosen. Signed-off-by: Jan Beulich <jbeulich@xxxxxxxx> Reviewed-by: Andrew Cooper <andrew.cooper3@xxxxxxxxxx> Reviewed-by: Kevin Tian <kevin.tian@xxxxxxxxx> Reviewed-by: Paul Durrant <paul.durrant@xxxxxxxxxx> [hvm/emulate.c] Reviewed-by: Boris Ostrovsky <boris.ostrovsky@xxxxxxxxxx> --- tools/fuzz/x86_instruction_emulator/fuzz-emul.c | 1 + tools/tests/x86_emulator/test_x86_emulator.c | 1 + tools/tests/x86_emulator/x86_emulate.c | 7 ++++ tools/tests/x86_emulator/x86_emulate.h | 4 +++ xen/arch/x86/hvm/emulate.c | 46 ++++++++++++++++++++++++- xen/arch/x86/hvm/svm/svm.c | 1 + xen/arch/x86/hvm/vmx/vmx.c | 1 + xen/arch/x86/x86_emulate/x86_emulate.c | 14 +++++--- xen/arch/x86/x86_emulate/x86_emulate.h | 5 ++- xen/include/asm-x86/hvm/hvm.h | 2 ++ 10 files changed, 75 insertions(+), 7 deletions(-) diff --git a/tools/fuzz/x86_instruction_emulator/fuzz-emul.c b/tools/fuzz/x86_instruction_emulator/fuzz-emul.c index 3b3041d..890642c 100644 --- a/tools/fuzz/x86_instruction_emulator/fuzz-emul.c +++ b/tools/fuzz/x86_instruction_emulator/fuzz-emul.c @@ -433,6 +433,7 @@ static struct x86_emulate_ops fuzz_emulops = { SET(wbinvd), SET(invlpg), .get_fpu = emul_test_get_fpu, + .put_fpu = emul_test_put_fpu, .cpuid = emul_test_cpuid, }; #undef SET diff --git a/tools/tests/x86_emulator/test_x86_emulator.c b/tools/tests/x86_emulator/test_x86_emulator.c index bfe7ea4..5be8ddc 100644 --- a/tools/tests/x86_emulator/test_x86_emulator.c +++ b/tools/tests/x86_emulator/test_x86_emulator.c @@ -298,6 +298,7 @@ static struct x86_emulate_ops emulops = { .read_cr = emul_test_read_cr, .read_msr = read_msr, .get_fpu = emul_test_get_fpu, + .put_fpu = emul_test_put_fpu, }; int main(int argc, char **argv) diff --git a/tools/tests/x86_emulator/x86_emulate.c b/tools/tests/x86_emulator/x86_emulate.c index f98f7f7..574be31 100644 --- a/tools/tests/x86_emulator/x86_emulate.c +++ b/tools/tests/x86_emulator/x86_emulate.c @@ -138,4 +138,11 @@ int emul_test_get_fpu( return X86EMUL_OKAY; } +void emul_test_put_fpu( + struct x86_emulate_ctxt *ctxt, + enum x86_emulate_fpu_type backout) +{ + /* TBD */ +} + #include "x86_emulate/x86_emulate.c" diff --git a/tools/tests/x86_emulator/x86_emulate.h b/tools/tests/x86_emulator/x86_emulate.h index e9a9e50..f1857d1 100644 --- a/tools/tests/x86_emulator/x86_emulate.h +++ b/tools/tests/x86_emulator/x86_emulate.h @@ -178,3 +178,7 @@ int emul_test_get_fpu( void *exception_callback_arg, enum x86_emulate_fpu_type type, struct x86_emulate_ctxt *ctxt); + +void emul_test_put_fpu( + struct x86_emulate_ctxt *ctxt, + enum x86_emulate_fpu_type backout); diff --git a/xen/arch/x86/hvm/emulate.c b/xen/arch/x86/hvm/emulate.c index f36d7c9..3f118a0 100644 --- a/xen/arch/x86/hvm/emulate.c +++ b/xen/arch/x86/hvm/emulate.c @@ -15,6 +15,7 @@ #include <xen/paging.h> #include <xen/trace.h> #include <asm/event.h> +#include <asm/i387.h> #include <asm/xstate.h> #include <asm/hvm/emulate.h> #include <asm/hvm/hvm.h> @@ -1621,6 +1622,35 @@ static int hvmemul_get_fpu( if ( !curr->fpu_dirtied ) hvm_funcs.fpu_dirty_intercept(); + else if ( type == X86EMUL_FPU_fpu ) + { + const typeof(curr->arch.xsave_area->fpu_sse) *fpu_ctxt = + curr->arch.fpu_ctxt; + + /* + * Latch current register state so that we can back out changes + * if needed (namely when a memory write fails after register state + * has already been updated). + * NB: We don't really need the "enable" part of the called function + * (->fpu_dirtied set implies CR0.TS clear), but the additional + * overhead should be low enough to not warrant introduction of yet + * another slightly different function. However, we need to undo the + * ->fpu_dirtied clearing the function does as well as the possible + * masking of all exceptions by FNSTENV.) + */ + save_fpu_enable(); + curr->fpu_dirtied = true; + if ( (fpu_ctxt->fcw & 0x3f) != 0x3f ) + { + uint16_t fcw; + + asm ( "fnstcw %0" : "=m" (fcw) ); + if ( (fcw & 0x3f) == 0x3f ) + asm ( "fldcw %0" :: "m" (fpu_ctxt->fcw) ); + else + ASSERT(fcw == fpu_ctxt->fcw); + } + } curr->arch.hvm_vcpu.fpu_exception_callback = exception_callback; curr->arch.hvm_vcpu.fpu_exception_callback_arg = exception_callback_arg; @@ -1629,10 +1659,24 @@ static int hvmemul_get_fpu( } static void hvmemul_put_fpu( - struct x86_emulate_ctxt *ctxt) + struct x86_emulate_ctxt *ctxt, + enum x86_emulate_fpu_type backout) { struct vcpu *curr = current; + curr->arch.hvm_vcpu.fpu_exception_callback = NULL; + + if ( backout == X86EMUL_FPU_fpu ) + { + /* + * To back out changes to the register file simply adjust state such + * that upon next FPU insn use by the guest we'll reload the state + * saved (or freshly loaded) by hvmemul_get_fpu(). + */ + curr->fpu_dirtied = false; + stts(); + hvm_funcs.fpu_leave(curr); + } } static int hvmemul_invlpg( diff --git a/xen/arch/x86/hvm/svm/svm.c b/xen/arch/x86/hvm/svm/svm.c index 43241d6..b69789b 100644 --- a/xen/arch/x86/hvm/svm/svm.c +++ b/xen/arch/x86/hvm/svm/svm.c @@ -2211,6 +2211,7 @@ static struct hvm_function_table __initdata svm_function_table = { .update_guest_cr = svm_update_guest_cr, .update_guest_efer = svm_update_guest_efer, .update_guest_vendor = svm_update_guest_vendor, + .fpu_leave = svm_fpu_leave, .set_guest_pat = svm_set_guest_pat, .get_guest_pat = svm_get_guest_pat, .set_tsc_offset = svm_set_tsc_offset, diff --git a/xen/arch/x86/hvm/vmx/vmx.c b/xen/arch/x86/hvm/vmx/vmx.c index 4565f5d..d201956 100644 --- a/xen/arch/x86/hvm/vmx/vmx.c +++ b/xen/arch/x86/hvm/vmx/vmx.c @@ -2211,6 +2211,7 @@ static struct hvm_function_table __initdata vmx_function_table = { .update_guest_cr = vmx_update_guest_cr, .update_guest_efer = vmx_update_guest_efer, .update_guest_vendor = vmx_update_guest_vendor, + .fpu_leave = vmx_fpu_leave, .set_guest_pat = vmx_set_guest_pat, .get_guest_pat = vmx_get_guest_pat, .set_tsc_offset = vmx_set_tsc_offset, diff --git a/xen/arch/x86/x86_emulate/x86_emulate.c b/xen/arch/x86/x86_emulate/x86_emulate.c index ce725d8..19c38b2 100644 --- a/xen/arch/x86/x86_emulate/x86_emulate.c +++ b/xen/arch/x86/x86_emulate/x86_emulate.c @@ -965,6 +965,7 @@ static int _get_fpu( { unsigned long cr0; + fail_if(type == X86EMUL_FPU_fpu && !ops->put_fpu); fic->type = type; fail_if(!ops->read_cr); @@ -1026,11 +1027,14 @@ do { \ static void put_fpu( struct fpu_insn_ctxt *fic, + bool failed_late, struct x86_emulate_ctxt *ctxt, const struct x86_emulate_ops *ops) { - if ( fic->type != X86EMUL_FPU_none && ops->put_fpu ) - ops->put_fpu(ctxt); + if ( unlikely(failed_late) && fic->type == X86EMUL_FPU_fpu ) + ops->put_fpu(ctxt, X86EMUL_FPU_fpu); + else if ( fic->type != X86EMUL_FPU_none && ops->put_fpu ) + ops->put_fpu(ctxt, X86EMUL_FPU_none); fic->type = X86EMUL_FPU_none; } @@ -3734,9 +3738,9 @@ x86_emulate( break; case 0x9b: /* wait/fwait */ - fic.insn_bytes = 1; host_and_vcpu_must_have(fpu); get_fpu(X86EMUL_FPU_wait, &fic); + fic.insn_bytes = 1; asm volatile ( "fwait" ::: "memory" ); check_fpu_exn(&fic); break; @@ -7910,7 +7914,7 @@ x86_emulate( } complete_insn: /* Commit shadow register state. */ - put_fpu(&fic, ctxt, ops); + put_fpu(&fic, false, ctxt, ops); /* Zero the upper 32 bits of %rip if not in 64-bit mode. */ if ( !mode_64bit() ) @@ -7934,7 +7938,7 @@ x86_emulate( ctxt->regs->eflags &= ~X86_EFLAGS_RF; done: - put_fpu(&fic, ctxt, ops); + put_fpu(&fic, fic.insn_bytes > 0 && dst.type == OP_MEM, ctxt, ops); put_stub(stub); return rc; #undef state diff --git a/xen/arch/x86/x86_emulate/x86_emulate.h b/xen/arch/x86/x86_emulate/x86_emulate.h index 3d92a47..00b3f82 100644 --- a/xen/arch/x86/x86_emulate/x86_emulate.h +++ b/xen/arch/x86/x86_emulate/x86_emulate.h @@ -437,9 +437,12 @@ struct x86_emulate_ops * put_fpu: Relinquish the FPU. Unhook from FPU/SIMD exception handlers. * The handler, if installed, must be prepared to get called without * the get_fpu one having got called before! + * @backout: Undo updates to the specified register file (can, besides + * X86EMUL_FPU_none, only be X86EMUL_FPU_fpu at present); */ void (*put_fpu)( - struct x86_emulate_ctxt *ctxt); + struct x86_emulate_ctxt *ctxt, + enum x86_emulate_fpu_type backout); /* invlpg: Invalidate paging structures which map addressed byte. */ int (*invlpg)( diff --git a/xen/include/asm-x86/hvm/hvm.h b/xen/include/asm-x86/hvm/hvm.h index 2b4e328..f9bb190 100644 --- a/xen/include/asm-x86/hvm/hvm.h +++ b/xen/include/asm-x86/hvm/hvm.h @@ -137,6 +137,8 @@ struct hvm_function_table { void (*update_guest_vendor)(struct vcpu *v); + void (*fpu_leave)(struct vcpu *v); + int (*get_guest_pat)(struct vcpu *v, u64 *); int (*set_guest_pat)(struct vcpu *v, u64); -- generated by git-patchbot for /home/xen/git/xen.git#master _______________________________________________ Xen-changelog mailing list Xen-changelog@xxxxxxxxxxxxx https://lists.xenproject.org/xen-changelog
|
Lists.xenproject.org is hosted with RackSpace, monitoring our |