1289 lines
52 KiB
C
Raw Permalink Normal View History

2021-07-24 21:11:47 -07:00
/*
TODO: add option to null out drawprim calls.
Also maybe hook the PIX_ENABLE stuff to Telemetry
In imaterialsystem.h:
#define PIX_ENABLE 1 // set this to 1 and build engine/studiorender to enable pix events in the engine
And in shaderapidx8.h:
#define PIX_ENABLE 1 // set this to 1 and build engine/studiorender to enable pix events in the engine
Might be interesting to make it so dx9hook.h paid attention to the PIX_ENABLE names
and allowed you to filter drawprim calls based on those?
*/
#ifndef _DX9HOOK_H_
#define _DX9HOOK_H_
#if D3D_BATCH_PERF_ANALYSIS
#include "../../thirdparty/miniz/miniz.c"
#include "../../thirdparty/miniz/simple_bitmap.h"
extern ConVar d3d_batch_vis, d3d_batch_vis_abs_scale, d3d_batch_vis_y_scale, d3d_present_vis_abs_scale;
extern uint64 g_nTotalD3DCalls, g_nTotalD3DCycles;
class CD3DCallTimer
{
public:
inline CD3DCallTimer() { g_nTotalD3DCalls++; g_nTotalD3DCycles -= TM_FAST_TIME(); }
inline ~CD3DCallTimer() { g_nTotalD3DCycles += TM_FAST_TIME(); }
};
#define D3D_BATCH_PERF(...) __VA_ARGS__
#else
#define D3D_BATCH_PERF(...)
#endif
#if D3D_BATCH_PERF_ANALYSIS
#define XXX \
tmZone( TELEMETRY_LEVEL3, TMZF_NONE, "D3D9: %s", __FUNCTION__ ); \
CD3DCallTimer scopedCallTimer;
#else
#define XXX \
if( ThreadInMainThread() ) \
{ \
tmMessage( TELEMETRY_LEVEL0, TMMF_ICON_NOTE | TMMF_SEVERITY_WARNING, "(dota/d3d)%s", __FUNCTION__ ); \
tmZoneFiltered( TELEMETRY_LEVEL0, 50, TMZF_NONE, "%s", __FUNCTION__ ); \
}
#endif
// Hooks for routines which return values.
#define _DOCALL0( _member ) ( m_Data.pHWObj->_member )()
#define _DOCALL( _member, ... ) ( m_Data.pHWObj->_member )( __VA_ARGS__ )
// And hooks for routines which return squatola.
#define _DOCALL0_NORET( _member ) ( m_Data.pHWObj->_member )()
#define _DOCALL_NORET( _member, ... ) ( m_Data.pHWObj->_member )( __VA_ARGS__)
#define DEF_HOOKCLASSES( X ) \
template<> class CDx9HookBase< struct I ## X > \
{ \
public: \
typedef I ## X _D3DINTERFACE; \
typedef class C ## X ## Hook _HOOKCLASS; \
\
public: \
CDx9HookBase() { memset( &m_Data, 0, sizeof( m_Data ) ); } \
virtual ~CDx9HookBase() {} \
\
public: \
struct DATA \
{ \
class CDirect3DDevice9Hook *pDevice; \
_D3DINTERFACE *pHWObj; \
}; \
\
DATA m_Data; \
}
template < class T > class CDx9HookBase;
DEF_HOOKCLASSES( Direct3DVertexDeclaration9 );
DEF_HOOKCLASSES( Direct3DPixelShader9 );
DEF_HOOKCLASSES( Direct3DVertexShader9 );
DEF_HOOKCLASSES( Direct3DVertexBuffer9 );
DEF_HOOKCLASSES( Direct3DIndexBuffer9 );
DEF_HOOKCLASSES( Direct3DQuery9 );
DEF_HOOKCLASSES( Direct3DStateBlock9 );
DEF_HOOKCLASSES( Direct3DSurface9 );
DEF_HOOKCLASSES( Direct3DBaseTexture9 );
DEF_HOOKCLASSES( Direct3DTexture9 );
DEF_HOOKCLASSES( Direct3DCubeTexture9 );
DEF_HOOKCLASSES( Direct3DVolume9 );
DEF_HOOKCLASSES( Direct3DVolumeTexture9 );
DEF_HOOKCLASSES( Direct3DSwapChain9 );
DEF_HOOKCLASSES( Direct3D9 );
DEF_HOOKCLASSES( Direct3DDevice9 );
template < class _D3DINTERFACE > HRESULT AllocOverride( HRESULT *hr, class CDirect3DDevice9Hook *pDevice, _D3DINTERFACE **ppHWObj )
{
if( SUCCEEDED(*hr) && ppHWObj && *ppHWObj )
{
CDx9HookBase< _D3DINTERFACE >::_HOOKCLASS *pClass = new CDx9HookBase< _D3DINTERFACE >::_HOOKCLASS;
if(!pClass)
{
( *ppHWObj )->Release();
*hr = E_OUTOFMEMORY;
return NULL;
}
pClass->m_Data.pDevice = pDevice;
pClass->m_Data.pHWObj = *ppHWObj;
*ppHWObj = pClass;
return *hr;
}
return *hr;
}
template <class _D3DINTERFACE> _D3DINTERFACE *GetHWPtr( _D3DINTERFACE *pD3DInterface )
{
if( pD3DInterface )
{
CDx9HookBase< _D3DINTERFACE >::_HOOKCLASS *pClass = ( CDx9HookBase< _D3DINTERFACE >::_HOOKCLASS * )pD3DInterface;
return pClass->m_Data.pHWObj;
}
return NULL;
}
template <class _D3DINTERFACE> CDirect3DDevice9Hook *GetHookDevice( _D3DINTERFACE *pD3DInterface )
{
if( pD3DInterface )
{
CDx9HookBase< _D3DINTERFACE >::_HOOKCLASS *pClass = ( CDx9HookBase< _D3DINTERFACE >::_HOOKCLASS * )pD3DInterface;
return pClass->m_Data.pDevice;
}
return NULL;
}
//$ TODO: if(riid == IID_IDirect3DDevice9Ex, IID_IDirect3DDevice9, etc.
#define IMPL_QUERYINTERFACE() \
STDMETHOD(QueryInterface)(THIS_ REFIID riid, void** ppvObj) \
{ \
__debugbreak(); \
XXX; return _DOCALL(QueryInterface, riid, ppvObj); \
}
#define IMPL_IUNKOWN() \
IMPL_QUERYINTERFACE(); \
STDMETHOD_(ULONG,AddRef)(THIS) \
{ XXX; return _DOCALL0(AddRef); } \
STDMETHOD_(ULONG,Release)(THIS) \
{ \
XXX; \
ULONG retval = _DOCALL0(Release); \
if(retval == 0) \
delete this; \
return retval; \
}
#define IMPL_GETDEVICE() \
STDMETHOD(GetDevice)(THIS_ IDirect3DDevice9** ppDevice) \
{ \
XXX; \
HRESULT hr = _DOCALL( GetDevice, ppDevice ); \
return AllocOverride( &hr, m_Data.pDevice, ppDevice ); \
}
#define IMPL_SETPRIVATEDATA() \
STDMETHOD(SetPrivateData)(THIS_ REFGUID refguid,CONST void* pData,DWORD SizeOfData,DWORD Flags) \
{ XXX; return _DOCALL(SetPrivateData, refguid, pData, SizeOfData, Flags); }
#define IMPL_GETPRIVATEDATA() \
STDMETHOD(GetPrivateData)(THIS_ REFGUID refguid,void* pData,DWORD* pSizeOfData) \
{ XXX; return _DOCALL(GetPrivateData, refguid, pData, pSizeOfData); }
#define IMPL_FREEPRIVATEDATA() \
STDMETHOD(FreePrivateData)(THIS_ REFGUID refguid) \
{ XXX; return _DOCALL(FreePrivateData, refguid); }
#define IMPL_IDIRECT3DRESOURCE9() \
IMPL_GETDEVICE(); \
IMPL_SETPRIVATEDATA(); \
IMPL_GETPRIVATEDATA(); \
IMPL_FREEPRIVATEDATA(); \
STDMETHOD_(DWORD, SetPriority)(THIS_ DWORD PriorityNew) \
{ XXX; return _DOCALL(SetPriority, PriorityNew); } \
STDMETHOD_(DWORD, GetPriority)(THIS) \
{ XXX; return _DOCALL0(GetPriority); } \
STDMETHOD_(void, PreLoad)(THIS) \
{ XXX; _DOCALL0_NORET(PreLoad); } \
STDMETHOD_(D3DRESOURCETYPE, GetType)(THIS) \
{ XXX; return _DOCALL0(GetType); }
class CDirect3DSwapChain9Hook : public CDx9HookBase<IDirect3DSwapChain9>, public IDirect3DSwapChain9
{
public:
/*** IUnknown methods ***/
IMPL_IUNKOWN();
/*** IDirect3DSwapChain9 methods ***/
STDMETHOD(Present)(THIS_ CONST RECT* pSourceRect,CONST RECT* pDestRect,HWND hDestWindowOverride,CONST RGNDATA* pDirtyRegion,DWORD dwFlags)
{ XXX; return _DOCALL(Present, pSourceRect, pDestRect, hDestWindowOverride, pDirtyRegion, dwFlags); }
STDMETHOD(GetFrontBufferData)(THIS_ IDirect3DSurface9* pDestSurface)
{
XXX; return _DOCALL(GetFrontBufferData, GetHWPtr(pDestSurface));
}
STDMETHOD(GetBackBuffer)(THIS_ UINT iBackBuffer,D3DBACKBUFFER_TYPE Type,IDirect3DSurface9** ppBackBuffer)
{
XXX;
HRESULT hr = _DOCALL(GetBackBuffer, iBackBuffer, Type, ppBackBuffer);
return AllocOverride( &hr, m_Data.pDevice, ppBackBuffer );
}
STDMETHOD(GetRasterStatus)(THIS_ D3DRASTER_STATUS* pRasterStatus)
{ XXX; return _DOCALL(GetRasterStatus, pRasterStatus); }
STDMETHOD(GetDisplayMode)(THIS_ D3DDISPLAYMODE* pMode)
{ XXX; return _DOCALL(GetDisplayMode, pMode); }
IMPL_GETDEVICE();
STDMETHOD(GetPresentParameters)(THIS_ D3DPRESENT_PARAMETERS* pPresentationParameters)
{ XXX; return _DOCALL(GetPresentParameters, pPresentationParameters); }
};
class CDirect3DVertexDeclaration9Hook : public CDx9HookBase<IDirect3DVertexDeclaration9>, public IDirect3DVertexDeclaration9
{
public:
/*** IUnknown methods ***/
IMPL_IUNKOWN();
// IDirect3DVertexDeclaration9 methods
IMPL_GETDEVICE();
STDMETHOD(GetDeclaration)(THIS_ D3DVERTEXELEMENT9 *pVertElem, UINT* pNumElements)
{ XXX; return _DOCALL(GetDeclaration, pVertElem, pNumElements); }
};
class CDirect3DPixelShader9Hook : public CDx9HookBase<IDirect3DPixelShader9>, public IDirect3DPixelShader9
{
public:
/*** IUnknown methods ***/
IMPL_IUNKOWN();
// IDirect3DPixelShader9 methods
IMPL_GETDEVICE();
STDMETHOD(GetFunction)(THIS_ void *pData,UINT* pSizeOfData)
{ XXX; return _DOCALL(GetFunction, pData, pSizeOfData); }
};
class CDirect3DVertexShader9Hook : public CDx9HookBase<IDirect3DVertexShader9>, public IDirect3DVertexShader9
{
public:
/*** IUnknown methods ***/
IMPL_IUNKOWN();
// IDirect3DVertexDeclaration9 methods
IMPL_GETDEVICE();
STDMETHOD(GetFunction)(THIS_ void *pData,UINT* pSizeOfData)
{ XXX; return _DOCALL(GetFunction, pData, pSizeOfData); }
};
class CDirect3DVertexBuffer9Hook : public CDx9HookBase<IDirect3DVertexBuffer9>, public IDirect3DVertexBuffer9
{
public:
/*** IUnknown methods ***/
IMPL_IUNKOWN();
// IDirect3DResource9 methods
IMPL_IDIRECT3DRESOURCE9();
// IDirect3DVertexBuffer9
STDMETHOD(Lock)(THIS_ UINT OffsetToLock,UINT SizeToLock,void** ppbData,DWORD Flags)
{ XXX; return _DOCALL(Lock, OffsetToLock, SizeToLock, ppbData, Flags); }
STDMETHOD(Unlock)(THIS)
{ XXX; return _DOCALL0(Unlock); }
STDMETHOD(GetDesc)(THIS_ D3DVERTEXBUFFER_DESC *pDesc)
{ XXX; return _DOCALL(GetDesc, pDesc); }
};
class CDirect3DIndexBuffer9Hook : public CDx9HookBase<IDirect3DIndexBuffer9>, public IDirect3DIndexBuffer9
{
public:
/*** IUnknown methods ***/
IMPL_IUNKOWN();
// IDirect3DResource9 methods
IMPL_IDIRECT3DRESOURCE9();
// IDirect3DIndexBuffer9
STDMETHOD(Lock)(THIS_ UINT OffsetToLock,UINT SizeToLock,void** ppbData,DWORD Flags)
{ XXX; return _DOCALL(Lock, OffsetToLock, SizeToLock, ppbData, Flags); }
STDMETHOD(Unlock)(THIS)
{ XXX; return _DOCALL0(Unlock); }
STDMETHOD(GetDesc)(THIS_ D3DINDEXBUFFER_DESC *pDesc)
{ XXX; return _DOCALL(GetDesc, pDesc); }
};
class CDirect3DQuery9Hook : public CDx9HookBase<IDirect3DQuery9>, public IDirect3DQuery9
{
public:
/*** IUnknown methods ***/
IMPL_IUNKOWN();
/*** IDirect3DQuery9 methods ***/
IMPL_GETDEVICE();
// IDirect3DQuery9
STDMETHOD_(D3DQUERYTYPE, GetType)(THIS)
{ XXX; return _DOCALL0(GetType); }
STDMETHOD_(DWORD, GetDataSize)(THIS)
{ XXX; return _DOCALL0(GetDataSize); }
STDMETHOD(Issue)(THIS_ DWORD dwIssueFlags)
{ XXX; return _DOCALL(Issue, dwIssueFlags); }
STDMETHOD(GetData)(THIS_ void* pData,DWORD dwSize,DWORD dwGetDataFlags)
{ XXX; return _DOCALL(GetData, pData, dwSize, dwGetDataFlags); }
};
class CDirect3DStateBlock9Hook : public CDx9HookBase<IDirect3DStateBlock9>, public IDirect3DStateBlock9
{
public:
CDirect3DStateBlock9Hook() {}
virtual ~CDirect3DStateBlock9Hook() {}
/*** IUnknown methods ***/
IMPL_IUNKOWN();
// IDirect3DStateBlock9 methods
IMPL_GETDEVICE();
STDMETHOD(Capture)(THIS)
{ XXX; return _DOCALL0(Capture); }
STDMETHOD(Apply)(THIS)
{ XXX; return _DOCALL0(Apply); }
};
class CDirect3DSurface9Hook : public CDx9HookBase<IDirect3DSurface9>, public IDirect3DSurface9
{
public:
/*** IUnknown methods ***/
IMPL_IUNKOWN();
// IDirect3DResource9 methods
IMPL_IDIRECT3DRESOURCE9();
STDMETHOD(GetContainer)(THIS_ REFIID riid,void** ppContainer)
{
//$ TODO: do the call, check riid, and wrap the returned ppContainer
__debugbreak();
XXX; return _DOCALL(GetContainer, riid, ppContainer);
}
STDMETHOD(GetDesc)(THIS_ D3DSURFACE_DESC *pDesc)
{ XXX; return _DOCALL(GetDesc, pDesc); }
STDMETHOD(LockRect)(THIS_ D3DLOCKED_RECT* pLockedRect,CONST RECT* pRect,DWORD Flags)
{ XXX; return _DOCALL(LockRect, pLockedRect, pRect, Flags); }
STDMETHOD(UnlockRect)(THIS)
{ XXX; return _DOCALL0(UnlockRect); }
STDMETHOD(GetDC)(THIS_ HDC *phdc)
{ XXX; return _DOCALL(GetDC, phdc); }
STDMETHOD(ReleaseDC)(THIS_ HDC hdc)
{ XXX; return _DOCALL(ReleaseDC, hdc); }
};
class CDirect3DBaseTexture9Hook : public CDx9HookBase<IDirect3DBaseTexture9>, public IDirect3DBaseTexture9
{
public:
/*** IUnknown methods ***/
IMPL_IUNKOWN();
/*** IDirect3DBaseTexture9 methods ***/
IMPL_IDIRECT3DRESOURCE9();
STDMETHOD_(DWORD, SetLOD)(THIS_ DWORD LODNew)
{ XXX; return _DOCALL(SetLOD, LODNew); }
STDMETHOD_(DWORD, GetLOD)(THIS)
{ XXX; return _DOCALL0(GetLOD); }
STDMETHOD_(DWORD, GetLevelCount)(THIS)
{ XXX; return _DOCALL0(GetLevelCount); }
STDMETHOD(SetAutoGenFilterType)(THIS_ D3DTEXTUREFILTERTYPE FilterType)
{ XXX; return _DOCALL(SetAutoGenFilterType, FilterType); }
STDMETHOD_(D3DTEXTUREFILTERTYPE, GetAutoGenFilterType)(THIS)
{ XXX; return _DOCALL0(GetAutoGenFilterType); }
STDMETHOD_(void, GenerateMipSubLevels)(THIS)
{ XXX; return _DOCALL0(GenerateMipSubLevels); }
};
class CDirect3DTexture9Hook : public CDx9HookBase<IDirect3DTexture9>, public IDirect3DTexture9
{
public:
/*** IUnknown methods ***/
IMPL_IUNKOWN();
/*** IDirect3DBaseTexture9 methods ***/
IMPL_IDIRECT3DRESOURCE9();
STDMETHOD_(DWORD, SetLOD)(THIS_ DWORD LODNew)
{ XXX; return _DOCALL(SetLOD, LODNew); }
STDMETHOD_(DWORD, GetLOD)(THIS)
{ XXX; return _DOCALL0(GetLOD); }
STDMETHOD_(DWORD, GetLevelCount)(THIS)
{ XXX; return _DOCALL0(GetLevelCount); }
STDMETHOD(SetAutoGenFilterType)(THIS_ D3DTEXTUREFILTERTYPE FilterType)
{ XXX; return _DOCALL(SetAutoGenFilterType, FilterType); }
STDMETHOD_(D3DTEXTUREFILTERTYPE, GetAutoGenFilterType)(THIS)
{ XXX; return _DOCALL0(GetAutoGenFilterType); }
STDMETHOD_(void, GenerateMipSubLevels)(THIS)
{ XXX; return _DOCALL0(GenerateMipSubLevels); }
STDMETHOD(GetLevelDesc)(THIS_ UINT Level,D3DSURFACE_DESC *pDesc)
{ XXX; return _DOCALL(GetLevelDesc, Level, pDesc); }
STDMETHOD(GetSurfaceLevel)(THIS_ UINT Level,IDirect3DSurface9** ppSurfaceLevel)
{
XXX;
HRESULT hr = _DOCALL(GetSurfaceLevel, Level, ppSurfaceLevel);
return AllocOverride( &hr, m_Data.pDevice, ppSurfaceLevel );
}
STDMETHOD(LockRect)(THIS_ UINT Level,D3DLOCKED_RECT* pLockedRect,CONST RECT* pRect,DWORD Flags)
{ XXX; return _DOCALL(LockRect, Level, pLockedRect, pRect, Flags); }
STDMETHOD(UnlockRect)(THIS_ UINT Level)
{ XXX; return _DOCALL(UnlockRect, Level); }
STDMETHOD(AddDirtyRect)(THIS_ CONST RECT* pDirtyRect)
{ XXX; return _DOCALL(AddDirtyRect, pDirtyRect); }
};
class CDirect3DCubeTexture9Hook : public CDx9HookBase<IDirect3DCubeTexture9>, public IDirect3DCubeTexture9
{
public:
/*** IUnknown methods ***/
IMPL_IUNKOWN();
/*** IDirect3DBaseTexture9 methods ***/
IMPL_IDIRECT3DRESOURCE9();
STDMETHOD_(DWORD, SetLOD)(THIS_ DWORD LODNew)
{ XXX; return _DOCALL(SetLOD, LODNew); }
STDMETHOD_(DWORD, GetLOD)(THIS)
{ XXX; return _DOCALL0(GetLOD); }
STDMETHOD_(DWORD, GetLevelCount)(THIS)
{ XXX; return _DOCALL0(GetLevelCount); }
STDMETHOD(SetAutoGenFilterType)(THIS_ D3DTEXTUREFILTERTYPE FilterType)
{ XXX; return _DOCALL(SetAutoGenFilterType, FilterType); }
STDMETHOD_(D3DTEXTUREFILTERTYPE, GetAutoGenFilterType)(THIS)
{ XXX; return _DOCALL0(GetAutoGenFilterType); }
STDMETHOD_(void, GenerateMipSubLevels)(THIS)
{ XXX; return _DOCALL0(GenerateMipSubLevels); }
STDMETHOD(GetLevelDesc)(THIS_ UINT Level,D3DSURFACE_DESC *pDesc)
{ XXX; return _DOCALL(GetLevelDesc, Level, pDesc); }
STDMETHOD(GetCubeMapSurface)(THIS_ D3DCUBEMAP_FACES FaceType,UINT Level,IDirect3DSurface9** ppCubeMapSurface)
{
XXX;
HRESULT hr = _DOCALL(GetCubeMapSurface, FaceType, Level, ppCubeMapSurface);
return AllocOverride(&hr, m_Data.pDevice, ppCubeMapSurface);
}
STDMETHOD(LockRect)(THIS_ D3DCUBEMAP_FACES FaceType,UINT Level,D3DLOCKED_RECT* pLockedRect,CONST RECT* pRect,DWORD Flags)
{ XXX; return _DOCALL(LockRect, FaceType, Level, pLockedRect, pRect, Flags); }
STDMETHOD(UnlockRect)(THIS_ D3DCUBEMAP_FACES FaceType,UINT Level)
{ XXX; return _DOCALL(UnlockRect, FaceType, Level); }
STDMETHOD(AddDirtyRect)(THIS_ D3DCUBEMAP_FACES FaceType,CONST RECT* pDirtyRect)
{ XXX; return _DOCALL(AddDirtyRect, FaceType, pDirtyRect); }
};
class CDirect3DVolume9Hook : public CDx9HookBase<IDirect3DVolume9>, public IDirect3DVolume9
{
public:
/*** IUnknown methods ***/
IMPL_IUNKOWN();
/*** IDirect3DVolume9 methods ***/
IMPL_GETDEVICE();
IMPL_SETPRIVATEDATA();
IMPL_GETPRIVATEDATA();
IMPL_FREEPRIVATEDATA()
STDMETHOD(GetContainer)(THIS_ REFIID riid,void** ppContainer)
{
//$ TODO: do the call, check riid, and wrap the returned ppContainer
__debugbreak();
XXX; return _DOCALL(GetContainer, riid, ppContainer);
}
STDMETHOD(GetDesc)(THIS_ D3DVOLUME_DESC *pDesc)
{ XXX; return _DOCALL(GetDesc, pDesc); }
STDMETHOD(LockBox)(THIS_ D3DLOCKED_BOX * pLockedVolume,CONST D3DBOX* pBox,DWORD Flags)
{ XXX; return _DOCALL(LockBox, pLockedVolume, pBox, Flags); }
STDMETHOD(UnlockBox)(THIS)
{ XXX; return _DOCALL0(UnlockBox); }
};
class CDirect3DVolumeTexture9Hook : public CDx9HookBase<IDirect3DVolumeTexture9>, public IDirect3DVolumeTexture9
{
public:
/*** IUnknown methods ***/
IMPL_IUNKOWN();
/*** IDirect3DBaseTexture9 methods ***/
IMPL_IDIRECT3DRESOURCE9();
STDMETHOD_(DWORD, SetLOD)(THIS_ DWORD LODNew)
{ XXX; return _DOCALL(SetLOD, LODNew); }
STDMETHOD_(DWORD, GetLOD)(THIS)
{ XXX; return _DOCALL0(GetLOD); }
STDMETHOD_(DWORD, GetLevelCount)(THIS)
{ XXX; return _DOCALL0(GetLevelCount); }
STDMETHOD(SetAutoGenFilterType)(THIS_ D3DTEXTUREFILTERTYPE FilterType)
{ XXX; return _DOCALL(SetAutoGenFilterType, FilterType); }
STDMETHOD_(D3DTEXTUREFILTERTYPE, GetAutoGenFilterType)(THIS)
{ XXX; return _DOCALL0(GetAutoGenFilterType); }
STDMETHOD_(void, GenerateMipSubLevels)(THIS)
{ XXX; return _DOCALL0(GenerateMipSubLevels); }
STDMETHOD(GetLevelDesc)(THIS_ UINT Level,D3DVOLUME_DESC *pDesc)
{ XXX; return _DOCALL(GetLevelDesc, Level, pDesc); }
STDMETHOD(GetVolumeLevel)(THIS_ UINT Level,IDirect3DVolume9** ppVolumeLevel)
{
XXX;
HRESULT hr = _DOCALL(GetVolumeLevel, Level, ppVolumeLevel);
return AllocOverride( &hr, m_Data.pDevice, ppVolumeLevel );
}
STDMETHOD(LockBox)(THIS_ UINT Level,D3DLOCKED_BOX* pLockedVolume,CONST D3DBOX* pBox,DWORD Flags)
{ XXX; return _DOCALL(LockBox, Level, pLockedVolume, pBox, Flags); }
STDMETHOD(UnlockBox)(THIS_ UINT Level)
{ XXX; return _DOCALL(UnlockBox, Level); }
STDMETHOD(AddDirtyBox)(THIS_ CONST D3DBOX* pDirtyBox)
{ XXX; return _DOCALL(AddDirtyBox, pDirtyBox); }
};
class CDirect3DDevice9Hook : public CDx9HookBase<IDirect3DDevice9>, public IDirect3DDevice9
{
public:
/*** IUnknown methods ***/
IMPL_IUNKOWN();
CDirect3DDevice9Hook() : CDx9HookBase<IDirect3DDevice9>(), IDirect3DDevice9()
{
D3D_BATCH_PERF( g_nTotalD3DCycles = 0; g_nTotalD3DCalls = 0; m_batch_state.Clear(); m_nTotalDraws = 0; m_nTotalPrims = 0; m_nTotalD3DCalls = 0; m_flTotalD3DTime = 0; m_nOverallDraws = 0; m_nOverallPrims = 0; m_nOverallD3DCalls = 0; m_flOverallD3DTime = 0; m_nTotalFrames = 0; m_pPrevRenderTarget0 = NULL; )
}
// IDirect3DDevice9 methods
STDMETHOD(TestCooperativeLevel)(THIS)
{ XXX; return _DOCALL0(TestCooperativeLevel); }
STDMETHOD_(UINT, GetAvailableTextureMem)(THIS)
{ XXX; return _DOCALL0(GetAvailableTextureMem); }
STDMETHOD(EvictManagedResources)(THIS)
{ XXX; return _DOCALL0(EvictManagedResources); }
STDMETHOD(GetDirect3D)(THIS_ IDirect3D9** ppD3D9)
{
XXX;
HRESULT hr = _DOCALL(GetDirect3D, ppD3D9);
return AllocOverride(&hr, this, ppD3D9);
}
STDMETHOD(GetDeviceCaps)(THIS_ D3DCAPS9* pCaps)
{ XXX; return _DOCALL(GetDeviceCaps, pCaps); }
STDMETHOD(GetDisplayMode)(THIS_ UINT iSwapChain,D3DDISPLAYMODE* pMode)
{ XXX; return _DOCALL(GetDisplayMode, iSwapChain, pMode); }
STDMETHOD(GetCreationParameters)(THIS_ D3DDEVICE_CREATION_PARAMETERS *pParameters)
{ XXX; return _DOCALL(GetCreationParameters, pParameters); }
STDMETHOD(SetCursorProperties)(THIS_ UINT XHotSpot,UINT YHotSpot,IDirect3DSurface9* pCursorBitmap)
{ XXX; return _DOCALL(SetCursorProperties, XHotSpot, YHotSpot, GetHWPtr(pCursorBitmap)); }
STDMETHOD_(void, SetCursorPosition)(THIS_ int X,int Y,DWORD Flags)
{ XXX; _DOCALL_NORET(SetCursorPosition, X, Y, Flags); }
STDMETHOD_(BOOL, ShowCursor)(THIS_ BOOL bShow)
{ XXX; return _DOCALL(ShowCursor, bShow); }
STDMETHOD(CreateAdditionalSwapChain)(THIS_ D3DPRESENT_PARAMETERS* pPresentationParameters,IDirect3DSwapChain9** ppSwapChain)
{
XXX;
HRESULT hr = _DOCALL(CreateAdditionalSwapChain, pPresentationParameters, ppSwapChain);
return AllocOverride(&hr, this, ppSwapChain);
}
STDMETHOD(GetSwapChain)(THIS_ UINT iSwapChain,IDirect3DSwapChain9** ppSwapChain)
{
XXX;
HRESULT hr = _DOCALL(GetSwapChain, iSwapChain, ppSwapChain);
return AllocOverride(&hr, this, ppSwapChain);
}
STDMETHOD_(UINT, GetNumberOfSwapChains)(THIS)
{ XXX; return _DOCALL0(GetNumberOfSwapChains); }
STDMETHOD(Reset)(THIS_ D3DPRESENT_PARAMETERS* pPresentationParameters)
{ XXX; return _DOCALL( Reset, pPresentationParameters ); }
STDMETHOD(Present)(THIS_ CONST RECT* pSourceRect,CONST RECT* pDestRect,HWND hDestWindowOverride,CONST RGNDATA* pDirtyRegion)
{
HRESULT hres;
#if D3D_BATCH_PERF_ANALYSIS
uint64 nStartCycles = g_nTotalD3DCycles;
CFastTimer tm;
tm.Start();
g_nTotalD3DCalls++;
#endif
{
XXX;
hres = _DOCALL(Present, pSourceRect, pDestRect, hDestWindowOverride, pDirtyRegion);
}
#if D3D_BATCH_PERF_ANALYSIS
const uint nFrameIndex = m_nTotalFrames;
m_nTotalFrames++;
if (nFrameIndex >= 5)
{
double flPresentTime = tm.GetDurationInProgress().GetMillisecondsF();
uint64 nEndCycles = g_nTotalD3DCycles;
double flTotalPresentTime = ( nEndCycles - nStartCycles ) * s_rdtsc_to_ms;
m_nTotalD3DCalls += g_nTotalD3DCalls;
m_flTotalD3DTime += g_nTotalD3DCycles * s_rdtsc_to_ms;
static int bPrevBatchVis = -1;
if ((bPrevBatchVis == 1) && m_batch_vis_bitmap.is_valid())
{
m_nOverallDraws += m_nTotalDraws;
m_nOverallPrims += m_nTotalPrims;
m_nOverallD3DCalls += m_nTotalD3DCalls;
m_flOverallD3DTime += m_flTotalD3DTime;
m_batch_vis_bitmap.fill_box(0, m_nBatchVisY, (uint)(.5f + flPresentTime / d3d_present_vis_abs_scale.GetFloat() * m_batch_vis_bitmap.width()), 10, 255, 16, 128);
m_batch_vis_bitmap.additive_fill_box(0, m_nBatchVisY, (uint)(.5f + flTotalPresentTime / d3d_present_vis_abs_scale.GetFloat() * m_batch_vis_bitmap.width()), 10, 0, 255, 128);
m_nBatchVisY += 10;
uint y = MAX( 600, m_nBatchVisY + 20 ), l = 0;
m_batch_vis_bitmap.draw_formatted_text(0, y+8*(l++), 1, 255, 255, 255, "Frame: %u, Batches: %u, Prims: %u", nFrameIndex, m_nTotalDraws, m_nTotalPrims );
m_batch_vis_bitmap.draw_formatted_text(0, y+8*(l++), 1, 255, 255, 255, "Frame: D3D Calls: %u, D3D Time: %3.3fms", m_nTotalD3DCalls, m_flTotalD3DTime);
l++;
m_batch_vis_bitmap.draw_formatted_text(0, y+8*(l++), 1, 255, 255, 255, "Overall: Batches: %u, Prims: %u", m_nOverallDraws, m_nOverallPrims );
m_batch_vis_bitmap.draw_formatted_text(0, y+8*(l++), 1, 255, 255, 255, "Overall: D3D Calls: %u D3D Time: %4.3fms", m_nOverallD3DCalls, m_flOverallD3DTime );
size_t png_size = 0;
void *pPNG_data = tdefl_write_image_to_png_file_in_memory(m_batch_vis_bitmap.get_ptr(), m_batch_vis_bitmap.width(), m_batch_vis_bitmap.height(), 3, &png_size, true);
if (pPNG_data)
{
char filename[256];
V_snprintf(filename, sizeof(filename), "left4dead2/batchvis_%u_%u.png", m_nBatchVisFileIdx, m_nBatchVisFrameIndex);
FILE* pFile = fopen(filename, "wb");
if (pFile)
{
fwrite(pPNG_data, png_size, 1, pFile);
fclose(pFile);
}
free(pPNG_data);
}
m_nBatchVisFrameIndex++;
m_nBatchVisY = 0;
m_batch_vis_bitmap.cls();
}
if (bPrevBatchVis != (int)d3d_batch_vis.GetBool())
{
bPrevBatchVis = d3d_batch_vis.GetBool();
if (!bPrevBatchVis)
{
m_batch_vis_bitmap.clear();
}
else
{
m_batch_vis_bitmap.init(768, 1024);
}
m_nBatchVisY = 0;
m_nBatchVisFrameIndex = 0;
m_nBatchVisFileIdx = (uint)time(NULL); //rand();
m_nOverallDraws = 0;
m_nOverallPrims = 0;
m_nOverallD3DCalls = 0;
m_flOverallD3DTime = 0;
}
}
m_nTotalD3DCalls = 0;
m_nTotalPrims = 0;
m_flTotalD3DTime = 0;
g_nTotalD3DCycles = 0;
g_nTotalD3DCalls = 0;
m_nTotalDraws = 0;
#else
if ( d3d_batch_vis.GetBool() )
{
d3d_batch_vis.SetValue( false );
ConMsg( "Must define D3D_BATCH_PERF_ANALYSIS to use this feature" );
}
#endif
return hres;
}
STDMETHOD(GetBackBuffer)(THIS_ UINT iSwapChain,UINT iBackBuffer,D3DBACKBUFFER_TYPE Type,IDirect3DSurface9** ppBackBuffer)
{
XXX;
HRESULT hr = _DOCALL(GetBackBuffer, iSwapChain, iBackBuffer, Type, ppBackBuffer);
return AllocOverride(&hr, this, ppBackBuffer);
}
STDMETHOD(GetRasterStatus)(THIS_ UINT iSwapChain,D3DRASTER_STATUS* pRasterStatus)
{ XXX; return _DOCALL(GetRasterStatus, iSwapChain, pRasterStatus); }
STDMETHOD(SetDialogBoxMode)(THIS_ BOOL bEnableDialogs)
{ XXX; return _DOCALL(SetDialogBoxMode, bEnableDialogs); }
STDMETHOD_(void, SetGammaRamp)(THIS_ UINT iSwapChain,DWORD Flags,CONST D3DGAMMARAMP* pRamp)
{ XXX; _DOCALL_NORET(SetGammaRamp, iSwapChain, Flags, pRamp); }
STDMETHOD_(void, GetGammaRamp)(THIS_ UINT iSwapChain,D3DGAMMARAMP* pRamp)
{ XXX; _DOCALL_NORET(GetGammaRamp, iSwapChain, pRamp); }
STDMETHOD(CreateTexture)(THIS_ UINT Width,UINT Height,UINT Levels,DWORD Usage,D3DFORMAT Format,D3DPOOL Pool,IDirect3DTexture9** ppTexture,HANDLE* pSharedHandle)
{
XXX;
HRESULT hr = _DOCALL(CreateTexture, Width, Height, Levels, Usage, Format, Pool, ppTexture, pSharedHandle);
return AllocOverride(&hr, this, ppTexture);
}
STDMETHOD(CreateVolumeTexture)(THIS_ UINT Width,UINT Height,UINT Depth,UINT Levels,DWORD Usage,D3DFORMAT Format,D3DPOOL Pool,IDirect3DVolumeTexture9** ppVolumeTexture,HANDLE* pSharedHandle)
{
XXX;
HRESULT hr = _DOCALL(CreateVolumeTexture, Width, Height, Depth, Levels, Usage, Format, Pool, ppVolumeTexture, pSharedHandle);
return AllocOverride(&hr, this, ppVolumeTexture);
}
STDMETHOD(CreateCubeTexture)(THIS_ UINT EdgeLength,UINT Levels,DWORD Usage,D3DFORMAT Format,D3DPOOL Pool,IDirect3DCubeTexture9** ppCubeTexture,HANDLE* pSharedHandle)
{
XXX;
HRESULT hr = _DOCALL(CreateCubeTexture, EdgeLength, Levels, Usage, Format, Pool, ppCubeTexture, pSharedHandle);
return AllocOverride(&hr, this, ppCubeTexture);
}
STDMETHOD(CreateVertexBuffer)(THIS_ UINT Length,DWORD Usage,DWORD FVF,D3DPOOL Pool,IDirect3DVertexBuffer9** ppVertexBuffer,HANDLE* pSharedHandle)
{
XXX;
HRESULT hr = _DOCALL(CreateVertexBuffer, Length, Usage, FVF, Pool, ppVertexBuffer, pSharedHandle);
return AllocOverride(&hr, this, ppVertexBuffer);
}
STDMETHOD(CreateIndexBuffer)(THIS_ UINT Length,DWORD Usage,D3DFORMAT Format,D3DPOOL Pool,IDirect3DIndexBuffer9** ppIndexBuffer,HANDLE* pSharedHandle)
{
XXX;
HRESULT hr = _DOCALL(CreateIndexBuffer, Length, Usage, Format, Pool, ppIndexBuffer, pSharedHandle);
return AllocOverride(&hr, this, ppIndexBuffer);
}
STDMETHOD(CreateRenderTarget)(THIS_ UINT Width,UINT Height,D3DFORMAT Format,D3DMULTISAMPLE_TYPE MultiSample,DWORD MultiSampleQuality,BOOL Lockable,IDirect3DSurface9** ppSurface,HANDLE* pSharedHandle)
{
XXX;
HRESULT hr = _DOCALL(CreateRenderTarget, Width, Height, Format, MultiSample, MultiSampleQuality, Lockable, ppSurface, pSharedHandle);
return AllocOverride(&hr, this, ppSurface);
}
STDMETHOD(CreateDepthStencilSurface)(THIS_ UINT Width,UINT Height,D3DFORMAT Format,D3DMULTISAMPLE_TYPE MultiSample,DWORD MultiSampleQuality,BOOL Discard,IDirect3DSurface9** ppSurface,HANDLE* pSharedHandle)
{
XXX;
HRESULT hr = _DOCALL(CreateDepthStencilSurface, Width, Height, Format, MultiSample, MultiSampleQuality, Discard, ppSurface, pSharedHandle);
return AllocOverride(&hr, this, ppSurface);
}
STDMETHOD(UpdateSurface)(THIS_ IDirect3DSurface9* pSourceSurface,CONST RECT* pSourceRect,IDirect3DSurface9* pDestinationSurface,CONST POINT* pDestPoint)
{ XXX; return _DOCALL(UpdateSurface, GetHWPtr(pSourceSurface), pSourceRect, GetHWPtr(pDestinationSurface), pDestPoint); }
STDMETHOD(UpdateTexture)(THIS_ IDirect3DBaseTexture9* pSourceTexture,IDirect3DBaseTexture9* pDestinationTexture)
{ XXX; return _DOCALL(UpdateTexture, GetHWPtr(pSourceTexture), GetHWPtr(pDestinationTexture)); }
STDMETHOD(GetRenderTargetData)(THIS_ IDirect3DSurface9* pRenderTarget,IDirect3DSurface9* pDestSurface)
{ XXX; return _DOCALL(GetRenderTargetData, GetHWPtr(pRenderTarget), GetHWPtr(pDestSurface)); }
STDMETHOD(GetFrontBufferData)(THIS_ UINT iSwapChain,IDirect3DSurface9* pDestSurface)
{ XXX; return _DOCALL(GetFrontBufferData, iSwapChain, GetHWPtr(pDestSurface)); }
STDMETHOD(StretchRect)(THIS_ IDirect3DSurface9* pSourceSurface,CONST RECT* pSourceRect,IDirect3DSurface9* pDestSurface,CONST RECT* pDestRect,D3DTEXTUREFILTERTYPE Filter)
{ XXX; return _DOCALL(StretchRect, GetHWPtr(pSourceSurface), pSourceRect, GetHWPtr(pDestSurface), pDestRect, Filter); }
STDMETHOD(ColorFill)(THIS_ IDirect3DSurface9* pSurface,CONST RECT* pRect,D3DCOLOR color)
{ XXX; return _DOCALL(ColorFill, GetHWPtr(pSurface), pRect, color); }
STDMETHOD(CreateOffscreenPlainSurface)(THIS_ UINT Width,UINT Height,D3DFORMAT Format,D3DPOOL Pool,IDirect3DSurface9** ppSurface,HANDLE* pSharedHandle)
{
XXX;
HRESULT hr = _DOCALL(CreateOffscreenPlainSurface, Width, Height, Format, Pool, ppSurface, pSharedHandle);
return AllocOverride(&hr, this, ppSurface);
}
STDMETHOD(SetRenderTarget)(THIS_ DWORD RenderTargetIndex,IDirect3DSurface9* pRenderTarget)
{
HRESULT hres;
{
XXX;
hres = _DOCALL(SetRenderTarget, RenderTargetIndex, GetHWPtr(pRenderTarget));
}
#if D3D_BATCH_PERF_ANALYSIS
if ( m_batch_vis_bitmap.is_valid() && !RenderTargetIndex )
{
if ( pRenderTarget != m_pPrevRenderTarget0 )
{
m_batch_vis_bitmap.fill_box(0, m_nBatchVisY, m_batch_vis_bitmap.width(), 1, 30, 20, 20);
m_nBatchVisY += 1;
}
}
if ( !RenderTargetIndex )
{
m_pPrevRenderTarget0 = pRenderTarget;
}
#endif
return hres;
}
STDMETHOD(GetRenderTarget)(THIS_ DWORD RenderTargetIndex,IDirect3DSurface9** ppRenderTarget)
{
XXX;
HRESULT hr = _DOCALL(GetRenderTarget, RenderTargetIndex, ppRenderTarget);
return AllocOverride(&hr, this, ppRenderTarget);
}
STDMETHOD(SetDepthStencilSurface)(THIS_ IDirect3DSurface9* pNewZStencil)
{ XXX; return _DOCALL(SetDepthStencilSurface, GetHWPtr(pNewZStencil)); }
STDMETHOD(GetDepthStencilSurface)(THIS_ IDirect3DSurface9** ppZStencilSurface)
{
XXX;
HRESULT hr = _DOCALL(GetDepthStencilSurface, ppZStencilSurface);
return AllocOverride(&hr, this, ppZStencilSurface);
}
STDMETHOD(BeginScene)(THIS)
{ XXX; return _DOCALL0(BeginScene); }
STDMETHOD(EndScene)(THIS)
{ XXX; return _DOCALL0(EndScene); }
STDMETHOD(Clear)(THIS_ DWORD Count,CONST D3DRECT* pRects,DWORD Flags,D3DCOLOR Color,float Z,DWORD Stencil)
{ XXX; return _DOCALL(Clear, Count, pRects, Flags, Color, Z, Stencil); }
STDMETHOD(SetTransform)(THIS_ D3DTRANSFORMSTATETYPE State,CONST D3DMATRIX* pMatrix)
{ XXX; return _DOCALL(SetTransform, State, pMatrix); }
STDMETHOD(GetTransform)(THIS_ D3DTRANSFORMSTATETYPE State,D3DMATRIX* pMatrix)
{ XXX; return _DOCALL(GetTransform, State, pMatrix); }
STDMETHOD(MultiplyTransform)(THIS_ D3DTRANSFORMSTATETYPE State,CONST D3DMATRIX *pMatrix)
{ XXX; return _DOCALL(MultiplyTransform, State, pMatrix); }
STDMETHOD(SetViewport)(THIS_ CONST D3DVIEWPORT9* pViewport)
{ XXX; return _DOCALL(SetViewport, pViewport); }
STDMETHOD(GetViewport)(THIS_ D3DVIEWPORT9* pViewport)
{ XXX; return _DOCALL(GetViewport, pViewport); }
STDMETHOD(SetMaterial)(THIS_ CONST D3DMATERIAL9* pMaterial)
{ XXX; return _DOCALL(SetMaterial, pMaterial); }
STDMETHOD(GetMaterial)(THIS_ D3DMATERIAL9* pMaterial)
{ XXX; return _DOCALL(GetMaterial, pMaterial); }
STDMETHOD(SetLight)(THIS_ DWORD Index,CONST D3DLIGHT9* pLight)
{ XXX; return _DOCALL(SetLight, Index, pLight); }
STDMETHOD(GetLight)(THIS_ DWORD Index,D3DLIGHT9* pLight)
{ XXX; return _DOCALL(GetLight, Index, pLight); }
STDMETHOD(LightEnable)(THIS_ DWORD Index,BOOL Enable)
{ XXX; return _DOCALL(LightEnable, Index, Enable); }
STDMETHOD(GetLightEnable)(THIS_ DWORD Index,BOOL* pEnable)
{ XXX; return _DOCALL(GetLightEnable, Index, pEnable); }
STDMETHOD(SetClipPlane)(THIS_ DWORD Index,CONST float* pPlane)
{ XXX; return _DOCALL(SetClipPlane, Index, pPlane);}
STDMETHOD(GetClipPlane)(THIS_ DWORD Index,float* pPlane)
{ XXX; return _DOCALL(GetClipPlane, Index, pPlane); }
STDMETHOD(SetRenderState)(THIS_ D3DRENDERSTATETYPE State,DWORD Value)
{ XXX; return _DOCALL(SetRenderState, State, Value); }
STDMETHOD(GetRenderState)(THIS_ D3DRENDERSTATETYPE State,DWORD* pValue)
{ XXX; return _DOCALL(GetRenderState, State, pValue); }
STDMETHOD(CreateStateBlock)(THIS_ D3DSTATEBLOCKTYPE Type,IDirect3DStateBlock9** ppSB)
{
XXX;
HRESULT hr = _DOCALL(CreateStateBlock, Type, ppSB);
return AllocOverride(&hr, this, ppSB);
}
STDMETHOD(BeginStateBlock)(THIS)
{ XXX; return _DOCALL0(BeginStateBlock); }
STDMETHOD(EndStateBlock)(THIS_ IDirect3DStateBlock9** ppSB)
{
XXX;
HRESULT hr = _DOCALL(EndStateBlock, ppSB);
return AllocOverride(&hr, this, ppSB);
}
STDMETHOD(SetClipStatus)(THIS_ CONST D3DCLIPSTATUS9* pClipStatus)
{ XXX; return _DOCALL(SetClipStatus, pClipStatus);}
STDMETHOD(GetClipStatus)(THIS_ D3DCLIPSTATUS9* pClipStatus)
{ XXX; return _DOCALL(GetClipStatus, pClipStatus);}
STDMETHOD(GetTexture)(THIS_ DWORD Stage,IDirect3DBaseTexture9** ppTexture)
{
XXX;
HRESULT hr = _DOCALL(GetTexture, Stage, ppTexture);
return AllocOverride(&hr, this, ppTexture);
}
STDMETHOD(SetTexture)(THIS_ DWORD Stage,IDirect3DBaseTexture9* pTexture)
{
HRESULT hres;
{
XXX;
hres = _DOCALL(SetTexture, Stage, GetHWPtr(pTexture));
}
D3D_BATCH_PERF( m_batch_state.m_nNumSamplersChanged++; )
return hres;
}
STDMETHOD(GetTextureStageState)(THIS_ DWORD Stage,D3DTEXTURESTAGESTATETYPE Type,DWORD* pValue)
{ XXX; return _DOCALL(GetTextureStageState, Stage, Type, pValue); }
STDMETHOD(SetTextureStageState)(THIS_ DWORD Stage,D3DTEXTURESTAGESTATETYPE Type,DWORD Value)
{ XXX; return _DOCALL(SetTextureStageState, Stage, Type, Value); }
STDMETHOD(GetSamplerState)(THIS_ DWORD Sampler,D3DSAMPLERSTATETYPE Type,DWORD* pValue)
{ XXX; return _DOCALL(GetSamplerState, Sampler, Type, pValue); }
STDMETHOD(SetSamplerState)(THIS_ DWORD Sampler,D3DSAMPLERSTATETYPE Type,DWORD Value)
{
HRESULT hres;
{
XXX;
hres = _DOCALL(SetSamplerState, Sampler, Type, Value);
}
D3D_BATCH_PERF( m_batch_state.m_nNumSamplerStatesChanged++; )
return hres;
}
STDMETHOD(ValidateDevice)(THIS_ DWORD* pNumPasses)
{ XXX; return _DOCALL(ValidateDevice, pNumPasses); }
STDMETHOD(SetPaletteEntries)(THIS_ UINT PaletteNumber,CONST PALETTEENTRY* pEntries)
{ XXX; return _DOCALL(SetPaletteEntries, PaletteNumber, pEntries); }
STDMETHOD(GetPaletteEntries)(THIS_ UINT PaletteNumber,PALETTEENTRY* pEntries)
{ XXX; return _DOCALL(GetPaletteEntries, PaletteNumber, pEntries); }
STDMETHOD(SetCurrentTexturePalette)(THIS_ UINT PaletteNumber)
{ XXX; return _DOCALL(SetCurrentTexturePalette, PaletteNumber);}
STDMETHOD(GetCurrentTexturePalette)(THIS_ UINT *PaletteNumber)
{ XXX; return _DOCALL(GetCurrentTexturePalette, PaletteNumber); }
STDMETHOD(SetScissorRect)(THIS_ CONST RECT* pRect)
{ XXX; return _DOCALL(SetScissorRect, pRect); }
STDMETHOD(GetScissorRect)(THIS_ RECT* pRect)
{ XXX; return _DOCALL(GetScissorRect, pRect); }
STDMETHOD(SetSoftwareVertexProcessing)(THIS_ BOOL bSoftware)
{ XXX; return _DOCALL(SetSoftwareVertexProcessing, bSoftware); }
STDMETHOD_(BOOL, GetSoftwareVertexProcessing)(THIS)
{ XXX; return _DOCALL0(GetSoftwareVertexProcessing); }
STDMETHOD(SetNPatchMode)(THIS_ float nSegments)
{ XXX; return _DOCALL(SetNPatchMode, nSegments); }
STDMETHOD_(float, GetNPatchMode)(THIS)
{ XXX; return _DOCALL0(GetNPatchMode);}
STDMETHOD(DrawPrimitive)(THIS_ D3DPRIMITIVETYPE PrimitiveType,UINT StartVertex,UINT PrimitiveCount)
{ XXX; m_nTotalDraws++; m_nTotalPrims += PrimitiveCount; return _DOCALL(DrawPrimitive, PrimitiveType, StartVertex, PrimitiveCount); }
STDMETHOD(DrawIndexedPrimitive)(THIS_ D3DPRIMITIVETYPE PrimitiveType,INT BaseVertexIndex,UINT MinVertexIndex,UINT NumVertices,UINT startIndex,UINT primCount)
{
m_nTotalDraws++;
m_nTotalPrims += primCount;
#if D3D_BATCH_PERF_ANALYSIS
CFastTimer tm;
if ( m_batch_vis_bitmap.is_valid() )
{
tm.Start();
}
#endif
HRESULT hres;
{
XXX;
hres = _DOCALL(DrawIndexedPrimitive, PrimitiveType, BaseVertexIndex, MinVertexIndex, NumVertices, startIndex, primCount);
}
#if D3D_BATCH_PERF_ANALYSIS
if ( m_batch_vis_bitmap.is_valid() )
{
double t = tm.GetDurationInProgress().GetMillisecondsF();
uint h = 1;
if ( d3d_batch_vis_y_scale.GetFloat() > 0.0f)
{
h = ceil( t / d3d_batch_vis_y_scale.GetFloat() );
h = MAX(h, 1);
}
m_batch_vis_bitmap.fill_box(0, m_nBatchVisY, (uint)(.5f + t / d3d_batch_vis_abs_scale.GetFloat() * m_batch_vis_bitmap.width()), h, 64, 64, 64);
if ( s_rdtsc_to_ms == 0.0f )
{
TmU64 t0 = TM_FAST_TIME();
double d0 = Plat_FloatTime();
ThreadSleep( 1 );
TmU64 t1 = TM_FAST_TIME();
double d1 = Plat_FloatTime();
s_rdtsc_to_ms = ( 1000.0f * ( d1 - d0 ) ) / ( t1 - t0 );
}
double flTotalD3DCallMS = g_nTotalD3DCycles * s_rdtsc_to_ms;
m_batch_vis_bitmap.additive_fill_box(0, m_nBatchVisY, (uint)(.5f + flTotalD3DCallMS / d3d_batch_vis_abs_scale.GetFloat() * m_batch_vis_bitmap.width()), h, 96, 96, 128);
if (m_batch_state.m_bVertexShaderChanged) m_batch_vis_bitmap.additive_fill_box(0, m_nBatchVisY, 8, h, 0, 0, 64);
if (m_batch_state.m_bPixelShaderChanged) m_batch_vis_bitmap.additive_fill_box(32, m_nBatchVisY, 8, h, 64, 0, 64);
int lm = 80;
m_batch_vis_bitmap.fill_box(lm+0+0, m_nBatchVisY, m_batch_state.m_nNumVSConstants, h, 64, 255, 255);
m_batch_vis_bitmap.fill_box(lm+64+256+0, m_nBatchVisY, m_batch_state.m_nNumPSConstants, h, 64, 64, 255);
m_batch_vis_bitmap.fill_box(lm+64+256+32, m_nBatchVisY, m_batch_state.m_nNumSamplersChanged, h, 255, 255, 255);
m_batch_vis_bitmap.fill_box(lm+64+256+32+16, m_nBatchVisY, m_batch_state.m_nNumSamplerStatesChanged, h, 92, 128, 255);
if ( m_batch_state.m_bStreamSourceChanged) m_batch_vis_bitmap.fill_box(lm+64+256+32+16+64, m_nBatchVisY, 16, h, 128, 128, 128);
if ( m_batch_state.m_bIndicesChanged ) m_batch_vis_bitmap.fill_box(lm+64+256+32+16+64+16, m_nBatchVisY, 16, h, 128, 128, 255);
m_nBatchVisY += h;
m_nTotalD3DCalls += g_nTotalD3DCalls;
m_flTotalD3DTime += flTotalD3DCallMS;
g_nTotalD3DCycles = 0;
g_nTotalD3DCalls = 0;
m_batch_state.Clear();
}
#endif
return hres;
}
STDMETHOD(DrawPrimitiveUP)(THIS_ D3DPRIMITIVETYPE PrimitiveType,UINT PrimitiveCount,CONST void* pVertexStreamZeroData,UINT VertexStreamZeroStride)
{ XXX; m_nTotalDraws++; m_nTotalPrims += PrimitiveCount; return _DOCALL(DrawPrimitiveUP, PrimitiveType, PrimitiveCount, pVertexStreamZeroData, VertexStreamZeroStride); }
STDMETHOD(DrawIndexedPrimitiveUP)(THIS_ D3DPRIMITIVETYPE PrimitiveType,UINT MinVertexIndex,UINT NumVertices,UINT PrimitiveCount,CONST void* pIndexData,D3DFORMAT IndexDataFormat,CONST void* pVertexStreamZeroData,UINT VertexStreamZeroStride)
{ XXX; m_nTotalDraws++; m_nTotalPrims += PrimitiveCount; return _DOCALL(DrawIndexedPrimitiveUP, PrimitiveType, MinVertexIndex, NumVertices, PrimitiveCount, pIndexData, IndexDataFormat, pVertexStreamZeroData, VertexStreamZeroStride); }
STDMETHOD(ProcessVertices)(THIS_ UINT SrcStartIndex,UINT DestIndex,UINT VertexCount,IDirect3DVertexBuffer9* pDestBuffer,IDirect3DVertexDeclaration9* pVertexDecl,DWORD Flags)
{ XXX; return _DOCALL(ProcessVertices, SrcStartIndex, DestIndex, VertexCount, GetHWPtr(pDestBuffer), GetHWPtr(pVertexDecl), Flags); }
STDMETHOD(CreateVertexDeclaration)(THIS_ CONST D3DVERTEXELEMENT9* pVertexElements,IDirect3DVertexDeclaration9** ppDecl)
{
XXX;
HRESULT hr = _DOCALL(CreateVertexDeclaration, pVertexElements, ppDecl);
return AllocOverride(&hr, this, ppDecl);
}
STDMETHOD(SetVertexDeclaration)(THIS_ IDirect3DVertexDeclaration9* pDecl)
{ XXX; return _DOCALL(SetVertexDeclaration, GetHWPtr(pDecl)); }
STDMETHOD(GetVertexDeclaration)(THIS_ IDirect3DVertexDeclaration9** ppDecl)
{
XXX;
HRESULT hr = _DOCALL(GetVertexDeclaration, ppDecl);
return AllocOverride(&hr, this, ppDecl);
}
STDMETHOD(SetFVF)(THIS_ DWORD FVF)
{ XXX; return _DOCALL(SetFVF, FVF); }
STDMETHOD(GetFVF)(THIS_ DWORD* pFVF)
{ XXX; return _DOCALL(GetFVF, pFVF); }
STDMETHOD(CreateVertexShader)(THIS_ CONST DWORD* pFunction,IDirect3DVertexShader9** ppShader)
{
XXX;
HRESULT hr = _DOCALL(CreateVertexShader, pFunction, ppShader);
return AllocOverride(&hr, this, ppShader);
}
STDMETHOD(SetVertexShader)(THIS_ IDirect3DVertexShader9* pShader)
{
HRESULT hres;
{
XXX;
hres = _DOCALL(SetVertexShader, GetHWPtr(pShader));
}
D3D_BATCH_PERF( m_batch_state.m_bVertexShaderChanged = true; )
return hres;
}
STDMETHOD(GetVertexShader)(THIS_ IDirect3DVertexShader9** ppShader)
{
XXX;
HRESULT hr = _DOCALL(GetVertexShader, ppShader);
return AllocOverride(&hr, this, ppShader);
}
STDMETHOD(SetVertexShaderConstantF)(THIS_ UINT StartRegister,CONST float* pConstantData,UINT Vector4fCount)
{
HRESULT hres;
{
XXX;
hres = _DOCALL(SetVertexShaderConstantF, StartRegister, pConstantData, Vector4fCount);
}
D3D_BATCH_PERF( m_batch_state.m_nNumVSConstants += Vector4fCount; )
return hres;
}
STDMETHOD(GetVertexShaderConstantF)(THIS_ UINT StartRegister,float* pConstantData,UINT Vector4fCount)
{ XXX; return _DOCALL(GetVertexShaderConstantF, StartRegister, pConstantData, Vector4fCount);}
STDMETHOD(SetVertexShaderConstantI)(THIS_ UINT StartRegister,CONST int* pConstantData,UINT Vector4iCount)
{ XXX; return _DOCALL(SetVertexShaderConstantI, StartRegister, pConstantData, Vector4iCount); }
STDMETHOD(GetVertexShaderConstantI)(THIS_ UINT StartRegister,int* pConstantData,UINT Vector4iCount)
{ XXX; return _DOCALL(GetVertexShaderConstantI, StartRegister, pConstantData, Vector4iCount); }
STDMETHOD(SetVertexShaderConstantB)(THIS_ UINT StartRegister,CONST BOOL* pConstantData,UINT BoolCount)
{ XXX; return _DOCALL(SetVertexShaderConstantB, StartRegister, pConstantData, BoolCount); }
STDMETHOD(GetVertexShaderConstantB)(THIS_ UINT StartRegister,BOOL* pConstantData,UINT BoolCount)
{ XXX; return _DOCALL(GetVertexShaderConstantB, StartRegister, pConstantData, BoolCount); }
STDMETHOD(SetStreamSource)(THIS_ UINT StreamNumber,IDirect3DVertexBuffer9* pStreamData,UINT OffsetInBytes,UINT Stride)
{
HRESULT hres;
{
XXX;
hres = _DOCALL(SetStreamSource, StreamNumber, GetHWPtr(pStreamData), OffsetInBytes, Stride);
}
D3D_BATCH_PERF( m_batch_state.m_bStreamSourceChanged = true; )
return hres;
}
STDMETHOD(GetStreamSource)(THIS_ UINT StreamNumber,IDirect3DVertexBuffer9** ppStreamData,UINT* OffsetInBytes,UINT* pStride)
{
XXX;
HRESULT hr = _DOCALL(GetStreamSource, StreamNumber, ppStreamData, OffsetInBytes, pStride);
return AllocOverride(&hr, this, ppStreamData);
}
STDMETHOD(SetStreamSourceFreq)(THIS_ UINT StreamNumber,UINT Divider)
{ XXX; return _DOCALL(SetStreamSourceFreq, StreamNumber, Divider); }
STDMETHOD(GetStreamSourceFreq)(THIS_ UINT StreamNumber,UINT* Divider)
{ XXX; return _DOCALL(GetStreamSourceFreq, StreamNumber, Divider); }
STDMETHOD(SetIndices)(THIS_ IDirect3DIndexBuffer9* pIndexData)
{
HRESULT hres;
{
XXX;
hres = _DOCALL(SetIndices, GetHWPtr(pIndexData));
}
D3D_BATCH_PERF( m_batch_state.m_bIndicesChanged = true; )
return hres;
}
STDMETHOD(GetIndices)(THIS_ IDirect3DIndexBuffer9** ppIndexData)
{
XXX;
HRESULT hr = _DOCALL(GetIndices, ppIndexData);
return AllocOverride(&hr, this, ppIndexData);
}
STDMETHOD(CreatePixelShader)(THIS_ CONST DWORD* pFunction,IDirect3DPixelShader9** ppShader)
{
XXX;
HRESULT hr = _DOCALL(CreatePixelShader, pFunction, ppShader);
return AllocOverride(&hr, this, ppShader);
}
STDMETHOD(SetPixelShader)(THIS_ IDirect3DPixelShader9* pShader)
{
HRESULT hres;
{
XXX;
hres = _DOCALL(SetPixelShader, GetHWPtr(pShader));
}
D3D_BATCH_PERF( m_batch_state.m_bPixelShaderChanged = true; )
return hres;
}
STDMETHOD(GetPixelShader)(THIS_ IDirect3DPixelShader9** ppShader)
{
XXX;
HRESULT hr = _DOCALL(GetPixelShader, ppShader);
return AllocOverride(&hr, this, ppShader);
}
STDMETHOD(SetPixelShaderConstantF)(THIS_ UINT StartRegister,CONST float* pConstantData,UINT Vector4fCount)
{
HRESULT hres;
{
XXX;
hres = _DOCALL(SetPixelShaderConstantF, StartRegister, pConstantData, Vector4fCount);
}
D3D_BATCH_PERF( m_batch_state.m_nNumPSConstants += Vector4fCount; )
return hres;
}
STDMETHOD(GetPixelShaderConstantF)(THIS_ UINT StartRegister,float* pConstantData,UINT Vector4fCount)
{ XXX; return _DOCALL(GetPixelShaderConstantF, StartRegister, pConstantData, Vector4fCount); }
STDMETHOD(SetPixelShaderConstantI)(THIS_ UINT StartRegister,CONST int* pConstantData,UINT Vector4iCount)
{ XXX; return _DOCALL(SetPixelShaderConstantI, StartRegister, pConstantData, Vector4iCount); }
STDMETHOD(GetPixelShaderConstantI)(THIS_ UINT StartRegister,int* pConstantData,UINT Vector4iCount)
{ XXX; return _DOCALL(GetPixelShaderConstantI, StartRegister, pConstantData, Vector4iCount); }
STDMETHOD(SetPixelShaderConstantB)(THIS_ UINT StartRegister,CONST BOOL* pConstantData,UINT BoolCount)
{ XXX; return _DOCALL(SetPixelShaderConstantB, StartRegister, pConstantData, BoolCount); }
STDMETHOD(GetPixelShaderConstantB)(THIS_ UINT StartRegister,BOOL* pConstantData,UINT BoolCount)
{ XXX; return _DOCALL(GetPixelShaderConstantB, StartRegister, pConstantData, BoolCount); }
STDMETHOD(DrawRectPatch)(THIS_ UINT Handle,CONST float* pNumSegs,CONST D3DRECTPATCH_INFO* pRectPatchInfo)
{ XXX; return _DOCALL(DrawRectPatch, Handle, pNumSegs, pRectPatchInfo);}
STDMETHOD(DrawTriPatch)(THIS_ UINT Handle,CONST float* pNumSegs,CONST D3DTRIPATCH_INFO* pTriPatchInfo)
{ XXX; return _DOCALL(DrawTriPatch, Handle, pNumSegs, pTriPatchInfo); }
STDMETHOD(DeletePatch)(THIS_ UINT Handle)
{ XXX; return _DOCALL(DeletePatch, Handle);}
STDMETHOD(CreateQuery)(THIS_ D3DQUERYTYPE Type,IDirect3DQuery9** ppQuery)
{
XXX;
HRESULT hr = _DOCALL(CreateQuery, Type, ppQuery);
return AllocOverride(&hr, this, ppQuery);
}
private:
#if D3D_BATCH_PERF_ANALYSIS
IDirect3DSurface9* m_pPrevRenderTarget0;
simple_bitmap m_batch_vis_bitmap;
uint m_nBatchVisY;
uint m_nBatchVisFrameIndex, m_nBatchVisFileIdx;
struct BatchState_t
{
void Clear() { memset(this, 0, sizeof(*this)); }
bool m_bStreamSourceChanged;
bool m_bIndicesChanged;
bool m_bPixelShaderChanged;
bool m_bVertexShaderChanged;
uint m_nNumPSConstants;
uint m_nNumVSConstants;
uint m_nNumSamplersChanged;
uint m_nNumSamplerStatesChanged;
};
BatchState_t m_batch_state;
uint m_nTotalFrames;
uint m_nTotalDraws;
uint m_nTotalPrims;
uint m_nTotalD3DCalls;
double m_flTotalD3DTime;
uint m_nOverallDraws;
uint m_nOverallPrims;
uint m_nOverallD3DCalls;
double m_flOverallD3DTime;
#endif
};
class CDirect3D9Hook : public CDx9HookBase<IDirect3D9>, public IDirect3D9
{
public:
/*** IUnknown methods ***/
IMPL_IUNKOWN();
// IDirect3D9 methods
STDMETHOD(RegisterSoftwareDevice)(THIS_ void* pInitializeFunction)
{ XXX; return _DOCALL(RegisterSoftwareDevice, pInitializeFunction); }
STDMETHOD_(UINT, GetAdapterCount)(THIS)
{ XXX; return _DOCALL0(GetAdapterCount); }
STDMETHOD(GetAdapterIdentifier)(THIS_ UINT Adapter,DWORD Flags,D3DADAPTER_IDENTIFIER9* pIdentifier)
{ XXX; return _DOCALL(GetAdapterIdentifier, Adapter, Flags, pIdentifier);}
STDMETHOD_(UINT, GetAdapterModeCount)(THIS_ UINT Adapter,D3DFORMAT Format)
{ XXX; return _DOCALL(GetAdapterModeCount, Adapter, Format);}
STDMETHOD(EnumAdapterModes)(THIS_ UINT Adapter,D3DFORMAT Format,UINT Mode,D3DDISPLAYMODE* pMode)
{ XXX; return _DOCALL(EnumAdapterModes, Adapter, Format, Mode, pMode); }
STDMETHOD(GetAdapterDisplayMode)(THIS_ UINT Adapter,D3DDISPLAYMODE* pMode)
{ XXX; return _DOCALL(GetAdapterDisplayMode, Adapter, pMode); }
STDMETHOD(CheckDeviceType)(THIS_ UINT iAdapter,D3DDEVTYPE DevType,D3DFORMAT DisplayFormat,D3DFORMAT BackBufferFormat,BOOL bWindowed)
{ XXX; return _DOCALL(CheckDeviceType, iAdapter, DevType, DisplayFormat, BackBufferFormat, bWindowed); }
STDMETHOD(CheckDeviceFormat)(THIS_ UINT Adapter,D3DDEVTYPE DeviceType,D3DFORMAT AdapterFormat,DWORD Usage,D3DRESOURCETYPE RType,D3DFORMAT CheckFormat)
{ XXX; return _DOCALL(CheckDeviceFormat, Adapter, DeviceType, AdapterFormat, Usage, RType, CheckFormat); }
STDMETHOD(CheckDeviceMultiSampleType)(THIS_ UINT Adapter,D3DDEVTYPE DeviceType,D3DFORMAT SurfaceFormat,BOOL Windowed,D3DMULTISAMPLE_TYPE MultiSampleType,DWORD* pQualityLevels)
{ XXX; return _DOCALL(CheckDeviceMultiSampleType, Adapter, DeviceType, SurfaceFormat, Windowed, MultiSampleType, pQualityLevels); }
STDMETHOD(CheckDepthStencilMatch)(THIS_ UINT Adapter,D3DDEVTYPE DeviceType,D3DFORMAT AdapterFormat,D3DFORMAT RenderTargetFormat,D3DFORMAT DepthStencilFormat)
{ XXX; return _DOCALL(CheckDepthStencilMatch, Adapter, DeviceType, AdapterFormat, RenderTargetFormat, DepthStencilFormat); }
STDMETHOD(CheckDeviceFormatConversion)(THIS_ UINT Adapter,D3DDEVTYPE DeviceType,D3DFORMAT SourceFormat,D3DFORMAT TargetFormat)
{ XXX; return _DOCALL(CheckDeviceFormatConversion, Adapter, DeviceType, SourceFormat, TargetFormat); }
STDMETHOD(GetDeviceCaps)(THIS_ UINT Adapter,D3DDEVTYPE DeviceType,D3DCAPS9* pCaps)
{ XXX; return _DOCALL(GetDeviceCaps, Adapter, DeviceType, pCaps); }
STDMETHOD_(HMONITOR, GetAdapterMonitor)(THIS_ UINT Adapter)
{ XXX; return _DOCALL(GetAdapterMonitor, Adapter); }
STDMETHOD(CreateDevice)(THIS_ UINT Adapter,D3DDEVTYPE DeviceType,HWND hFocusWindow,DWORD BehaviorFlags,D3DPRESENT_PARAMETERS* pPresentationParameters,IDirect3DDevice9** ppReturnedDeviceInterface)
{
XXX;
HRESULT hr = _DOCALL(CreateDevice, Adapter, DeviceType, hFocusWindow, BehaviorFlags, pPresentationParameters, ppReturnedDeviceInterface);
if( SUCCEEDED( hr ) && ppReturnedDeviceInterface && *ppReturnedDeviceInterface )
{
CDirect3DDevice9Hook *pDevice = new CDirect3DDevice9Hook;
if(!pDevice)
{
( *ppReturnedDeviceInterface )->Release();
hr = E_OUTOFMEMORY;
return NULL;
}
pDevice->m_Data.pDevice = pDevice;
pDevice->m_Data.pHWObj = *ppReturnedDeviceInterface;
*ppReturnedDeviceInterface = pDevice;
}
return hr;
}
};
inline IDirect3D9 *Direct3DCreate9Hook( UINT SDKVersion )
{
HRESULT hr = S_OK;
IDirect3D9 *pD3D = Direct3DCreate9( D3D_SDK_VERSION );
AllocOverride( &hr, NULL, &pD3D );
return pD3D;
}
#endif // _DX9HOOK_H_