VirtualBox

Changeset 73983 in vbox for trunk/src


Ignore:
Timestamp:
Aug 31, 2018 8:17:31 AM (6 years ago)
Author:
vboxsync
Message:

VMM/IEM, HM: Nested VMX: bugref:9180 Implement VMREAD, added using decoded IEM APIs for
VMXON, VMREAD, VMWRITE in VMX R0 code.

Location:
trunk/src/VBox/VMM
Files:
5 edited

Legend:

Unmodified
Added
Removed
  • trunk/src/VBox/VMM/VMMAll/HMVMXAll.cpp

    r73937 r73983  
    102102    VMX_INSTR_DIAG_DESC(kVmxVInstrDiag_Vmwrite_PtrInvalid    , "PtrInvalid"    ),
    103103    VMX_INSTR_DIAG_DESC(kVmxVInstrDiag_Vmwrite_PtrMap        , "PtrMap"        ),
    104     VMX_INSTR_DIAG_DESC(kVmxVInstrDiag_Vmwrite_Success       , "Success"       )
     104    VMX_INSTR_DIAG_DESC(kVmxVInstrDiag_Vmwrite_Success       , "Success"       ),
     105    /* VMREAD. */
     106    VMX_INSTR_DIAG_DESC(kVmxVInstrDiag_Vmread_Cpl            , "Cpl"           ),
     107    VMX_INSTR_DIAG_DESC(kVmxVInstrDiag_Vmread_FieldInvalid   , "FieldInvalid"  ),
     108    VMX_INSTR_DIAG_DESC(kVmxVInstrDiag_Vmread_LinkPtrInvalid , "LinkPtrInvalid"),
     109    VMX_INSTR_DIAG_DESC(kVmxVInstrDiag_Vmread_PtrInvalid     , "PtrInvalid"    ),
     110    VMX_INSTR_DIAG_DESC(kVmxVInstrDiag_Vmread_PtrMap         , "PtrMap"        ),
     111    VMX_INSTR_DIAG_DESC(kVmxVInstrDiag_Vmread_Success        , "Success"       )
    105112    /* kVmxVInstrDiag_Last */
    106113};
  • trunk/src/VBox/VMM/VMMAll/IEMAll.cpp

    r73961 r73983  
    1554015540
    1554115541/**
    15542  * Interface for HM and EM to emulate the VMWRITE instruction.
     15542 * Interface for HM and EM to emulate the VMREAD instruction.
    1554315543 *
    1554415544 * @returns Strict VBox status code.
    1554515545 * @param   pVCpu           The cross context virtual CPU structure of the calling EMT.
    15546  * @param   cbInstr         The instruction length in bytes.
    15547  * @param   u64Val          The value to write or guest linear address of the value
    15548  *                          to write.
    15549  * @param   uFieldEnc       The VMCS field encoding.
    1555015546 * @param   pExitInfo       Pointer to the VM-exit information struct.
    1555115547 * @thread  EMT(pVCpu)
    1555215548 */
    15553 VMM_INT_DECL(VBOXSTRICTRC) IEMExecDecodedVmwrite(PVMCPU pVCpu, uint8_t cbInstr, uint64_t u64Val, uint32_t uFieldEnc,
    15554                                                  PCVMXVEXITINFO pExitInfo)
    15555 {
    15556     IEMEXEC_ASSERT_INSTR_LEN_RETURN(cbInstr, 3);
     15549VMM_INT_DECL(VBOXSTRICTRC) IEMExecDecodedVmread(PVMCPU pVCpu, PCVMXVEXITINFO pExitInfo)
     15550{
     15551    IEMEXEC_ASSERT_INSTR_LEN_RETURN(pExitInfo->cbInstr, 3);
    1555715552    IEM_CTX_ASSERT(pVCpu, CPUMCTX_EXTRN_HWVIRT);
    1555815553    Assert(pExitInfo);
    1555915554
    15560     uint8_t const iEffSeg        = pExitInfo->ExitInstrInfo.VmreadVmwrite.iSegReg;
    15561     IEMMODE const enmEffAddrMode = (IEMMODE)pExitInfo->ExitInstrInfo.VmreadVmwrite.u3AddrSize;
    15562 
    1556315555    iemInitExec(pVCpu, false /*fBypassHandlers*/);
     15556
     15557    VBOXSTRICTRC   rcStrict;
     15558    uint8_t const  cbInstr   = pExitInfo->cbInstr;
     15559    uint32_t const uFieldEnc = iemGRegFetchU64(pVCpu, pExitInfo->InstrInfo.VmreadVmwrite.iReg2);
     15560    if (pExitInfo->InstrInfo.VmreadVmwrite.fIsRegOperand)
     15561    {
     15562        if (pVCpu->iem.s.enmCpuMode == IEMMODE_64BIT)
     15563        {
     15564            uint64_t *pu64Dst = iemGRegRefU64(pVCpu, pExitInfo->InstrInfo.VmreadVmwrite.iReg1);
     15565            rcStrict = iemVmxVmreadReg64(pVCpu, cbInstr, pu64Dst, uFieldEnc, pExitInfo);
     15566        }
     15567        else
     15568        {
     15569            uint32_t *pu32Dst = iemGRegRefU32(pVCpu, pExitInfo->InstrInfo.VmreadVmwrite.iReg1);
     15570            rcStrict = iemVmxVmreadReg32(pVCpu, cbInstr, pu32Dst, uFieldEnc, pExitInfo);
     15571        }
     15572    }
     15573    else
     15574    {
     15575        RTGCPTR GCPtrDst       = pExitInfo->GCPtrEffAddr;
     15576        uint8_t iEffSeg        = pExitInfo->InstrInfo.VmreadVmwrite.iSegReg;
     15577        IEMMODE enmEffAddrMode = (IEMMODE)pExitInfo->InstrInfo.VmreadVmwrite.u3AddrSize;
     15578        rcStrict = iemVmxVmreadMem(pVCpu, cbInstr, iEffSeg, enmEffAddrMode, GCPtrDst, uFieldEnc, pExitInfo);
     15579    }
     15580    if (pVCpu->iem.s.cActiveMappings)
     15581        iemMemRollback(pVCpu);
     15582    return iemExecStatusCodeFiddling(pVCpu, rcStrict);
     15583}
     15584
     15585
     15586/**
     15587 * Interface for HM and EM to emulate the VMWRITE instruction.
     15588 *
     15589 * @returns Strict VBox status code.
     15590 * @param   pVCpu           The cross context virtual CPU structure of the calling EMT.
     15591 * @param   pExitInfo       Pointer to the VM-exit information struct.
     15592 * @thread  EMT(pVCpu)
     15593 */
     15594VMM_INT_DECL(VBOXSTRICTRC) IEMExecDecodedVmwrite(PVMCPU pVCpu, PCVMXVEXITINFO pExitInfo)
     15595{
     15596    IEMEXEC_ASSERT_INSTR_LEN_RETURN(pExitInfo->cbInstr, 3);
     15597    IEM_CTX_ASSERT(pVCpu, CPUMCTX_EXTRN_HWVIRT);
     15598    Assert(pExitInfo);
     15599
     15600    iemInitExec(pVCpu, false /*fBypassHandlers*/);
     15601
     15602    uint64_t u64Val;
     15603    uint8_t  iEffSeg;
     15604    IEMMODE  enmEffAddrMode;
     15605    if (pExitInfo->InstrInfo.VmreadVmwrite.fIsRegOperand)
     15606    {
     15607        u64Val         = iemGRegFetchU64(pVCpu, pExitInfo->InstrInfo.VmreadVmwrite.iReg1);
     15608        iEffSeg        = UINT8_MAX;
     15609        enmEffAddrMode = UINT8_MAX;
     15610    }
     15611    else
     15612    {
     15613        u64Val         = pExitInfo->GCPtrEffAddr;
     15614        iEffSeg        = pExitInfo->InstrInfo.VmreadVmwrite.iSegReg;
     15615        enmEffAddrMode = (IEMMODE)pExitInfo->InstrInfo.VmreadVmwrite.u3AddrSize;
     15616    }
     15617    uint8_t const  cbInstr   = pExitInfo->cbInstr;
     15618    uint32_t const uFieldEnc = iemGRegFetchU64(pVCpu, pExitInfo->InstrInfo.VmreadVmwrite.iReg2);
    1556415619    VBOXSTRICTRC rcStrict = iemVmxVmwrite(pVCpu, cbInstr, iEffSeg, enmEffAddrMode, u64Val, uFieldEnc, pExitInfo);
    1556515620    if (pVCpu->iem.s.cActiveMappings)
     
    1557415629 * @returns Strict VBox status code.
    1557515630 * @param   pVCpu           The cross context virtual CPU structure of the calling EMT.
    15576  * @param   cbInstr         The instruction length in bytes.
    15577  * @param   iEffSeg         The effective segment register to use with @a GCPtrVmcs.
    15578  * @param   GCPtrVmcs       The linear address of the VMCS pointer.
    1557915631 * @param   pExitInfo       Pointer to the VM-exit information struct.
    1558015632 * @thread  EMT(pVCpu)
    1558115633 */
    15582 VMM_INT_DECL(VBOXSTRICTRC) IEMExecDecodedVmptrld(PVMCPU pVCpu, uint8_t cbInstr, uint8_t iEffSeg, RTGCPHYS GCPtrVmcs,
    15583                                                  PCVMXVEXITINFO pExitInfo)
    15584 {
    15585     IEMEXEC_ASSERT_INSTR_LEN_RETURN(cbInstr, 3);
     15634VMM_INT_DECL(VBOXSTRICTRC) IEMExecDecodedVmptrld(PVMCPU pVCpu, PCVMXVEXITINFO pExitInfo)
     15635{
     15636    Assert(pExitInfo);
     15637    IEMEXEC_ASSERT_INSTR_LEN_RETURN(pExitInfo->cbInstr, 3);
    1558615638    IEM_CTX_ASSERT(pVCpu, CPUMCTX_EXTRN_HWVIRT);
    15587     Assert(pExitInfo);
    1558815639
    1558915640    iemInitExec(pVCpu, false /*fBypassHandlers*/);
     15641
     15642    uint8_t const iEffSeg   = pExitInfo->InstrInfo.VmxXsave.iSegReg;
     15643    uint8_t const cbInstr   = pExitInfo->cbInstr;
     15644    RTGCPTR const GCPtrVmcs = pExitInfo->GCPtrEffAddr;
    1559015645    VBOXSTRICTRC rcStrict = iemVmxVmptrld(pVCpu, cbInstr, iEffSeg, GCPtrVmcs, pExitInfo);
    1559115646    if (pVCpu->iem.s.cActiveMappings)
     
    1560015655 * @returns Strict VBox status code.
    1560115656 * @param   pVCpu           The cross context virtual CPU structure of the calling EMT.
    15602  * @param   cbInstr         The instruction length in bytes.
    15603  * @param   iEffSeg         The effective segment register to use with @a GCPtrVmcs.
    15604  * @param   GCPtrVmcs       The linear address of the VMCS pointer.
    1560515657 * @param   pExitInfo       Pointer to the VM-exit information struct.
    1560615658 * @thread  EMT(pVCpu)
    1560715659 */
    15608 VMM_INT_DECL(VBOXSTRICTRC) IEMExecDecodedVmptrst(PVMCPU pVCpu, uint8_t cbInstr, uint8_t iEffSeg, RTGCPHYS GCPtrVmcs,
    15609                                                  PCVMXVEXITINFO pExitInfo)
    15610 {
    15611     IEMEXEC_ASSERT_INSTR_LEN_RETURN(cbInstr, 3);
     15660VMM_INT_DECL(VBOXSTRICTRC) IEMExecDecodedVmptrst(PVMCPU pVCpu, PCVMXVEXITINFO pExitInfo)
     15661{
     15662    Assert(pExitInfo);
     15663    IEMEXEC_ASSERT_INSTR_LEN_RETURN(pExitInfo->cbInstr, 3);
    1561215664    IEM_CTX_ASSERT(pVCpu, CPUMCTX_EXTRN_HWVIRT);
    15613     Assert(pExitInfo);
    1561415665
    1561515666    iemInitExec(pVCpu, false /*fBypassHandlers*/);
     15667
     15668    uint8_t const iEffSeg   = pExitInfo->InstrInfo.VmxXsave.iSegReg;
     15669    uint8_t const cbInstr   = pExitInfo->cbInstr;
     15670    RTGCPTR const GCPtrVmcs = pExitInfo->GCPtrEffAddr;
    1561615671    VBOXSTRICTRC rcStrict = iemVmxVmptrst(pVCpu, cbInstr, iEffSeg, GCPtrVmcs, pExitInfo);
    1561715672    if (pVCpu->iem.s.cActiveMappings)
     
    1562615681 * @returns Strict VBox status code.
    1562715682 * @param   pVCpu           The cross context virtual CPU structure of the calling EMT.
    15628  * @param   cbInstr         The instruction length in bytes.
    15629  * @param   iEffSeg         The effective segment register to use with @a GCPtrVmcs.
    15630  * @param   GCPtrVmcs       The linear address of the VMCS pointer.
    1563115683 * @param   pExitInfo       Pointer to the VM-exit information struct.
    1563215684 * @thread  EMT(pVCpu)
    1563315685 */
    15634 VMM_INT_DECL(VBOXSTRICTRC) IEMExecDecodedVmclear(PVMCPU pVCpu, uint8_t cbInstr, uint8_t iEffSeg, RTGCPHYS GCPtrVmcs,
    15635                                                  PCVMXVEXITINFO pExitInfo)
    15636 {
    15637     IEMEXEC_ASSERT_INSTR_LEN_RETURN(cbInstr, 3);
     15686VMM_INT_DECL(VBOXSTRICTRC) IEMExecDecodedVmclear(PVMCPU pVCpu, PCVMXVEXITINFO pExitInfo)
     15687{
     15688    Assert(pExitInfo);
     15689    IEMEXEC_ASSERT_INSTR_LEN_RETURN(pExitInfo->cbInstr, 3);
    1563815690    IEM_CTX_ASSERT(pVCpu, CPUMCTX_EXTRN_HWVIRT);
    15639     Assert(pExitInfo);
    1564015691
    1564115692    iemInitExec(pVCpu, false /*fBypassHandlers*/);
     15693
     15694    uint8_t const iEffSeg   = pExitInfo->InstrInfo.VmxXsave.iSegReg;
     15695    uint8_t const cbInstr   = pExitInfo->cbInstr;
     15696    RTGCPTR const GCPtrVmcs = pExitInfo->GCPtrEffAddr;
    1564215697    VBOXSTRICTRC rcStrict = iemVmxVmclear(pVCpu, cbInstr, iEffSeg, GCPtrVmcs, pExitInfo);
    1564315698    if (pVCpu->iem.s.cActiveMappings)
     
    1565215707 * @returns Strict VBox status code.
    1565315708 * @param   pVCpu           The cross context virtual CPU structure of the calling EMT.
    15654  * @param   cbInstr         The instruction length in bytes.
    15655  * @param   iEffSeg         The effective segment register to use with @a
    15656  *                          GCPtrVmxon.
    15657  * @param   GCPtrVmxon      The linear address of the VMXON pointer.
    15658  * @param   pExitInfo       The VM-exit instruction information struct.
     15709 * @param   pExitInfo       Pointer to the VM-exit information struct.
    1565915710 * @thread  EMT(pVCpu)
    1566015711 */
    15661 VMM_INT_DECL(VBOXSTRICTRC) IEMExecDecodedVmxon(PVMCPU pVCpu, uint8_t cbInstr, uint8_t iEffSeg, RTGCPHYS GCPtrVmxon,
    15662                                                PCVMXVEXITINFO pExitInfo)
    15663 {
    15664     IEMEXEC_ASSERT_INSTR_LEN_RETURN(cbInstr, 3);
     15712VMM_INT_DECL(VBOXSTRICTRC) IEMExecDecodedVmxon(PVMCPU pVCpu, PCVMXVEXITINFO pExitInfo)
     15713{
     15714    Assert(pExitInfo);
     15715    IEMEXEC_ASSERT_INSTR_LEN_RETURN(pExitInfo->cbInstr, 3);
    1566515716    IEM_CTX_ASSERT(pVCpu, CPUMCTX_EXTRN_HWVIRT);
    15666     Assert(pExitInfo);
    1566715717
    1566815718    iemInitExec(pVCpu, false /*fBypassHandlers*/);
     15719
     15720    uint8_t const iEffSeg    = pExitInfo->InstrInfo.VmxXsave.iSegReg;
     15721    uint8_t const cbInstr    = pExitInfo->cbInstr;
     15722    RTGCPTR const GCPtrVmxon = pExitInfo->GCPtrEffAddr;
    1566915723    VBOXSTRICTRC rcStrict = iemVmxVmxon(pVCpu, cbInstr, iEffSeg, GCPtrVmxon, pExitInfo);
    1567015724    if (pVCpu->iem.s.cActiveMappings)
  • trunk/src/VBox/VMM/VMMAll/IEMAllCImplVmxInstr.cpp.h

    r73970 r73983  
    654654 * @param   pVCpu           The cross context virtual CPU structure.
    655655 * @param   uExitReason     The VM-exit reason.
    656  * @param   InstrId         The VM-exit instruction identity (VMX_INSTR_ID_XXX) if
     656 * @param   uInstrId        The VM-exit instruction identity (VMX_INSTR_ID_XXX) if
    657657 *                          any. Pass VMX_INSTR_ID_NONE otherwise.
     658 * @param   fPrimaryOpRead  If the primary operand of the ModR/M byte (bits 0:3) is
     659 *                          a read or write.
    658660 * @param   pGCPtrDisp      Where to store the displacement field. Optional, can be
    659661 *                          NULL.
    660662 */
    661 IEM_STATIC uint32_t iemVmxGetExitInstrInfo(PVMCPU pVCpu, uint32_t uExitReason, VMXINSTRID InstrId, PRTGCPTR pGCPtrDisp)
     663IEM_STATIC uint32_t iemVmxGetExitInstrInfo(PVMCPU pVCpu, uint32_t uExitReason, VMXINSTRID uInstrId, bool fPrimaryOpRead,
     664                                           PRTGCPTR pGCPtrDisp)
    662665{
    663666    RTGCPTR          GCPtrDisp;
     
    675678        /*
    676679         * ModR/M indicates register addressing.
     680         *
     681         * The primary/secondary register operands are reported in the iReg1 or iReg2
     682         * fields depending on whether it is a read/write form.
    677683         */
     684        uint8_t idxReg1;
     685        uint8_t idxReg2;
     686        if (fPrimaryOpRead)
     687        {
     688            idxReg1 = ((bRm >> X86_MODRM_REG_SHIFT) & X86_MODRM_REG_SMASK) | pVCpu->iem.s.uRexReg;
     689            idxReg2 = (bRm & X86_MODRM_RM_MASK) | pVCpu->iem.s.uRexB;
     690        }
     691        else
     692        {
     693            idxReg1 = (bRm & X86_MODRM_RM_MASK) | pVCpu->iem.s.uRexB;
     694            idxReg2 = ((bRm >> X86_MODRM_REG_SHIFT) & X86_MODRM_REG_SMASK) | pVCpu->iem.s.uRexReg;
     695        }
    678696        ExitInstrInfo.All.u2Scaling       = 0;
    679         ExitInstrInfo.All.iReg1           = (bRm & X86_MODRM_RM_MASK) | pVCpu->iem.s.uRexB;
     697        ExitInstrInfo.All.iReg1           = idxReg1;
    680698        ExitInstrInfo.All.u3AddrSize      = pVCpu->iem.s.enmEffAddrMode;
    681699        ExitInstrInfo.All.fIsRegOperand   = 1;
     
    686704        ExitInstrInfo.All.iBaseReg        = 0;
    687705        ExitInstrInfo.All.fBaseRegInvalid = 1;
    688         ExitInstrInfo.All.iReg2           = ((bRm >> X86_MODRM_REG_SHIFT) & X86_MODRM_REG_SMASK) | pVCpu->iem.s.uRexReg;
     706        ExitInstrInfo.All.iReg2           = idxReg2;
    689707
    690708        /* Displacement not applicable for register addressing. */
     
    701719        uint8_t  iBaseReg      = 0;
    702720        uint8_t  iIdxReg       = 0;
    703         uint8_t  iReg2         = 0;
    704721        if (pVCpu->iem.s.enmEffAddrMode == IEMMODE_16BIT)
    705722        {
     
    747764            Assert(!uScale);                    /* There's no scaling/SIB byte for 16-bit addressing. */
    748765            GCPtrDisp = (int16_t)u16Disp;       /* Sign-extend the displacement. */
    749             iReg2     = ((bRm >> X86_MODRM_REG_SHIFT) & X86_MODRM_REG_SMASK);
    750766        }
    751767        else if (pVCpu->iem.s.enmEffAddrMode == IEMMODE_32BIT)
     
    819835
    820836            GCPtrDisp = (int32_t)u32Disp;       /* Sign-extend the displacement. */
    821             iReg2     = ((bRm >> X86_MODRM_REG_SHIFT) & X86_MODRM_REG_SMASK);
    822837        }
    823838        else
     
    908923
    909924            GCPtrDisp = fRipRelativeAddr ? pVCpu->cpum.GstCtx.rip + u64Disp : u64Disp;
    910             iReg2 = ((bRm >> X86_MODRM_REG_SHIFT) & X86_MODRM_REG_SMASK) | pVCpu->iem.s.uRexReg;
    911         }
    912 
     925        }
     926
     927        /*
     928         * The primary or secondary register operand is reported in iReg2 depending
     929         * on whether the primary operand is in read/write form.
     930         */
     931        uint8_t idxReg2;
     932        if (fPrimaryOpRead)
     933        {
     934            idxReg2 = bRm & X86_MODRM_RM_MASK;
     935            if (pVCpu->iem.s.enmEffAddrMode == IEMMODE_64BIT)
     936                idxReg2 |= pVCpu->iem.s.uRexB;
     937        }
     938        else
     939        {
     940            idxReg2 = (bRm >> X86_MODRM_REG_SHIFT) & X86_MODRM_REG_SMASK;
     941            if (pVCpu->iem.s.enmEffAddrMode == IEMMODE_64BIT)
     942                 idxReg2 |= pVCpu->iem.s.uRexReg;
     943        }
    913944        ExitInstrInfo.All.u2Scaling      = uScale;
    914         ExitInstrInfo.All.iReg1          = 0;   /* Not applicable for memory instructions. */
     945        ExitInstrInfo.All.iReg1          = 0;   /* Not applicable for memory addressing. */
    915946        ExitInstrInfo.All.u3AddrSize     = pVCpu->iem.s.enmEffAddrMode;
    916947        ExitInstrInfo.All.fIsRegOperand  = 0;
     
    921952        ExitInstrInfo.All.iBaseReg       = iBaseReg;
    922953        ExitInstrInfo.All.iIdxReg        = !fBaseRegValid;
    923         ExitInstrInfo.All.iReg2          = iReg2;
     954        ExitInstrInfo.All.iReg2          = idxReg2;
    924955    }
    925956
     
    932963        case VMX_EXIT_XDTR_ACCESS:
    933964        {
    934             Assert(VMX_INSTR_ID_IS_VALID(InstrId));
    935             ExitInstrInfo.GdtIdt.u2InstrId = VMX_INSTR_ID_GET_ID(InstrId);
     965            Assert(VMX_INSTR_ID_IS_VALID(uInstrId));
     966            ExitInstrInfo.GdtIdt.u2InstrId = VMX_INSTR_ID_GET_ID(uInstrId);
    936967            ExitInstrInfo.GdtIdt.u2Undef0  = 0;
    937968            break;
     
    940971        case VMX_EXIT_TR_ACCESS:
    941972        {
    942             Assert(VMX_INSTR_ID_IS_VALID(InstrId));
    943             ExitInstrInfo.LdtTr.u2InstrId = VMX_INSTR_ID_GET_ID(InstrId);
     973            Assert(VMX_INSTR_ID_IS_VALID(uInstrId));
     974            ExitInstrInfo.LdtTr.u2InstrId = VMX_INSTR_ID_GET_ID(uInstrId);
    944975            ExitInstrInfo.LdtTr.u2Undef0 = 0;
    945976            break;
     
    10331064    IEM_VMX_CLEAR_CURRENT_VMCS(pVCpu);
    10341065    return rc;
     1066}
     1067
     1068
     1069/**
     1070 * Implements VMSucceed for the VMREAD instruction and increments the guest RIP.
     1071 *
     1072 * @param   pVCpu           The cross context virtual CPU structure.
     1073 */
     1074DECLINLINE(void) iemVmxVmreadSuccess(PVMCPU pVCpu, uint8_t cbInstr)
     1075{
     1076    pVCpu->cpum.GstCtx.hwvirt.vmx.enmInstrDiag = kVmxVInstrDiag_Vmread_Success;
     1077    iemVmxVmSucceed(pVCpu);
     1078    iemRegAddToRipAndClearRF(pVCpu, cbInstr);
     1079}
     1080
     1081
     1082/**
     1083 * VMREAD common (memory/register) instruction execution worker
     1084 *
     1085 * @param   pVCpu           The cross context virtual CPU structure.
     1086 * @param   cbInstr         The instruction length.
     1087 * @param   pu64Dst         Where to write the VMCS value (only updated when
     1088 *                          VINF_SUCCESS is returned).
     1089 * @param   uFieldEnc       The VMCS field encoding.
     1090 * @param   pExitInfo       Pointer to the VM-exit information struct. Optional, can
     1091 *                          be NULL.
     1092 */
     1093IEM_STATIC VBOXSTRICTRC iemVmxVmreadCommon(PVMCPU pVCpu, uint8_t cbInstr, uint64_t *pu64Dst, uint32_t uFieldEnc,
     1094                                           PCVMXVEXITINFO pExitInfo)
     1095{
     1096    if (IEM_IS_VMX_NON_ROOT_MODE(pVCpu))
     1097    {
     1098        RT_NOREF(pExitInfo); RT_NOREF(cbInstr);
     1099        /** @todo NSTVMX: intercept. */
     1100        /** @todo NSTVMX: VMCS shadowing intercept (VMREAD bitmap). */
     1101    }
     1102
     1103    /* CPL. */
     1104    if (CPUMGetGuestCPL(pVCpu) > 0)
     1105    {
     1106        Log(("vmread: CPL %u -> #GP(0)\n", pVCpu->iem.s.uCpl));
     1107        pVCpu->cpum.GstCtx.hwvirt.vmx.enmInstrDiag = kVmxVInstrDiag_Vmread_Cpl;
     1108        return iemRaiseGeneralProtectionFault0(pVCpu);
     1109    }
     1110
     1111    /* VMCS pointer in root mode. */
     1112    if (    IEM_IS_VMX_ROOT_MODE(pVCpu)
     1113        && !IEM_VMX_HAS_CURRENT_VMCS(pVCpu))
     1114    {
     1115        Log(("vmread: VMCS pointer %#RGp invalid -> VMFailInvalid\n", IEM_VMX_GET_CURRENT_VMCS(pVCpu)));
     1116        pVCpu->cpum.GstCtx.hwvirt.vmx.enmInstrDiag = kVmxVInstrDiag_Vmread_PtrInvalid;
     1117        iemVmxVmFailInvalid(pVCpu);
     1118        iemRegAddToRipAndClearRF(pVCpu, cbInstr);
     1119        return VINF_SUCCESS;
     1120    }
     1121
     1122    /* VMCS-link pointer in non-root mode. */
     1123    if (    IEM_IS_VMX_NON_ROOT_MODE(pVCpu)
     1124        && !IEM_VMX_HAS_SHADOW_VMCS(pVCpu))
     1125    {
     1126        Log(("vmread: VMCS-link pointer %#RGp invalid -> VMFailInvalid\n", IEM_VMX_GET_SHADOW_VMCS(pVCpu)));
     1127        pVCpu->cpum.GstCtx.hwvirt.vmx.enmInstrDiag = kVmxVInstrDiag_Vmread_LinkPtrInvalid;
     1128        iemVmxVmFailInvalid(pVCpu);
     1129        iemRegAddToRipAndClearRF(pVCpu, cbInstr);
     1130        return VINF_SUCCESS;
     1131    }
     1132
     1133    /* Supported VMCS field. */
     1134    if (!iemVmxIsVmcsFieldValid(pVCpu, uFieldEnc))
     1135    {
     1136        Log(("vmread: VMCS field %#x invalid -> VMFail\n", uFieldEnc));
     1137        pVCpu->cpum.GstCtx.hwvirt.vmx.enmInstrDiag = kVmxVInstrDiag_Vmread_FieldInvalid;
     1138        iemVmxVmFail(pVCpu, VMXINSTRERR_VMREAD_INVALID_COMPONENT);
     1139        iemRegAddToRipAndClearRF(pVCpu, cbInstr);
     1140        return VINF_SUCCESS;
     1141    }
     1142
     1143    /*
     1144     * Setup reading from the current or shadow VMCS.
     1145     */
     1146    uint8_t *pbVmcs;
     1147    if (IEM_IS_VMX_NON_ROOT_MODE(pVCpu))
     1148        pbVmcs = (uint8_t *)pVCpu->cpum.GstCtx.hwvirt.vmx.CTX_SUFF(pShadowVmcs);
     1149    else
     1150        pbVmcs = (uint8_t *)pVCpu->cpum.GstCtx.hwvirt.vmx.CTX_SUFF(pVmcs);
     1151    Assert(pbVmcs);
     1152
     1153    PCVMXVMCSFIELDENC pFieldEnc = (PCVMXVMCSFIELDENC)&uFieldEnc;
     1154    uint8_t  const uWidth     = pFieldEnc->n.u2Width;
     1155    uint8_t  const uType      = pFieldEnc->n.u2Type;
     1156    uint8_t  const uWidthType = (uWidth << 2) | uType;
     1157    uint8_t  const uIndex     = pFieldEnc->n.u8Index;
     1158    AssertRCReturn(uIndex <= VMX_V_VMCS_MAX_INDEX, VERR_IEM_IPE_2);
     1159    uint16_t const offField  = g_aoffVmcsMap[uWidthType][uIndex];
     1160
     1161    /*
     1162     * Read the VMCS component based on the field's effective width.
     1163     *
     1164     * The effective width is 64-bit fields adjusted to 32-bits if the access-type
     1165     * indicates high bits (little endian).
     1166     *
     1167     * Note! The caller is responsible to trim the result and update registers
     1168     * or memory locations are required. Here we just zero-extend to the largest
     1169     * type (i.e. 64-bits).
     1170     */
     1171    uint8_t      *pbField = pbVmcs + offField;
     1172    uint8_t const uEffWidth = HMVmxGetVmcsFieldWidthEff(uFieldEnc);
     1173    switch (uEffWidth)
     1174    {
     1175        case VMX_VMCS_ENC_WIDTH_64BIT:
     1176        case VMX_VMCS_ENC_WIDTH_NATURAL: *pu64Dst = *(uint64_t *)pbField; break;
     1177        case VMX_VMCS_ENC_WIDTH_32BIT:   *pu64Dst = *(uint32_t *)pbField; break;
     1178        case VMX_VMCS_ENC_WIDTH_16BIT:   *pu64Dst = *(uint16_t *)pbField; break;
     1179    }
     1180    return VINF_SUCCESS;
     1181}
     1182
     1183
     1184/**
     1185 * VMREAD (64-bit register) instruction execution worker.
     1186 *
     1187 * @param   pVCpu           The cross context virtual CPU structure.
     1188 * @param   cbInstr         The instruction length.
     1189 * @param   pu64Dst         Where to store the VMCS field's value.
     1190 * @param   uFieldEnc       The VMCS field encoding.
     1191 * @param   pExitInfo       Pointer to the VM-exit information struct. Optional, can
     1192 *                          be NULL.
     1193 */
     1194IEM_STATIC VBOXSTRICTRC iemVmxVmreadReg64(PVMCPU pVCpu, uint8_t cbInstr, uint64_t *pu64Dst, uint32_t uFieldEnc,
     1195                                          PCVMXVEXITINFO pExitInfo)
     1196{
     1197    VBOXSTRICTRC rcStrict = iemVmxVmreadCommon(pVCpu, cbInstr, pu64Dst, uFieldEnc, pExitInfo);
     1198    if (rcStrict == VINF_SUCCESS)
     1199    {
     1200        iemVmxVmreadSuccess(pVCpu, cbInstr);
     1201        return VINF_SUCCESS;
     1202    }
     1203
     1204    Log(("vmread/reg: iemVmxVmreadCommon failed rc=%Rrc\n", VBOXSTRICTRC_VAL(rcStrict)));
     1205    return rcStrict;
     1206}
     1207
     1208
     1209/**
     1210 * VMREAD (32-bit register) instruction execution worker.
     1211 *
     1212 * @param   pVCpu           The cross context virtual CPU structure.
     1213 * @param   cbInstr         The instruction length.
     1214 * @param   pu32Dst         Where to store the VMCS field's value.
     1215 * @param   uFieldEnc       The VMCS field encoding.
     1216 * @param   pExitInfo       Pointer to the VM-exit information struct. Optional, can
     1217 *                          be NULL.
     1218 */
     1219IEM_STATIC VBOXSTRICTRC iemVmxVmreadReg32(PVMCPU pVCpu, uint8_t cbInstr, uint32_t *pu32Dst, uint32_t uFieldEnc,
     1220                                          PCVMXVEXITINFO pExitInfo)
     1221{
     1222    uint64_t u64Dst;
     1223    VBOXSTRICTRC rcStrict = iemVmxVmreadCommon(pVCpu, cbInstr, &u64Dst, uFieldEnc, pExitInfo);
     1224    if (rcStrict == VINF_SUCCESS)
     1225    {
     1226        *pu32Dst = u64Dst;
     1227        iemVmxVmreadSuccess(pVCpu, cbInstr);
     1228        return VINF_SUCCESS;
     1229    }
     1230
     1231    Log(("vmread/reg: iemVmxVmreadCommon failed rc=%Rrc\n", VBOXSTRICTRC_VAL(rcStrict)));
     1232    return rcStrict;
     1233}
     1234
     1235
     1236/**
     1237 * VMREAD (memory) instruction execution worker.
     1238 *
     1239 * @param   pVCpu           The cross context virtual CPU structure.
     1240 * @param   cbInstr         The instruction length.
     1241 * @param   iEffSeg         The effective segment register to use with @a u64Val.
     1242 *                          Pass UINT8_MAX if it is a register access.
     1243 * @param   enmEffAddrMode  The effective addressing mode (only used with memory
     1244 *                          operand).
     1245 * @param   GCPtrDst        The guest linear address to store the VMCS field's
     1246 *                          value.
     1247 * @param   uFieldEnc       The VMCS field encoding.
     1248 * @param   pExitInfo       Pointer to the VM-exit information struct. Optional, can
     1249 *                          be NULL.
     1250 */
     1251IEM_STATIC VBOXSTRICTRC iemVmxVmreadMem(PVMCPU pVCpu, uint8_t cbInstr, uint8_t iEffSeg, IEMMODE enmEffAddrMode,
     1252                                        RTGCPTR GCPtrDst, uint32_t uFieldEnc, PCVMXVEXITINFO pExitInfo)
     1253{
     1254    uint64_t u64Dst;
     1255    VBOXSTRICTRC rcStrict = iemVmxVmreadCommon(pVCpu, cbInstr, &u64Dst, uFieldEnc, pExitInfo);
     1256    if (rcStrict == VINF_SUCCESS)
     1257    {
     1258        /*
     1259         * Write the VMCS field's value to the location specified in guest-memory.
     1260         *
     1261         * The pointer size depends on the address size (address-size prefix allowed).
     1262         * The operand size depends on IA-32e mode (operand-size prefix not allowed).
     1263         */
     1264        static uint64_t const s_auAddrSizeMasks[] = { UINT64_C(0xffff), UINT64_C(0xffffffff), UINT64_C(0xffffffffffffffff) };
     1265        Assert(enmEffAddrMode < RT_ELEMENTS(s_auAddrSizeMasks));
     1266        GCPtrDst &= s_auAddrSizeMasks[enmEffAddrMode];
     1267
     1268        if (pVCpu->iem.s.enmCpuMode == IEMMODE_64BIT)
     1269            rcStrict = iemMemStoreDataU64(pVCpu, iEffSeg, GCPtrDst, u64Dst);
     1270        else
     1271            rcStrict = iemMemStoreDataU32(pVCpu, iEffSeg, GCPtrDst, u64Dst);
     1272        if (rcStrict == VINF_SUCCESS)
     1273        {
     1274            iemVmxVmreadSuccess(pVCpu, cbInstr);
     1275            return VINF_SUCCESS;
     1276        }
     1277
     1278        Log(("vmread/mem: Failed to write to memory operand at %#RGv, rc=%Rrc\n", GCPtrDst, VBOXSTRICTRC_VAL(rcStrict)));
     1279        pVCpu->cpum.GstCtx.hwvirt.vmx.enmInstrDiag = kVmxVInstrDiag_Vmread_PtrMap;
     1280        return rcStrict;
     1281    }
     1282
     1283    Log(("vmread/mem: iemVmxVmreadCommon failed rc=%Rrc\n", VBOXSTRICTRC_VAL(rcStrict)));
     1284    return rcStrict;
    10351285}
    10361286
     
    10491299 *                          operand.
    10501300 * @param   uFieldEnc       The VMCS field encoding.
    1051  * @param   pExitInfo       Pointer to the VM-exit information struct.
     1301 * @param   pExitInfo       Pointer to the VM-exit information struct. Optional, can
     1302 *                          be NULL.
    10521303 */
    10531304IEM_STATIC VBOXSTRICTRC iemVmxVmwrite(PVMCPU pVCpu, uint8_t cbInstr, uint8_t iEffSeg, IEMMODE enmEffAddrMode, uint64_t u64Val,
     
    10581309        RT_NOREF(pExitInfo);
    10591310        /** @todo NSTVMX: intercept. */
    1060         /** @todo NSTVMX: VMCS shadowing intercept (VMREAD/VMWRITE bitmap). */
     1311        /** @todo NSTVMX: VMCS shadowing intercept (VMWRITE bitmap). */
    10611312    }
    10621313
     
    10851336    {
    10861337        Log(("vmwrite: VMCS-link pointer %#RGp invalid -> VMFailInvalid\n", IEM_VMX_GET_SHADOW_VMCS(pVCpu)));
    1087         pVCpu->cpum.GstCtx.hwvirt.vmx.enmInstrDiag = kVmxVInstrDiag_Vmwrite_PtrInvalid;
     1338        pVCpu->cpum.GstCtx.hwvirt.vmx.enmInstrDiag = kVmxVInstrDiag_Vmwrite_LinkPtrInvalid;
    10881339        iemVmxVmFailInvalid(pVCpu);
    10891340        iemRegAddToRipAndClearRF(pVCpu, cbInstr);
     
    10911342    }
    10921343
    1093     /* If the VMWRITE instruction references memory, access the specified in memory operand. */
     1344    /* If the VMWRITE instruction references memory, access the specified memory operand. */
    10941345    bool const fIsRegOperand = iEffSeg == UINT8_MAX;
    10951346    if (!fIsRegOperand)
     
    11161367        }
    11171368    }
     1369    else
     1370        Assert(!pExitInfo || pExitInfo->InstrInfo.VmreadVmwrite.fIsRegOperand);
    11181371
    11191372    /* Supported VMCS field. */
     
    11321385        && !IEM_GET_GUEST_CPU_FEATURES(pVCpu)->fVmxVmwriteAll)
    11331386    {
    1134         Log(("vmwrite: Write to read-only VMCS component -> VMFail\n", uFieldEnc));
     1387        Log(("vmwrite: Write to read-only VMCS component %#x -> VMFail\n", uFieldEnc));
    11351388        pVCpu->cpum.GstCtx.hwvirt.vmx.enmInstrDiag = kVmxVInstrDiag_Vmwrite_FieldRo;
    11361389        iemVmxVmFail(pVCpu, VMXINSTRERR_VMWRITE_RO_COMPONENT);
     
    13461599 * @param   cbInstr         The instruction length.
    13471600 * @param   GCPtrVmcs       The linear address of the current VMCS pointer.
    1348  * @param   pExitInfo       Pointer to the virtual VM-exit information struct.
    1349  *                          Optional, can be NULL.
     1601 * @param   pExitInfo       Pointer to the VM-exit information struct. Optional, can
     1602 *                          be NULL.
    13501603 *
    13511604 * @remarks Common VMX instruction checks are already expected to by the caller,
     
    17572010IEM_CIMPL_DEF_2(iemCImpl_vmwrite_reg, uint64_t, u64Val, uint32_t, uFieldEnc)
    17582011{
    1759     return iemVmxVmwrite(pVCpu, cbInstr, UINT8_MAX /*iEffSeg*/, IEMMODE_64BIT /* N/A */, u64Val, uFieldEnc, NULL /* pExitInfo */);
     2012    return iemVmxVmwrite(pVCpu, cbInstr, UINT8_MAX /* iEffSeg */, IEMMODE_64BIT /* N/A */, u64Val, uFieldEnc,
     2013                         NULL /* pExitInfo */);
    17602014}
    17612015
     
    17692023}
    17702024
     2025
     2026/**
     2027 * Implements 'VMREAD' 64-bit register.
     2028 */
     2029IEM_CIMPL_DEF_2(iemCImpl_vmread64_reg, uint64_t *, pu64Dst, uint32_t, uFieldEnc)
     2030{
     2031    return iemVmxVmreadReg64(pVCpu, cbInstr, pu64Dst, uFieldEnc, NULL /* pExitInfo */);
     2032}
     2033
     2034
     2035/**
     2036 * Implements 'VMREAD' 32-bit register.
     2037 */
     2038IEM_CIMPL_DEF_2(iemCImpl_vmread32_reg, uint32_t *, pu32Dst, uint32_t, uFieldEnc)
     2039{
     2040    return iemVmxVmreadReg32(pVCpu, cbInstr, pu32Dst, uFieldEnc, NULL /* pExitInfo */);
     2041}
     2042
     2043
     2044/**
     2045 * Implements 'VMREAD' memory.
     2046 */
     2047IEM_CIMPL_DEF_4(iemCImpl_vmread_mem, uint8_t, iEffSeg, IEMMODE, enmEffAddrMode, RTGCPTR, GCPtrDst, uint32_t, uFieldEnc)
     2048{
     2049    return iemVmxVmreadMem(pVCpu, cbInstr, iEffSeg, enmEffAddrMode, GCPtrDst, uFieldEnc, NULL /* pExitInfo */);
     2050}
     2051
     2052
    17712053#endif
    17722054
  • trunk/src/VBox/VMM/VMMAll/IEMAllInstructionsTwoByte0f.cpp.h

    r73959 r73983  
    42794279
    42804280/** Opcode      0x0f 0x78 - VMREAD Ey, Gy */
     4281#ifdef VBOX_WITH_NESTED_HWVIRT_VMX
     4282FNIEMOP_DEF(iemOp_vmread_Ey_Gy)
     4283{
     4284    IEMOP_MNEMONIC(vmread, "vmread Ey,Gy");
     4285    IEMOP_HLP_IN_VMX_OPERATION();
     4286    IEMOP_HLP_VMX_INSTR();
     4287    IEMMODE const enmEffOpSize = pVCpu->iem.s.enmCpuMode == IEMMODE_64BIT ? IEMMODE_64BIT : IEMMODE_32BIT;
     4288
     4289    uint8_t bRm; IEM_OPCODE_GET_NEXT_U8(&bRm);
     4290    if ((bRm & X86_MODRM_MOD_MASK) == (3 << X86_MODRM_MOD_SHIFT))
     4291    {
     4292        /*
     4293         * Register, register.
     4294         */
     4295        IEMOP_HLP_DONE_DECODING_NO_SIZE_OP_REPZ_OR_REPNZ_PREFIXES();
     4296        if (enmEffOpSize == IEMMODE_64BIT)
     4297        {
     4298            IEM_MC_BEGIN(2, 0);
     4299            IEM_MC_ARG(uint64_t *, pu64Dst, 0);
     4300            IEM_MC_ARG(uint64_t,   u64Enc,  1);
     4301            IEM_MC_FETCH_GREG_U64(u64Enc, (bRm & X86_MODRM_RM_MASK) | pVCpu->iem.s.uRexB);
     4302            IEM_MC_REF_GREG_U64(pu64Dst, ((bRm >> X86_MODRM_REG_SHIFT) & X86_MODRM_REG_SMASK) | pVCpu->iem.s.uRexReg);
     4303            IEM_MC_CALL_CIMPL_2(iemCImpl_vmread64_reg, pu64Dst, u64Enc);
     4304            IEM_MC_END();
     4305        }
     4306        else
     4307        {
     4308            IEM_MC_BEGIN(2, 0);
     4309            IEM_MC_ARG(uint32_t *, pu32Dst, 0);
     4310            IEM_MC_ARG(uint32_t,   u32Enc,  1);
     4311            IEM_MC_FETCH_GREG_U32(u32Enc, (bRm & X86_MODRM_RM_MASK) | pVCpu->iem.s.uRexB);
     4312            IEM_MC_REF_GREG_U32(pu32Dst, ((bRm >> X86_MODRM_REG_SHIFT) & X86_MODRM_REG_SMASK) | pVCpu->iem.s.uRexReg);
     4313            IEM_MC_CALL_CIMPL_2(iemCImpl_vmread32_reg, pu32Dst, u32Enc);
     4314            IEM_MC_END();
     4315        }
     4316    }
     4317    else
     4318    {
     4319        /*
     4320         * Register, memory.
     4321         */
     4322        if (enmEffOpSize == IEMMODE_64BIT)
     4323        {
     4324            IEM_MC_BEGIN(4, 0);
     4325            IEM_MC_ARG(uint8_t,       iEffSeg,                                          0);
     4326            IEM_MC_ARG_CONST(IEMMODE, enmEffAddrMode,/*=*/pVCpu->iem.s.enmEffAddrMode,  1);
     4327            IEM_MC_ARG(RTGCPTR,       GCPtrVal,                                         2);
     4328            IEM_MC_ARG(uint64_t,      u64Enc,                                           3);
     4329            IEM_MC_FETCH_GREG_U64(u64Enc, (bRm & X86_MODRM_RM_MASK) | pVCpu->iem.s.uRexB);
     4330            IEM_MC_CALC_RM_EFF_ADDR(GCPtrVal, bRm, 0);
     4331            IEMOP_HLP_DONE_DECODING_NO_SIZE_OP_REPZ_OR_REPNZ_PREFIXES();
     4332            IEM_MC_ASSIGN(iEffSeg, pVCpu->iem.s.iEffSeg);
     4333            IEM_MC_CALL_CIMPL_4(iemCImpl_vmread_mem, iEffSeg, enmEffAddrMode, GCPtrVal, u64Enc);
     4334            IEM_MC_END();
     4335        }
     4336        else
     4337        {
     4338            IEM_MC_BEGIN(4, 0);
     4339            IEM_MC_ARG(uint8_t,       iEffSeg,                                          0);
     4340            IEM_MC_ARG_CONST(IEMMODE, enmEffAddrMode,/*=*/pVCpu->iem.s.enmEffAddrMode,  1);
     4341            IEM_MC_ARG(RTGCPTR,       GCPtrVal,                                         2);
     4342            IEM_MC_ARG(uint32_t,      u32Enc,                                           3);
     4343            IEM_MC_FETCH_GREG_U32(u32Enc, (bRm & X86_MODRM_RM_MASK) | pVCpu->iem.s.uRexB);
     4344            IEM_MC_CALC_RM_EFF_ADDR(GCPtrVal, bRm, 0);
     4345            IEMOP_HLP_DONE_DECODING_NO_SIZE_OP_REPZ_OR_REPNZ_PREFIXES();
     4346            IEM_MC_ASSIGN(iEffSeg, pVCpu->iem.s.iEffSeg);
     4347            IEM_MC_CALL_CIMPL_4(iemCImpl_vmread_mem, iEffSeg, enmEffAddrMode, GCPtrVal, u32Enc);
     4348            IEM_MC_END();
     4349        }
     4350    }
     4351    return VINF_SUCCESS;
     4352}
     4353#else
    42814354FNIEMOP_STUB(iemOp_vmread_Ey_Gy);
     4355#endif
     4356
    42824357/*  Opcode 0x66 0x0f 0x78 - AMD Group 17 */
    42834358FNIEMOP_STUB(iemOp_AmdGrp17);
  • trunk/src/VBox/VMM/VMMR0/HMVMXR0.cpp

    r73959 r73983  
    183183    hmR0VmxExportGuestSegmentReg(pVCpu, VMX_VMCS16_GUEST_##Sel##_SEL, VMX_VMCS32_GUEST_##Sel##_LIMIT, \
    184184                                 VMX_VMCS_GUEST_##Sel##_BASE, VMX_VMCS32_GUEST_##Sel##_ACCESS_RIGHTS, (a_pCtxSelReg))
     185
     186#ifdef VBOX_WITH_NESTED_HWVIRT_VMX
     187/** Macro that does the necessary privilege checks and intercepted VM-exits for
     188 *  guests that attempted to execute a VMX instruction. */
     189#define HMVMX_CHECK_EXIT_DUE_TO_VMX_INSTR(a_pVCpu, a_uExitReason) \
     190    do \
     191    { \
     192        VBOXSTRICTRC rcStrictTmp = hmR0VmxCheckExitDueToVmxInstr((a_pVCpu), (a_uExitReason)); \
     193        if (rcStrictTmp == VINF_SUCCESS) \
     194        { /* likely */ } \
     195        else if (rcStrictTmp == VINF_HM_PENDING_XCPT) \
     196        { \
     197            Assert((a_pVCpu)->hm.s.Event.fPending); \
     198            Log4Func(("Privilege checks failed -> %#x\n", VMX_ENTRY_INT_INFO_VECTOR((a_pVCpu)->hm.s.Event.u64IntInfo))); \
     199            return VINF_SUCCESS; \
     200        } \
     201        else \
     202        { \
     203            int rcTmp = VBOXSTRICTRC_VAL(rcStrictTmp); \
     204            AssertMsgFailedReturn(("Unexpected failure. rc=%Rrc", rcTmp), rcTmp); \
     205        } \
     206    } while (0)
     207#endif  /* VBOX_WITH_NESTED_HWVIRT_VMX */
    185208
    186209
     
    58465869
    58475870/**
    5848  * Decodes the memory operand of a VM-exit due to instruction execution.
     5871 * Decodes the memory operand of an instruction that caused a VM-exit.
    58495872 *
    58505873 * The VM-exit qualification field provides the displacement field for memory
     
    58575880 * @param   pVCpu           The cross context virtual CPU structure.
    58585881 * @param   pExitInstrInfo  Pointer to the VM-exit instruction information.
    5859  * @param   fIsWrite        Whether the operand is a destination memory operand
    5860  *                          (i.e. writeable memory location) or not.
     5882 * @param   fIsDstOperand   Whether the operand is a destination memory
     5883 *                          operand (i.e. writeable memory location) or not.
    58615884 * @param   GCPtrDisp       The instruction displacement field, if any. For
    58625885 *                          RIP-relative addressing pass RIP + displacement here.
    5863  * @param   pGCPtrMem       Where to store the destination memory operand.
    5864  */
    5865 static VBOXSTRICTRC hmR0VmxDecodeMemOperand(PVMCPU pVCpu, PCVMXEXITINSTRINFO pExitInstrInfo, RTGCPTR GCPtrDisp, bool fIsWrite,
    5866                                             PRTGCPTR pGCPtrMem)
     5886 * @param   pGCPtrMem       Where to store the effective destination memory address.
     5887 */
     5888static VBOXSTRICTRC hmR0VmxDecodeMemOperand(PVMCPU pVCpu, PCVMXEXITINSTRINFO pExitInstrInfo, RTGCPTR GCPtrDisp,
     5889                                            bool fIsDstOperand, PRTGCPTR pGCPtrMem)
    58675890{
    58685891    Assert(pExitInstrInfo);
     
    59515974        {
    59525975            /* Check permissions for the data segment. */
    5953             if (   fIsWrite
     5976            if (   fIsDstOperand
    59545977                && !(pSel->Attr.n.u4Type & X86_SEL_TYPE_WRITE))
    59555978            {
     
    59966019        {
    59976020            /* Check permissions for the code segment. */
    5998             if (   fIsWrite
     6021            if (   fIsDstOperand
    59996022                || !(pSel->Attr.n.u4Type & X86_SEL_TYPE_READ))
    60006023            {
     
    60406063 *
    60416064 * @param   pVCpu           The cross context virtual CPU structure.
    6042  * @param   pVmxTransient   Pointer to the VMX transient structure.
     6065 * @param   uExitReason     The VM-exit reason.
    60436066 *
    60446067 * @todo    NstVmx: Document other error codes when VM-exit is implemented.
    60456068 * @remarks No-long-jump zone!!!
    60466069 */
    6047 static VBOXSTRICTRC hmR0VmxCheckExitDueToVmxInstr(PVMCPU pVCpu, PVMXTRANSIENT pVmxTransient)
     6070static VBOXSTRICTRC hmR0VmxCheckExitDueToVmxInstr(PVMCPU pVCpu, uint32_t uExitReason)
    60486071{
    60496072    HMVMX_CPUMCTX_ASSERT(pVCpu, CPUMCTX_EXTRN_CR4 | CPUMCTX_EXTRN_CR0 | CPUMCTX_EXTRN_RFLAGS | CPUMCTX_EXTRN_SS
     
    60596082    }
    60606083
    6061     if (pVmxTransient->uExitReason == VMX_EXIT_VMXON)
     6084    if (uExitReason == VMX_EXIT_VMXON)
    60626085    {
    60636086        /*
     
    1342213445    HMVMX_VALIDATE_EXIT_HANDLER_PARAMS(pVCpu, pVmxTransient);
    1342313446
    13424     /** @todo NSTVMX: Vmwrite. */
    13425     hmR0VmxSetPendingXcptUD(pVCpu);
    13426     return VINF_SUCCESS;
    13427 }
    13428 
    13429 
    13430 /**
    13431  * VM-exit handler for VMXOFF (VMX_EXIT_VMXOFF). Unconditional VM-exit.
    13432  */
    13433 HMVMX_EXIT_DECL hmR0VmxExitVmxoff(PVMCPU pVCpu, PVMXTRANSIENT pVmxTransient)
    13434 {
    13435     HMVMX_VALIDATE_EXIT_HANDLER_PARAMS(pVCpu, pVmxTransient);
    13436 
    13437     int rc = hmR0VmxReadExitInstrLenVmcs(pVmxTransient);
    13438     rc    |= hmR0VmxImportGuestState(pVCpu, IEM_CPUMCTX_EXTRN_EXEC_DECODED_MEM_MASK);
    13439     AssertRCReturn(rc, rc);
    13440 
    13441     VBOXSTRICTRC rcStrict = IEMExecDecodedVmxoff(pVCpu, pVmxTransient->cbInstr);
    13442     if (RT_LIKELY(rcStrict == VINF_SUCCESS))
    13443     {
    13444         /* VMXOFF on success changes the internal hwvirt state but not anything that's visible to the guest. */
    13445         ASMAtomicUoOrU64(&pVCpu->hm.s.fCtxChanged, HM_CHANGED_GUEST_HWVIRT);
    13446     }
    13447     else if (rcStrict == VINF_IEM_RAISED_XCPT)
    13448     {
    13449         ASMAtomicUoOrU64(&pVCpu->hm.s.fCtxChanged, HM_CHANGED_RAISED_XCPT_MASK);
    13450         rcStrict = VINF_SUCCESS;
    13451     }
    13452     return rcStrict;
    13453 }
    13454 
    13455 
    13456 /**
    13457  * VM-exit handler for VMXON (VMX_EXIT_VMXON). Unconditional VM-exit.
    13458  */
    13459 HMVMX_EXIT_DECL hmR0VmxExitVmxon(PVMCPU pVCpu, PVMXTRANSIENT pVmxTransient)
    13460 {
    13461     HMVMX_VALIDATE_EXIT_HANDLER_PARAMS(pVCpu, pVmxTransient);
    13462 
    1346313447    int rc = hmR0VmxReadExitInstrLenVmcs(pVmxTransient);
    1346413448    rc    |= hmR0VmxImportGuestState(pVCpu, CPUMCTX_EXTRN_SREG_MASK | IEM_CPUMCTX_EXTRN_EXEC_DECODED_MEM_MASK);
     
    1346713451    AssertRCReturn(rc, rc);
    1346813452
    13469     VBOXSTRICTRC rcStrict = hmR0VmxCheckExitDueToVmxInstr(pVCpu, pVmxTransient);
     13453    HMVMX_CHECK_EXIT_DUE_TO_VMX_INSTR(pVCpu, pVmxTransient->uExitReason);
     13454
     13455    VMXVEXITINFO ExitInfo;
     13456    RT_ZERO(ExitInfo);
     13457    ExitInfo.uReason     = pVmxTransient->uExitReason;
     13458    ExitInfo.u64Qual     = pVmxTransient->uExitQual;
     13459    ExitInfo.InstrInfo.u = pVmxTransient->ExitInstrInfo.u;
     13460    ExitInfo.cbInstr     = pVmxTransient->cbInstr;
     13461    if (!ExitInfo.InstrInfo.VmreadVmwrite.fIsRegOperand)
     13462    {
     13463        RTGCPTR GCPtrVal;
     13464        VBOXSTRICTRC rcStrict = hmR0VmxDecodeMemOperand(pVCpu, &ExitInfo.InstrInfo, ExitInfo.u64Qual, false /* fIsDstOperand */,
     13465                                                         &GCPtrVal);
     13466        if (rcStrict == VINF_SUCCESS)
     13467        { /* likely */ }
     13468        else if (rcStrict == VINF_HM_PENDING_XCPT)
     13469        {
     13470            Assert(pVCpu->hm.s.Event.fPending);
     13471            Log4Func(("Memory operand decoding failed, raising xcpt %#x\n",
     13472                      VMX_ENTRY_INT_INFO_VECTOR(pVCpu->hm.s.Event.u64IntInfo)));
     13473            return VINF_SUCCESS;
     13474        }
     13475        else
     13476        {
     13477            Log4Func(("hmR0VmxCheckExitDueToVmxInstr failed. rc=%Rrc\n", VBOXSTRICTRC_VAL(rcStrict)));
     13478            return rcStrict;
     13479        }
     13480        ExitInfo.GCPtrEffAddr = GCPtrVal;
     13481    }
     13482
     13483    VBOXSTRICTRC rcStrict = IEMExecDecodedVmwrite(pVCpu, &ExitInfo);
     13484    if (RT_LIKELY(rcStrict == VINF_SUCCESS))
     13485        ASMAtomicUoOrU64(&pVCpu->hm.s.fCtxChanged, HM_CHANGED_GUEST_RIP | HM_CHANGED_GUEST_RFLAGS | HM_CHANGED_GUEST_HWVIRT);
     13486    else if (rcStrict == VINF_IEM_RAISED_XCPT)
     13487    {
     13488        ASMAtomicUoOrU64(&pVCpu->hm.s.fCtxChanged, HM_CHANGED_RAISED_XCPT_MASK);
     13489        rcStrict = VINF_SUCCESS;
     13490    }
     13491    return rcStrict;
     13492}
     13493
     13494
     13495/**
     13496 * VM-exit handler for VMXOFF (VMX_EXIT_VMXOFF). Unconditional VM-exit.
     13497 */
     13498HMVMX_EXIT_DECL hmR0VmxExitVmxoff(PVMCPU pVCpu, PVMXTRANSIENT pVmxTransient)
     13499{
     13500    HMVMX_VALIDATE_EXIT_HANDLER_PARAMS(pVCpu, pVmxTransient);
     13501
     13502    int rc = hmR0VmxReadExitInstrLenVmcs(pVmxTransient);
     13503    rc    |= hmR0VmxImportGuestState(pVCpu, IEM_CPUMCTX_EXTRN_EXEC_DECODED_MEM_MASK);
     13504    AssertRCReturn(rc, rc);
     13505
     13506    HMVMX_CHECK_EXIT_DUE_TO_VMX_INSTR(pVCpu, pVmxTransient->uExitReason);
     13507
     13508    VBOXSTRICTRC rcStrict = IEMExecDecodedVmxoff(pVCpu, pVmxTransient->cbInstr);
     13509    if (RT_LIKELY(rcStrict == VINF_SUCCESS))
     13510    {
     13511        /* VMXOFF on success changes the internal hwvirt state but not anything that's visible to the guest. */
     13512        ASMAtomicUoOrU64(&pVCpu->hm.s.fCtxChanged, HM_CHANGED_GUEST_HWVIRT);
     13513    }
     13514    else if (rcStrict == VINF_IEM_RAISED_XCPT)
     13515    {
     13516        ASMAtomicUoOrU64(&pVCpu->hm.s.fCtxChanged, HM_CHANGED_RAISED_XCPT_MASK);
     13517        rcStrict = VINF_SUCCESS;
     13518    }
     13519    return rcStrict;
     13520}
     13521
     13522
     13523/**
     13524 * VM-exit handler for VMXON (VMX_EXIT_VMXON). Unconditional VM-exit.
     13525 */
     13526HMVMX_EXIT_DECL hmR0VmxExitVmxon(PVMCPU pVCpu, PVMXTRANSIENT pVmxTransient)
     13527{
     13528    HMVMX_VALIDATE_EXIT_HANDLER_PARAMS(pVCpu, pVmxTransient);
     13529
     13530    int rc = hmR0VmxReadExitInstrLenVmcs(pVmxTransient);
     13531    rc    |= hmR0VmxImportGuestState(pVCpu, CPUMCTX_EXTRN_SREG_MASK | IEM_CPUMCTX_EXTRN_EXEC_DECODED_MEM_MASK);
     13532    rc    |= hmR0VmxReadExitInstrInfoVmcs(pVmxTransient);
     13533    rc    |= hmR0VmxReadExitQualVmcs(pVCpu, pVmxTransient);
     13534    AssertRCReturn(rc, rc);
     13535
     13536    HMVMX_CHECK_EXIT_DUE_TO_VMX_INSTR(pVCpu, pVmxTransient->uExitReason);
     13537
     13538    VMXVEXITINFO ExitInfo;
     13539    RT_ZERO(ExitInfo);
     13540    ExitInfo.uReason     = pVmxTransient->uExitReason;
     13541    ExitInfo.u64Qual     = pVmxTransient->uExitQual;
     13542    ExitInfo.InstrInfo.u = pVmxTransient->ExitInstrInfo.u;
     13543    ExitInfo.cbInstr     = pVmxTransient->cbInstr;
     13544
     13545    RTGCPTR GCPtrVmxon;
     13546    VBOXSTRICTRC rcStrict = hmR0VmxDecodeMemOperand(pVCpu, &ExitInfo.InstrInfo, ExitInfo.u64Qual, false /* fIsDstOperand */,
     13547                                                    &GCPtrVmxon);
    1347013548    if (rcStrict == VINF_SUCCESS)
    1347113549    { /* likely */ }
    1347213550    else if (rcStrict == VINF_HM_PENDING_XCPT)
    1347313551    {
    13474         Log4Func(("Privilege checks failed, raising xcpt %#x!\n", VMX_ENTRY_INT_INFO_VECTOR(pVCpu->hm.s.Event.u64IntInfo)));
     13552        Log4Func(("Memory operand decoding failed, raising xcpt %#x\n", VMX_ENTRY_INT_INFO_VECTOR(pVCpu->hm.s.Event.u64IntInfo)));
    1347513553        return VINF_SUCCESS;
    1347613554    }
     
    1348013558        return rcStrict;
    1348113559    }
    13482 
    13483     RTGCPTR            GCPtrVmxon;
    13484     PCVMXEXITINSTRINFO pExitInstrInfo = &pVmxTransient->ExitInstrInfo;
    13485     RTGCPTR const      GCPtrDisp      =  pVmxTransient->uExitQual;
    13486     rcStrict = hmR0VmxDecodeMemOperand(pVCpu, pExitInstrInfo, GCPtrDisp, false /*fIsWrite*/,  &GCPtrVmxon);
    13487     if (rcStrict == VINF_SUCCESS)
    13488     { /* likely */ }
    13489     else if (rcStrict == VINF_HM_PENDING_XCPT)
    13490     {
    13491         Log4Func(("Memory operand decoding failed, raising xcpt %#x\n", VMX_ENTRY_INT_INFO_VECTOR(pVCpu->hm.s.Event.u64IntInfo)));
    13492         return VINF_SUCCESS;
    13493     }
    13494     else
    13495     {
    13496         Log4Func(("hmR0VmxCheckExitDueToVmxInstr failed. rc=%Rrc\n", VBOXSTRICTRC_VAL(rcStrict)));
    13497         return rcStrict;
    13498     }
    13499 
    13500     VMXVEXITINFO ExitInfo;
    13501     RT_ZERO(ExitInfo);
    13502     ExitInfo.ExitInstrInfo.u = pExitInstrInfo->u;
    13503     ExitInfo.u64ExitQual     = GCPtrDisp;
    13504     uint8_t const iEffSeg    = pExitInstrInfo->VmreadVmwrite.iSegReg;
    13505     rcStrict = IEMExecDecodedVmxon(pVCpu, pVmxTransient->cbInstr, iEffSeg, GCPtrVmxon, &ExitInfo);
     13560    ExitInfo.GCPtrEffAddr = GCPtrVmxon;
     13561
     13562    rcStrict = IEMExecDecodedVmxon(pVCpu, &ExitInfo);
    1350613563    if (RT_LIKELY(rcStrict == VINF_SUCCESS))
    1350713564        ASMAtomicUoOrU64(&pVCpu->hm.s.fCtxChanged, HM_CHANGED_GUEST_RIP | HM_CHANGED_GUEST_RFLAGS | HM_CHANGED_GUEST_HWVIRT);
Note: See TracChangeset for help on using the changeset viewer.

© 2024 Oracle Support Privacy / Do Not Sell My Info Terms of Use Trademark Policy Automated Access Etiquette