2018-01-23 09:23:31 +01:00
|
|
|
#include "d3d11_context_def.h"
|
|
|
|
|
|
|
|
namespace dxvk {
|
|
|
|
|
|
|
|
D3D11DeferredContext::D3D11DeferredContext(
|
|
|
|
D3D11Device* pParent,
|
|
|
|
Rc<DxvkDevice> Device,
|
|
|
|
UINT ContextFlags)
|
|
|
|
: D3D11DeviceContext(pParent, Device),
|
2018-01-23 12:03:26 +01:00
|
|
|
m_contextFlags(ContextFlags),
|
|
|
|
m_commandList (CreateCommandList()) {
|
2018-03-03 20:59:17 +01:00
|
|
|
ClearState();
|
2018-01-23 09:23:31 +01:00
|
|
|
}
|
|
|
|
|
|
|
|
|
|
|
|
D3D11_DEVICE_CONTEXT_TYPE STDMETHODCALLTYPE D3D11DeferredContext::GetType() {
|
|
|
|
return D3D11_DEVICE_CONTEXT_DEFERRED;
|
|
|
|
}
|
|
|
|
|
|
|
|
|
|
|
|
UINT STDMETHODCALLTYPE D3D11DeferredContext::GetContextFlags() {
|
|
|
|
return m_contextFlags;
|
|
|
|
}
|
|
|
|
|
|
|
|
|
|
|
|
void STDMETHODCALLTYPE D3D11DeferredContext::Flush() {
|
|
|
|
Logger::err("D3D11: Flush called on a deferred context");
|
|
|
|
}
|
|
|
|
|
|
|
|
|
|
|
|
void STDMETHODCALLTYPE D3D11DeferredContext::ExecuteCommandList(
|
|
|
|
ID3D11CommandList* pCommandList,
|
2018-03-06 18:34:34 +01:00
|
|
|
BOOL RestoreContextState) {
|
2018-03-12 23:36:55 +01:00
|
|
|
FlushCsChunk();
|
|
|
|
|
2018-03-03 20:59:17 +01:00
|
|
|
static_cast<D3D11CommandList*>(pCommandList)->EmitToCommandList(m_commandList.ptr());
|
|
|
|
|
|
|
|
if (RestoreContextState)
|
|
|
|
RestoreState();
|
|
|
|
else
|
|
|
|
ClearState();
|
2018-01-23 09:23:31 +01:00
|
|
|
}
|
|
|
|
|
|
|
|
|
|
|
|
HRESULT STDMETHODCALLTYPE D3D11DeferredContext::FinishCommandList(
|
2018-03-06 18:34:34 +01:00
|
|
|
BOOL RestoreDeferredContextState,
|
2018-01-23 09:23:31 +01:00
|
|
|
ID3D11CommandList **ppCommandList) {
|
2018-03-12 23:36:55 +01:00
|
|
|
FlushCsChunk();
|
|
|
|
|
2018-03-03 22:28:30 +01:00
|
|
|
if (ppCommandList != nullptr)
|
|
|
|
*ppCommandList = m_commandList.ref();
|
2018-03-03 20:59:17 +01:00
|
|
|
m_commandList = CreateCommandList();
|
|
|
|
|
2018-03-10 14:41:06 +01:00
|
|
|
if (RestoreDeferredContextState)
|
2018-03-03 22:28:30 +01:00
|
|
|
RestoreState();
|
2018-03-10 14:41:06 +01:00
|
|
|
else
|
|
|
|
ClearState();
|
2018-03-03 20:59:17 +01:00
|
|
|
|
2018-03-19 03:19:13 +01:00
|
|
|
m_mappedResources.clear();
|
2018-03-03 20:59:17 +01:00
|
|
|
return S_OK;
|
2018-01-23 09:23:31 +01:00
|
|
|
}
|
|
|
|
|
|
|
|
|
|
|
|
HRESULT STDMETHODCALLTYPE D3D11DeferredContext::Map(
|
|
|
|
ID3D11Resource* pResource,
|
|
|
|
UINT Subresource,
|
|
|
|
D3D11_MAP MapType,
|
|
|
|
UINT MapFlags,
|
|
|
|
D3D11_MAPPED_SUBRESOURCE* pMappedResource) {
|
2018-03-19 03:19:13 +01:00
|
|
|
D3D11_RESOURCE_DIMENSION resourceDim = D3D11_RESOURCE_DIMENSION_UNKNOWN;
|
|
|
|
pResource->GetType(&resourceDim);
|
|
|
|
|
|
|
|
if (MapType != D3D11_MAP_WRITE_DISCARD
|
|
|
|
&& MapType != D3D11_MAP_WRITE_NO_OVERWRITE)
|
|
|
|
return E_INVALIDARG;
|
|
|
|
|
|
|
|
if (resourceDim == D3D11_RESOURCE_DIMENSION_BUFFER) {
|
|
|
|
return MapBuffer(
|
|
|
|
static_cast<D3D11Buffer*>(pResource),
|
|
|
|
MapType, MapFlags, pMappedResource);
|
|
|
|
} else {
|
|
|
|
return MapImage(
|
|
|
|
GetCommonTexture(pResource),
|
|
|
|
Subresource, MapType, MapFlags,
|
|
|
|
pMappedResource);
|
|
|
|
}
|
2018-01-23 09:23:31 +01:00
|
|
|
}
|
|
|
|
|
|
|
|
|
|
|
|
void STDMETHODCALLTYPE D3D11DeferredContext::Unmap(
|
|
|
|
ID3D11Resource* pResource,
|
|
|
|
UINT Subresource) {
|
2018-03-19 03:19:13 +01:00
|
|
|
D3D11_RESOURCE_DIMENSION resourceDim = D3D11_RESOURCE_DIMENSION_UNKNOWN;
|
|
|
|
pResource->GetType(&resourceDim);
|
|
|
|
|
|
|
|
if (resourceDim == D3D11_RESOURCE_DIMENSION_BUFFER) {
|
|
|
|
UnmapBuffer(static_cast<D3D11Buffer*>(pResource));
|
|
|
|
} else {
|
|
|
|
UnmapImage(
|
|
|
|
GetCommonTexture(pResource),
|
|
|
|
Subresource);
|
|
|
|
}
|
|
|
|
}
|
|
|
|
|
|
|
|
|
|
|
|
HRESULT D3D11DeferredContext::MapBuffer(
|
|
|
|
D3D11Buffer* pResource,
|
|
|
|
D3D11_MAP MapType,
|
|
|
|
UINT MapFlags,
|
|
|
|
D3D11_MAPPED_SUBRESOURCE* pMappedResource) {
|
|
|
|
Rc<DxvkBuffer> buffer = pResource->GetBuffer();
|
|
|
|
|
|
|
|
if (!(buffer->memFlags() & VK_MEMORY_PROPERTY_HOST_VISIBLE_BIT)) {
|
|
|
|
Logger::err("D3D11: Cannot map a device-local buffer");
|
|
|
|
return E_INVALIDARG;
|
|
|
|
}
|
|
|
|
|
|
|
|
if (pMappedResource == nullptr)
|
|
|
|
return S_FALSE;
|
|
|
|
|
|
|
|
auto entry = FindMapEntry(pResource, 0);
|
|
|
|
|
|
|
|
if (MapType == D3D11_MAP_WRITE_DISCARD) {
|
|
|
|
if (entry != m_mappedResources.end())
|
|
|
|
m_mappedResources.erase(entry);
|
|
|
|
|
|
|
|
D3D11DeferredContextMapEntry mapEntry;
|
|
|
|
mapEntry.pResource = pResource;
|
|
|
|
mapEntry.Subresource = 0;
|
|
|
|
mapEntry.MapType = D3D11_MAP_WRITE_DISCARD;
|
|
|
|
mapEntry.RowPitch = pResource->GetSize();
|
|
|
|
mapEntry.DepthPitch = pResource->GetSize();
|
|
|
|
mapEntry.DataSlice = AllocUpdateBufferSlice(pResource->GetSize());
|
|
|
|
m_mappedResources.push_back(mapEntry);
|
|
|
|
|
|
|
|
pMappedResource->pData = mapEntry.DataSlice.ptr();
|
|
|
|
pMappedResource->RowPitch = mapEntry.RowPitch;
|
|
|
|
pMappedResource->DepthPitch = mapEntry.DepthPitch;
|
|
|
|
return S_OK;
|
|
|
|
} else {
|
|
|
|
// The resource must be mapped with D3D11_MAP_WRITE_DISCARD
|
|
|
|
// before it can be mapped with D3D11_MAP_WRITE_NO_OVERWRITE.
|
|
|
|
if (entry == m_mappedResources.end())
|
|
|
|
return E_INVALIDARG;
|
|
|
|
|
|
|
|
// Return same memory region as earlier
|
|
|
|
entry->MapType = D3D11_MAP_WRITE_NO_OVERWRITE;
|
|
|
|
|
|
|
|
pMappedResource->pData = entry->DataSlice.ptr();
|
|
|
|
pMappedResource->RowPitch = entry->RowPitch;
|
|
|
|
pMappedResource->DepthPitch = entry->DepthPitch;
|
|
|
|
return S_OK;
|
|
|
|
}
|
|
|
|
}
|
|
|
|
|
|
|
|
|
|
|
|
HRESULT D3D11DeferredContext::MapImage(
|
|
|
|
D3D11CommonTexture* pResource,
|
|
|
|
UINT Subresource,
|
|
|
|
D3D11_MAP MapType,
|
|
|
|
UINT MapFlags,
|
|
|
|
D3D11_MAPPED_SUBRESOURCE* pMappedResource) {
|
|
|
|
Logger::err("D3D11DeferredContext::MapImage: Not implemented");
|
|
|
|
return E_NOTIMPL;
|
|
|
|
}
|
|
|
|
|
|
|
|
|
|
|
|
void D3D11DeferredContext::UnmapBuffer(
|
|
|
|
D3D11Buffer* pResource) {
|
|
|
|
auto entry = FindMapEntry(pResource, 0);
|
|
|
|
|
|
|
|
if (entry == m_mappedResources.end()) {
|
|
|
|
Logger::err("D3D11DeferredContext::Unmap: Buffer not mapped");
|
|
|
|
return;
|
|
|
|
}
|
|
|
|
|
|
|
|
if (entry->MapType == D3D11_MAP_WRITE_DISCARD) {
|
|
|
|
EmitCs([
|
|
|
|
cDstBuffer = pResource->GetBuffer(),
|
|
|
|
cDataSlice = entry->DataSlice
|
|
|
|
] (DxvkContext* ctx) {
|
|
|
|
DxvkPhysicalBufferSlice slice = cDstBuffer->allocPhysicalSlice();
|
|
|
|
std::memcpy(slice.mapPtr(0), cDataSlice.ptr(), cDataSlice.length());
|
|
|
|
ctx->invalidateBuffer(cDstBuffer, slice);
|
|
|
|
});
|
|
|
|
}
|
|
|
|
}
|
|
|
|
|
|
|
|
|
|
|
|
void D3D11DeferredContext::UnmapImage(
|
|
|
|
D3D11CommonTexture* pResource,
|
|
|
|
UINT Subresource) {
|
|
|
|
Logger::err("D3D11DeferredContext::UnmapImage: Not implemented");
|
2018-01-23 09:23:31 +01:00
|
|
|
}
|
|
|
|
|
|
|
|
|
2018-01-23 12:03:26 +01:00
|
|
|
Com<D3D11CommandList> D3D11DeferredContext::CreateCommandList() {
|
|
|
|
return new D3D11CommandList(m_parent, m_contextFlags);
|
|
|
|
}
|
|
|
|
|
|
|
|
|
|
|
|
void D3D11DeferredContext::EmitCsChunk(Rc<DxvkCsChunk>&& chunk) {
|
|
|
|
m_commandList->AddChunk(std::move(chunk));
|
2018-01-23 09:23:31 +01:00
|
|
|
}
|
|
|
|
|
|
|
|
}
|