VirtualBox

Changeset 101581 in vbox for trunk/src


Ignore:
Timestamp:
Oct 24, 2023 2:50:49 PM (18 months ago)
Author:
vboxsync
svn:sync-xref-src-repo-rev:
159662
Message:

VMM/IEM: Native IEM_MC_IF_EFL_BIT_SET_OR_BITS_NE and IEM_MC_IF_EFL_BIT_NOT_SET_AND_BITS_EQ translations. Emitter naming fixes. bugref:10371

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

Legend:

Unmodified
Added
Removed
  • TabularUnified trunk/src/VBox/VMM/VMMAll/IEMAllInstPython.py

    r101580 r101581  
    28882888    'IEM_MC_IF_EFL_ANY_BITS_SET':                                (McBlock.parseMcGenericCond,       True,  True,  ),
    28892889    'IEM_MC_IF_EFL_BIT_NOT_SET':                                 (McBlock.parseMcGenericCond,       True,  True,  ),
    2890     'IEM_MC_IF_EFL_BIT_NOT_SET_AND_BITS_EQ':                     (McBlock.parseMcGenericCond,       True,  False, ),
     2890    'IEM_MC_IF_EFL_BIT_NOT_SET_AND_BITS_EQ':                     (McBlock.parseMcGenericCond,       True,  True, ),
    28912891    'IEM_MC_IF_EFL_BIT_SET':                                     (McBlock.parseMcGenericCond,       True,  True,  ),
    2892     'IEM_MC_IF_EFL_BIT_SET_OR_BITS_NE':                          (McBlock.parseMcGenericCond,       True,  False, ),
     2892    'IEM_MC_IF_EFL_BIT_SET_OR_BITS_NE':                          (McBlock.parseMcGenericCond,       True,  True, ),
    28932893    'IEM_MC_IF_EFL_BITS_EQ':                                     (McBlock.parseMcGenericCond,       True,  True,  ),
    28942894    'IEM_MC_IF_EFL_BITS_NE':                                     (McBlock.parseMcGenericCond,       True,  True,  ),
  • TabularUnified trunk/src/VBox/VMM/VMMAll/IEMAllN8veRecompiler.cpp

    r101580 r101581  
    40894089 * Start of the if-block, snapshotting the register and variable state.
    40904090 */
    4091 DECLINLINE(void) iemNativeCondStartIfBlock(PIEMRECOMPILERSTATE pReNative, uint32_t offIfBlock)
     4091DECLINLINE(void) iemNativeCondStartIfBlock(PIEMRECOMPILERSTATE pReNative, uint32_t offIfBlock, uint32_t idxLabelIf = UINT32_MAX)
    40924092{
    40934093    Assert(offIfBlock != UINT32_MAX);
     
    40964096    Assert(!pEntry->fInElse);
    40974097
    4098     /* Define the start of the IF block for disassembly. */
     4098    /* Define the start of the IF block if request or for disassembly purposes. */
     4099    if (idxLabelIf != UINT32_MAX)
     4100        iemNativeLabelDefine(pReNative, idxLabelIf, offIfBlock);
    40994101#ifdef IEMNATIVE_WITH_TB_DEBUG_INFO
    4100     iemNativeLabelCreate(pReNative, kIemNativeLabelType_If, offIfBlock, pReNative->paLabels[pEntry->idxLabelElse].uData);
     4102    else
     4103        iemNativeLabelCreate(pReNative, kIemNativeLabelType_If, offIfBlock, pReNative->paLabels[pEntry->idxLabelElse].uData);
    41014104#else
    41024105    RT_NOREF(offIfBlock);
     
    43314334
    43324335    /* Test and jump. */
    4333     off = iemNativeEmitTestBitInGprAndJmpToLabelIfSet(pReNative, off, idxEflReg, iBitNo, pEntry->idxLabelElse);
     4336    off = iemNativeEmitTestBitInGprAndJmpToLabelIfNotSet(pReNative, off, idxEflReg, iBitNo, pEntry->idxLabelElse);
    43344337
    43354338    /* Free but don't flush the EFlags register. */
     
    43634366
    43644367    /* Test and jump. */
    4365     off = iemNativeEmitTestBitInGprAndJmpToLabelIfNotSet(pReNative, off, idxEflReg, iBitNo, pEntry->idxLabelElse);
     4368    off = iemNativeEmitTestBitInGprAndJmpToLabelIfSet(pReNative, off, idxEflReg, iBitNo, pEntry->idxLabelElse);
    43664369
    43674370    /* Free but don't flush the EFlags register. */
     
    43764379
    43774380#define IEM_MC_IF_EFL_BITS_EQ(a_fBit1, a_fBit2)         \
    4378     off = iemNativeEmitIfEflagsTwoBitsComp(pReNative, off, a_fBit1, a_fBit2, false /*fNotEqual*/); \
     4381    off = iemNativeEmitIfEflagsTwoBitsEqual(pReNative, off, a_fBit1, a_fBit2, false /*fInverted*/); \
    43794382    AssertReturn(off != UINT32_MAX, UINT32_MAX); \
    43804383    do {
    43814384
    43824385#define IEM_MC_IF_EFL_BITS_NE(a_fBit1, a_fBit2)         \
    4383     off = iemNativeEmitIfEflagsTwoBitsComp(pReNative, off, a_fBit1, a_fBit2, true /*fNotEqual*/); \
     4386    off = iemNativeEmitIfEflagsTwoBitsEqual(pReNative, off, a_fBit1, a_fBit2, true /*fInverted*/); \
    43844387    AssertReturn(off != UINT32_MAX, UINT32_MAX); \
    43854388    do {
    43864389
    43874390/** Emits code for IEM_MC_IF_EFL_BITS_EQ and IEM_MC_IF_EFL_BITS_NE. */
    4388 DECLINLINE(uint32_t) iemNativeEmitIfEflagsTwoBitsComp(PIEMRECOMPILERSTATE pReNative, uint32_t off,
    4389                                                       uint32_t fBit1InEfl, uint32_t fBit2InEfl, bool fNotEqual)
     4391DECLINLINE(uint32_t) iemNativeEmitIfEflagsTwoBitsEqual(PIEMRECOMPILERSTATE pReNative, uint32_t off,
     4392                                                       uint32_t fBit1InEfl, uint32_t fBit2InEfl, bool fInverted)
    43904393{
    43914394    PIEMNATIVECOND pEntry = iemNativeCondPushIf(pReNative);
     
    44394442#endif
    44404443
    4441     /* Test and jump. */
    4442     if (fNotEqual)
    4443         off = iemNativeEmitTestBitInGprAndJmpToLabelIfSet(pReNative, off, idxTmpReg, iBitNo2, pEntry->idxLabelElse);
    4444     else
    4445         off = iemNativeEmitTestBitInGprAndJmpToLabelIfNotSet(pReNative, off, idxTmpReg, iBitNo2, pEntry->idxLabelElse);
     4444    /* Test (bit #2 is set in tmpreg if not-equal) and jump. */
     4445    off = iemNativeEmitTestBitInGprAndJmpToLabelIfCc(pReNative, off, idxTmpReg, iBitNo2,
     4446                                                     pEntry->idxLabelElse, !fInverted /*fJmpIfSet*/);
    44464447
    44474448    /* Free but don't flush the EFlags and tmp registers. */
     
    44514452    /* Make a copy of the core state now as we start the if-block. */
    44524453    iemNativeCondStartIfBlock(pReNative, off);
     4454
     4455    return off;
     4456}
     4457
     4458
     4459#define IEM_MC_IF_EFL_BIT_NOT_SET_AND_BITS_EQ(a_fBit, a_fBit1, a_fBit2) \
     4460    off = iemNativeEmitIfEflagsBitNotSetAndTwoBitsEqual(pReNative, off, a_fBit, a_fBit1, a_fBit2, false /*fInverted*/); \
     4461    AssertReturn(off != UINT32_MAX, UINT32_MAX); \
     4462    do {
     4463
     4464#define IEM_MC_IF_EFL_BIT_SET_OR_BITS_NE(a_fBit, a_fBit1, a_fBit2) \
     4465    off = iemNativeEmitIfEflagsBitNotSetAndTwoBitsEqual(pReNative, off, a_fBit, a_fBit1, a_fBit2, true /*fInverted*/); \
     4466    AssertReturn(off != UINT32_MAX, UINT32_MAX); \
     4467    do {
     4468
     4469/** Emits code for IEM_MC_IF_EFL_BIT_NOT_SET_AND_BITS_EQ and
     4470 *  IEM_MC_IF_EFL_BIT_SET_OR_BITS_NE. */
     4471DECLINLINE(uint32_t) iemNativeEmitIfEflagsBitNotSetAndTwoBitsEqual(PIEMRECOMPILERSTATE pReNative, uint32_t off, uint32_t fBitInEfl,
     4472                                                                  uint32_t fBit1InEfl, uint32_t fBit2InEfl, bool fInverted)
     4473{
     4474    PIEMNATIVECOND pEntry = iemNativeCondPushIf(pReNative);
     4475    AssertReturn(pEntry, UINT32_MAX);
     4476
     4477    /* We need an if-block label for the non-inverted variant. */
     4478    uint32_t const idxLabelIf = fInverted ? iemNativeLabelCreate(pReNative, kIemNativeLabelType_If, UINT32_MAX,
     4479                                                                 pReNative->paLabels[pEntry->idxLabelElse].uData) : UINT32_MAX;
     4480    AssertReturn(idxLabelIf != UINT32_MAX || !fInverted, UINT32_MAX);
     4481
     4482    /* Get the eflags. */
     4483    uint8_t const idxEflReg = iemNativeRegAllocTmpForGuestReg(pReNative, &off, kIemNativeGstReg_EFlags,
     4484                                                              kIemNativeGstRegUse_ReadOnly);
     4485    AssertReturn(idxEflReg != UINT8_MAX, UINT32_MAX);
     4486
     4487    /* Translate the flag masks to bit numbers. */
     4488    unsigned const iBitNo = ASMBitFirstSetU32(fBitInEfl) - 1;
     4489    Assert(RT_BIT_32(iBitNo) == fBitInEfl);
     4490
     4491    unsigned const iBitNo1 = ASMBitFirstSetU32(fBit1InEfl) - 1;
     4492    Assert(RT_BIT_32(iBitNo1) == fBit1InEfl);
     4493    Assert(iBitNo1 != iBitNo);
     4494
     4495    unsigned const iBitNo2 = ASMBitFirstSetU32(fBit2InEfl) - 1;
     4496    Assert(RT_BIT_32(iBitNo2) == fBit2InEfl);
     4497    Assert(iBitNo2 != iBitNo);
     4498    Assert(iBitNo2 != iBitNo1);
     4499
     4500#ifdef RT_ARCH_AMD64
     4501    uint8_t const idxTmpReg = iemNativeRegAllocTmpImm(pReNative, &off, fBit1InEfl); /* This must come before we jump anywhere! */
     4502#elif defined(RT_ARCH_ARM64)
     4503    uint8_t const idxTmpReg = iemNativeRegAllocTmp(pReNative, &off);
     4504#endif
     4505    AssertReturn(idxTmpReg != UINT8_MAX, UINT32_MAX);
     4506
     4507    /* Check for the lone bit first. */
     4508    if (!fInverted)
     4509        off = iemNativeEmitTestBitInGprAndJmpToLabelIfSet(pReNative, off, idxEflReg, iBitNo, pEntry->idxLabelElse);
     4510    else
     4511        off = iemNativeEmitTestBitInGprAndJmpToLabelIfSet(pReNative, off, idxEflReg, iBitNo, idxLabelIf);
     4512
     4513    /* Then extract and compare the other two bits. */
     4514#ifdef RT_ARCH_AMD64
     4515    off = iemNativeEmitAndGpr32ByGpr32(pReNative, off, idxTmpReg, idxEflReg);
     4516    if (iBitNo1 > iBitNo2)
     4517        off = iemNativeEmitShiftGpr32Right(pReNative, off, idxTmpReg, iBitNo1 - iBitNo2);
     4518    else
     4519        off = iemNativeEmitShiftGpr32Left(pReNative, off, idxTmpReg, iBitNo2 - iBitNo1);
     4520    off = iemNativeEmitXorGpr32ByGpr32(pReNative, off, idxTmpReg, idxEflReg);
     4521
     4522#elif defined(RT_ARCH_ARM64)
     4523    uint32_t *pu32CodeBuf = iemNativeInstrBufEnsure(pReNative, off, 2);
     4524    AssertReturn(pu32CodeBuf, UINT32_MAX);
     4525
     4526    /* and tmpreg, eflreg, #1<<iBitNo1 */
     4527    pu32CodeBuf[off++] = Armv8A64MkInstrAndImm(idxTmpReg, idxEflReg, 0 /*uImm7SizeLen -> 32*/, 32 - iBitNo1, false /*f64Bit*/);
     4528
     4529    /* eeyore tmpreg, eflreg, tmpreg, LSL/LSR, #abs(iBitNo2 - iBitNo1) */
     4530    if (iBitNo1 > iBitNo2)
     4531        pu32CodeBuf[off++] = Armv8A64MkInstrEor(idxTmpReg, idxEflReg, idxTmpReg, false /*64bit*/,
     4532                                                iBitNo1 - iBitNo2, kArmv8A64InstrShift_Lsr);
     4533    else
     4534        pu32CodeBuf[off++] = Armv8A64MkInstrEor(idxTmpReg, idxEflReg, idxTmpReg, false /*64bit*/,
     4535                                                iBitNo2 - iBitNo1, kArmv8A64InstrShift_Lsl);
     4536
     4537    IEMNATIVE_ASSERT_INSTR_BUF_ENSURE(pReNative, off);
     4538
     4539#else
     4540# error "Port me"
     4541#endif
     4542
     4543    /* Test (bit #2 is set in tmpreg if not-equal) and jump. */
     4544    off = iemNativeEmitTestBitInGprAndJmpToLabelIfCc(pReNative, off, idxTmpReg, iBitNo2,
     4545                                                     pEntry->idxLabelElse, !fInverted /*fJmpIfSet*/);
     4546
     4547    /* Free but don't flush the EFlags and tmp registers. */
     4548    iemNativeRegFreeTmp(pReNative, idxTmpReg);
     4549    iemNativeRegFreeTmp(pReNative, idxEflReg);
     4550
     4551    /* Make a copy of the core state now as we start the if-block. */
     4552    iemNativeCondStartIfBlock(pReNative, off, idxLabelIf);
    44534553
    44544554    return off;
  • TabularUnified trunk/src/VBox/VMM/include/IEMN8veRecompiler.h

    r101576 r101581  
    264264    kIemNativeLabelType_Invalid = 0,
    265265    kIemNativeLabelType_Return,
    266 #ifdef IEMNATIVE_WITH_TB_DEBUG_INFO
    267266    kIemNativeLabelType_If,
    268 #endif
    269267    kIemNativeLabelType_Else,
    270268    kIemNativeLabelType_Endif,
     
    24832481                                                                 uint8_t iGprSrc, uint8_t iBitNo, uint32_t idxLabel)
    24842482{
    2485     return iemNativeEmitTestBitInGprAndJmpToLabelIfCc(pReNative, off, iGprSrc, iBitNo, idxLabel, false /*fJmpIfSet*/);
     2483    return iemNativeEmitTestBitInGprAndJmpToLabelIfCc(pReNative, off, iGprSrc, iBitNo, idxLabel, true /*fJmpIfSet*/);
    24862484}
    24872485
     
    24962494                                                                    uint8_t iGprSrc, uint8_t iBitNo, uint32_t idxLabel)
    24972495{
    2498     return iemNativeEmitTestBitInGprAndJmpToLabelIfCc(pReNative, off, iGprSrc, iBitNo, idxLabel, true /*fJmpIfSet*/);
     2496    return iemNativeEmitTestBitInGprAndJmpToLabelIfCc(pReNative, off, iGprSrc, iBitNo, idxLabel, false /*fJmpIfSet*/);
    24992497}
    25002498
Note: See TracChangeset for help on using the changeset viewer.

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