Changeset 103554 in vbox for trunk/src/VBox
- Timestamp:
- Feb 23, 2024 11:26:09 PM (13 months ago)
- svn:sync-xref-src-repo-rev:
- 161904
- Location:
- trunk/src/VBox/VMM
- Files:
-
- 3 edited
Legend:
- Unmodified
- Added
- Removed
-
trunk/src/VBox/VMM/VMMAll/IEMAllThrdRecompiler.cpp
r103535 r103554 1275 1275 *********************************************************************************************************************************/ 1276 1276 1277 /** @callback_method_impl{FNDISREADBYTES, Dummy.} */ 1278 static DECLCALLBACK(int) iemThreadedDisasReadBytesDummy(PDISSTATE pDis, uint8_t offInstr, uint8_t cbMinRead, uint8_t cbMaxRead) 1279 { 1280 RT_BZERO(&pDis->Instr.ab[offInstr], cbMaxRead); 1281 pDis->cbCachedInstr += cbMaxRead; 1282 RT_NOREF(cbMinRead); 1283 return VERR_NO_DATA; 1284 } 1285 1286 1287 DECLHIDDEN(void) iemThreadedDisassembleTb(PCIEMTB pTb, PCDBGFINFOHLP pHlp) RT_NOEXCEPT 1288 { 1289 AssertReturnVoid((pTb->fFlags & IEMTB_F_TYPE_MASK) == IEMTB_F_TYPE_THREADED); 1290 1291 char szDisBuf[512]; 1292 1293 /* 1294 * Print TB info. 1295 */ 1296 pHlp->pfnPrintf(pHlp, 1297 "pTb=%p: GCPhysPc=%RGp cInstructions=%u LB %#x cRanges=%u\n" 1298 "pTb=%p: cUsed=%u msLastUsed=%u fFlags=%#010x %s\n", 1299 pTb, pTb->GCPhysPc, pTb->cInstructions, pTb->cbOpcodes, pTb->cRanges, 1300 pTb, pTb->cUsed, pTb->msLastUsed, pTb->fFlags, iemTbFlagsToString(pTb->fFlags, szDisBuf, sizeof(szDisBuf))); 1301 1302 /* 1303 * This disassembly is driven by the debug info which follows the native 1304 * code and indicates when it starts with the next guest instructions, 1305 * where labels are and such things. 1306 */ 1307 DISSTATE Dis; 1308 PCIEMTHRDEDCALLENTRY const paCalls = pTb->Thrd.paCalls; 1309 uint32_t const cCalls = pTb->Thrd.cCalls; 1310 DISCPUMODE enmGstCpuMode = (pTb->fFlags & IEM_F_MODE_CPUMODE_MASK) == IEMMODE_16BIT ? DISCPUMODE_16BIT 1311 : (pTb->fFlags & IEM_F_MODE_CPUMODE_MASK) == IEMMODE_32BIT ? DISCPUMODE_32BIT 1312 : DISCPUMODE_64BIT; 1313 uint32_t fExec = pTb->fFlags & UINT32_C(0x00ffffff); 1314 uint8_t idxRange = UINT8_MAX; 1315 uint8_t const cRanges = RT_MIN(pTb->cRanges, RT_ELEMENTS(pTb->aRanges)); 1316 uint32_t offRange = 0; 1317 uint32_t offOpcodes = 0; 1318 uint32_t const cbOpcodes = pTb->cbOpcodes; 1319 RTGCPHYS GCPhysPc = pTb->GCPhysPc; 1320 1321 for (uint32_t iCall = 0; iCall < cCalls; iCall++) 1322 { 1323 /* 1324 * New opcode range? 1325 */ 1326 if ( idxRange == UINT8_MAX 1327 || idxRange >= cRanges 1328 || offRange >= pTb->aRanges[idxRange].cbOpcodes) 1329 { 1330 idxRange += 1; 1331 if (idxRange < cRanges) 1332 offRange = !idxRange ? 0 : offRange - pTb->aRanges[idxRange - 1].cbOpcodes; 1333 else 1334 continue; 1335 GCPhysPc = pTb->aRanges[idxRange].offPhysPage 1336 + (pTb->aRanges[idxRange].idxPhysPage == 0 1337 ? pTb->GCPhysPc & ~(RTGCPHYS)GUEST_PAGE_OFFSET_MASK 1338 : pTb->aGCPhysPages[pTb->aRanges[idxRange].idxPhysPage - 1]); 1339 pHlp->pfnPrintf(pHlp, " Range #%u: GCPhysPc=%RGp LB %#x [idxPg=%d]\n", 1340 idxRange, GCPhysPc, pTb->aRanges[idxRange].cbOpcodes, 1341 pTb->aRanges[idxRange].idxPhysPage); 1342 GCPhysPc += offRange; 1343 } 1344 1345 /* 1346 * Disassemble another guest instruction? 1347 */ 1348 if ( paCalls[iCall].offOpcode != offOpcodes 1349 && paCalls[iCall].cbOpcode > 0 1350 && (uint32_t)(cbOpcodes - paCalls[iCall].offOpcode) <= cbOpcodes /* paranoia^2 */ ) 1351 { 1352 offOpcodes = paCalls[iCall].offOpcode; 1353 uint8_t const cbInstrMax = RT_MIN(cbOpcodes - offOpcodes, 15); 1354 uint32_t cbInstr = 1; 1355 int rc = DISInstrWithPrefetchedBytes(GCPhysPc, enmGstCpuMode, DISOPTYPE_ALL, 1356 &pTb->pabOpcodes[offOpcodes], cbInstrMax, 1357 iemThreadedDisasReadBytesDummy, NULL, &Dis, &cbInstr); 1358 if (RT_SUCCESS(rc)) 1359 { 1360 DISFormatYasmEx(&Dis, szDisBuf, sizeof(szDisBuf), 1361 DIS_FMT_FLAGS_BYTES_WIDTH_MAKE(10) | DIS_FMT_FLAGS_BYTES_LEFT 1362 | DIS_FMT_FLAGS_RELATIVE_BRANCH | DIS_FMT_FLAGS_C_HEX, 1363 NULL /*pfnGetSymbol*/, NULL /*pvUser*/); 1364 pHlp->pfnPrintf(pHlp, " %%%%%RGp: %s\n", GCPhysPc, szDisBuf); 1365 } 1366 else 1367 { 1368 pHlp->pfnPrintf(pHlp, " %%%%%RGp: %.*Rhxs - guest disassembly failure %Rrc\n", 1369 GCPhysPc, cbInstrMax, &pTb->pabOpcodes[offOpcodes], rc); 1370 cbInstr = paCalls[iCall].cbOpcode; 1371 } 1372 GCPhysPc += cbInstr; 1373 offRange += cbInstr; 1374 } 1375 1376 /* 1377 * Dump call details. 1378 */ 1379 pHlp->pfnPrintf(pHlp, 1380 " Call #%u to %s (%u args)\n", 1381 iCall, g_apszIemThreadedFunctions[paCalls[iCall].enmFunction], 1382 g_acIemThreadedFunctionUsedArgs[paCalls[iCall].enmFunction]); 1383 1384 /* 1385 * Snoop fExec. 1386 */ 1387 switch (paCalls[iCall].enmFunction) 1388 { 1389 default: 1390 break; 1391 case kIemThreadedFunc_BltIn_CheckMode: 1392 fExec = paCalls[iCall].auParams[0]; 1393 break; 1394 } 1395 } 1396 } 1397 1398 1399 1277 1400 /** 1278 1401 * Allocate a translation block for threadeded recompilation. -
trunk/src/VBox/VMM/VMMR3/IEMR3.cpp
r103516 r103554 31 31 *********************************************************************************************************************************/ 32 32 #define LOG_GROUP LOG_GROUP_EM 33 #define VMCPU_INCL_CPUM_GST_CTX 33 34 #include <VBox/vmm/iem.h> 34 35 #include <VBox/vmm/cpum.h> … … 51 52 #include <iprt/string.h> 52 53 53 #if defined(VBOX_WITH_ STATISTICS) && defined(VBOX_WITH_IEM_RECOMPILER) && !defined(VBOX_VMM_TARGET_ARMV8)54 #if defined(VBOX_WITH_IEM_RECOMPILER) && !defined(VBOX_VMM_TARGET_ARMV8) 54 55 # include "IEMN8veRecompiler.h" 55 56 # include "IEMThreadedFunctions.h" 57 # include "IEMInline.h" 56 58 #endif 57 59 … … 62 64 static FNDBGFINFOARGVINT iemR3InfoITlb; 63 65 static FNDBGFINFOARGVINT iemR3InfoDTlb; 66 #if defined(VBOX_WITH_IEM_RECOMPILER) && !defined(VBOX_VMM_TARGET_ARMV8) 67 static FNDBGFINFOARGVINT iemR3InfoTb; 68 #endif 64 69 #ifdef VBOX_WITH_DEBUGGER 65 70 static void iemR3RegisterDebuggerCommands(void); … … 575 580 DBGFR3InfoRegisterInternalArgv(pVM, "itlb", "IEM instruction TLB", iemR3InfoITlb, DBGFINFO_FLAGS_RUN_ON_EMT); 576 581 DBGFR3InfoRegisterInternalArgv(pVM, "dtlb", "IEM instruction TLB", iemR3InfoDTlb, DBGFINFO_FLAGS_RUN_ON_EMT); 582 #if defined(VBOX_WITH_IEM_RECOMPILER) && !defined(VBOX_VMM_TARGET_ARMV8) 583 DBGFR3InfoRegisterInternalArgv(pVM, "tb", "IEM translation block", iemR3InfoTb, DBGFINFO_FLAGS_RUN_ON_EMT); 584 #endif 577 585 #ifdef VBOX_WITH_DEBUGGER 578 586 iemR3RegisterDebuggerCommands(); … … 898 906 } 899 907 908 #if defined(VBOX_WITH_IEM_RECOMPILER) && !defined(VBOX_VMM_TARGET_ARMV8) 909 /** 910 * @callback_method_impl{FNDBGFINFOARGVINT, dtlb} 911 */ 912 static DECLCALLBACK(void) iemR3InfoTb(PVM pVM, PCDBGFINFOHLP pHlp, int cArgs, char **papszArgs) 913 { 914 /* 915 * Parse arguments. 916 */ 917 static RTGETOPTDEF const s_aOptions[] = 918 { 919 { "--cpu", 'c', RTGETOPT_REQ_UINT32 }, 920 { "--vcpu", 'c', RTGETOPT_REQ_UINT32 }, 921 { "--addr", 'a', RTGETOPT_REQ_UINT64 | RTGETOPT_FLAG_HEX }, 922 { "--address", 'a', RTGETOPT_REQ_UINT64 | RTGETOPT_FLAG_HEX }, 923 { "--phys", 'p', RTGETOPT_REQ_UINT64 | RTGETOPT_FLAG_HEX }, 924 { "--physical", 'p', RTGETOPT_REQ_UINT64 | RTGETOPT_FLAG_HEX }, 925 { "--phys-addr", 'p', RTGETOPT_REQ_UINT64 | RTGETOPT_FLAG_HEX }, 926 { "--phys-address", 'p', RTGETOPT_REQ_UINT64 | RTGETOPT_FLAG_HEX }, 927 { "--physical-address", 'p', RTGETOPT_REQ_UINT64 | RTGETOPT_FLAG_HEX }, 928 { "--flags", 'f', RTGETOPT_REQ_UINT32 | RTGETOPT_FLAG_HEX }, 929 }; 930 931 RTGETOPTSTATE State; 932 int rc = RTGetOptInit(&State, cArgs, papszArgs, s_aOptions, RT_ELEMENTS(s_aOptions), 0 /*iFirst*/, 0 /*fFlags*/); 933 AssertRCReturnVoid(rc); 934 935 PVMCPU const pVCpuThis = VMMGetCpu(pVM); 936 PVMCPU pVCpu = pVCpuThis ? pVCpuThis : VMMGetCpuById(pVM, 0); 937 RTGCPHYS GCPhysPc = NIL_RTGCPHYS; 938 RTGCPHYS GCVirt = NIL_RTGCPTR; 939 uint32_t fFlags = UINT32_MAX; 940 941 RTGETOPTUNION ValueUnion; 942 while ((rc = RTGetOpt(&State, &ValueUnion)) != 0) 943 { 944 switch (rc) 945 { 946 case 'c': 947 if (ValueUnion.u32 >= pVM->cCpus) 948 pHlp->pfnPrintf(pHlp, "error: Invalid CPU ID: %u\n", ValueUnion.u32); 949 else if (!pVCpu || pVCpu->idCpu != ValueUnion.u32) 950 pVCpu = VMMGetCpuById(pVM, ValueUnion.u32); 951 break; 952 953 case 'a': 954 GCVirt = ValueUnion.u64; 955 GCPhysPc = NIL_RTGCPHYS; 956 break; 957 958 case 'p': 959 GCVirt = NIL_RTGCPHYS; 960 GCPhysPc = ValueUnion.u64; 961 break; 962 963 case 'f': 964 fFlags = ValueUnion.u32; 965 break; 966 967 case 'h': 968 pHlp->pfnPrintf(pHlp, 969 "Usage: info %ctlb [options]\n" 970 "\n" 971 "Options:\n" 972 " -c<n>, --cpu=<n>, --vcpu=<n>\n" 973 " Selects the CPU which TBs we're looking at. Default: Caller / 0\n" 974 " -a<virt>, --address=<virt>\n" 975 " Shows the TB for the specified guest virtual address.\n" 976 " -p<phys>, --phys=<phys>, --phys-addr=<phys>\n" 977 " Shows the TB for the specified guest physical address.\n" 978 " -f<flags>,--flags=<flags>\n" 979 " The TB flags value (hex) to use when looking up the TB.\n" 980 "\n" 981 "The default is to use CS:RIP and derive flags from the CPU mode.\n"); 982 return; 983 984 default: 985 pHlp->pfnGetOptError(pHlp, rc, &ValueUnion, &State); 986 return; 987 } 988 } 989 990 /* Currently, only do work on the same EMT. */ 991 if (pVCpu != pVCpuThis) 992 { 993 pHlp->pfnPrintf(pHlp, "TODO: Cross EMT calling not supported yet: targeting %u, caller on %d\n", 994 pVCpu->idCpu, pVCpuThis ? (int)pVCpuThis->idCpu : -1); 995 return; 996 } 997 998 /* 999 * Defaults. 1000 */ 1001 if (GCPhysPc == NIL_RTGCPHYS) 1002 { 1003 if (GCVirt == NIL_RTGCPTR) 1004 GCVirt = CPUMGetGuestFlatPC(pVCpu); 1005 rc = PGMPhysGCPtr2GCPhys(pVCpu, GCVirt, &GCPhysPc); 1006 if (RT_FAILURE(rc)) 1007 { 1008 pHlp->pfnPrintf(pHlp, "Failed to convert %%%RGv to an guest physical address: %Rrc\n", GCVirt, rc); 1009 return; 1010 } 1011 } 1012 if (fFlags == UINT32_MAX) 1013 { 1014 /* Note! This is duplicating code in IEMAllThrdRecompiler. */ 1015 fFlags = iemCalcExecFlags(pVCpu); 1016 if (pVM->cCpus == 1) 1017 fFlags |= IEM_F_X86_DISREGARD_LOCK; 1018 if (CPUMIsInInterruptShadow(&pVCpu->cpum.GstCtx)) 1019 fFlags |= IEMTB_F_INHIBIT_SHADOW; 1020 if (CPUMAreInterruptsInhibitedByNmiEx(&pVCpu->cpum.GstCtx)) 1021 fFlags |= IEMTB_F_INHIBIT_NMI; 1022 if ((IEM_F_MODE_CPUMODE_MASK & fFlags) != IEMMODE_64BIT) 1023 { 1024 int64_t const offFromLim = (int64_t)pVCpu->cpum.GstCtx.cs.u32Limit - (int64_t)pVCpu->cpum.GstCtx.eip; 1025 if (offFromLim < X86_PAGE_SIZE + 16 - (int32_t)(pVCpu->cpum.GstCtx.cs.u64Base & GUEST_PAGE_OFFSET_MASK)) 1026 fFlags |= IEMTB_F_CS_LIM_CHECKS; 1027 } 1028 } 1029 1030 /* 1031 * Do the lookup... 1032 * 1033 * Note! This is also duplicating code in IEMAllThrdRecompiler. We don't 1034 * have much choice since we don't want to increase use counters and 1035 * trigger native recompilation. 1036 */ 1037 fFlags &= IEMTB_F_KEY_MASK; 1038 IEMTBCACHE const * const pTbCache = pVCpu->iem.s.pTbCacheR3; 1039 uint32_t const idxHash = IEMTBCACHE_HASH_NO_KEY_MASK(pTbCache, fFlags, GCPhysPc); 1040 PCIEMTB pTb = IEMTBCACHE_PTR_GET_TB(pTbCache->apHash[idxHash]); 1041 while (pTb) 1042 { 1043 if (pTb->GCPhysPc == GCPhysPc) 1044 { 1045 if ((pTb->fFlags & IEMTB_F_KEY_MASK) == fFlags) 1046 { 1047 /// @todo if (pTb->x86.fAttr == (uint16_t)pVCpu->cpum.GstCtx.cs.Attr.u) 1048 break; 1049 } 1050 } 1051 pTb = pTb->pNext; 1052 } 1053 if (!pTb) 1054 pHlp->pfnPrintf(pHlp, "PC=%RGp fFlags=%#x - no TB found on #%u\n", GCPhysPc, fFlags, pVCpu->idCpu); 1055 else 1056 { 1057 /* 1058 * 1059 */ 1060 switch (pTb->fFlags & IEMTB_F_TYPE_MASK) 1061 { 1062 case IEMTB_F_TYPE_NATIVE: 1063 pHlp->pfnPrintf(pHlp, "PC=%RGp fFlags=%#x on #%u: %p - native\n", GCPhysPc, fFlags, pVCpu->idCpu, pTb); 1064 iemNativeDisassembleTb(pTb, pHlp); 1065 break; 1066 1067 case IEMTB_F_TYPE_THREADED: 1068 pHlp->pfnPrintf(pHlp, "PC=%RGp fFlags=%#x on #%u: %p - threaded\n", GCPhysPc, fFlags, pVCpu->idCpu, pTb); 1069 iemThreadedDisassembleTb(pTb, pHlp); 1070 break; 1071 1072 default: 1073 pHlp->pfnPrintf(pHlp, "PC=%RGp fFlags=%#x on #%u: %p - ??? %#x\n", 1074 GCPhysPc, fFlags, pVCpu->idCpu, pTb, pTb->fFlags); 1075 break; 1076 } 1077 } 1078 } 1079 #endif /* VBOX_WITH_IEM_RECOMPILER && !VBOX_VMM_TARGET_ARMV8 */ 1080 900 1081 901 1082 #ifdef VBOX_WITH_DEBUGGER … … 945 1126 } 946 1127 947 #endif 948 1128 #endif /* VBOX_WITH_DEBUGGER */ 1129 -
trunk/src/VBox/VMM/include/IEMInternal.h
r103514 r103554 5822 5822 void iemTbAllocatorProcessDelayedFrees(PVMCPUCC pVCpu, PIEMTBALLOCATOR pTbAllocator); 5823 5823 void iemTbAllocatorFreeupNativeSpace(PVMCPUCC pVCpu, uint32_t cNeededInstrs); 5824 DECLHIDDEN(const char *) iemTbFlagsToString(uint32_t fFlags, char *pszBuf, size_t cbBuf) RT_NOEXCEPT; 5825 DECLHIDDEN(void) iemThreadedDisassembleTb(PCIEMTB pTb, PCDBGFINFOHLP pHlp) RT_NOEXCEPT; 5824 5826 5825 5827
Note:
See TracChangeset
for help on using the changeset viewer.