mirror of
https://gitlab.freedesktop.org/gstreamer/gstreamer.git
synced 2025-01-01 21:18:52 +00:00
d3d12frame: Extract external fence from memory and wait helper function
Adding gst_d3d12_frame_fence_{gpu,cpu}_wait() methods Part-of: <https://gitlab.freedesktop.org/gstreamer/gstreamer/-/merge_requests/6749>
This commit is contained in:
parent
478e49dd73
commit
87f43c25cc
2 changed files with 145 additions and 1 deletions
|
@ -27,6 +27,12 @@
|
||||||
#include "gstd3d12-private.h"
|
#include "gstd3d12-private.h"
|
||||||
#include <string.h>
|
#include <string.h>
|
||||||
#include <directx/d3dx12.h>
|
#include <directx/d3dx12.h>
|
||||||
|
#include <vector>
|
||||||
|
#include <wrl.h>
|
||||||
|
|
||||||
|
/* *INDENT-OFF* */
|
||||||
|
using namespace Microsoft::WRL;
|
||||||
|
/* *INDENT-ON* */
|
||||||
|
|
||||||
#ifndef GST_DISABLE_GST_DEBUG
|
#ifndef GST_DISABLE_GST_DEBUG
|
||||||
#define GST_CAT_DEFAULT ensure_debug_category()
|
#define GST_CAT_DEFAULT ensure_debug_category()
|
||||||
|
@ -90,7 +96,7 @@ gst_d3d12_frame_map (GstD3D12Frame * frame, const GstVideoInfo * info,
|
||||||
for (guint i = 0; i < num_mem; i++) {
|
for (guint i = 0; i < num_mem; i++) {
|
||||||
auto mem = gst_buffer_peek_memory (buffer, i);
|
auto mem = gst_buffer_peek_memory (buffer, i);
|
||||||
if (!gst_is_d3d12_memory (mem)) {
|
if (!gst_is_d3d12_memory (mem)) {
|
||||||
GST_WARNING ("memory %u is not a d3d12 memory", i);
|
GST_LOG ("memory %u is not a d3d12 memory", i);
|
||||||
return FALSE;
|
return FALSE;
|
||||||
}
|
}
|
||||||
|
|
||||||
|
@ -217,6 +223,9 @@ gst_d3d12_frame_map (GstD3D12Frame * frame, const GstVideoInfo * info,
|
||||||
rtv_handle.Offset (rtv_inc_size);
|
rtv_handle.Offset (rtv_inc_size);
|
||||||
}
|
}
|
||||||
|
|
||||||
|
gst_d3d12_memory_get_external_fence (dmem, &frame->fence[plane_idx].fence,
|
||||||
|
&frame->fence[plane_idx].fence_value);
|
||||||
|
|
||||||
plane_idx++;
|
plane_idx++;
|
||||||
}
|
}
|
||||||
}
|
}
|
||||||
|
@ -266,6 +275,11 @@ gst_d3d12_frame_unmap (GstD3D12Frame * frame)
|
||||||
{
|
{
|
||||||
g_return_if_fail (frame);
|
g_return_if_fail (frame);
|
||||||
|
|
||||||
|
for (guint i = 0; i < G_N_ELEMENTS (frame->fence); i++) {
|
||||||
|
if (frame->fence[i].fence)
|
||||||
|
frame->fence[i].fence->Release ();
|
||||||
|
}
|
||||||
|
|
||||||
for (guint i = 0; i < G_N_ELEMENTS (frame->map); i++) {
|
for (guint i = 0; i < G_N_ELEMENTS (frame->map); i++) {
|
||||||
auto mem = frame->map[i].memory;
|
auto mem = frame->map[i].memory;
|
||||||
if (!mem)
|
if (!mem)
|
||||||
|
@ -337,6 +351,12 @@ gst_d3d12_frame_copy (GstD3D12Frame * dest, const GstD3D12Frame * src,
|
||||||
gst_d3d12_fence_data_add_notify_mini_object (fence_data,
|
gst_d3d12_fence_data_add_notify_mini_object (fence_data,
|
||||||
gst_buffer_ref (src->buffer));
|
gst_buffer_ref (src->buffer));
|
||||||
|
|
||||||
|
auto cq = gst_d3d12_device_get_command_queue (src->device,
|
||||||
|
D3D12_COMMAND_LIST_TYPE_DIRECT);
|
||||||
|
auto cq_handle = gst_d3d12_command_queue_get_handle (cq);
|
||||||
|
gst_d3d12_frame_fence_gpu_wait (src, cq_handle);
|
||||||
|
gst_d3d12_frame_fence_gpu_wait (dest, cq_handle);
|
||||||
|
|
||||||
return gst_d3d12_device_copy_texture_region (dest->device,
|
return gst_d3d12_device_copy_texture_region (dest->device,
|
||||||
GST_VIDEO_INFO_N_PLANES (&dest->info), args, fence_data,
|
GST_VIDEO_INFO_N_PLANES (&dest->info), args, fence_data,
|
||||||
nullptr, 0, D3D12_COMMAND_LIST_TYPE_DIRECT, fence_value);
|
nullptr, 0, D3D12_COMMAND_LIST_TYPE_DIRECT, fence_value);
|
||||||
|
@ -383,6 +403,113 @@ gst_d3d12_frame_copy_plane (GstD3D12Frame * dest, const GstD3D12Frame * src,
|
||||||
gst_d3d12_fence_data_add_notify_mini_object (fence_data,
|
gst_d3d12_fence_data_add_notify_mini_object (fence_data,
|
||||||
gst_buffer_ref (src->buffer));
|
gst_buffer_ref (src->buffer));
|
||||||
|
|
||||||
|
auto cq = gst_d3d12_device_get_command_queue (src->device,
|
||||||
|
D3D12_COMMAND_LIST_TYPE_DIRECT);
|
||||||
|
auto cq_handle = gst_d3d12_command_queue_get_handle (cq);
|
||||||
|
|
||||||
|
if (src->fence[plane].fence)
|
||||||
|
cq_handle->Wait (src->fence[plane].fence, src->fence[plane].fence_value);
|
||||||
|
|
||||||
|
if (dest->fence[plane].fence)
|
||||||
|
cq_handle->Wait (dest->fence[plane].fence, dest->fence[plane].fence_value);
|
||||||
|
|
||||||
return gst_d3d12_device_copy_texture_region (dest->device, 1, &args,
|
return gst_d3d12_device_copy_texture_region (dest->device, 1, &args,
|
||||||
fence_data, nullptr, 0, D3D12_COMMAND_LIST_TYPE_DIRECT, fence_value);
|
fence_data, nullptr, 0, D3D12_COMMAND_LIST_TYPE_DIRECT, fence_value);
|
||||||
}
|
}
|
||||||
|
|
||||||
|
/**
|
||||||
|
* gst_d3d12_frame_fence_gpu_wait:
|
||||||
|
* @frame: a #GstD3D12Frame
|
||||||
|
* @queue: a ID3D12CommandQueue
|
||||||
|
*
|
||||||
|
* Executes ID3D12CommandQueue::Wait() if external fence exists
|
||||||
|
*
|
||||||
|
* Returns: %TRUE on success.
|
||||||
|
*
|
||||||
|
* Since: 1.26
|
||||||
|
*/
|
||||||
|
gboolean
|
||||||
|
gst_d3d12_frame_fence_gpu_wait (const GstD3D12Frame * frame,
|
||||||
|
ID3D12CommandQueue * queue)
|
||||||
|
{
|
||||||
|
g_return_val_if_fail (frame, FALSE);
|
||||||
|
g_return_val_if_fail (GST_IS_D3D12_DEVICE (frame->device), FALSE);
|
||||||
|
g_return_val_if_fail (queue, FALSE);
|
||||||
|
|
||||||
|
ID3D12Fence *last_fence = nullptr;
|
||||||
|
guint64 last_fence_val = 0;
|
||||||
|
for (guint i = 0; i < G_N_ELEMENTS (frame->fence); i++) {
|
||||||
|
if (frame->fence[i].fence) {
|
||||||
|
if (frame->fence[i].fence == last_fence &&
|
||||||
|
frame->fence[i].fence_value <= last_fence_val) {
|
||||||
|
continue;
|
||||||
|
}
|
||||||
|
last_fence = frame->fence[i].fence;
|
||||||
|
last_fence_val = frame->fence[i].fence_value;
|
||||||
|
|
||||||
|
auto hr = queue->Wait (frame->fence[i].fence,
|
||||||
|
frame->fence[i].fence_value);
|
||||||
|
if (!gst_d3d12_result (hr, frame->device))
|
||||||
|
return FALSE;
|
||||||
|
}
|
||||||
|
}
|
||||||
|
|
||||||
|
return TRUE;
|
||||||
|
}
|
||||||
|
|
||||||
|
/**
|
||||||
|
* gst_d3d12_frame_fence_cpu_wait:
|
||||||
|
* @frame: a #GstD3D12Frame
|
||||||
|
*
|
||||||
|
* Waits for external fence objects
|
||||||
|
*
|
||||||
|
* Returns: %TRUE on success.
|
||||||
|
*
|
||||||
|
* Since: 1.26
|
||||||
|
*/
|
||||||
|
gboolean
|
||||||
|
gst_d3d12_frame_fence_cpu_wait (const GstD3D12Frame * frame)
|
||||||
|
{
|
||||||
|
g_return_val_if_fail (frame, FALSE);
|
||||||
|
g_return_val_if_fail (GST_IS_D3D12_DEVICE (frame->device), FALSE);
|
||||||
|
|
||||||
|
ID3D12Fence *last_fence = nullptr;
|
||||||
|
guint64 last_fence_val = 0;
|
||||||
|
std::vector < ID3D12Fence * >fences;
|
||||||
|
std::vector < UINT64 > fence_vals;
|
||||||
|
|
||||||
|
for (guint i = 0; i < G_N_ELEMENTS (frame->fence); i++) {
|
||||||
|
if (frame->fence[i].fence) {
|
||||||
|
if (frame->fence[i].fence == last_fence &&
|
||||||
|
frame->fence[i].fence_value <= last_fence_val) {
|
||||||
|
continue;
|
||||||
|
}
|
||||||
|
last_fence = frame->fence[i].fence;
|
||||||
|
last_fence_val = frame->fence[i].fence_value;
|
||||||
|
|
||||||
|
fences.push_back (frame->fence[i].fence);
|
||||||
|
fence_vals.push_back (frame->fence[i].fence_value);
|
||||||
|
}
|
||||||
|
}
|
||||||
|
|
||||||
|
if (fences.empty ())
|
||||||
|
return TRUE;
|
||||||
|
|
||||||
|
ComPtr < ID3D12Device1 > device1;
|
||||||
|
auto device = gst_d3d12_device_get_device_handle (frame->device);
|
||||||
|
auto hr = device->QueryInterface (IID_PPV_ARGS (&device1));
|
||||||
|
|
||||||
|
if (SUCCEEDED (hr)) {
|
||||||
|
hr = device1->SetEventOnMultipleFenceCompletion (fences.data (),
|
||||||
|
fence_vals.data (), fences.size (), D3D12_MULTIPLE_FENCE_WAIT_FLAG_ALL,
|
||||||
|
nullptr);
|
||||||
|
} else {
|
||||||
|
for (size_t i = 0; i < fences.size (); i++) {
|
||||||
|
hr = fences[i]->SetEventOnCompletion (fence_vals[i], nullptr);
|
||||||
|
if (FAILED (hr))
|
||||||
|
break;
|
||||||
|
}
|
||||||
|
}
|
||||||
|
|
||||||
|
return gst_d3d12_result (hr, frame->device);
|
||||||
|
}
|
||||||
|
|
|
@ -25,6 +25,8 @@
|
||||||
|
|
||||||
G_BEGIN_DECLS
|
G_BEGIN_DECLS
|
||||||
|
|
||||||
|
typedef struct _GstD3D12FrameFence GstD3D12FrameFence;
|
||||||
|
|
||||||
/**
|
/**
|
||||||
* GstD3D12FrameMapFlags:
|
* GstD3D12FrameMapFlags:
|
||||||
* @GST_D3D12_FRAME_MAP_FLAG_NONE: No flags
|
* @GST_D3D12_FRAME_MAP_FLAG_NONE: No flags
|
||||||
|
@ -42,6 +44,12 @@ typedef enum
|
||||||
|
|
||||||
DEFINE_ENUM_FLAG_OPERATORS (GstD3D12FrameMapFlags);
|
DEFINE_ENUM_FLAG_OPERATORS (GstD3D12FrameMapFlags);
|
||||||
|
|
||||||
|
struct _GstD3D12FrameFence
|
||||||
|
{
|
||||||
|
ID3D12Fence *fence;
|
||||||
|
guint64 fence_value;
|
||||||
|
};
|
||||||
|
|
||||||
/**
|
/**
|
||||||
* GstD3D12Frame:
|
* GstD3D12Frame:
|
||||||
* @info: the #GstVideoInfo
|
* @info: the #GstVideoInfo
|
||||||
|
@ -53,6 +61,7 @@ DEFINE_ENUM_FLAG_OPERATORS (GstD3D12FrameMapFlags);
|
||||||
* @data: pointers to the plane data
|
* @data: pointers to the plane data
|
||||||
* @subresource_index: subresource index of the plane
|
* @subresource_index: subresource index of the plane
|
||||||
* @plane_rect: plane rectangle
|
* @plane_rect: plane rectangle
|
||||||
|
* @fence: external fences
|
||||||
* @srv_desc_handle: shader resource view descriptor handle
|
* @srv_desc_handle: shader resource view descriptor handle
|
||||||
* @rtb_desc_handle: render target view descriptor handle
|
* @rtb_desc_handle: render target view descriptor handle
|
||||||
*
|
*
|
||||||
|
@ -72,6 +81,7 @@ struct _GstD3D12Frame
|
||||||
ID3D12Resource *data[GST_VIDEO_MAX_PLANES];
|
ID3D12Resource *data[GST_VIDEO_MAX_PLANES];
|
||||||
guint subresource_index[GST_VIDEO_MAX_PLANES];
|
guint subresource_index[GST_VIDEO_MAX_PLANES];
|
||||||
D3D12_RECT plane_rect[GST_VIDEO_MAX_PLANES];
|
D3D12_RECT plane_rect[GST_VIDEO_MAX_PLANES];
|
||||||
|
GstD3D12FrameFence fence[GST_VIDEO_MAX_PLANES];
|
||||||
D3D12_CPU_DESCRIPTOR_HANDLE srv_desc_handle[GST_VIDEO_MAX_PLANES];
|
D3D12_CPU_DESCRIPTOR_HANDLE srv_desc_handle[GST_VIDEO_MAX_PLANES];
|
||||||
D3D12_CPU_DESCRIPTOR_HANDLE rtv_desc_handle[GST_VIDEO_MAX_PLANES];
|
D3D12_CPU_DESCRIPTOR_HANDLE rtv_desc_handle[GST_VIDEO_MAX_PLANES];
|
||||||
|
|
||||||
|
@ -100,5 +110,12 @@ gboolean gst_d3d12_frame_copy_plane (GstD3D12Frame * dest,
|
||||||
guint plane,
|
guint plane,
|
||||||
guint64 * fence_value);
|
guint64 * fence_value);
|
||||||
|
|
||||||
|
GST_D3D12_API
|
||||||
|
gboolean gst_d3d12_frame_fence_gpu_wait (const GstD3D12Frame * frame,
|
||||||
|
ID3D12CommandQueue * queue);
|
||||||
|
|
||||||
|
GST_D3D12_API
|
||||||
|
gboolean gst_d3d12_frame_fence_cpu_wait (const GstD3D12Frame * frame);
|
||||||
|
|
||||||
G_END_DECLS
|
G_END_DECLS
|
||||||
|
|
||||||
|
|
Loading…
Reference in a new issue