VirtualBox

Changeset 41786 in vbox


Ignore:
Timestamp:
Jun 16, 2012 7:58:22 PM (13 years ago)
Author:
vboxsync
Message:

DIS: Forgotten FNDISPARSE change in previous commit + DISInstWithPrefetchedBytes.

Location:
trunk
Files:
2 edited

Legend:

Unmodified
Added
Removed
  • trunk/include/VBox/dis.h

    r41760 r41786  
    520520
    521521/** Parser callback.
    522  * @remark no DECLCALLBACK() here because it's considered to be internal (really, I'm too lazy to update all the functions). */
    523 typedef unsigned FNDISPARSE(RTUINTPTR pu8CodeBlock, PCDISOPCODE pOp, PDISOPPARAM pParam, PDISCPUSTATE pCpu);
     522 * @remark no DECLCALLBACK() here because it's considered to be internal and
     523 *         there is no point in enforcing CDECL. */
     524typedef size_t FNDISPARSE(size_t offInstr, PCDISOPCODE pOp, PDISOPPARAM pParam, PDISCPUSTATE pCpu);
    524525/** Pointer to a disassembler parser function. */
    525526typedef FNDISPARSE *PFNDISPARSE;
     
    657658                       PFNDISREADBYTES pfnReadBytes, void *pvUser,
    658659                       PDISCPUSTATE pCpu, uint32_t *pcbInstr);
     660DISDECL(int) DISInstWithPrefetchedBytes(RTUINTPTR uInstrAddr, DISCPUMODE enmCpuMode, uint32_t fFilter,
     661                                        void const *pvPrefetched, size_t cbPretched,
     662                                        PFNDISREADBYTES pfnReadBytes, void *pvUser,
     663                                        PDISCPUSTATE pCpu, uint32_t *pcbInstr);
    659664
    660665DISDECL(int)        DISGetParamSize(PDISCPUSTATE pCpu, PDISOPPARAM pParam);
  • trunk/src/VBox/Disassembler/DisasmCore.cpp

    r41784 r41786  
    26422642
    26432643/**
     2644 * Inlined worker that initializes the disassembler state.
     2645 *
     2646 * @returns The primary opcode map to use.
     2647 * @param   pCpu            The disassembler state.
     2648 * @param   uInstrAddr      The instruction address.
     2649 * @param   enmCpuMode      The CPU mode.
     2650 * @param   fFilter         The instruction filter settings.
     2651 * @param   pfnReadBytes    The byte reader, can be NULL.
     2652 * @param   pvUser          The the user data for the reader.
     2653 */
     2654DECL_FORCE_INLINE(PCDISOPCODE)
     2655disInitializeState(PDISCPUSTATE pCpu, RTUINTPTR uInstrAddr, DISCPUMODE enmCpuMode, uint32_t fFilter,
     2656                   PFNDISREADBYTES pfnReadBytes, void *pvUser)
     2657{
     2658
     2659    /*
     2660     * Initialize the CPU state.
     2661     * Note! The RT_BZERO make ASSUMPTIONS about the placement of pvUser2.
     2662     */
     2663    RT_BZERO(pCpu, RT_OFFSETOF(DISCPUSTATE, pvUser2));
     2664
     2665#ifdef VBOX_STRICT /* poison */
     2666    pCpu->Param1.Base.idxGenReg  = 0xc1;
     2667    pCpu->Param2.Base.idxGenReg  = 0xc2;
     2668    pCpu->Param3.Base.idxGenReg  = 0xc3;
     2669    pCpu->Param1.Index.idxGenReg = 0xc4;
     2670    pCpu->Param2.Index.idxGenReg = 0xc5;
     2671    pCpu->Param3.Index.idxGenReg = 0xc6;
     2672    pCpu->Param1.uDisp.u64 = UINT64_C(0xd1d1d1d1d1d1d1d1);
     2673    pCpu->Param2.uDisp.u64 = UINT64_C(0xd2d2d2d2d2d2d2d2);
     2674    pCpu->Param3.uDisp.u64 = UINT64_C(0xd3d3d3d3d3d3d3d3);
     2675    pCpu->Param1.uValue    = UINT64_C(0xb1b1b1b1b1b1b1b1);
     2676    pCpu->Param2.uValue    = UINT64_C(0xb2b2b2b2b2b2b2b2);
     2677    pCpu->Param3.uValue    = UINT64_C(0xb3b3b3b3b3b3b3b3);
     2678    pCpu->Param1.uScale    = 28;
     2679    pCpu->Param2.uScale    = 29;
     2680    pCpu->Param3.uScale    = 30;
     2681#endif
     2682
     2683    pCpu->fPrefix           = DISPREFIX_NONE;
     2684    pCpu->idxSegPrefix      = DISSELREG_DS;
     2685    pCpu->rc                = VINF_SUCCESS;
     2686    pCpu->pfnDisasmFnTable  = g_apfnFullDisasm;
     2687
     2688    pCpu->uInstrAddr        = uInstrAddr;
     2689    pCpu->fFilter           = fFilter;
     2690    pCpu->pfnReadBytes      = pfnReadBytes ? pfnReadBytes : disReadBytesDefault;
     2691    pCpu->pvUser            = pvUser;
     2692    pCpu->uCpuMode          = enmCpuMode;
     2693    PCDISOPCODE paOneByteMap;
     2694    if (enmCpuMode == DISCPUMODE_64BIT)
     2695    {
     2696        pCpu->uAddrMode     = DISCPUMODE_64BIT;
     2697        pCpu->uOpMode       = DISCPUMODE_32BIT;
     2698        paOneByteMap        = g_aOneByteMapX64;
     2699    }
     2700    else
     2701    {
     2702        pCpu->uAddrMode     = enmCpuMode;
     2703        pCpu->uOpMode       = enmCpuMode;
     2704        paOneByteMap        = g_aOneByteMapX86;
     2705    }
     2706    return paOneByteMap;
     2707}
     2708
     2709
     2710/**
     2711 * Reads some bytes into the cache.
     2712 *
     2713 * While this will set DISCPUSTATE::rc on failure, the caller should disregard
     2714 * this since that is what would happen if we didn't prefetch bytes prior to the
     2715 * instruction parsing.
     2716 *
     2717 * @param   pCpu                The disassembler state.
     2718 */
     2719DECL_FORCE_INLINE(void) disPrefetchBytes(PDISCPUSTATE pCpu)
     2720{
     2721    /*
     2722     * Read some bytes into the cache.  (If this fail we continue as nothing
     2723     * has gone wrong since this is what would happen if we didn't precharge
     2724     * the cache here.)
     2725     */
     2726    int rc = pCpu->pfnReadBytes(pCpu, 0, 1, sizeof(pCpu->abInstr));
     2727    if (RT_SUCCESS(rc))
     2728    {
     2729        Assert(pCpu->cbCachedInstr >= 1);
     2730        Assert(pCpu->cbCachedInstr <= sizeof(pCpu->abInstr));
     2731    }
     2732    else
     2733    {
     2734        Log(("Initial read failed with rc=%Rrc!!\n", rc));
     2735        pCpu->rc = VERR_DIS_MEM_READ;
     2736    }
     2737}
     2738
     2739
     2740/**
    26442741 * Disassembles on instruction, details in @a pCpu and length in @a pcbInstr.
    26452742 *
     
    26622759                       PDISCPUSTATE pCpu, uint32_t *pcbInstr)
    26632760{
    2664     PCDISOPCODE paOneByteMap;
    2665 
    2666     /*
    2667      * Initialize the CPU state.
    2668      * Note! The RT_BZERO make ASSUMPTIONS about the placement of pvUser2.
    2669      */
    2670     RT_BZERO(pCpu, RT_OFFSETOF(DISCPUSTATE, pvUser2));
    2671 
    2672 #ifdef VBOX_STRICT /* poison */
    2673     pCpu->Param1.Base.idxGenReg  = 0xc1;
    2674     pCpu->Param2.Base.idxGenReg  = 0xc2;
    2675     pCpu->Param3.Base.idxGenReg  = 0xc3;
    2676     pCpu->Param1.Index.idxGenReg = 0xc4;
    2677     pCpu->Param2.Index.idxGenReg = 0xc5;
    2678     pCpu->Param3.Index.idxGenReg = 0xc6;
    2679     pCpu->Param1.uDisp.u64 = UINT64_C(0xd1d1d1d1d1d1d1d1);
    2680     pCpu->Param2.uDisp.u64 = UINT64_C(0xd2d2d2d2d2d2d2d2);
    2681     pCpu->Param3.uDisp.u64 = UINT64_C(0xd3d3d3d3d3d3d3d3);
    2682     pCpu->Param1.uValue    = UINT64_C(0xb1b1b1b1b1b1b1b1);
    2683     pCpu->Param2.uValue    = UINT64_C(0xb2b2b2b2b2b2b2b2);
    2684     pCpu->Param3.uValue    = UINT64_C(0xb3b3b3b3b3b3b3b3);
    2685     pCpu->Param1.uScale    = 28;
    2686     pCpu->Param2.uScale    = 29;
    2687     pCpu->Param3.uScale    = 30;
    2688 #endif
    2689 
    2690     pCpu->uCpuMode          = enmCpuMode;
    2691     if (enmCpuMode == DISCPUMODE_64BIT)
    2692     {
    2693         paOneByteMap        = g_aOneByteMapX64;
    2694         pCpu->uAddrMode     = DISCPUMODE_64BIT;
    2695         pCpu->uOpMode       = DISCPUMODE_32BIT;
    2696     }
    2697     else
    2698     {
    2699         paOneByteMap        = g_aOneByteMapX86;
    2700         pCpu->uAddrMode     = enmCpuMode;
    2701         pCpu->uOpMode       = enmCpuMode;
    2702     }
    2703     pCpu->fPrefix           = DISPREFIX_NONE;
    2704     pCpu->idxSegPrefix      = DISSELREG_DS;
    2705     pCpu->uInstrAddr        = uInstrAddr;
    2706     pCpu->pfnDisasmFnTable  = g_apfnFullDisasm;
    2707     pCpu->fFilter           = fFilter;
    2708     pCpu->rc                = VINF_SUCCESS;
    2709     pCpu->pfnReadBytes      = pfnReadBytes ? pfnReadBytes : disReadBytesDefault;
    2710     pCpu->pvUser            = pvUser;
    2711 
    2712     /*
    2713      * Read some bytes into the cache.  (If this fail we continue as nothing
    2714      * has gone wrong since this is what would happen if we didn't precharge
    2715      * the cache here.)
    2716      */
    2717     int rc = pCpu->pfnReadBytes(pCpu, 0, 1, sizeof(pCpu->abInstr));
    2718     if (RT_SUCCESS(rc))
    2719     {
    2720         Assert(pCpu->cbCachedInstr >= 1);
    2721         Assert(pCpu->cbCachedInstr <= sizeof(pCpu->abInstr));
    2722     }
    2723     else
    2724     {
    2725         Log(("Initial read failed with rc=%Rrc!!\n", rc));
    2726         pCpu->rc = VERR_DIS_MEM_READ;
    2727     }
    2728 
     2761
     2762    PCDISOPCODE paOneByteMap = disInitializeState(pCpu, uInstrAddr, enmCpuMode, fFilter, pfnReadBytes, pvUser);
     2763    disPrefetchBytes(pCpu);
    27292764    return disInstrWorker(pCpu, paOneByteMap, pcbInstr);
    27302765}
     2766
     2767
     2768/**
     2769 * Disassembles on instruction partially or fully from prefetched bytes, details
     2770 * in @a pCpu and length in @a pcbInstr.
     2771 *
     2772 * @returns VBox status code.
     2773 * @param   uInstrAddr      Address of the instruction to decode. What this means
     2774 *                          is left to the pfnReadBytes function.
     2775 * @param   enmCpuMode      The CPU mode. DISCPUMODE_32BIT, DISCPUMODE_16BIT, or DISCPUMODE_64BIT.
     2776 * @param   pvPrefetched    Pointer to the prefetched bytes.
     2777 * @param   cbPrefetched    The number of valid bytes pointed to by @a
     2778 *                          pbPrefetched.
     2779 * @param   pfnReadBytes    Callback for reading instruction bytes.
     2780 * @param   fFilter         Instruction type filter.
     2781 * @param   pvUser          User argument for the instruction reader. (Ends up in pvUser.)
     2782 * @param   pCpu            Pointer to CPU structure. With the exception of
     2783 *                          DISCPUSTATE::pvUser2, the structure will be
     2784 *                          completely initialized by this API, i.e. no input is
     2785 *                          taken from it.
     2786 * @param   pcbInstr        Where to store the size of the instruction.  (This
     2787 *                          is also stored in PDISCPUSTATE::cbInstr.)  Optional.
     2788 */
     2789DISDECL(int) DISInstWithPrefetchedBytes(RTUINTPTR uInstrAddr, DISCPUMODE enmCpuMode, uint32_t fFilter,
     2790                                        void const *pvPrefetched, size_t cbPretched,
     2791                                        PFNDISREADBYTES pfnReadBytes, void *pvUser,
     2792                                        PDISCPUSTATE pCpu, uint32_t *pcbInstr)
     2793{
     2794    PCDISOPCODE paOneByteMap = disInitializeState(pCpu, uInstrAddr, enmCpuMode, fFilter, pfnReadBytes, pvUser);
     2795
     2796    if (!cbPretched)
     2797        disPrefetchBytes(pCpu);
     2798    else
     2799    {
     2800        if (cbPretched >= sizeof(pCpu->abInstr))
     2801        {
     2802            memcpy(pCpu->abInstr, pvPrefetched, sizeof(pCpu->abInstr));
     2803            pCpu->cbCachedInstr = (uint8_t)sizeof(pCpu->abInstr);
     2804        }
     2805        else
     2806        {
     2807            memcpy(pCpu->abInstr, pvPrefetched, cbPretched);
     2808            pCpu->cbCachedInstr = (uint8_t)cbPretched;
     2809        }
     2810    }
     2811
     2812    return disInstrWorker(pCpu, paOneByteMap, pcbInstr);
     2813}
     2814
    27312815
    27322816
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