- html - 出于某种原因,IE8 对我的 Sass 文件中继承的 html5 CSS 不友好?
- JMeter 在响应断言中使用 span 标签的问题
- html - 在 :hover and :active? 上具有不同效果的 CSS 动画
- html - 相对于居中的 html 内容固定的 CSS 重复背景?
我正在编写一个基于 DirectX 11 的引擎。所有 DirectX Com 对象都包含在 Microsoft::WRL::ComPtr 中。但不幸的是,当我在关机时调用 ID3D11Debug::ReportLiveDeviceObjects 时,它报告了超过 1 个对象的引用计数非零。
我很困惑,哪些代码实际上增加了那些引用计数,以防止引用最后为零。例如,我在我的代码中提供了 ID3D11ShaderResourceView 的示例用法:
ID3D11ShaderResourceView 在我的渲染目标类中声明为成员,
Microsoft::WRL::ComPtr<ID3D11ShaderResourceView> m_pShaderResourceView
它在构造函数中被初始化为 nullptr:
m_pShaderResourceView(wwNULLPTR)
界面创建如下:
device->CreateShaderResourceView(m_pDXTexture.Get(), &shaderResourceDesc, m_pShaderResourceView.ReleaseAndGetAddressOf());
GenerateMips函数中使用的接口(interface):
deviceContext->GenerateMips(m_pShaderResourceView.Get());
接口(interface)有一个getter函数:
ID3D11ShaderResourceView* GetShaderResourceView()
{
return m_pShaderResourceView.Get();
}
最后在 descstructor 中重置了界面:
m_pShaderResourceView.Reset();
最后,我在事件对象摘要中至少获得了 8 个引用。
我的问题是:
我的代码的哪一部分实际上增加了引用计数?为什么调用 Reset 不会使引用为零?可能我遗漏了什么?
谢谢。
最佳答案
ID3D11Debug::ReportLiveDeviceObjects
是一个非常有用的调试工具,但它确实有一些怪癖。具体来说,一些对象具有您无法直接控制的生命,它们是 ID3D11Device 本身的一部分。您无法在不关闭设备的情况下删除引用,此时您无法获得事件对象报告。
在DXUT for Direct3D 11框架,我通过调用 ClearState
尽可能地降低它,然后在调用报告事件对象之前在即时上下文中调用 Flush
。
即使在“干净”的导出中,您仍然会有一些挥之不去的“事件”对象:
D3D11 WARNING: Live ID3D11Device at 0x025CA03C, Refcount: 5 [ STATE_CREATION WARNING #441: LIVE_DEVICE]
D3D11 WARNING: Live ID3D11Context at 0x02831030, Refcount: 0, IntRef: 1 [ STATE_CREATION WARNING #2097226: LIVE_CONTEXT]
D3D11 WARNING: Live ID3DDeviceContextState at 0x02828038, Refcount: 0, IntRef: 1 [ STATE_CREATION WARNING #3145742: LIVE_DEVICECONTEXTSTATE]
D3D11 WARNING: Live ID3D11BlendState at 0x005B766C, Refcount: 0, IntRef: 1 [ STATE_CREATION WARNING #435: LIVE_BLENDSTATE]
D3D11 WARNING: Live ID3D11DepthStencilState at 0x0283ECAC, Refcount: 0, IntRef: 1 [ STATE_CREATION WARNING #436: LIVE_DEPTHSTENCILSTATE]
D3D11 WARNING: Live ID3D11RasterizerState at 0x027006F4, Refcount: 0, IntRef: 1 [ STATE_CREATION WARNING #437: LIVE_RASTERIZERSTATE]
D3D11 WARNING: Live ID3D11Sampler at 0x0270082C, Refcount: 0, IntRef: 1 [ STATE_CREATION WARNING #434: LIVE_SAMPLER]
D3D11 WARNING: Live ID3D11Query at 0x005BB904, Refcount: 0, IntRef: 1 [ STATE_CREATION WARNING #438: LIVE_QUERY]
D3D11 WARNING: Live ID3D11Context : 1 [ STATE_CREATION WARNING #422: LIVE_OBJECT_SUMMARY]
D3D11 WARNING: Live ID3DDeviceContextState : 1 [ STATE_CREATION WARNING #422: LIVE_OBJECT_SUMMARY]
D3D11 WARNING: Live ID3D11BlendState : 1 [ STATE_CREATION WARNING #422: LIVE_OBJECT_SUMMARY]
D3D11 WARNING: Live ID3D11DepthStencilState : 1 [ STATE_CREATION WARNING #422: LIVE_OBJECT_SUMMARY]
D3D11 WARNING: Live ID3D11RasterizerState : 1 [ STATE_CREATION WARNING #422: LIVE_OBJECT_SUMMARY]
D3D11 WARNING: Live ID3D11Sampler : 1 [ STATE_CREATION WARNING #422: LIVE_OBJECT_SUMMARY]
D3D11 WARNING: Live ID3D11Query : 1 [ STATE_CREATION WARNING #422: LIVE_OBJECT_SUMMARY]
这些是运行时本身使用的“默认”对象。理想情况下,调试运行时会在生成此报告时忽略它们,但事实并非如此。
也就是说,由于您有 8 个而不是 5 个事件对象,因此您可能会有一些残留物。首先尝试 ClearState
和 Flush
以确保您没有因为绑定(bind)到管道或由于延迟销毁而使任何东西保持事件状态。如果仍有对象,找到它们的下一步是使用“调试对象命名”,这样您就可以找出您控制的对象仍然存在。调试命名可以使用宏或模板完成:
模板
template<UINT TNameLength>
inline void SetDebugObjectName(_In_ ID3D11DeviceChild* resource, _In_z_ const char (&name)[TNameLength])
{
#if defined(_DEBUG)
resource->SetPrivateData(WKPDID_D3DDebugObjectName, TNameLength - 1, name);
#else
UNREFERENCED_PARAMETER(resource);
UNREFERENCED_PARAMETER(name);
#endif
}
宏
#ifdef _DEBUG
inline void DXUT_SetDebugName( _In_ ID3D11DeviceChild* pObj,
_In_z_ const CHAR* pstrName )
{
if ( pObj )
pObj->SetPrivateData( WKPDID_D3DDebugObjectName,
(UINT)strlen(pstrName), pstrName );
}
#else
#define DXUT_SetDebugName( pObj, pstrName )
#endif
You need to link to
dxguid.lib
to get the symbolWKPDID_D3DDebugObjectName
. "WKPDID" stands for Well Known Private Data ID. It's ironic since it isn't all that 'well known'.
关于com - 在 DirectX 代码中正确使用 ComPtr,我们在Stack Overflow上找到一个类似的问题: https://stackoverflow.com/questions/33730537/
我正在努力从一些示例中学习 DirectX 12,但我无法理解 ComPtr.As() 方法的作用。 ComPtr device; ComPtr device2; // Do Stuff with D
我正在使用 ComPtr (Microsoft::WRL) 来管理一些 DirectX11 资源。如何手动释放? 如果我理解正确的话,“ReleaseAndGetAddressOf”方法只会释放指针而
我正在使用智能指针或 ComPtr。我在我的 directX 应用程序中使用它,我还没有看到其他人使用 ComPtr 在他们的代码中调用发布函数。那么智能指针是释放智能指针指向的数据还是需要我手动释放
我正在编写一个基于 DirectX 11 的引擎。所有 DirectX Com 对象都包含在 Microsoft::WRL::ComPtr 中。但不幸的是,当我在关机时调用 ID3D11Debug::
我正在编写一个基于 DirectX 11 的引擎。所有 DirectX Com 对象都包含在 Microsoft::WRL::ComPtr 中。但不幸的是,当我在关机时调用 ID3D11Debug::
我在映射顶点缓冲区时遇到问题。当我这样做时: ID3D11Buffer* pD3DSingleVertexBuffer; ... pD3DImmediateContext->Map(pD3DSingl
我只想知道 ComPtr 之间的确切区别和 CComPtr ,以及是否 ComPtr::As()是 CComPtr::QueryInterface() 的类似物? 我阅读了两者的文档,但这个问题没有明
我正在努力连接两个 API。从一个我得到一个 Windows::Storage::IBuffer ,另一个要求 ComPtr.我似乎没有找到一种方法来桥接这两者,即使我想在流中复制数据。 到目前为止,
我刚开始使用适用于 Windows 8 应用程序的 DirectX 11.1,例如我得到了以下 ComPtr: ComPtr constantBuffer; 我想知道的是,使用 &constantBu
尝试启动并运行 DX11 项目。为渲染的基础知识启动并运行大部分所需的对象,但是当我调用 OMSetRenderTargets 或 IASetVertexBuffers 时,我为这些函数传递的对象在函
我一直在我的代码中反复使用 ComPtr,因为我到处都需要它们,但我一直这样做: HRESULT Material::Initialize(aiMaterial* pMaterial,
我正在查看 WinRT 的一些 Direct3D 代码并注意那里的“引用类”使用“ComPtr”而不是“ID3D11Device1*”。所以我的问题是你必须在“ref 类”中使用“ComPtr”还是可
我正致力于从 DirectX12 Win32 桌面示例创建 WPF Assets (如 D3D11Image):D3D12PipelineStateCache ,所以我可以将它作为 XAML 元素嵌入
我正在学习教程,并将通常的初始化转换为使用 ComPtrs 直到这一行: ID3D11Device* g_pd3dDevice = nullptr; ID3D11Device1*
我是一名优秀的程序员,十分优秀!