D3D11:映射顶点缓冲区时E_OUTOFMEMORY

D3D11: E_OUTOFMEMORY when mapping vertex buffer

本文关键字:OUTOFMEMORY 缓冲区 映射 顶点 D3D11      更新时间:2023-10-16

在我的Unity游戏中,我必须通过本机代码修改许多图形资源,如纹理和顶点缓冲区,以保持良好的性能。

当代码在很短的时间内多次调用 ID3D11ImmediateContext::Map 时,问题就开始了(我的意思是非常短 - 从并行运行的不同线程调用(。映射是否成功没有规则。方法调用看起来像

ID3D11DeviceContext* sU_m_D_context;

void* BeginModifyingVBO(void* bufferHandle)
{
ID3D11Buffer* d3dbuf = static_cast<ID3D11Buffer*>(bufferHandle);
D3D11_MAPPED_SUBRESOURCE mapped;
HRESULT res = sU_m_D_context->Map(d3dbuf, 0, D3D11_MAP_WRITE_DISCARD, 0, &mapped);
assert(mapped.pData);
return mapped.pData;
}

void FinishModifyingVBO(void* bufferHandle)
{
ID3D11Buffer* d3dbuf = static_cast<ID3D11Buffer*>(bufferHandle);
sU_m_D_context->Unmap(d3dbuf, 0);
}

std::mutex sU_m_D_locker;
void Mesh::ApplyBuffer()
{
sU_m_D_locker.lock();
// map buffer
VBVertex* mappedBuffer = (VBVertex*)BeginModifyingVBO(this->currentBufferPtr);
memcpy(mappedBuffer, this->mainBuffer, this->mainBufferLength * sizeof(VBVertex));
// unmap buffer
FinishModifyingVBO(this->currentBufferPtr);
sU_m_D_locker.unlock();
this->markedAsChanged = false;
}

其中 d3dbuf 是动态顶点缓冲区。我不知道为什么,但有时结果是E_OUTOFMEMORY,尽管有很多空闲内存。我试图用互斥体包围代码 - 没有效果。

这真的是记忆问题还是不太明显?

没有一个设备上下文方法是线程安全的。如果要从多个线程使用它们,则需要手动同步所有调用,或使用多个(延迟(上下文,每个线程一个。请参阅 Direct3D 11 中的多线程处理简介。

此外,错误检查应该更好:您需要始终检查返回的HRESULT值,因为在失败的情况下,像assert(mapped.pData);这样的东西仍然可能成功。