[D3D12] Await UpdateTileMappings when syncing with the GPU
This commit is contained in:
parent
c2e8c5554d
commit
0156d3ef26
|
@ -802,7 +802,7 @@ std::unique_ptr<xe::ui::RawImage> D3D12CommandProcessor::Capture() {
|
||||||
if (!EndSubmission(false)) {
|
if (!EndSubmission(false)) {
|
||||||
return nullptr;
|
return nullptr;
|
||||||
}
|
}
|
||||||
AwaitAllSubmissionsCompletion();
|
AwaitAllQueueOperationsCompletion();
|
||||||
D3D12_RANGE readback_range;
|
D3D12_RANGE readback_range;
|
||||||
readback_range.Begin = swap_texture_copy_footprint_.Offset;
|
readback_range.Begin = swap_texture_copy_footprint_.Offset;
|
||||||
readback_range.End = swap_texture_copy_size_;
|
readback_range.End = swap_texture_copy_size_;
|
||||||
|
@ -850,26 +850,25 @@ bool D3D12CommandProcessor::SetupContext() {
|
||||||
auto device = provider.GetDevice();
|
auto device = provider.GetDevice();
|
||||||
auto direct_queue = provider.GetDirectQueue();
|
auto direct_queue = provider.GetDirectQueue();
|
||||||
|
|
||||||
submission_open_ = false;
|
fence_completion_event_ = CreateEvent(nullptr, false, false, nullptr);
|
||||||
submission_current_ = 1;
|
if (fence_completion_event_ == nullptr) {
|
||||||
submission_completed_ = 0;
|
XELOGE("Failed to create the fence completion event");
|
||||||
|
return false;
|
||||||
|
}
|
||||||
if (FAILED(device->CreateFence(0, D3D12_FENCE_FLAG_NONE,
|
if (FAILED(device->CreateFence(0, D3D12_FENCE_FLAG_NONE,
|
||||||
IID_PPV_ARGS(&submission_fence_)))) {
|
IID_PPV_ARGS(&submission_fence_)))) {
|
||||||
XELOGE("Failed to create the submission fence");
|
XELOGE("Failed to create the submission fence");
|
||||||
return false;
|
return false;
|
||||||
}
|
}
|
||||||
submission_fence_completion_event_ =
|
if (FAILED(device->CreateFence(
|
||||||
CreateEvent(nullptr, false, false, nullptr);
|
0, D3D12_FENCE_FLAG_NONE,
|
||||||
if (submission_fence_completion_event_ == nullptr) {
|
IID_PPV_ARGS(&queue_operations_since_submission_signal_fence_)))) {
|
||||||
XELOGE("Failed to create the submission fence completion event");
|
XELOGE(
|
||||||
|
"Failed to create the fence for awaiting queue operations done since "
|
||||||
|
"the latest submission");
|
||||||
return false;
|
return false;
|
||||||
}
|
}
|
||||||
|
|
||||||
frame_open_ = false;
|
|
||||||
frame_current_ = 1;
|
|
||||||
frame_completed_ = 0;
|
|
||||||
std::memset(closed_frame_submissions_, 0, sizeof(closed_frame_submissions_));
|
|
||||||
|
|
||||||
// Create the command list and one allocator because it's needed for a command
|
// Create the command list and one allocator because it's needed for a command
|
||||||
// list.
|
// list.
|
||||||
ID3D12CommandAllocator* command_allocator;
|
ID3D12CommandAllocator* command_allocator;
|
||||||
|
@ -883,8 +882,6 @@ bool D3D12CommandProcessor::SetupContext() {
|
||||||
command_allocator_writable_first_->last_usage_submission = 0;
|
command_allocator_writable_first_->last_usage_submission = 0;
|
||||||
command_allocator_writable_first_->next = nullptr;
|
command_allocator_writable_first_->next = nullptr;
|
||||||
command_allocator_writable_last_ = command_allocator_writable_first_;
|
command_allocator_writable_last_ = command_allocator_writable_first_;
|
||||||
command_allocator_submitted_first_ = nullptr;
|
|
||||||
command_allocator_submitted_last_ = nullptr;
|
|
||||||
if (FAILED(device->CreateCommandList(0, D3D12_COMMAND_LIST_TYPE_DIRECT,
|
if (FAILED(device->CreateCommandList(0, D3D12_COMMAND_LIST_TYPE_DIRECT,
|
||||||
command_allocator, nullptr,
|
command_allocator, nullptr,
|
||||||
IID_PPV_ARGS(&command_list_)))) {
|
IID_PPV_ARGS(&command_list_)))) {
|
||||||
|
@ -1469,7 +1466,7 @@ bool D3D12CommandProcessor::SetupContext() {
|
||||||
}
|
}
|
||||||
|
|
||||||
void D3D12CommandProcessor::ShutdownContext() {
|
void D3D12CommandProcessor::ShutdownContext() {
|
||||||
AwaitAllSubmissionsCompletion();
|
AwaitAllQueueOperationsCompletion();
|
||||||
|
|
||||||
ui::d3d12::util::ReleaseAndNull(readback_buffer_);
|
ui::d3d12::util::ReleaseAndNull(readback_buffer_);
|
||||||
readback_buffer_size_ = 0;
|
readback_buffer_size_ = 0;
|
||||||
|
@ -1557,16 +1554,22 @@ void D3D12CommandProcessor::ShutdownContext() {
|
||||||
frame_completed_ = 0;
|
frame_completed_ = 0;
|
||||||
std::memset(closed_frame_submissions_, 0, sizeof(closed_frame_submissions_));
|
std::memset(closed_frame_submissions_, 0, sizeof(closed_frame_submissions_));
|
||||||
|
|
||||||
// First release the fence since it may reference the event.
|
// First release the fences since they may reference fence_completion_event_.
|
||||||
|
|
||||||
|
queue_operations_done_since_submission_signal_ = false;
|
||||||
|
ui::d3d12::util::ReleaseAndNull(
|
||||||
|
queue_operations_since_submission_signal_fence_);
|
||||||
|
|
||||||
ui::d3d12::util::ReleaseAndNull(submission_fence_);
|
ui::d3d12::util::ReleaseAndNull(submission_fence_);
|
||||||
if (submission_fence_completion_event_) {
|
|
||||||
CloseHandle(submission_fence_completion_event_);
|
|
||||||
submission_fence_completion_event_ = nullptr;
|
|
||||||
}
|
|
||||||
submission_open_ = false;
|
submission_open_ = false;
|
||||||
submission_current_ = 1;
|
submission_current_ = 1;
|
||||||
submission_completed_ = 0;
|
submission_completed_ = 0;
|
||||||
|
|
||||||
|
if (fence_completion_event_) {
|
||||||
|
CloseHandle(fence_completion_event_);
|
||||||
|
fence_completion_event_ = nullptr;
|
||||||
|
}
|
||||||
|
|
||||||
CommandProcessor::ShutdownContext();
|
CommandProcessor::ShutdownContext();
|
||||||
}
|
}
|
||||||
|
|
||||||
|
@ -2283,7 +2286,7 @@ bool D3D12CommandProcessor::IssueDraw(xenos::PrimitiveType primitive_type,
|
||||||
memexport_range.base_address_dwords << 2, memexport_range_size);
|
memexport_range.base_address_dwords << 2, memexport_range_size);
|
||||||
readback_buffer_offset += memexport_range_size;
|
readback_buffer_offset += memexport_range_size;
|
||||||
}
|
}
|
||||||
AwaitAllSubmissionsCompletion();
|
AwaitAllQueueOperationsCompletion();
|
||||||
D3D12_RANGE readback_range;
|
D3D12_RANGE readback_range;
|
||||||
readback_range.Begin = 0;
|
readback_range.Begin = 0;
|
||||||
readback_range.End = memexport_total_size;
|
readback_range.End = memexport_total_size;
|
||||||
|
@ -2322,7 +2325,7 @@ void D3D12CommandProcessor::InitializeTrace() {
|
||||||
if (!EndSubmission(false)) {
|
if (!EndSubmission(false)) {
|
||||||
return;
|
return;
|
||||||
}
|
}
|
||||||
AwaitAllSubmissionsCompletion();
|
AwaitAllQueueOperationsCompletion();
|
||||||
if (render_target_cache_submitted) {
|
if (render_target_cache_submitted) {
|
||||||
render_target_cache_->InitializeTraceCompleteDownloads();
|
render_target_cache_->InitializeTraceCompleteDownloads();
|
||||||
}
|
}
|
||||||
|
@ -2355,7 +2358,7 @@ bool D3D12CommandProcessor::IssueCopy() {
|
||||||
readback_buffer, 0, shared_memory_buffer, written_address,
|
readback_buffer, 0, shared_memory_buffer, written_address,
|
||||||
written_length);
|
written_length);
|
||||||
if (EndSubmission(false)) {
|
if (EndSubmission(false)) {
|
||||||
AwaitAllSubmissionsCompletion();
|
AwaitAllQueueOperationsCompletion();
|
||||||
D3D12_RANGE readback_range;
|
D3D12_RANGE readback_range;
|
||||||
readback_range.Begin = 0;
|
readback_range.Begin = 0;
|
||||||
readback_range.End = written_length;
|
readback_range.End = written_length;
|
||||||
|
@ -2384,8 +2387,8 @@ void D3D12CommandProcessor::CheckSubmissionFence(uint64_t await_submission) {
|
||||||
submission_completed_ = submission_fence_->GetCompletedValue();
|
submission_completed_ = submission_fence_->GetCompletedValue();
|
||||||
if (submission_completed_ < await_submission) {
|
if (submission_completed_ < await_submission) {
|
||||||
submission_fence_->SetEventOnCompletion(await_submission,
|
submission_fence_->SetEventOnCompletion(await_submission,
|
||||||
submission_fence_completion_event_);
|
fence_completion_event_);
|
||||||
WaitForSingleObject(submission_fence_completion_event_, INFINITE);
|
WaitForSingleObject(fence_completion_event_, INFINITE);
|
||||||
submission_completed_ = submission_fence_->GetCompletedValue();
|
submission_completed_ = submission_fence_->GetCompletedValue();
|
||||||
}
|
}
|
||||||
if (submission_completed_ <= submission_completed_before) {
|
if (submission_completed_ <= submission_completed_before) {
|
||||||
|
@ -2622,6 +2625,10 @@ bool D3D12CommandProcessor::EndSubmission(bool is_swap) {
|
||||||
direct_queue->Signal(submission_fence_, submission_current_++);
|
direct_queue->Signal(submission_fence_, submission_current_++);
|
||||||
|
|
||||||
submission_open_ = false;
|
submission_open_ = false;
|
||||||
|
|
||||||
|
// Queue operations done directly (like UpdateTileMappings) will be awaited
|
||||||
|
// alongside the last submission if needed.
|
||||||
|
queue_operations_done_since_submission_signal_ = false;
|
||||||
}
|
}
|
||||||
|
|
||||||
if (is_closing_frame) {
|
if (is_closing_frame) {
|
||||||
|
@ -2640,7 +2647,7 @@ bool D3D12CommandProcessor::EndSubmission(bool is_swap) {
|
||||||
|
|
||||||
if (cache_clear_requested_) {
|
if (cache_clear_requested_) {
|
||||||
cache_clear_requested_ = false;
|
cache_clear_requested_ = false;
|
||||||
AwaitAllSubmissionsCompletion();
|
AwaitAllQueueOperationsCompletion();
|
||||||
|
|
||||||
ClearCommandAllocatorCache();
|
ClearCommandAllocatorCache();
|
||||||
|
|
||||||
|
@ -2685,15 +2692,31 @@ bool D3D12CommandProcessor::CanEndSubmissionImmediately() const {
|
||||||
return !submission_open_ || !pipeline_cache_->IsCreatingPipelineStates();
|
return !submission_open_ || !pipeline_cache_->IsCreatingPipelineStates();
|
||||||
}
|
}
|
||||||
|
|
||||||
void D3D12CommandProcessor::AwaitAllSubmissionsCompletion() {
|
void D3D12CommandProcessor::AwaitAllQueueOperationsCompletion() {
|
||||||
// May be called if shutting down without everything set up.
|
ui::d3d12::D3D12Context& context = GetD3D12Context();
|
||||||
if ((submission_completed_ + 1) >= submission_current_ ||
|
if (context.WasLost()) {
|
||||||
!submission_fence_ || GetD3D12Context().WasLost()) {
|
|
||||||
return;
|
return;
|
||||||
}
|
}
|
||||||
|
if (queue_operations_done_since_submission_signal_) {
|
||||||
|
assert_not_null(queue_operations_since_submission_signal_fence_);
|
||||||
|
UINT64 fence_value =
|
||||||
|
queue_operations_since_submission_signal_fence_->GetCompletedValue() +
|
||||||
|
1;
|
||||||
|
context.GetD3D12Provider().GetDirectQueue()->Signal(
|
||||||
|
queue_operations_since_submission_signal_fence_, fence_value);
|
||||||
|
queue_operations_since_submission_signal_fence_->SetEventOnCompletion(
|
||||||
|
fence_value, fence_completion_event_);
|
||||||
|
queue_operations_done_since_submission_signal_ = false;
|
||||||
|
// Since this indicates operations made after the last submission was done,
|
||||||
|
// it will await all the preceding submissions too.
|
||||||
|
} else if ((submission_completed_ + 1) < submission_current_) {
|
||||||
|
assert_not_null(submission_fence_);
|
||||||
submission_fence_->SetEventOnCompletion(submission_current_ - 1,
|
submission_fence_->SetEventOnCompletion(submission_current_ - 1,
|
||||||
submission_fence_completion_event_);
|
fence_completion_event_);
|
||||||
WaitForSingleObject(submission_fence_completion_event_, INFINITE);
|
} else {
|
||||||
|
return;
|
||||||
|
}
|
||||||
|
WaitForSingleObject(fence_completion_event_, INFINITE);
|
||||||
submission_completed_ = submission_current_ - 1;
|
submission_completed_ = submission_current_ - 1;
|
||||||
}
|
}
|
||||||
|
|
||||||
|
|
|
@ -68,6 +68,13 @@ class D3D12CommandProcessor : public CommandProcessor {
|
||||||
uint64_t GetCurrentSubmission() const { return submission_current_; }
|
uint64_t GetCurrentSubmission() const { return submission_current_; }
|
||||||
uint64_t GetCompletedSubmission() const { return submission_completed_; }
|
uint64_t GetCompletedSubmission() const { return submission_completed_; }
|
||||||
|
|
||||||
|
// Must be called when a subsystem does something like UpdateTileMappings so
|
||||||
|
// it can be awaited in AwaitAllQueueOperationsCompletion if it was done after
|
||||||
|
// the latest ExecuteCommandLists + Signal.
|
||||||
|
void NotifyQueueOperationsDoneDirectly() {
|
||||||
|
queue_operations_done_since_submission_signal_ = true;
|
||||||
|
}
|
||||||
|
|
||||||
uint64_t GetCurrentFrame() const { return frame_current_; }
|
uint64_t GetCurrentFrame() const { return frame_current_; }
|
||||||
uint64_t GetCompletedFrame() const { return frame_completed_; }
|
uint64_t GetCompletedFrame() const { return frame_completed_; }
|
||||||
|
|
||||||
|
@ -307,7 +314,7 @@ class D3D12CommandProcessor : public CommandProcessor {
|
||||||
// as when there are unfinished graphics pipeline state creation requests that
|
// as when there are unfinished graphics pipeline state creation requests that
|
||||||
// would need to be fulfilled before actually submitting the command list.
|
// would need to be fulfilled before actually submitting the command list.
|
||||||
bool CanEndSubmissionImmediately() const;
|
bool CanEndSubmissionImmediately() const;
|
||||||
void AwaitAllSubmissionsCompletion();
|
void AwaitAllQueueOperationsCompletion();
|
||||||
// Need to await submission completion before calling.
|
// Need to await submission completion before calling.
|
||||||
void ClearCommandAllocatorCache();
|
void ClearCommandAllocatorCache();
|
||||||
|
|
||||||
|
@ -350,13 +357,21 @@ class D3D12CommandProcessor : public CommandProcessor {
|
||||||
|
|
||||||
bool cache_clear_requested_ = false;
|
bool cache_clear_requested_ = false;
|
||||||
|
|
||||||
|
HANDLE fence_completion_event_ = nullptr;
|
||||||
|
|
||||||
bool submission_open_ = false;
|
bool submission_open_ = false;
|
||||||
// Values of submission_fence_.
|
// Values of submission_fence_.
|
||||||
uint64_t submission_current_ = 1;
|
uint64_t submission_current_ = 1;
|
||||||
uint64_t submission_completed_ = 0;
|
uint64_t submission_completed_ = 0;
|
||||||
HANDLE submission_fence_completion_event_ = nullptr;
|
|
||||||
ID3D12Fence* submission_fence_ = nullptr;
|
ID3D12Fence* submission_fence_ = nullptr;
|
||||||
|
|
||||||
|
// For awaiting non-submission queue operations such as UpdateTileMappings in
|
||||||
|
// AwaitAllQueueOperationsCompletion when they're queued after the latest
|
||||||
|
// ExecuteCommandLists + Signal, thus won't be awaited by just awaiting the
|
||||||
|
// submission.
|
||||||
|
ID3D12Fence* queue_operations_since_submission_signal_fence_ = nullptr;
|
||||||
|
bool queue_operations_done_since_submission_signal_ = false;
|
||||||
|
|
||||||
bool frame_open_ = false;
|
bool frame_open_ = false;
|
||||||
// Guest frame index, since some transient resources can be reused across
|
// Guest frame index, since some transient resources can be reused across
|
||||||
// submissions. Values updated in the beginning of a frame.
|
// submissions. Values updated in the beginning of a frame.
|
||||||
|
|
|
@ -392,14 +392,11 @@ bool SharedMemory::EnsureTilesResident(uint32_t start, uint32_t length) {
|
||||||
D3D12_TILE_RANGE_FLAGS range_flags = D3D12_TILE_RANGE_FLAG_NONE;
|
D3D12_TILE_RANGE_FLAGS range_flags = D3D12_TILE_RANGE_FLAG_NONE;
|
||||||
UINT heap_range_start_offset = 0;
|
UINT heap_range_start_offset = 0;
|
||||||
UINT range_tile_count = kHeapSize / D3D12_TILED_RESOURCE_TILE_SIZE_IN_BYTES;
|
UINT range_tile_count = kHeapSize / D3D12_TILED_RESOURCE_TILE_SIZE_IN_BYTES;
|
||||||
// FIXME(Triang3l): This may cause issues if the emulator is shut down
|
|
||||||
// mid-frame and the heaps are destroyed before tile mappings are updated
|
|
||||||
// (awaiting the fence won't catch this then). Defer this until the actual
|
|
||||||
// command list submission at the end of the frame.
|
|
||||||
direct_queue->UpdateTileMappings(
|
direct_queue->UpdateTileMappings(
|
||||||
buffer_, 1, ®ion_start_coordinates, ®ion_size, heaps_[i], 1,
|
buffer_, 1, ®ion_start_coordinates, ®ion_size, heaps_[i], 1,
|
||||||
&range_flags, &heap_range_start_offset, &range_tile_count,
|
&range_flags, &heap_range_start_offset, &range_tile_count,
|
||||||
D3D12_TILE_MAPPING_FLAG_NONE);
|
D3D12_TILE_MAPPING_FLAG_NONE);
|
||||||
|
command_processor_.NotifyQueueOperationsDoneDirectly();
|
||||||
}
|
}
|
||||||
return true;
|
return true;
|
||||||
}
|
}
|
||||||
|
|
|
@ -1658,14 +1658,11 @@ bool TextureCache::EnsureScaledResolveBufferResident(uint32_t start_unscaled,
|
||||||
UINT heap_range_start_offset = 0;
|
UINT heap_range_start_offset = 0;
|
||||||
UINT range_tile_count =
|
UINT range_tile_count =
|
||||||
kScaledResolveHeapSize / D3D12_TILED_RESOURCE_TILE_SIZE_IN_BYTES;
|
kScaledResolveHeapSize / D3D12_TILED_RESOURCE_TILE_SIZE_IN_BYTES;
|
||||||
// FIXME(Triang3l): This may cause issues if the emulator is shut down
|
|
||||||
// mid-frame and the heaps are destroyed before tile mappings are updated
|
|
||||||
// (awaiting the fence won't catch this then). Defer this until the actual
|
|
||||||
// command list submission.
|
|
||||||
direct_queue->UpdateTileMappings(
|
direct_queue->UpdateTileMappings(
|
||||||
scaled_resolve_buffer_, 1, ®ion_start_coordinates, ®ion_size,
|
scaled_resolve_buffer_, 1, ®ion_start_coordinates, ®ion_size,
|
||||||
scaled_resolve_heaps_[i], 1, &range_flags, &heap_range_start_offset,
|
scaled_resolve_heaps_[i], 1, &range_flags, &heap_range_start_offset,
|
||||||
&range_tile_count, D3D12_TILE_MAPPING_FLAG_NONE);
|
&range_tile_count, D3D12_TILE_MAPPING_FLAG_NONE);
|
||||||
|
command_processor_.NotifyQueueOperationsDoneDirectly();
|
||||||
}
|
}
|
||||||
return true;
|
return true;
|
||||||
}
|
}
|
||||||
|
|
Loading…
Reference in New Issue