VirtualBox

Changeset 73710 in vbox for trunk


Ignore:
Timestamp:
Aug 16, 2018 10:23:05 AM (6 years ago)
Author:
vboxsync
Message:

VMM/IEM: Nested VMX: bugref:9180 Implement construction of VM-exit instruction info. for VM-exits.

File:
1 edited

Legend:

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

    r73628 r73710  
    3131
    3232/**
     33 * Gets the ModR/M and displacement byte(s) from decoded opcodes given their
     34 * relative offsets.
     35 */
     36# ifdef IEM_WITH_CODE_TLB
     37#  define IEM_MODRM_GET_U8(a_pVCpu, a_bModRm, a_offModRm)         do { } while (0)
     38#  define IEM_SIB_GET_U8(a_pVCpu, a_bSib, a_offSib)               do { } while (0)
     39#  define IEM_DISP_GET_U16(a_pVCpu, a_u16Disp, a_offDisp)         do { } while (0)
     40#  define IEM_DISP_GET_S8_SX_U16(a_pVCpu, a_u16Disp, a_offDisp)   do { } while (0)
     41#  define IEM_DISP_GET_U32(a_pVCpu, a_u32Disp, a_offDisp)         do { } while (0)
     42#  define IEM_DISP_GET_S8_SX_U32(a_pVCpu, a_u32Disp, a_offDisp)   do { } while (0)
     43#  define IEM_DISP_GET_S32_SX_U64(a_pVCpu, a_u64Disp, a_offDisp)  do { } while (0)
     44#  define IEM_DISP_GET_S8_SX_U64(a_pVCpu, a_u64Disp, a_offDisp)   do { } while (0)
     45#  error "Implement me: Getting ModR/M, disp. has to work even when the instruction crosses a page boundary."
     46# else  /* !IEM_WITH_CODE_TLB */
     47#  define IEM_MODRM_GET_U8(a_pVCpu, a_bModRm, a_offModRm) \
     48    do \
     49    { \
     50        Assert((a_offModRm) < (a_pVCpu)->iem.s.cbOpcode); \
     51        (a_bModRm) = (a_pVCpu)->iem.s.abOpcode[(a_offModRm)]; \
     52    } while (0)
     53
     54#  define IEM_SIB_GET_U8(a_pVCpu, a_bSib, a_offSib)      IEM_MODRM_GET_U8(a_pVCpu, a_bSib, a_offSib)
     55
     56#  define IEM_DISP_GET_U16(a_pVCpu, a_u16Disp, a_offDisp) \
     57    do \
     58    { \
     59        Assert((a_offDisp) + 1 < (a_pVCpu)->iem.s.cbOpcode); \
     60        uint8_t const bTmpLo = (a_pVCpu)->iem.s.abOpcode[(a_offDisp)]; \
     61        uint8_t const bTmpHi = (a_pVCpu)->iem.s.abOpcode[(a_offDisp) + 1]; \
     62        (a_u16Disp) = RT_MAKE_U16(bTmpLo, bTmpHi); \
     63    } while (0)
     64
     65#  define IEM_DISP_GET_S8_SX_U16(a_pVCpu, a_u16Disp, a_offDisp) \
     66    do \
     67    { \
     68        Assert((a_offDisp) < (a_pVCpu)->iem.s.cbOpcode); \
     69        (a_u16Disp) = (int8_t)((a_pVCpu)->iem.s.abOpcode[(a_offDisp)]); \
     70    } while (0)
     71
     72#  define IEM_DISP_GET_U32(a_pVCpu, a_u32Disp, a_offDisp) \
     73    do \
     74    { \
     75        Assert((a_offDisp) + 3 < (a_pVCpu)->iem.s.cbOpcode); \
     76        uint8_t const bTmp0 = (a_pVCpu)->iem.s.abOpcode[(a_offDisp)]; \
     77        uint8_t const bTmp1 = (a_pVCpu)->iem.s.abOpcode[(a_offDisp) + 1]; \
     78        uint8_t const bTmp2 = (a_pVCpu)->iem.s.abOpcode[(a_offDisp) + 2]; \
     79        uint8_t const bTmp3 = (a_pVCpu)->iem.s.abOpcode[(a_offDisp) + 3]; \
     80        (a_u32Disp) = RT_MAKE_U32_FROM_U8(bTmp0, bTmp1, bTmp2, bTmp3); \
     81    } while (0)
     82
     83#  define IEM_DISP_GET_S8_SX_U32(a_pVCpu, a_u32Disp, a_offDisp) \
     84    do \
     85    { \
     86        Assert((a_offDisp) + 1 < (a_pVCpu)->iem.s.cbOpcode); \
     87        (a_u32Disp) = (int8_t)((a_pVCpu)->iem.s.abOpcode[(a_offDisp)]); \
     88    } while (0)
     89
     90#  define IEM_DISP_GET_S8_SX_U64(a_pVCpu, a_u64Disp, a_offDisp) \
     91    do \
     92    { \
     93        Assert((a_offDisp) + 1 < (a_pVCpu)->iem.s.cbOpcode); \
     94        (a_u64Disp) = (int8_t)((a_pVCpu)->iem.s.abOpcode[(a_offDisp)]); \
     95    } while (0)
     96
     97#  define IEM_DISP_GET_S32_SX_U64(a_pVCpu, a_u64Disp, a_offDisp) \
     98    do \
     99    { \
     100        Assert((a_offDisp) + 3 < (a_pVCpu)->iem.s.cbOpcode); \
     101        uint8_t const bTmp0 = (a_pVCpu)->iem.s.abOpcode[(a_offDisp)]; \
     102        uint8_t const bTmp1 = (a_pVCpu)->iem.s.abOpcode[(a_offDisp) + 1]; \
     103        uint8_t const bTmp2 = (a_pVCpu)->iem.s.abOpcode[(a_offDisp) + 2]; \
     104        uint8_t const bTmp3 = (a_pVCpu)->iem.s.abOpcode[(a_offDisp) + 3]; \
     105        (a_u64Disp) = (int32_t)RT_MAKE_U32_FROM_U8(bTmp0, bTmp1, bTmp2, bTmp3); \
     106    } while (0)
     107# endif /* !IEM_WITH_CODE_TLB */
     108
     109/**
     110 * Gets VM-exit instruction information along with any displacement for an
     111 * instruction VM-exit.
     112 *
     113 * @returns The VM-exit instruction information.
     114 * @param   pVCpu           The cross context virtual CPU structure.
     115 * @param   uExitReason     The VM-exit reason.
     116 * @param   InstrId         The VM-exit instruction identity (VMX_INSTR_ID_XXX) if
     117 *                          any. Pass VMX_INSTR_ID_NONE otherwise.
     118 * @param   pGCPtrDisp      Where to store the displacement field. Optional, can be
     119 *                          NULL.
     120 */
     121IEM_STATIC uint32_t iemVmxGetExitInstrInfo(PVMCPU pVCpu, uint32_t uExitReason, VMXINSTRID InstrId, PRTGCPTR pGCPtrDisp)
     122{
     123    RTGCPTR          GCPtrDisp;
     124    VMXEXITINSTRINFO ExitInstrInfo;
     125    ExitInstrInfo.u = 0;
     126
     127    /*
     128     * Get and parse the ModR/M byte from our decoded opcodes.
     129     */
     130    uint8_t bRm;
     131    uint8_t const offModRm = pVCpu->iem.s.offModRm;
     132    IEM_MODRM_GET_U8(pVCpu, bRm, offModRm);
     133    if ((bRm & X86_MODRM_MOD_MASK) == (3 << X86_MODRM_MOD_SHIFT))
     134    {
     135        /*
     136         * ModR/M indicates register addressing.
     137         */
     138        ExitInstrInfo.All.u2Scaling       = 0;
     139        ExitInstrInfo.All.iReg1           = (bRm & X86_MODRM_RM_MASK) | pVCpu->iem.s.uRexB;
     140        ExitInstrInfo.All.u3AddrSize      = pVCpu->iem.s.enmEffAddrMode;
     141        ExitInstrInfo.All.fIsRegOperand   = 1;
     142        ExitInstrInfo.All.uOperandSize    = pVCpu->iem.s.enmEffOpSize;
     143        ExitInstrInfo.All.iSegReg         = 0;
     144        ExitInstrInfo.All.iIdxReg         = 0;
     145        ExitInstrInfo.All.fIdxRegInvalid  = 1;
     146        ExitInstrInfo.All.iBaseReg        = 0;
     147        ExitInstrInfo.All.fBaseRegInvalid = 1;
     148        ExitInstrInfo.All.iReg2           = ((bRm >> X86_MODRM_REG_SHIFT) & X86_MODRM_REG_SMASK) | pVCpu->iem.s.uRexReg;
     149
     150        /* Displacement not applicable for register addressing. */
     151        GCPtrDisp = 0;
     152    }
     153    else
     154    {
     155        /*
     156         * ModR/M indicates memory addressing.
     157         */
     158        uint8_t  uScale        = 0;
     159        bool     fBaseRegValid = false;
     160        bool     fIdxRegValid  = false;
     161        uint8_t  iBaseReg      = 0;
     162        uint8_t  iIdxReg       = 0;
     163        uint8_t  iReg2         = 0;
     164        if (pVCpu->iem.s.enmEffAddrMode == IEMMODE_16BIT)
     165        {
     166            /*
     167             * Parse the ModR/M, displacement for 16-bit addressing mode.
     168             * See Intel instruction spec. Table 2-1. "16-Bit Addressing Forms with the ModR/M Byte".
     169             */
     170            uint16_t u16Disp = 0;
     171            uint8_t const offDisp = offModRm + sizeof(bRm);
     172            if ((bRm & (X86_MODRM_MOD_MASK | X86_MODRM_RM_MASK)) == 6)
     173            {
     174                /* Displacement without any registers. */
     175                IEM_DISP_GET_U16(pVCpu, u16Disp, offDisp);
     176            }
     177            else
     178            {
     179                /* Register (index and base). */
     180                switch (bRm & X86_MODRM_RM_MASK)
     181                {
     182                    case 0: fBaseRegValid = true; iBaseReg = X86_GREG_xBX; fIdxRegValid = true; iIdxReg = X86_GREG_xSI; break;
     183                    case 1: fBaseRegValid = true; iBaseReg = X86_GREG_xBX; fIdxRegValid = true; iIdxReg = X86_GREG_xDI; break;
     184                    case 2: fBaseRegValid = true; iBaseReg = X86_GREG_xBP; fIdxRegValid = true; iIdxReg = X86_GREG_xSI; break;
     185                    case 3: fBaseRegValid = true; iBaseReg = X86_GREG_xBP; fIdxRegValid = true; iIdxReg = X86_GREG_xDI; break;
     186                    case 4: fIdxRegValid  = true; iIdxReg  = X86_GREG_xSI; break;
     187                    case 5: fIdxRegValid  = true; iIdxReg  = X86_GREG_xDI; break;
     188                    case 6: fBaseRegValid = true; iBaseReg = X86_GREG_xBP; break;
     189                    case 7: fBaseRegValid = true; iBaseReg = X86_GREG_xBX; break;
     190                }
     191
     192                /* Register + displacement. */
     193                switch ((bRm >> X86_MODRM_MOD_SHIFT) & X86_MODRM_MOD_SMASK)
     194                {
     195                    case 0:                                                  break;
     196                    case 1: IEM_DISP_GET_S8_SX_U16(pVCpu, u16Disp, offDisp); break;
     197                    case 2: IEM_DISP_GET_U16(pVCpu, u16Disp, offDisp);       break;
     198                    default:
     199                    {
     200                        /* Register addressing, handled at the beginning. */
     201                        AssertMsgFailed(("ModR/M %#x implies register addressing, memory addressing expected!", bRm));
     202                        break;
     203                    }
     204                }
     205            }
     206
     207            Assert(!uScale);                    /* There's no scaling/SIB byte for 16-bit addressing. */
     208            GCPtrDisp = (int16_t)u16Disp;       /* Sign-extend the displacement. */
     209            iReg2     = ((bRm >> X86_MODRM_REG_SHIFT) & X86_MODRM_REG_SMASK);
     210        }
     211        else if (pVCpu->iem.s.enmEffAddrMode == IEMMODE_32BIT)
     212        {
     213            /*
     214             * Parse the ModR/M, SIB, displacement for 32-bit addressing mode.
     215             * See Intel instruction spec. Table 2-2. "32-Bit Addressing Forms with the ModR/M Byte".
     216             */
     217            uint32_t u32Disp = 0;
     218            if ((bRm & (X86_MODRM_MOD_MASK | X86_MODRM_RM_MASK)) == 5)
     219            {
     220                /* Displacement without any registers. */
     221                uint8_t const offDisp = offModRm + sizeof(bRm);
     222                IEM_DISP_GET_U32(pVCpu, u32Disp, offDisp);
     223            }
     224            else
     225            {
     226                /* Register (and perhaps scale, index and base). */
     227                uint8_t offDisp = offModRm + sizeof(bRm);
     228                iBaseReg = (bRm & X86_MODRM_RM_MASK);
     229                if (iBaseReg == 4)
     230                {
     231                    /* An SIB byte follows the ModR/M byte, parse it. */
     232                    uint8_t bSib;
     233                    uint8_t const offSib = offModRm + sizeof(bRm);
     234                    IEM_SIB_GET_U8(pVCpu, bSib, offSib);
     235
     236                    /* A displacement may follow SIB, update its offset. */
     237                    offDisp += sizeof(bSib);
     238
     239                    /* Get the scale. */
     240                    uScale = (bSib >> X86_SIB_SCALE_SHIFT) & X86_SIB_SCALE_SMASK;
     241
     242                    /* Get the index register. */
     243                    iIdxReg = (bSib >> X86_SIB_INDEX_SHIFT) & X86_SIB_INDEX_SMASK;
     244                    fIdxRegValid = RT_BOOL(iIdxReg != 4);
     245
     246                    /* Get the base register. */
     247                    iBaseReg = bSib & X86_SIB_BASE_MASK;
     248                    fBaseRegValid = true;
     249                    if (iBaseReg == 5)
     250                    {
     251                        if ((bRm & X86_MODRM_MOD_MASK) == 0)
     252                        {
     253                            /* Mod is 0 implies a 32-bit displacement with no base. */
     254                            fBaseRegValid = false;
     255                            IEM_DISP_GET_U32(pVCpu, u32Disp, offDisp);
     256                        }
     257                        else
     258                        {
     259                            /* Mod is not 0 implies an 8-bit/32-bit displacement (handled below) with an EBP base. */
     260                            iBaseReg = X86_GREG_xBP;
     261                        }
     262                    }
     263                }
     264
     265                /* Register + displacement. */
     266                switch ((bRm >> X86_MODRM_MOD_SHIFT) & X86_MODRM_MOD_SMASK)
     267                {
     268                    case 0: /* Handled above */                              break;
     269                    case 1: IEM_DISP_GET_S8_SX_U32(pVCpu, u32Disp, offDisp); break;
     270                    case 2: IEM_DISP_GET_U32(pVCpu, u32Disp, offDisp);       break;
     271                    default:
     272                    {
     273                        /* Register addressing, handled at the beginning. */
     274                        AssertMsgFailed(("ModR/M %#x implies register addressing, memory addressing expected!", bRm));
     275                        break;
     276                    }
     277                }
     278            }
     279
     280            GCPtrDisp = (int32_t)u32Disp;       /* Sign-extend the displacement. */
     281            iReg2     = ((bRm >> X86_MODRM_REG_SHIFT) & X86_MODRM_REG_SMASK);
     282        }
     283        else
     284        {
     285            Assert(pVCpu->iem.s.enmEffAddrMode == IEMMODE_64BIT);
     286
     287            /*
     288             * Parse the ModR/M, SIB, displacement for 64-bit addressing mode.
     289             * See Intel instruction spec. 2.2 "IA-32e Mode".
     290             */
     291            uint64_t u64Disp = 0;
     292            bool const fRipRelativeAddr = RT_BOOL((bRm & (X86_MODRM_MOD_MASK | X86_MODRM_RM_MASK)) == 5);
     293            if (fRipRelativeAddr)
     294            {
     295                /*
     296                 * RIP-relative addressing mode.
     297                 *
     298                 * The displacment is 32-bit signed implying an offset range of +/-2G.
     299                 * See Intel instruction spec. 2.2.1.6 "RIP-Relative Addressing".
     300                 */
     301                uint8_t const offDisp = offModRm + sizeof(bRm);
     302                IEM_DISP_GET_S32_SX_U64(pVCpu, u64Disp, offDisp);
     303            }
     304            else
     305            {
     306                uint8_t offDisp = offModRm + sizeof(bRm);
     307
     308                /*
     309                 * Register (and perhaps scale, index and base).
     310                 *
     311                 * REX.B extends the most-significant bit of the base register. However, REX.B
     312                 * is ignored while determining whether an SIB follows the opcode. Hence, we
     313                 * shall OR any REX.B bit -after- inspecting for an SIB byte below.
     314                 *
     315                 * See Intel instruction spec. Table 2-5. "Special Cases of REX Encodings".
     316                 */
     317                iBaseReg = (bRm & X86_MODRM_RM_MASK);
     318                if (iBaseReg == 4)
     319                {
     320                    /* An SIB byte follows the ModR/M byte, parse it. Displacement (if any) follows SIB. */
     321                    uint8_t bSib;
     322                    uint8_t const offSib = offModRm + sizeof(bRm);
     323                    IEM_SIB_GET_U8(pVCpu, bSib, offSib);
     324
     325                    /* Displacement may follow SIB, update its offset. */
     326                    offDisp += sizeof(bSib);
     327
     328                    /* Get the scale. */
     329                    uScale = (bSib >> X86_SIB_SCALE_SHIFT) & X86_SIB_SCALE_SMASK;
     330
     331                    /* Get the index. */
     332                    iIdxReg = ((bSib >> X86_SIB_INDEX_SHIFT) & X86_SIB_INDEX_SMASK) | pVCpu->iem.s.uRexIndex;
     333                    fIdxRegValid = RT_BOOL(iIdxReg != 4);   /* R12 -can- be used as an index register. */
     334
     335                    /* Get the base. */
     336                    iBaseReg = (bSib & X86_SIB_BASE_MASK);
     337                    fBaseRegValid = true;
     338                    if (iBaseReg == 5)
     339                    {
     340                        if ((bRm & X86_MODRM_MOD_MASK) == 0)
     341                        {
     342                            /* Mod is 0 implies a signed 32-bit displacement with no base. */
     343                            IEM_DISP_GET_S32_SX_U64(pVCpu, u64Disp, offDisp);
     344                        }
     345                        else
     346                        {
     347                            /* Mod is non-zero implies an 8-bit/32-bit displacement (handled below) with RBP or R13 as base. */
     348                            iBaseReg = pVCpu->iem.s.uRexB ? X86_GREG_x13 : X86_GREG_xBP;
     349                        }
     350                    }
     351                }
     352                iBaseReg |= pVCpu->iem.s.uRexB;
     353
     354                /* Register + displacement. */
     355                switch ((bRm >> X86_MODRM_MOD_SHIFT) & X86_MODRM_MOD_SMASK)
     356                {
     357                    case 0: /* Handled above */                               break;
     358                    case 1: IEM_DISP_GET_S8_SX_U64(pVCpu, u64Disp, offDisp);  break;
     359                    case 2: IEM_DISP_GET_S32_SX_U64(pVCpu, u64Disp, offDisp); break;
     360                    default:
     361                    {
     362                        /* Register addressing, handled at the beginning. */
     363                        AssertMsgFailed(("ModR/M %#x implies register addressing, memory addressing expected!", bRm));
     364                        break;
     365                    }
     366                }
     367            }
     368
     369            GCPtrDisp = fRipRelativeAddr ? pVCpu->cpum.GstCtx.rip + u64Disp : u64Disp;
     370            iReg2 = ((bRm >> X86_MODRM_REG_SHIFT) & X86_MODRM_REG_SMASK) | pVCpu->iem.s.uRexReg;
     371        }
     372
     373        ExitInstrInfo.All.u2Scaling      = uScale;
     374        ExitInstrInfo.All.iReg1          = 0;   /* Not applicable for memory instructions. */
     375        ExitInstrInfo.All.u3AddrSize     = pVCpu->iem.s.enmEffAddrMode;
     376        ExitInstrInfo.All.fIsRegOperand  = 0;
     377        ExitInstrInfo.All.uOperandSize   = pVCpu->iem.s.enmEffOpSize;
     378        ExitInstrInfo.All.iSegReg        = pVCpu->iem.s.iEffSeg;
     379        ExitInstrInfo.All.iIdxReg        = iIdxReg;
     380        ExitInstrInfo.All.fIdxRegInvalid = !fIdxRegValid;
     381        ExitInstrInfo.All.iBaseReg       = iBaseReg;
     382        ExitInstrInfo.All.iIdxReg        = !fBaseRegValid;
     383        ExitInstrInfo.All.iReg2          = iReg2;
     384    }
     385
     386    /*
     387     * Handle exceptions for certain instructions.
     388     * (e.g. some instructions convey an instruction identity).
     389     */
     390    switch (uExitReason)
     391    {
     392        case VMX_EXIT_XDTR_ACCESS:
     393        {
     394            Assert(VMX_INSTR_ID_IS_VALID(InstrId));
     395            ExitInstrInfo.GdtIdt.u2InstrId = VMX_INSTR_ID_GET_ID(InstrId);
     396            ExitInstrInfo.GdtIdt.u2Undef0  = 0;
     397            break;
     398        }
     399
     400        case VMX_EXIT_TR_ACCESS:
     401        {
     402            Assert(VMX_INSTR_ID_IS_VALID(InstrId));
     403            ExitInstrInfo.LdtTr.u2InstrId = VMX_INSTR_ID_GET_ID(InstrId);
     404            ExitInstrInfo.GdtIdt.u2Undef0 = 0;
     405            break;
     406        }
     407
     408        case VMX_EXIT_RDRAND:
     409        case VMX_EXIT_RDSEED:
     410        {
     411            Assert(ExitInstrInfo.RdrandRdseed.u2OperandSize != 3);
     412            break;
     413        }
     414    }
     415
     416    /* Update displacement and return the constructed VM-exit instruction information field. */
     417    if (pGCPtrDisp)
     418        *pGCPtrDisp = GCPtrDisp;
     419    return ExitInstrInfo.u;
     420}
     421
     422
     423/**
    33424 * Implements VMSucceed for VMX instruction success.
    34425 *
     
    95486 * @param   cbInstr         The instruction length.
    96487 * @param   GCPtrVmxon      The linear address of the VMXON pointer.
    97  * @param   ExitInstrInfo   The VM-exit instruction information field.
     488 * @param   pExitInstrInfo  Pointer to the VM-exit instruction information field.
    98489 * @param   GCPtrDisp       The displacement field for @a GCPtrVmxon if any.
    99490 *
     
    155546        /* Get the VMXON pointer from the location specified by the source memory operand. */
    156547        RTGCPHYS GCPhysVmxon;
    157         VBOXSTRICTRC rcStrict = iemMemFetchDataU64(pVCpu, &GCPhysVmxon, pExitInstrInfo->InvVmxXsaves.iSegReg, GCPtrVmxon);
     548        VBOXSTRICTRC rcStrict = iemMemFetchDataU64(pVCpu, &GCPhysVmxon, pExitInstrInfo->VmxXsave.iSegReg, GCPtrVmxon);
    158549        if (RT_UNLIKELY(rcStrict != VINF_SUCCESS))
    159550        {
     
    275666IEM_CIMPL_DEF_1(iemCImpl_vmxon, RTGCPTR, GCPtrVmxon)
    276667{
    277     /** @todo NSTVMX: Parse ModR/M, SIB, disp.  */
    278     RTGCPTR GCPtrDisp = 0;
     668    RTGCPTR GCPtrDisp;
    279669    VMXEXITINSTRINFO ExitInstrInfo;
    280     ExitInstrInfo.u = 0;
    281     ExitInstrInfo.InvVmxXsaves.u2Scaling       = 0;
    282     ExitInstrInfo.InvVmxXsaves.u3AddrSize      = pVCpu->iem.s.enmEffAddrMode;
    283     ExitInstrInfo.InvVmxXsaves.fIsRegOperand   = 0;
    284     ExitInstrInfo.InvVmxXsaves.iSegReg         = pVCpu->iem.s.iEffSeg;
    285     ExitInstrInfo.InvVmxXsaves.iIdxReg         = 0;
    286     ExitInstrInfo.InvVmxXsaves.fIdxRegInvalid  = 0;
    287     ExitInstrInfo.InvVmxXsaves.iBaseReg        = 0;
    288     ExitInstrInfo.InvVmxXsaves.fBaseRegInvalid = 0;
    289     ExitInstrInfo.InvVmxXsaves.iReg2           = 0;
     670    ExitInstrInfo.u = iemVmxGetExitInstrInfo(pVCpu, VMX_EXIT_VMXON, VMX_INSTR_ID_NONE, &GCPtrDisp);
    290671    return iemVmxVmxon(pVCpu, cbInstr, GCPtrVmxon, &ExitInstrInfo, GCPtrDisp);
    291672}
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