Changeset 84552 in vbox for trunk/src/VBox/VMM
- Timestamp:
- May 27, 2020 7:34:40 AM (5 years ago)
- Location:
- trunk/src/VBox/VMM
- Files:
-
- 2 edited
Legend:
- Unmodified
- Added
- Removed
-
trunk/src/VBox/VMM/VMMAll/DBGFAllTracer.cpp
r84494 r84552 288 288 289 289 /** 290 * Handles a I/O port string transfer event. 291 * 292 * @returns VBox status code. 293 * @param pVM The current context VM instance data. 294 * @param pThisCC The event tracer instance current context data. 295 * @param enmTraceEvt The trace event type posted. 296 * @param hEvtSrc The event source for the posted event. 297 * @param hIoPorts The I/O port region handle for the transfer. 298 * @param offPort The offset into the region where the transfer happened. 299 * @param pv The data being transfered. 300 * @param cb Number of bytes of valid data in the buffer. 301 * @param cbItem Item size in bytes. 302 * @param cTransfersReq Number of transfers requested. 303 * @param cTransfersRet Number of transfers done. 304 */ 305 static int dbgfTracerEvtIoPortStr(PVMCC pVM, PDBGFTRACERINSCC pThisCC, DBGFTRACEREVT enmTraceEvt, DBGFTRACEREVTSRC hEvtSrc, 306 uint64_t hIoPorts, RTIOPORT offPort, const void *pv, size_t cb, size_t cbItem, uint32_t cTransfersReq, 307 uint32_t cTransfersRet) 308 { 309 /* Fast path for really small transfers where everything fits into the descriptor. */ 310 DBGFTRACEREVTIOPORTSTR EvtIoPortStr; 311 EvtIoPortStr.hIoPorts = hIoPorts; 312 EvtIoPortStr.cbItem = cbItem; 313 EvtIoPortStr.cTransfersReq = cTransfersReq; 314 EvtIoPortStr.cTransfersRet = cTransfersRet; 315 EvtIoPortStr.offPort = offPort; 316 if (cb <= sizeof(EvtIoPortStr.abData)) 317 { 318 memcpy(&EvtIoPortStr.abData[0], pv, cb); 319 return dbgfTracerEvtPostSingle(pVM, pThisCC, hEvtSrc, enmTraceEvt, &EvtIoPortStr, NULL /*pidEvt*/); 320 } 321 322 /* 323 * Slow path where we have to split the data into multiple entries. 324 * Each one is linked to the previous one by the previous event ID. 325 */ 326 const uint8_t *pbBuf = (const uint8_t *)pv; 327 size_t cbLeft = cb; 328 uint64_t idEvtPrev = 0; 329 memcpy(&EvtIoPortStr.abData[0], pbBuf, sizeof(EvtIoPortStr.abData)); 330 pbBuf += sizeof(EvtIoPortStr.abData); 331 cbLeft -= sizeof(EvtIoPortStr.abData); 332 333 int rc = dbgfTracerEvtPostSingle(pVM, pThisCC, hEvtSrc, enmTraceEvt, &EvtIoPortStr, &idEvtPrev); 334 while ( RT_SUCCESS(rc) 335 && cbLeft) 336 { 337 size_t cbThisXfer = RT_MIN(cbLeft, DBGF_TRACER_EVT_PAYLOAD_SZ); 338 rc = dbgfTracerEvtPostEx(pVM, pThisCC, hEvtSrc, enmTraceEvt, idEvtPrev, 339 pbBuf, cbThisXfer, &idEvtPrev); 340 341 pbBuf += cbThisXfer; 342 cbLeft -= cbThisXfer; 343 } 344 345 return rc; 346 } 347 348 349 /** 290 350 * Registers an MMIO region mapping event for the given event source. 291 351 * … … 490 550 491 551 /** 552 * Registers an I/O region string read event for the given event source. 553 * 554 * @returns VBox status code. 555 * @param pVM The current context VM instance data. 556 * @param hEvtSrc The event source for the posted event. 557 * @param hIoPorts The I/O port region handle being read from. 558 * @param offPort The offset into the region where the read happened. 559 * @param pv The data being read. 560 * @param cb Item size in bytes. 561 * @param cTransfersReq Number of transfers requested. 562 * @param cTransfersRet Number of transfers done. 563 */ 564 VMM_INT_DECL(int) DBGFTracerEvtIoPortReadStr(PVMCC pVM, DBGFTRACEREVTSRC hEvtSrc, uint64_t hIoPorts, RTIOPORT offPort, const void *pv, size_t cb, 565 uint32_t cTransfersReq, uint32_t cTransfersRet) 566 { 567 PDBGFTRACERINSCC pThisCC = dbgfTracerGetInstance(pVM); 568 AssertReturn(pThisCC, VERR_DBGF_TRACER_IPE_1); 569 570 return dbgfTracerEvtIoPortStr(pVM, pThisCC, DBGFTRACEREVT_IOPORT_READ_STR, hEvtSrc, hIoPorts, offPort, pv, cTransfersRet * cb, 571 cb, cTransfersReq, cTransfersRet); 572 } 573 574 575 /** 492 576 * Registers an I/O region write event for the given event source. 493 577 * … … 517 601 518 602 /** 603 * Registers an I/O region string write event for the given event source. 604 * 605 * @returns VBox status code. 606 * @param pVM The current context VM instance data. 607 * @param hEvtSrc The event source for the posted event. 608 * @param hIoPorts The I/O port region handle being written to. 609 * @param offPort The offset into the region where the write happened. 610 * @param pv The data being written. 611 * @param cb Item size in bytes. 612 * @param cTransfersReq Number of transfers requested. 613 * @param cTransfersRet Number of transfers done. 614 */ 615 VMM_INT_DECL(int) DBGFTracerEvtIoPortWriteStr(PVMCC pVM, DBGFTRACEREVTSRC hEvtSrc, uint64_t hIoPorts, RTIOPORT offPort, const void *pv, size_t cb, 616 uint32_t cTransfersReq, uint32_t cTransfersRet) 617 { 618 PDBGFTRACERINSCC pThisCC = dbgfTracerGetInstance(pVM); 619 AssertReturn(pThisCC, VERR_DBGF_TRACER_IPE_1); 620 621 return dbgfTracerEvtIoPortStr(pVM, pThisCC, DBGFTRACEREVT_IOPORT_WRITE_STR, hEvtSrc, hIoPorts, offPort, pv, cTransfersReq * cb, 622 cb, cTransfersReq, cTransfersRet); 623 } 624 625 626 /** 519 627 * Registers an IRQ change event for the given event source. 520 628 * -
trunk/src/VBox/VMM/VMMR3/DBGFR3Tracer.cpp
r84494 r84552 185 185 static const RTTRACELOGEVTITEMDESC g_DevIoPortRwEvtItems[] = 186 186 { 187 {"hIoPorts", "The MMIO region handle being unmapped",RTTRACELOGTYPE_UINT64, 0},187 {"hIoPorts", "The I/O region handle being accessed", RTTRACELOGTYPE_UINT64, 0}, 188 188 {"offPort", "The offset in the I/O port region being accessed", RTTRACELOGTYPE_UINT16, 0}, 189 189 {"cbXfer", "Number of bytes being transfered", RTTRACELOGTYPE_UINT64, 0}, … … 210 210 211 211 212 static const RTTRACELOGEVTITEMDESC g_DevIoPortRwStrEvtItems[] = 213 { 214 {"hIoPorts", "The I/O region handle being accesses", RTTRACELOGTYPE_UINT64, 0}, 215 {"offPort", "The offset in the I/O port region being accessed", RTTRACELOGTYPE_UINT16, 0}, 216 {"cbItem", "Item size for the access", RTTRACELOGTYPE_UINT32, 0}, 217 {"cTransfersReq", "Number of transfers requested by the guest", RTTRACELOGTYPE_UINT32, 0}, 218 {"cTransfersRet", "Number of transfers executed by the device", RTTRACELOGTYPE_UINT32, 0} 219 }; 220 221 static const RTTRACELOGEVTDESC g_DevIoPortReadStrEvtDesc = 222 { 223 "Dev.IoPortReadStr", 224 "I/O port region of a device is being read using REP INS", 225 RTTRACELOGEVTSEVERITY_DEBUG, 226 RT_ELEMENTS(g_DevIoPortRwStrEvtItems), 227 &g_DevIoPortRwStrEvtItems[0] 228 }; 229 230 static const RTTRACELOGEVTDESC g_DevIoPortWriteStrEvtDesc = 231 { 232 "Dev.IoPortWriteStr", 233 "I/O port region of a device is being written using REP OUTS", 234 RTTRACELOGEVTSEVERITY_DEBUG, 235 RT_ELEMENTS(g_DevIoPortRwStrEvtItems), 236 &g_DevIoPortRwStrEvtItems[0] 237 }; 238 239 212 240 static const RTTRACELOGEVTITEMDESC g_DevIrqEvtItems[] = 213 241 { … … 269 297 270 298 271 static const RTTRACELOGEVTITEMDESC g_Dev GCPhysRwDataEvtItems[] =299 static const RTTRACELOGEVTITEMDESC g_DevRwDataEvtItems[] = 272 300 { 273 301 {"abData", "The data being read/written", RTTRACELOGTYPE_RAWDATA, 0} 274 302 }; 275 303 276 static const RTTRACELOGEVTDESC g_Dev GCPhysRwDataEvtDesc =277 { 278 "Dev. GCPhysRwData",304 static const RTTRACELOGEVTDESC g_DevRwDataEvtDesc = 305 { 306 "Dev.RwData", 279 307 "The data being read or written", 280 308 RTTRACELOGEVTSEVERITY_DEBUG, 281 RT_ELEMENTS(g_Dev GCPhysRwDataEvtItems),282 &g_Dev GCPhysRwDataEvtItems[0]309 RT_ELEMENTS(g_DevRwDataEvtItems), 310 &g_DevRwDataEvtItems[0] 283 311 }; 284 312 … … 295 323 * @param pThis The DBGF tracer instance. 296 324 */ 297 static PDBGFTRACERGCPHYSRWAGG dbgfTracerR3Evt GCPhysRwAggNew(PDBGFTRACERINSR3 pThis)325 static PDBGFTRACERGCPHYSRWAGG dbgfTracerR3EvtRwAggNew(PDBGFTRACERINSR3 pThis) 298 326 { 299 327 for (uint32_t i = 0; i < RT_ELEMENTS(pThis->aGstMemRwData); i++) … … 314 342 * @param idEvtPrev The event ID to look for. 315 343 */ 316 static PDBGFTRACERGCPHYSRWAGG dbgfTracerR3Evt GCPhysRwAggFind(PDBGFTRACERINSR3 pThis, uint64_t idEvtPrev)344 static PDBGFTRACERGCPHYSRWAGG dbgfTracerR3EvtRwAggFind(PDBGFTRACERINSR3 pThis, uint64_t idEvtPrev) 317 345 { 318 346 for (uint32_t i = 0; i < RT_ELEMENTS(pThis->aGstMemRwData); i++) … … 324 352 325 353 return NULL; 354 } 355 356 357 /** 358 * Common code for the guest memory and string I/O port read/write events. 359 * 360 * @returns VBox status code. 361 * @param pThis The DBGF tracer instance. 362 * @param pEvtHdr The event header. 363 * @param cbXfer Overall number of bytes of data for this event. 364 * @param pvData Initial data supplied in the event starting the aggregation. 365 * @param cbData Number of initial bytes of data. 366 */ 367 static int dbgfTracerR3EvtRwStartCommon(PDBGFTRACERINSR3 pThis, PCDBGFTRACEREVTHDR pEvtHdr, size_t cbXfer, const void *pvData, size_t cbData) 368 { 369 /* Slow path, find an empty aggregation structure. */ 370 int rc = VINF_SUCCESS; 371 PDBGFTRACERGCPHYSRWAGG pDataAgg = dbgfTracerR3EvtRwAggNew(pThis); 372 if (RT_LIKELY(pDataAgg)) 373 { 374 /* Initialize it. */ 375 pDataAgg->idEvtStart = pEvtHdr->idEvt; 376 pDataAgg->idEvtPrev = pEvtHdr->idEvt; 377 pDataAgg->cbXfer = cbXfer; 378 pDataAgg->cbLeft = pDataAgg->cbXfer; 379 pDataAgg->offBuf = 0; 380 381 /* Need to reallocate the buffer to hold the complete data? */ 382 if (RT_UNLIKELY(pDataAgg->cbBufMax < pDataAgg->cbXfer)) 383 { 384 uint8_t *pbBufNew = (uint8_t *)RTMemRealloc(pDataAgg->pbBuf, pDataAgg->cbXfer); 385 if (RT_LIKELY(pbBufNew)) 386 { 387 pDataAgg->pbBuf = pbBufNew; 388 pDataAgg->cbBufMax = pDataAgg->cbXfer; 389 } 390 else 391 rc = VERR_NO_MEMORY; 392 } 393 394 if (RT_SUCCESS(rc)) 395 { 396 memcpy(pDataAgg->pbBuf, pvData, cbData); 397 pDataAgg->offBuf += cbData; 398 pDataAgg->cbLeft -= cbData; 399 } 400 } 401 else 402 rc = VERR_NO_MEMORY; 403 404 if (RT_FAILURE(rc)) 405 { 406 LogRelMax(10, ("DBGF: Creating new data aggregation structure for memory read/write failed with %Rrc, trace log will not contain data for this event!\n", rc)); 407 408 /* Write out the finish event without any data. */ 409 size_t cbEvtData = 0; 410 rc = RTTraceLogWrEvtAdd(pThis->hTraceLog, &g_DevRwDataEvtDesc, RTTRACELOG_WR_ADD_EVT_F_GRP_FINISH, 411 pEvtHdr->idEvt, pEvtHdr->hEvtSrc, NULL, &cbEvtData); 412 if (pDataAgg) /* Reset the aggregation event. */ 413 pDataAgg->idEvtStart = DBGF_TRACER_EVT_HDR_ID_INVALID; 414 } 415 416 return rc; 326 417 } 327 418 … … 352 443 size_t cbEvtData = pEvtGCPhysRw->cbXfer; 353 444 354 rc = RTTraceLogWrEvtAdd(pThis->hTraceLog, &g_Dev GCPhysRwDataEvtDesc, RTTRACELOG_WR_ADD_EVT_F_GRP_FINISH,445 rc = RTTraceLogWrEvtAdd(pThis->hTraceLog, &g_DevRwDataEvtDesc, RTTRACELOG_WR_ADD_EVT_F_GRP_FINISH, 355 446 pEvtHdr->idEvt, pEvtHdr->hEvtSrc, &pEvtGCPhysRw->abData[0], &cbEvtData); 356 447 } 357 448 else 358 { 359 /* Slow path, find an empty aggregation structure. */ 360 PDBGFTRACERGCPHYSRWAGG pDataAgg = dbgfTracerR3EvtGCPhysRwAggNew(pThis); 361 if (RT_LIKELY(pDataAgg)) 362 { 363 /* Initialize it. */ 364 pDataAgg->idEvtStart = pEvtHdr->idEvt; 365 pDataAgg->idEvtPrev = pEvtHdr->idEvt; 366 pDataAgg->cbXfer = pEvtGCPhysRw->cbXfer; 367 pDataAgg->cbLeft = pDataAgg->cbXfer; 368 pDataAgg->offBuf = 0; 369 370 /* Need to reallocate the buffer to hold the complete data? */ 371 if (RT_UNLIKELY(pDataAgg->cbBufMax < pDataAgg->cbXfer)) 372 { 373 uint8_t *pbBufNew = (uint8_t *)RTMemRealloc(pDataAgg->pbBuf, pDataAgg->cbXfer); 374 if (RT_LIKELY(pbBufNew)) 375 { 376 pDataAgg->pbBuf = pbBufNew; 377 pDataAgg->cbBufMax = pDataAgg->cbXfer; 378 } 379 else 380 rc = VERR_NO_MEMORY; 381 } 382 383 if (RT_SUCCESS(rc)) 384 { 385 memcpy(pDataAgg->pbBuf, &pEvtGCPhysRw->abData[0], sizeof(pEvtGCPhysRw->abData)); 386 pDataAgg->offBuf += sizeof(pEvtGCPhysRw->abData); 387 pDataAgg->cbLeft -= sizeof(pEvtGCPhysRw->abData); 388 } 389 } 390 else 391 rc = VERR_NO_MEMORY; 392 393 if (RT_FAILURE(rc)) 394 { 395 LogRelMax(10, ("DBGF: Creating new data aggregation structure for guest memory read/write failed with %Rrc, trace log will not contain data for this event!\n", rc)); 396 397 /* Write out the finish event without any data. */ 398 size_t cbEvtData = 0; 399 rc = RTTraceLogWrEvtAdd(pThis->hTraceLog, &g_DevGCPhysRwDataEvtDesc, RTTRACELOG_WR_ADD_EVT_F_GRP_FINISH, 400 pEvtHdr->idEvt, pEvtHdr->hEvtSrc, NULL, &cbEvtData); 401 if (pDataAgg) /* Reset the aggregation event. */ 402 pDataAgg->idEvtStart = DBGF_TRACER_EVT_HDR_ID_INVALID; 403 } 404 } 449 rc = dbgfTracerR3EvtRwStartCommon(pThis, pEvtHdr, pEvtGCPhysRw->cbXfer, &pEvtGCPhysRw->abData[0], sizeof(pEvtGCPhysRw->abData)); 405 450 } 406 451 … … 410 455 411 456 /** 412 * Continues a previously started guest memory read/write event. 457 * Starts a new I/O port string read/write event. 458 * 459 * @returns VBox status code. 460 * @param pThis The DBGF tracer instance. 461 * @param pEvtHdr The event header. 462 * @param pEvtIoPortStrRw The I/O port string read/write event descriptor. 463 * @param cbXfer Number of bytes of valid data for this event. 464 * @param pEvtDesc The event descriptor written to the trace log. 465 */ 466 static int dbgfTracerR3EvtIoPortStrRwStart(PDBGFTRACERINSR3 pThis, PCDBGFTRACEREVTHDR pEvtHdr, 467 PCDBGFTRACEREVTIOPORTSTR pEvtIoPortStrRw, size_t cbXfer, 468 PCRTTRACELOGEVTDESC pEvtDesc) 469 { 470 /* Write out the event header first in any case. */ 471 int rc = RTTraceLogWrEvtAddL(pThis->hTraceLog, pEvtDesc, RTTRACELOG_WR_ADD_EVT_F_GRP_START, 472 pEvtHdr->idEvt, pEvtHdr->hEvtSrc, pEvtIoPortStrRw->hIoPorts, pEvtIoPortStrRw->offPort, 473 pEvtIoPortStrRw->cbItem, pEvtIoPortStrRw->cTransfersReq, pEvtIoPortStrRw->cTransfersRet); 474 if (RT_SUCCESS(rc)) 475 { 476 /* 477 * If the amount of data is small enough to fit into the single event descriptor we can skip allocating 478 * an aggregation tracking structure and write the event containing the complete data out immediately. 479 */ 480 if (cbXfer <= sizeof(pEvtIoPortStrRw->abData)) 481 { 482 size_t cbEvtData = cbXfer; 483 484 rc = RTTraceLogWrEvtAdd(pThis->hTraceLog, &g_DevRwDataEvtDesc, RTTRACELOG_WR_ADD_EVT_F_GRP_FINISH, 485 pEvtHdr->idEvt, pEvtHdr->hEvtSrc, &pEvtIoPortStrRw->abData[0], &cbEvtData); 486 } 487 else 488 rc = dbgfTracerR3EvtRwStartCommon(pThis, pEvtHdr, cbXfer, &pEvtIoPortStrRw->abData[0], sizeof(pEvtIoPortStrRw->abData)); 489 } 490 491 return rc; 492 } 493 494 495 /** 496 * Continues a previously started guest memory or string I/O port read/write event. 413 497 * 414 498 * @returns VBox status code. … … 417 501 * @param pvData The data to log. 418 502 */ 419 static int dbgfTracerR3Evt GCPhysRwContinue(PDBGFTRACERINSR3 pThis, PCDBGFTRACEREVTHDR pEvtHdr, void *pvData)503 static int dbgfTracerR3EvtRwContinue(PDBGFTRACERINSR3 pThis, PCDBGFTRACEREVTHDR pEvtHdr, void *pvData) 420 504 { 421 505 int rc = VINF_SUCCESS; 422 PDBGFTRACERGCPHYSRWAGG pDataAgg = dbgfTracerR3Evt GCPhysRwAggFind(pThis, pEvtHdr->idEvtPrev);506 PDBGFTRACERGCPHYSRWAGG pDataAgg = dbgfTracerR3EvtRwAggFind(pThis, pEvtHdr->idEvtPrev); 423 507 424 508 if (RT_LIKELY(pDataAgg)) … … 433 517 { 434 518 /* All data aggregated, write it out and reset the structure. */ 435 rc = RTTraceLogWrEvtAdd(pThis->hTraceLog, &g_Dev GCPhysRwDataEvtDesc, RTTRACELOG_WR_ADD_EVT_F_GRP_FINISH,519 rc = RTTraceLogWrEvtAdd(pThis->hTraceLog, &g_DevRwDataEvtDesc, RTTRACELOG_WR_ADD_EVT_F_GRP_FINISH, 436 520 pDataAgg->idEvtStart, pEvtHdr->hEvtSrc, pDataAgg->pbBuf, &pDataAgg->cbXfer); 437 521 pDataAgg->offBuf = 0; … … 543 627 pEvtHdr->idEvt, pEvtHdr->hEvtSrc, pEvtIoPortRw->hIoPorts, pEvtIoPortRw->offPort, 544 628 pEvtIoPortRw->cbXfer, pEvtIoPortRw->u32Val); 629 break; 630 } 631 case DBGFTRACEREVT_IOPORT_READ_STR: 632 case DBGFTRACEREVT_IOPORT_WRITE_STR: 633 { 634 PCRTTRACELOGEVTDESC pEvtDesc = pEvtHdr->enmEvt == DBGFTRACEREVT_IOPORT_WRITE_STR 635 ? &g_DevIoPortWriteStrEvtDesc 636 : &g_DevIoPortReadStrEvtDesc; 637 638 /* If the previous event ID is invalid this starts a new read/write we have to aggregate all the data for. */ 639 if (pEvtHdr->idEvtPrev == DBGF_TRACER_EVT_HDR_ID_INVALID) 640 { 641 PCDBGFTRACEREVTIOPORTSTR pEvtIoPortStrRw = (PCDBGFTRACEREVTIOPORTSTR)(pEvtHdr + 1); 642 size_t cbXfer = pEvtHdr->enmEvt == DBGFTRACEREVT_IOPORT_WRITE_STR 643 ? pEvtIoPortStrRw->cTransfersReq * pEvtIoPortStrRw->cbItem 644 : pEvtIoPortStrRw->cTransfersRet * pEvtIoPortStrRw->cbItem; 645 646 rc = dbgfTracerR3EvtIoPortStrRwStart(pThis, pEvtHdr, pEvtIoPortStrRw, cbXfer, pEvtDesc); 647 } 648 else 649 { 650 /* Continuation of a started read or write, look up the right tracking structure and process the new data. */ 651 void *pvData = pEvtHdr + 1; 652 rc = dbgfTracerR3EvtRwContinue(pThis, pEvtHdr, pvData); 653 } 545 654 break; 546 655 } … … 578 687 /* Continuation of a started read or write, look up the right tracking structure and process the new data. */ 579 688 void *pvData = pEvtHdr + 1; 580 rc = dbgfTracerR3Evt GCPhysRwContinue(pThis, pEvtHdr, pvData);689 rc = dbgfTracerR3EvtRwContinue(pThis, pEvtHdr, pvData); 581 690 } 582 691 break;
Note:
See TracChangeset
for help on using the changeset viewer.