[D3D12] Copy index buffer to a scratch buffer for memexporting draws

This commit is contained in:
Triang3l 2019-01-09 13:19:49 +03:00
parent aabe6dec9c
commit 8ad12480a4
3 changed files with 42 additions and 24 deletions

View File

@ -1456,16 +1456,8 @@ bool D3D12CommandProcessor::IssueDraw(PrimitiveType primitive_type,
render_target_cache_->UseEDRAMAsUAV(); render_target_cache_->UseEDRAMAsUAV();
} }
// TODO(Triang3l): Copy the index buffer to a scratch buffer if using
// memexport with an index buffer, because a resource can't be an index buffer
// (read-only) and a UAV (read/write) at once.
// Actually draw. // Actually draw.
if (indexed) { if (indexed) {
if (memexport_used) {
// TODO(Triang3l): Index buffer copying for memexport.
return false;
}
uint32_t index_size = index_buffer_info->format == IndexFormat::kInt32 uint32_t index_size = index_buffer_info->format == IndexFormat::kInt32
? sizeof(uint32_t) ? sizeof(uint32_t)
: sizeof(uint16_t); : sizeof(uint16_t);
@ -1489,6 +1481,7 @@ bool D3D12CommandProcessor::IssueDraw(PrimitiveType primitive_type,
PrimitiveConverter::ConversionResult::kPrimitiveEmpty) { PrimitiveConverter::ConversionResult::kPrimitiveEmpty) {
return true; return true;
} }
ID3D12Resource* scratch_index_buffer = nullptr;
if (conversion_result == PrimitiveConverter::ConversionResult::kConverted) { if (conversion_result == PrimitiveConverter::ConversionResult::kConverted) {
index_buffer_view.SizeInBytes = converted_index_count * index_size; index_buffer_view.SizeInBytes = converted_index_count * index_size;
index_count = converted_index_count; index_count = converted_index_count;
@ -1501,8 +1494,29 @@ bool D3D12CommandProcessor::IssueDraw(PrimitiveType primitive_type,
index_base, index_buffer_size); index_base, index_buffer_size);
return false; return false;
} }
index_buffer_view.BufferLocation = if (memexport_used && !adaptive_tessellation) {
shared_memory_->GetGPUAddress() + index_base; // If the shared memory is a UAV, it can't be used as an index buffer
// (UAV is a read/write state, index buffer is a read-only state). Need
// to copy the indices to a buffer in the index buffer state.
scratch_index_buffer = RequestScratchGPUBuffer(
index_buffer_size, D3D12_RESOURCE_STATE_COPY_DEST);
if (scratch_index_buffer == nullptr) {
return false;
}
shared_memory_->UseAsCopySource();
SubmitBarriers();
deferred_command_list_->D3DCopyBufferRegion(
scratch_index_buffer, 0, shared_memory_->GetBuffer(), index_base,
index_buffer_size);
PushTransitionBarrier(scratch_index_buffer,
D3D12_RESOURCE_STATE_COPY_DEST,
D3D12_RESOURCE_STATE_INDEX_BUFFER);
index_buffer_view.BufferLocation =
scratch_index_buffer->GetGPUVirtualAddress();
} else {
index_buffer_view.BufferLocation =
shared_memory_->GetGPUAddress() + index_base;
}
index_buffer_view.SizeInBytes = index_buffer_size; index_buffer_view.SizeInBytes = index_buffer_size;
} }
if (memexport_used) { if (memexport_used) {
@ -1510,14 +1524,18 @@ bool D3D12CommandProcessor::IssueDraw(PrimitiveType primitive_type,
} else { } else {
shared_memory_->UseForReading(); shared_memory_->UseForReading();
} }
deferred_command_list_->D3DIASetIndexBuffer(&index_buffer_view);
SubmitBarriers(); SubmitBarriers();
if (adaptive_tessellation) { if (adaptive_tessellation) {
// Index buffer used for per-edge factors. // Index buffer used for per-edge factors.
deferred_command_list_->D3DDrawInstanced(index_count, 1, 0, 0); deferred_command_list_->D3DDrawInstanced(index_count, 1, 0, 0);
} else { } else {
deferred_command_list_->D3DIASetIndexBuffer(&index_buffer_view);
deferred_command_list_->D3DDrawIndexedInstanced(index_count, 1, 0, 0, 0); deferred_command_list_->D3DDrawIndexedInstanced(index_count, 1, 0, 0, 0);
} }
if (scratch_index_buffer != nullptr) {
ReleaseScratchGPUBuffer(scratch_index_buffer,
D3D12_RESOURCE_STATE_INDEX_BUFFER);
}
} else { } else {
// Check if need to draw using a conversion index buffer. // Check if need to draw using a conversion index buffer.
uint32_t converted_index_count; uint32_t converted_index_count;

View File

@ -573,17 +573,6 @@ void SharedMemory::TransitionBuffer(D3D12_RESOURCE_STATES new_state) {
buffer_state_ = new_state; buffer_state_ = new_state;
} }
void SharedMemory::UseForReading() {
// Vertex fetch also seems to be allowed in pixel shaders.
TransitionBuffer(D3D12_RESOURCE_STATE_INDEX_BUFFER |
D3D12_RESOURCE_STATE_NON_PIXEL_SHADER_RESOURCE |
D3D12_RESOURCE_STATE_PIXEL_SHADER_RESOURCE);
}
void SharedMemory::UseForWriting() {
TransitionBuffer(D3D12_RESOURCE_STATE_UNORDERED_ACCESS);
}
void SharedMemory::CreateSRV(D3D12_CPU_DESCRIPTOR_HANDLE handle) { void SharedMemory::CreateSRV(D3D12_CPU_DESCRIPTOR_HANDLE handle) {
ui::d3d12::util::CreateRawBufferSRV( ui::d3d12::util::CreateRawBufferSRV(
command_processor_->GetD3D12Context()->GetD3D12Provider()->GetDevice(), command_processor_->GetD3D12Context()->GetD3D12Provider()->GetDevice(),

View File

@ -104,9 +104,20 @@ class SharedMemory {
void RangeWrittenByGPU(uint32_t start, uint32_t length); void RangeWrittenByGPU(uint32_t start, uint32_t length);
// Makes the buffer usable for vertices, indices and texture untiling. // Makes the buffer usable for vertices, indices and texture untiling.
void UseForReading(); inline void UseForReading() {
// Vertex fetch is also allowed in pixel shaders.
TransitionBuffer(D3D12_RESOURCE_STATE_INDEX_BUFFER |
D3D12_RESOURCE_STATE_NON_PIXEL_SHADER_RESOURCE |
D3D12_RESOURCE_STATE_PIXEL_SHADER_RESOURCE);
}
// Makes the buffer usable for texture tiling after a resolve. // Makes the buffer usable for texture tiling after a resolve.
void UseForWriting(); inline void UseForWriting() {
TransitionBuffer(D3D12_RESOURCE_STATE_UNORDERED_ACCESS);
}
// Makes the buffer usable as a source for copy commands.
inline void UseAsCopySource() {
TransitionBuffer(D3D12_RESOURCE_STATE_COPY_SOURCE);
}
void CreateSRV(D3D12_CPU_DESCRIPTOR_HANDLE handle); void CreateSRV(D3D12_CPU_DESCRIPTOR_HANDLE handle);
void CreateRawUAV(D3D12_CPU_DESCRIPTOR_HANDLE handle); void CreateRawUAV(D3D12_CPU_DESCRIPTOR_HANDLE handle);