- Timestamp:
- Oct 24, 2023 2:50:49 PM (18 months ago)
- svn:sync-xref-src-repo-rev:
- 159662
- Location:
- trunk/src/VBox/VMM
- Files:
-
- 3 edited
Legend:
- Unmodified
- Added
- Removed
-
TabularUnified trunk/src/VBox/VMM/VMMAll/IEMAllInstPython.py ¶
r101580 r101581 2888 2888 'IEM_MC_IF_EFL_ANY_BITS_SET': (McBlock.parseMcGenericCond, True, True, ), 2889 2889 '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, ), 2891 2891 '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, ), 2893 2893 'IEM_MC_IF_EFL_BITS_EQ': (McBlock.parseMcGenericCond, True, True, ), 2894 2894 'IEM_MC_IF_EFL_BITS_NE': (McBlock.parseMcGenericCond, True, True, ), -
TabularUnified trunk/src/VBox/VMM/VMMAll/IEMAllN8veRecompiler.cpp ¶
r101580 r101581 4089 4089 * Start of the if-block, snapshotting the register and variable state. 4090 4090 */ 4091 DECLINLINE(void) iemNativeCondStartIfBlock(PIEMRECOMPILERSTATE pReNative, uint32_t offIfBlock )4091 DECLINLINE(void) iemNativeCondStartIfBlock(PIEMRECOMPILERSTATE pReNative, uint32_t offIfBlock, uint32_t idxLabelIf = UINT32_MAX) 4092 4092 { 4093 4093 Assert(offIfBlock != UINT32_MAX); … … 4096 4096 Assert(!pEntry->fInElse); 4097 4097 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); 4099 4101 #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); 4101 4104 #else 4102 4105 RT_NOREF(offIfBlock); … … 4331 4334 4332 4335 /* Test and jump. */ 4333 off = iemNativeEmitTestBitInGprAndJmpToLabelIf Set(pReNative, off, idxEflReg, iBitNo, pEntry->idxLabelElse);4336 off = iemNativeEmitTestBitInGprAndJmpToLabelIfNotSet(pReNative, off, idxEflReg, iBitNo, pEntry->idxLabelElse); 4334 4337 4335 4338 /* Free but don't flush the EFlags register. */ … … 4363 4366 4364 4367 /* Test and jump. */ 4365 off = iemNativeEmitTestBitInGprAndJmpToLabelIf NotSet(pReNative, off, idxEflReg, iBitNo, pEntry->idxLabelElse);4368 off = iemNativeEmitTestBitInGprAndJmpToLabelIfSet(pReNative, off, idxEflReg, iBitNo, pEntry->idxLabelElse); 4366 4369 4367 4370 /* Free but don't flush the EFlags register. */ … … 4376 4379 4377 4380 #define IEM_MC_IF_EFL_BITS_EQ(a_fBit1, a_fBit2) \ 4378 off = iemNativeEmitIfEflagsTwoBits Comp(pReNative, off, a_fBit1, a_fBit2, false /*fNotEqual*/); \4381 off = iemNativeEmitIfEflagsTwoBitsEqual(pReNative, off, a_fBit1, a_fBit2, false /*fInverted*/); \ 4379 4382 AssertReturn(off != UINT32_MAX, UINT32_MAX); \ 4380 4383 do { 4381 4384 4382 4385 #define IEM_MC_IF_EFL_BITS_NE(a_fBit1, a_fBit2) \ 4383 off = iemNativeEmitIfEflagsTwoBits Comp(pReNative, off, a_fBit1, a_fBit2, true /*fNotEqual*/); \4386 off = iemNativeEmitIfEflagsTwoBitsEqual(pReNative, off, a_fBit1, a_fBit2, true /*fInverted*/); \ 4384 4387 AssertReturn(off != UINT32_MAX, UINT32_MAX); \ 4385 4388 do { 4386 4389 4387 4390 /** Emits code for IEM_MC_IF_EFL_BITS_EQ and IEM_MC_IF_EFL_BITS_NE. */ 4388 DECLINLINE(uint32_t) iemNativeEmitIfEflagsTwoBits Comp(PIEMRECOMPILERSTATE pReNative, uint32_t off,4389 uint32_t fBit1InEfl, uint32_t fBit2InEfl, bool fNotEqual)4391 DECLINLINE(uint32_t) iemNativeEmitIfEflagsTwoBitsEqual(PIEMRECOMPILERSTATE pReNative, uint32_t off, 4392 uint32_t fBit1InEfl, uint32_t fBit2InEfl, bool fInverted) 4390 4393 { 4391 4394 PIEMNATIVECOND pEntry = iemNativeCondPushIf(pReNative); … … 4439 4442 #endif 4440 4443 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*/); 4446 4447 4447 4448 /* Free but don't flush the EFlags and tmp registers. */ … … 4451 4452 /* Make a copy of the core state now as we start the if-block. */ 4452 4453 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. */ 4471 DECLINLINE(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); 4453 4553 4454 4554 return off; -
TabularUnified trunk/src/VBox/VMM/include/IEMN8veRecompiler.h ¶
r101576 r101581 264 264 kIemNativeLabelType_Invalid = 0, 265 265 kIemNativeLabelType_Return, 266 #ifdef IEMNATIVE_WITH_TB_DEBUG_INFO267 266 kIemNativeLabelType_If, 268 #endif269 267 kIemNativeLabelType_Else, 270 268 kIemNativeLabelType_Endif, … … 2483 2481 uint8_t iGprSrc, uint8_t iBitNo, uint32_t idxLabel) 2484 2482 { 2485 return iemNativeEmitTestBitInGprAndJmpToLabelIfCc(pReNative, off, iGprSrc, iBitNo, idxLabel, false /*fJmpIfSet*/);2483 return iemNativeEmitTestBitInGprAndJmpToLabelIfCc(pReNative, off, iGprSrc, iBitNo, idxLabel, true /*fJmpIfSet*/); 2486 2484 } 2487 2485 … … 2496 2494 uint8_t iGprSrc, uint8_t iBitNo, uint32_t idxLabel) 2497 2495 { 2498 return iemNativeEmitTestBitInGprAndJmpToLabelIfCc(pReNative, off, iGprSrc, iBitNo, idxLabel, true /*fJmpIfSet*/);2496 return iemNativeEmitTestBitInGprAndJmpToLabelIfCc(pReNative, off, iGprSrc, iBitNo, idxLabel, false /*fJmpIfSet*/); 2499 2497 } 2500 2498
Note:
See TracChangeset
for help on using the changeset viewer.