d3d12converter: Gamma LUT related enhancements

* Build gamma LUT using shader, instead of CPU side math then uploading
* Make gamma LUT sharable across multiple converters

Part-of: <https://gitlab.freedesktop.org/gstreamer/gstreamer/-/merge_requests/7951>
This commit is contained in:
Seungha Yang 2024-11-23 22:08:56 +09:00 committed by GStreamer Marge Bot
parent f3191aca58
commit 6bcedb0d8b
3 changed files with 476 additions and 229 deletions

View file

@ -166,14 +166,7 @@ struct VertexData
} texture; } texture;
}; };
struct GammaLut
{
guint16 lut[GAMMA_LUT_SIZE];
};
/* *INDENT-OFF* */ /* *INDENT-OFF* */
typedef std::shared_ptr<GammaLut> GammaLutPtr;
static const XMFLOAT4X4A g_matrix_identity = XMFLOAT4X4A ( static const XMFLOAT4X4A g_matrix_identity = XMFLOAT4X4A (
1.0f, 0.0f, 0.0f, 0.0f, 1.0f, 0.0f, 0.0f, 0.0f,
0.0f, 1.0f, 0.0f, 0.0f, 0.0f, 1.0f, 0.0f, 0.0f,
@ -293,6 +286,12 @@ struct _GstD3D12ConverterPrivate
if (fence_val > 0 && cq) if (fence_val > 0 && cq)
gst_d3d12_cmd_queue_fence_wait (cq, fence_val); gst_d3d12_cmd_queue_fence_wait (cq, fence_val);
if (setup_fence) {
auto completed = setup_fence->GetCompletedValue ();
if (completed < setup_fence_val)
setup_fence->SetEventOnCompletion (setup_fence_val, nullptr);
}
gst_clear_object (&srv_heap_pool); gst_clear_object (&srv_heap_pool);
gst_clear_object (&cq); gst_clear_object (&cq);
gst_clear_object (&pack); gst_clear_object (&pack);
@ -323,7 +322,7 @@ struct _GstD3D12ConverterPrivate
gboolean have_lut = FALSE; gboolean have_lut = FALSE;
D3D12_VERTEX_BUFFER_VIEW vbv; D3D12_VERTEX_BUFFER_VIEW vbv;
D3D12_INDEX_BUFFER_VIEW idv; D3D12_INDEX_BUFFER_VIEW ibv;
D3D12_GPU_VIRTUAL_ADDRESS const_buf_addr[2]; D3D12_GPU_VIRTUAL_ADDRESS const_buf_addr[2];
ComPtr<ID3D12Resource> shader_buf; ComPtr<ID3D12Resource> shader_buf;
ComPtr<ID3D12Resource> vertex_upload; ComPtr<ID3D12Resource> vertex_upload;
@ -360,6 +359,9 @@ struct _GstD3D12ConverterPrivate
std::mutex prop_lock; std::mutex prop_lock;
guint64 fence_val = 0; guint64 fence_val = 0;
ComPtr<ID3D12Fence> setup_fence;
guint64 setup_fence_val = 0;
/* properties */ /* properties */
gint src_x = 0; gint src_x = 0;
gint src_y = 0; gint src_y = 0;
@ -695,54 +697,6 @@ gst_d3d12_converter_get_property (GObject * object, guint prop_id,
} }
} }
static GammaLutPtr
gst_d3d12_converter_get_gamma_dec_table (GstVideoTransferFunction func)
{
static std::mutex lut_lock;
static std::map < GstVideoTransferFunction, GammaLutPtr > g_gamma_dec_table;
std::lock_guard < std::mutex > lk (lut_lock);
auto lut = g_gamma_dec_table.find (func);
if (lut != g_gamma_dec_table.end ())
return lut->second;
const gdouble scale = (gdouble) 1 / (GAMMA_LUT_SIZE - 1);
auto table = std::make_shared < GammaLut > ();
for (guint i = 0; i < GAMMA_LUT_SIZE; i++) {
gdouble val = gst_video_transfer_function_decode (func, i * scale);
val = rint (val * 65535);
val = CLAMP (val, 0, 65535);
table->lut[i] = (guint16) val;
}
g_gamma_dec_table[func] = table;
return table;
}
static GammaLutPtr
gst_d3d12_converter_get_gamma_enc_table (GstVideoTransferFunction func)
{
static std::mutex lut_lock;
static std::map < GstVideoTransferFunction, GammaLutPtr > g_gamma_enc_table;
std::lock_guard < std::mutex > lk (lut_lock);
auto lut = g_gamma_enc_table.find (func);
if (lut != g_gamma_enc_table.end ())
return lut->second;
const gdouble scale = (gdouble) 1 / (GAMMA_LUT_SIZE - 1);
auto table = std::make_shared < GammaLut > ();
for (guint i = 0; i < GAMMA_LUT_SIZE; i++) {
gdouble val = gst_video_transfer_function_encode (func, i * scale);
val = rint (val * 65535);
val = CLAMP (val, 0, 65535);
table->lut[i] = (guint16) val;
}
g_gamma_enc_table[func] = table;
return table;
}
static guint static guint
reorder_rtv_index (GstVideoFormat output_format, guint index) reorder_rtv_index (GstVideoFormat output_format, guint index)
{ {
@ -820,8 +774,6 @@ gst_d3d12_converter_setup_resource (GstD3D12Converter * self,
HRESULT hr; HRESULT hr;
VertexData vertex_data[4]; VertexData vertex_data[4];
ComPtr < ID3D12Resource > upload_buf; ComPtr < ID3D12Resource > upload_buf;
ComPtr < ID3D12Resource > gamma_dec_lut_upload;
ComPtr < ID3D12Resource > gamma_enc_lut_upload;
auto device = gst_d3d12_device_get_device_handle (self->device); auto device = gst_d3d12_device_get_device_handle (self->device);
@ -1023,9 +975,9 @@ gst_d3d12_converter_setup_resource (GstD3D12Converter * self,
priv->vbv.SizeInBytes = g_vertex_buf_size; priv->vbv.SizeInBytes = g_vertex_buf_size;
priv->vbv.StrideInBytes = sizeof (VertexData); priv->vbv.StrideInBytes = sizeof (VertexData);
priv->idv.BufferLocation = priv->vbv.BufferLocation + g_vertex_buf_size; priv->ibv.BufferLocation = priv->vbv.BufferLocation + g_vertex_buf_size;
priv->idv.SizeInBytes = g_index_buf_size; priv->ibv.SizeInBytes = g_index_buf_size;
priv->idv.Format = DXGI_FORMAT_R16_UINT; priv->ibv.Format = DXGI_FORMAT_R16_UINT;
priv->const_buf_addr[0] = priv->vbv.BufferLocation + vertex_index_size; priv->const_buf_addr[0] = priv->vbv.BufferLocation + vertex_index_size;
priv->const_buf_addr[1] = priv->vbv.BufferLocation + other_const_off; priv->const_buf_addr[1] = priv->vbv.BufferLocation + other_const_off;
@ -1053,106 +1005,12 @@ gst_d3d12_converter_setup_resource (GstD3D12Converter * self,
upload_buf->Unmap (0, nullptr); upload_buf->Unmap (0, nullptr);
} }
if (priv->have_lut) { auto in_trc = in_info->colorimetry.transfer;
heap_prop = CD3DX12_HEAP_PROPERTIES (D3D12_HEAP_TYPE_DEFAULT); auto out_trc = in_info->colorimetry.transfer;
resource_desc = CD3DX12_RESOURCE_DESC::Tex1D (DXGI_FORMAT_R16_UNORM,
GAMMA_LUT_SIZE, 1, 1);
hr = device->CreateCommittedResource (&heap_prop, if (priv->convert_type[0] == CONVERT_TYPE::GAMMA ||
heap_flags, &resource_desc, D3D12_RESOURCE_STATE_COPY_DEST, nullptr, priv->convert_type[0] == CONVERT_TYPE::PRIMARY) {
IID_PPV_ARGS (&priv->gamma_dec_lut)); out_trc = out_info->colorimetry.transfer;
if (!gst_d3d12_result (hr, self->device)) {
GST_ERROR_OBJECT (self, "Couldn't create gamma decoding LUT");
return FALSE;
}
hr = device->CreateCommittedResource (&heap_prop,
heap_flags, &resource_desc, D3D12_RESOURCE_STATE_COPY_DEST, nullptr,
IID_PPV_ARGS (&priv->gamma_enc_lut));
if (!gst_d3d12_result (hr, self->device)) {
GST_ERROR_OBJECT (self, "Couldn't create gamma encoding LUT");
return FALSE;
}
UINT64 gamma_lut_size;
device->GetCopyableFootprints (&resource_desc, 0, 1, 0,
&priv->gamma_lut_layout, nullptr, nullptr, &gamma_lut_size);
heap_prop = CD3DX12_HEAP_PROPERTIES (D3D12_HEAP_TYPE_UPLOAD);
resource_desc = CD3DX12_RESOURCE_DESC::Buffer (gamma_lut_size);
hr = device->CreateCommittedResource (&heap_prop,
heap_flags, &resource_desc, D3D12_RESOURCE_STATE_GENERIC_READ, nullptr,
IID_PPV_ARGS (&gamma_dec_lut_upload));
if (!gst_d3d12_result (hr, self->device)) {
GST_ERROR_OBJECT (self, "Couldn't create gamma decoding LUT upload");
return FALSE;
}
hr = device->CreateCommittedResource (&heap_prop,
heap_flags, &resource_desc, D3D12_RESOURCE_STATE_GENERIC_READ, nullptr,
IID_PPV_ARGS (&gamma_enc_lut_upload));
if (!gst_d3d12_result (hr, self->device)) {
GST_ERROR_OBJECT (self, "Couldn't create gamma encoding LUT upload");
return FALSE;
}
auto in_trc = in_info->colorimetry.transfer;
auto out_trc = in_info->colorimetry.transfer;
if (priv->convert_type[0] == CONVERT_TYPE::GAMMA ||
priv->convert_type[0] == CONVERT_TYPE::PRIMARY) {
out_trc = out_info->colorimetry.transfer;
}
auto gamma_dec_table = gst_d3d12_converter_get_gamma_dec_table (in_trc);
auto gamma_enc_table = gst_d3d12_converter_get_gamma_enc_table (out_trc);
hr = gamma_dec_lut_upload->Map (0, &range, (void **) &data);
if (!gst_d3d12_result (hr, self->device)) {
GST_ERROR_OBJECT (self, "Couldn't map gamma lut upload buffer");
return FALSE;
}
memcpy (data, gamma_dec_table->lut, GAMMA_LUT_SIZE * sizeof (guint16));
gamma_dec_lut_upload->Unmap (0, nullptr);
hr = gamma_enc_lut_upload->Map (0, &range, (void **) &data);
if (!gst_d3d12_result (hr, self->device)) {
GST_ERROR_OBJECT (self, "Couldn't map gamma lut upload buffer");
return FALSE;
}
memcpy (data, gamma_enc_table->lut, GAMMA_LUT_SIZE * sizeof (guint16));
gamma_enc_lut_upload->Unmap (0, nullptr);
D3D12_DESCRIPTOR_HEAP_DESC desc = { };
desc.Type = D3D12_DESCRIPTOR_HEAP_TYPE_CBV_SRV_UAV;
desc.NumDescriptors = 2;
desc.Flags = D3D12_DESCRIPTOR_HEAP_FLAG_NONE;
auto hr = device->CreateDescriptorHeap (&desc,
IID_PPV_ARGS (&priv->gamma_lut_heap));
if (!gst_d3d12_result (hr, self->device)) {
GST_ERROR_OBJECT (self, "Couldn't map gamma lut upload buffer");
return FALSE;
}
auto cpu_handle =
CD3DX12_CPU_DESCRIPTOR_HANDLE (GetCPUDescriptorHandleForHeapStart
(priv->gamma_lut_heap));
D3D12_SHADER_RESOURCE_VIEW_DESC srv_desc = { };
srv_desc.ViewDimension = D3D12_SRV_DIMENSION_TEXTURE1D;
srv_desc.Shader4ComponentMapping = D3D12_DEFAULT_SHADER_4_COMPONENT_MAPPING;
srv_desc.Texture1D.MipLevels = 1;
device->CreateShaderResourceView (priv->gamma_dec_lut.Get (), &srv_desc,
cpu_handle);
cpu_handle.Offset (priv->srv_inc_size);
device->CreateShaderResourceView (priv->gamma_enc_lut.Get (), &srv_desc,
cpu_handle);
} }
priv->input_texture_width = GST_VIDEO_INFO_WIDTH (in_info); priv->input_texture_width = GST_VIDEO_INFO_WIDTH (in_info);
@ -1172,84 +1030,52 @@ gst_d3d12_converter_setup_resource (GstD3D12Converter * self,
priv->scissor_rect[i].bottom = GST_VIDEO_INFO_COMP_HEIGHT (out_info, i); priv->scissor_rect[i].bottom = GST_VIDEO_INFO_COMP_HEIGHT (out_info, i);
} }
ComPtr < ID3D12CommandAllocator > ca;
hr = device->CreateCommandAllocator (D3D12_COMMAND_LIST_TYPE_DIRECT,
IID_PPV_ARGS (&ca));
if (!gst_d3d12_result (hr, self->device))
return FALSE;
ComPtr < ID3D12GraphicsCommandList > cl;
hr = device->CreateCommandList (0, D3D12_COMMAND_LIST_TYPE_DIRECT,
ca.Get (), nullptr, IID_PPV_ARGS (&cl));
if (!gst_d3d12_result (hr, self->device))
return FALSE;
std::vector < D3D12_RESOURCE_BARRIER > barriers;
cl->CopyResource (priv->shader_buf.Get (), upload_buf.Get ());
barriers.push_back (CD3DX12_RESOURCE_BARRIER::Transition (priv->shader_buf.
Get (), D3D12_RESOURCE_STATE_COPY_DEST, STATE_VERTEX_AND_INDEX));
if (priv->have_lut) { if (priv->have_lut) {
D3D12_TEXTURE_COPY_LOCATION src; hr = gst_d3d12_device_get_converter_resources (self->device,
D3D12_TEXTURE_COPY_LOCATION dst; priv->shader_buf.Get (), upload_buf.Get (), &priv->vbv, &priv->ibv,
src = in_trc, &priv->gamma_dec_lut, out_trc, &priv->gamma_enc_lut,
CD3DX12_TEXTURE_COPY_LOCATION (gamma_dec_lut_upload.Get (), &priv->setup_fence, &priv->setup_fence_val);
priv->gamma_lut_layout); } else {
dst = CD3DX12_TEXTURE_COPY_LOCATION (priv->gamma_dec_lut.Get ()); hr = gst_d3d12_device_get_converter_resources (self->device,
cl->CopyTextureRegion (&dst, 0, 0, 0, &src, nullptr); priv->shader_buf.Get (), upload_buf.Get (), &priv->vbv, &priv->ibv,
in_trc, nullptr, out_trc, nullptr, &priv->setup_fence,
src = &priv->setup_fence_val);
CD3DX12_TEXTURE_COPY_LOCATION (gamma_enc_lut_upload.Get (),
priv->gamma_lut_layout);
dst = CD3DX12_TEXTURE_COPY_LOCATION (priv->gamma_enc_lut.Get ());
cl->CopyTextureRegion (&dst, 0, 0, 0, &src, nullptr);
barriers.
push_back (CD3DX12_RESOURCE_BARRIER::Transition (priv->gamma_dec_lut.
Get (), D3D12_RESOURCE_STATE_COPY_DEST,
D3D12_RESOURCE_STATE_PIXEL_SHADER_RESOURCE));
barriers.
push_back (CD3DX12_RESOURCE_BARRIER::Transition (priv->gamma_enc_lut.
Get (), D3D12_RESOURCE_STATE_COPY_DEST,
D3D12_RESOURCE_STATE_PIXEL_SHADER_RESOURCE));
} }
cl->ResourceBarrier (barriers.size (), barriers.data ());
hr = cl->Close ();
if (!gst_d3d12_result (hr, self->device)) {
GST_ERROR_OBJECT (self, "Couldn't close upload command list");
return FALSE;
}
ID3D12CommandList *cmd_list[] = { cl.Get () };
hr = gst_d3d12_cmd_queue_execute_command_lists (priv->cq, 1, cmd_list,
&priv->fence_val);
if (!gst_d3d12_result (hr, self->device)) { if (!gst_d3d12_result (hr, self->device)) {
GST_ERROR_OBJECT (self, "Couldn't execute command list"); GST_ERROR_OBJECT (self, "Couldn't execute command list");
return FALSE; return FALSE;
} }
GstD3D12FenceData *fence_data; if (priv->have_lut) {
gst_d3d12_device_acquire_fence_data (self->device, &fence_data); D3D12_DESCRIPTOR_HEAP_DESC desc = { };
gst_d3d12_fence_data_push (fence_data, FENCE_NOTIFY_COM (cl.Detach ())); desc.Type = D3D12_DESCRIPTOR_HEAP_TYPE_CBV_SRV_UAV;
gst_d3d12_fence_data_push (fence_data, FENCE_NOTIFY_COM (ca.Detach ())); desc.NumDescriptors = 2;
gst_d3d12_fence_data_push (fence_data, desc.Flags = D3D12_DESCRIPTOR_HEAP_FLAG_NONE;
FENCE_NOTIFY_COM (upload_buf.Detach ()));
if (gamma_dec_lut_upload) {
gst_d3d12_fence_data_push (fence_data,
FENCE_NOTIFY_COM (gamma_dec_lut_upload.Detach ()));
}
if (gamma_enc_lut_upload) { auto hr = device->CreateDescriptorHeap (&desc,
gst_d3d12_fence_data_push (fence_data, IID_PPV_ARGS (&priv->gamma_lut_heap));
FENCE_NOTIFY_COM (gamma_enc_lut_upload.Detach ())); if (!gst_d3d12_result (hr, self->device)) {
} GST_ERROR_OBJECT (self, "Couldn't create gamma lut heap");
return FALSE;
}
gst_d3d12_cmd_queue_set_notify (priv->cq, priv->fence_val, auto cpu_handle =
FENCE_NOTIFY_MINI_OBJECT (fence_data)); CD3DX12_CPU_DESCRIPTOR_HANDLE (GetCPUDescriptorHandleForHeapStart
(priv->gamma_lut_heap));
D3D12_SHADER_RESOURCE_VIEW_DESC srv_desc = { };
srv_desc.ViewDimension = D3D12_SRV_DIMENSION_TEXTURE1D;
srv_desc.Shader4ComponentMapping = D3D12_DEFAULT_SHADER_4_COMPONENT_MAPPING;
srv_desc.Texture1D.MipLevels = 1;
device->CreateShaderResourceView (priv->gamma_dec_lut.Get (), &srv_desc,
cpu_handle);
cpu_handle.Offset (priv->srv_inc_size);
device->CreateShaderResourceView (priv->gamma_enc_lut.Get (), &srv_desc,
cpu_handle);
}
return TRUE; return TRUE;
} }
@ -2410,7 +2236,7 @@ gst_d3d12_converter_execute (GstD3D12Converter * self, GstD3D12Frame * in_frame,
cl->SetGraphicsRootConstantBufferView (pipeline_data.crs->GetPsCbvIdx (), cl->SetGraphicsRootConstantBufferView (pipeline_data.crs->GetPsCbvIdx (),
priv->const_buf_addr[pipeline_index]); priv->const_buf_addr[pipeline_index]);
cl->IASetIndexBuffer (&priv->idv); cl->IASetIndexBuffer (&priv->ibv);
cl->IASetVertexBuffers (0, 1, &priv->vbv); cl->IASetVertexBuffers (0, 1, &priv->vbv);
cl->IASetPrimitiveTopology (D3D_PRIMITIVE_TOPOLOGY_TRIANGLELIST); cl->IASetPrimitiveTopology (D3D_PRIMITIVE_TOPOLOGY_TRIANGLELIST);
cl->RSSetViewports (1, priv->viewport); cl->RSSetViewports (1, priv->viewport);
@ -2534,6 +2360,20 @@ gst_d3d12_converter_convert_buffer (GstD3D12Converter * converter,
gst_d3d12_frame_fence_gpu_wait (&out_frame, priv->cq); gst_d3d12_frame_fence_gpu_wait (&out_frame, priv->cq);
} }
if (priv->setup_fence) {
auto default_queue = gst_d3d12_device_get_cmd_queue (converter->device,
D3D12_COMMAND_LIST_TYPE_DIRECT);
if (priv->cq != default_queue) {
auto completed = priv->setup_fence->GetCompletedValue ();
if (completed < priv->setup_fence_val) {
gst_d3d12_cmd_queue_execute_wait (priv->cq, priv->setup_fence.Get (),
priv->setup_fence_val);
}
}
priv->setup_fence = nullptr;
}
gst_d3d12_frame_unmap (&in_frame); gst_d3d12_frame_unmap (&in_frame);
gst_d3d12_frame_unmap (&out_frame); gst_d3d12_frame_unmap (&out_frame);

View file

@ -100,5 +100,18 @@ HRESULT gst_d3d12_device_get_sampler_state (GstD3D12Device * device,
GST_D3D12_API GST_D3D12_API
gboolean gst_d3d12_device_non_zeroed_supported (GstD3D12Device * device); gboolean gst_d3d12_device_non_zeroed_supported (GstD3D12Device * device);
GST_D3D12_API
HRESULT gst_d3d12_device_get_converter_resources (GstD3D12Device * device,
ID3D12Resource * index_buf,
ID3D12Resource * index_upload,
const D3D12_VERTEX_BUFFER_VIEW * vbv,
const D3D12_INDEX_BUFFER_VIEW * ibv,
GstVideoTransferFunction gamma_dec_func,
ID3D12Resource ** gamma_dec,
GstVideoTransferFunction gamma_enc_func,
ID3D12Resource ** gamma_enc,
ID3D12Fence ** fence,
guint64 * fence_val);
G_END_DECLS G_END_DECLS

View file

@ -26,6 +26,7 @@
#include "gstd3d12cmdlistpool.h" #include "gstd3d12cmdlistpool.h"
#include <directx/d3dx12.h> #include <directx/d3dx12.h>
#include <d3d11on12.h> #include <d3d11on12.h>
#include <gst/d3dshader/gstd3dshader.h>
#include <wrl.h> #include <wrl.h>
#include <vector> #include <vector>
#include <string.h> #include <string.h>
@ -142,8 +143,13 @@ struct DeviceInner
gst_clear_object (&copy_cl_pool); gst_clear_object (&copy_cl_pool);
gst_clear_object (&fence_data_pool); gst_clear_object (&fence_data_pool);
gst_clear_object (&rtv_heap_pool);
gamma_dec_lut.clear();
gamma_enc_lut.clear();
samplers.clear (); samplers.clear ();
gamma_lut_pso = nullptr;
gamma_lut_rs = nullptr;
factory = nullptr; factory = nullptr;
adapter = nullptr; adapter = nullptr;
@ -264,6 +270,12 @@ struct DeviceInner
GstD3D12FenceDataPool *fence_data_pool = nullptr; GstD3D12FenceDataPool *fence_data_pool = nullptr;
ComPtr<ID3D12RootSignature> gamma_lut_rs;
ComPtr<ID3D12PipelineState> gamma_lut_pso;
std::unordered_map<DWORD, ComPtr<ID3D12Resource>> gamma_dec_lut;
std::unordered_map<DWORD, ComPtr<ID3D12Resource>> gamma_enc_lut;
GstD3D12DescHeapPool *rtv_heap_pool = nullptr;
guint rtv_inc_size; guint rtv_inc_size;
guint adapter_index = 0; guint adapter_index = 0;
@ -1392,6 +1404,17 @@ gst_d3d12_device_new_internal (const GstD3D12DeviceConstructData * data)
priv->non_zeroed_supported = TRUE; priv->non_zeroed_supported = TRUE;
} }
{
D3D12_DESCRIPTOR_HEAP_DESC rtv_desc = { };
rtv_desc.Type = D3D12_DESCRIPTOR_HEAP_TYPE_RTV;
rtv_desc.NumDescriptors = 2;
rtv_desc.Flags = D3D12_DESCRIPTOR_HEAP_FLAG_NONE;
priv->rtv_heap_pool = gst_d3d12_desc_heap_pool_new (device.Get (),
&rtv_desc);
GST_OBJECT_FLAG_SET (priv->rtv_heap_pool, GST_OBJECT_FLAG_MAY_BE_LEAKED);
}
return self; return self;
error: error:
@ -2243,3 +2266,374 @@ gst_d3d12_flush_all_devices (void)
auto manager = DeviceCacheManager::GetInstance (); auto manager = DeviceCacheManager::GetInstance ();
manager->FlushAll (); manager->FlushAll ();
} }
static inline DWORD
gst_d3d12_transfer_func_to_gamma_func (GstVideoTransferFunction func)
{
enum class GammaFuncType
{
GAMMA10,
GAMMA18,
GAMMA20,
GAMMA22,
BT709,
SMPTE240M,
SRGB,
GAMMA28,
LOG100,
LOG316,
BT2020,
ADOBERGB,
PQ,
HLG,
};
switch (func) {
case GST_VIDEO_TRANSFER_GAMMA18:
return (DWORD) GammaFuncType::GAMMA18;
case GST_VIDEO_TRANSFER_GAMMA20:
return (DWORD) GammaFuncType::GAMMA20;
case GST_VIDEO_TRANSFER_GAMMA22:
return (DWORD) GammaFuncType::GAMMA22;
case GST_VIDEO_TRANSFER_BT601:
case GST_VIDEO_TRANSFER_BT709:
case GST_VIDEO_TRANSFER_BT2020_10:
return (DWORD) GammaFuncType::BT709;
case GST_VIDEO_TRANSFER_SMPTE240M:
return (DWORD) GammaFuncType::SMPTE240M;
case GST_VIDEO_TRANSFER_SRGB:
return (DWORD) GammaFuncType::SRGB;
case GST_VIDEO_TRANSFER_GAMMA28:
return (DWORD) GammaFuncType::GAMMA28;
case GST_VIDEO_TRANSFER_LOG100:
return (DWORD) GammaFuncType::LOG100;
case GST_VIDEO_TRANSFER_LOG316:
return (DWORD) GammaFuncType::LOG316;
case GST_VIDEO_TRANSFER_BT2020_12:
return (DWORD) GammaFuncType::BT2020;
case GST_VIDEO_TRANSFER_ADOBERGB:
return (DWORD) GammaFuncType::ADOBERGB;
case GST_VIDEO_TRANSFER_SMPTE2084:
return (DWORD) GammaFuncType::PQ;
case GST_VIDEO_TRANSFER_ARIB_STD_B67:
return (DWORD) GammaFuncType::HLG;
default:
break;
}
return (DWORD) GammaFuncType::GAMMA10;
}
HRESULT
gst_d3d12_device_get_converter_resources (GstD3D12Device * device,
ID3D12Resource * index_buf, ID3D12Resource * index_upload,
const D3D12_VERTEX_BUFFER_VIEW * vbv, const D3D12_INDEX_BUFFER_VIEW * ibv,
GstVideoTransferFunction gamma_dec_func, ID3D12Resource ** gamma_dec,
GstVideoTransferFunction gamma_enc_func, ID3D12Resource ** gamma_enc,
ID3D12Fence ** fence, guint64 * fence_val)
{
auto priv = device->priv->inner;
GstD3D12FenceData *fence_data = nullptr;
GstD3D12CmdAlloc *gst_ca = nullptr;
GstD3D12CmdList *gst_cl = nullptr;
bool need_lut = false;
HRESULT hr = S_OK;
DWORD gamma_dec_func_d3d12 = 0;
DWORD gamma_enc_func_d3d12 = 0;
if (gamma_dec != nullptr && gamma_enc != nullptr)
need_lut = true;
if (need_lut) {
gamma_dec_func_d3d12 =
gst_d3d12_transfer_func_to_gamma_func (gamma_dec_func);
gamma_enc_func_d3d12 =
gst_d3d12_transfer_func_to_gamma_func (gamma_enc_func);
}
std::lock_guard < std::mutex > lk (priv->lock);
if (!priv->gamma_lut_rs) {
D3D12_ROOT_SIGNATURE_FLAGS rs_flags =
D3D12_ROOT_SIGNATURE_FLAG_ALLOW_INPUT_ASSEMBLER_INPUT_LAYOUT |
D3D12_ROOT_SIGNATURE_FLAG_DENY_HULL_SHADER_ROOT_ACCESS |
D3D12_ROOT_SIGNATURE_FLAG_DENY_DOMAIN_SHADER_ROOT_ACCESS |
D3D12_ROOT_SIGNATURE_FLAG_DENY_GEOMETRY_SHADER_ROOT_ACCESS |
D3D12_ROOT_SIGNATURE_FLAG_DENY_AMPLIFICATION_SHADER_ROOT_ACCESS |
D3D12_ROOT_SIGNATURE_FLAG_DENY_MESH_SHADER_ROOT_ACCESS;
D3D12_VERSIONED_ROOT_SIGNATURE_DESC rs_desc = { };
CD3DX12_ROOT_PARAMETER root_params[1];
root_params[0].InitAsConstants (2, 0);
CD3DX12_VERSIONED_ROOT_SIGNATURE_DESC::Init_1_0 (rs_desc, 1, root_params,
0, nullptr, rs_flags);
ComPtr < ID3DBlob > rs_blob;
ComPtr < ID3DBlob > error_blob;
hr = D3DX12SerializeVersionedRootSignature (&rs_desc,
D3D_ROOT_SIGNATURE_VERSION_1, &rs_blob, &error_blob);
if (!gst_d3d12_result (hr, device)) {
const gchar *error_msg = nullptr;
if (error_blob)
error_msg = (const gchar *) error_blob->GetBufferPointer ();
GST_ERROR_OBJECT (device,
"Couldn't serialize root signature, hr: 0x%x, error detail: %s",
(guint) hr, GST_STR_NULL (error_msg));
return hr;
}
hr = priv->device->CreateRootSignature (0, rs_blob->GetBufferPointer (),
rs_blob->GetBufferSize (), IID_PPV_ARGS (&priv->gamma_lut_rs));
if (!gst_d3d12_result (hr, device)) {
GST_ERROR_OBJECT (device, "Couldn't create root signature");
return hr;
}
}
if (!priv->gamma_lut_pso) {
GstD3DShaderByteCode vs_blob;
GstD3DShaderByteCode ps_blob;
if (!gst_d3d12_shader_cache_get_gamma_lut_blob (&vs_blob, &ps_blob)) {
GST_ERROR_OBJECT (device, "Couldn't get gamma decode byte code");
return E_FAIL;
}
D3D12_INPUT_ELEMENT_DESC input_desc[2];
input_desc[0].SemanticName = "POSITION";
input_desc[0].SemanticIndex = 0;
input_desc[0].Format = DXGI_FORMAT_R32G32B32_FLOAT;
input_desc[0].InputSlot = 0;
input_desc[0].AlignedByteOffset = D3D12_APPEND_ALIGNED_ELEMENT;
input_desc[0].InputSlotClass = D3D12_INPUT_CLASSIFICATION_PER_VERTEX_DATA;
input_desc[0].InstanceDataStepRate = 0;
input_desc[1].SemanticName = "TEXCOORD";
input_desc[1].SemanticIndex = 0;
input_desc[1].Format = DXGI_FORMAT_R32G32_FLOAT;
input_desc[1].InputSlot = 0;
input_desc[1].AlignedByteOffset = D3D12_APPEND_ALIGNED_ELEMENT;
input_desc[1].InputSlotClass = D3D12_INPUT_CLASSIFICATION_PER_VERTEX_DATA;
input_desc[1].InstanceDataStepRate = 0;
D3D12_GRAPHICS_PIPELINE_STATE_DESC pso_desc = { };
pso_desc.pRootSignature = priv->gamma_lut_rs.Get ();
pso_desc.VS.pShaderBytecode = vs_blob.byte_code;
pso_desc.VS.BytecodeLength = vs_blob.byte_code_len;
pso_desc.PS.pShaderBytecode = ps_blob.byte_code;
pso_desc.PS.BytecodeLength = ps_blob.byte_code_len;
pso_desc.BlendState = CD3DX12_BLEND_DESC (D3D12_DEFAULT);
pso_desc.SampleMask = UINT_MAX;
pso_desc.RasterizerState = CD3DX12_RASTERIZER_DESC (D3D12_DEFAULT);
pso_desc.RasterizerState.CullMode = D3D12_CULL_MODE_NONE;
pso_desc.DepthStencilState.DepthEnable = FALSE;
pso_desc.DepthStencilState.StencilEnable = FALSE;
pso_desc.InputLayout.pInputElementDescs = input_desc;
pso_desc.InputLayout.NumElements = 2;
pso_desc.PrimitiveTopologyType = D3D12_PRIMITIVE_TOPOLOGY_TYPE_TRIANGLE;
pso_desc.NumRenderTargets = 1;
pso_desc.RTVFormats[0] = DXGI_FORMAT_R16_UNORM;
pso_desc.SampleDesc.Count = 1;
pso_desc.SampleDesc.Quality = 0;
hr = priv->device->CreateGraphicsPipelineState (&pso_desc,
IID_PPV_ARGS (&priv->gamma_lut_pso));
if (!gst_d3d12_result (hr, device)) {
GST_ERROR_OBJECT (device, "Couldn't create gamma decode pso");
return hr;
}
}
gst_d3d12_fence_data_pool_acquire (priv->fence_data_pool, &fence_data);
gst_d3d12_cmd_alloc_pool_acquire (priv->direct_ca_pool, &gst_ca);
if (!gst_ca) {
GST_ERROR_OBJECT (device, "Couldn't acquire command allocator");
gst_d3d12_fence_data_unref (fence_data);
return E_FAIL;
}
gst_d3d12_fence_data_push (fence_data, FENCE_NOTIFY_MINI_OBJECT (gst_ca));
auto ca = gst_d3d12_cmd_alloc_get_handle (gst_ca);
gst_d3d12_cmd_list_pool_acquire (priv->direct_cl_pool, ca, &gst_cl);
if (!gst_cl) {
GST_ERROR_OBJECT (device, "Couldn't acquire command list");
gst_d3d12_fence_data_unref (fence_data);
return E_FAIL;
}
ComPtr < ID3D12CommandList > cl_base;
ComPtr < ID3D12GraphicsCommandList > cl;
cl_base = gst_d3d12_cmd_list_get_handle (gst_cl);
cl_base.As (&cl);
cl->CopyResource (index_buf, index_upload);
index_buf->AddRef ();
gst_d3d12_fence_data_push (fence_data, FENCE_NOTIFY_COM (index_buf));
index_upload->AddRef ();
gst_d3d12_fence_data_push (fence_data, FENCE_NOTIFY_COM (index_upload));
D3D12_RESOURCE_BARRIER copy_barrier =
CD3DX12_RESOURCE_BARRIER::Transition (index_buf,
D3D12_RESOURCE_STATE_COPY_DEST,
D3D12_RESOURCE_STATE_VERTEX_AND_CONSTANT_BUFFER |
D3D12_RESOURCE_STATE_INDEX_BUFFER,
D3D12_RESOURCE_BARRIER_ALL_SUBRESOURCES);
cl->ResourceBarrier (1, &copy_barrier);
bool store_dec = false;
bool store_enc = false;
ComPtr < ID3D12Resource > gamma_dec_resource;
ComPtr < ID3D12Resource > gamma_enc_resource;
if (need_lut) {
const UINT lut_size = 4096;
auto dec_iter = priv->gamma_dec_lut.find (gamma_dec_func_d3d12);
if (dec_iter != priv->gamma_dec_lut.end ()) {
GST_LOG_OBJECT (device, "Reuse gamma decode LUT");
gamma_dec_resource = dec_iter->second;
} else {
GST_DEBUG_OBJECT (device, "Need to build gamma decode LUT");
}
auto enc_iter = priv->gamma_enc_lut.find (gamma_enc_func_d3d12);
if (enc_iter != priv->gamma_enc_lut.end ()) {
GST_LOG_OBJECT (device, "Reuse gamma encode LUT");
gamma_enc_resource = enc_iter->second;
} else {
GST_DEBUG_OBJECT (device, "Need to build gamma encode LUT");
}
if (!gamma_dec_resource || !gamma_enc_resource) {
GstD3D12DescHeap *desc_heap = nullptr;
D3D12_HEAP_FLAGS heap_flags = D3D12_HEAP_FLAG_NONE;
if (gst_d3d12_device_non_zeroed_supported (device))
heap_flags = D3D12_HEAP_FLAG_CREATE_NOT_ZEROED;
auto heap_prop = CD3DX12_HEAP_PROPERTIES (D3D12_HEAP_TYPE_DEFAULT);
auto resource_desc = CD3DX12_RESOURCE_DESC::Tex1D (DXGI_FORMAT_R16_UNORM,
lut_size, 1, 1, D3D12_RESOURCE_FLAG_ALLOW_RENDER_TARGET |
D3D12_RESOURCE_FLAG_ALLOW_SIMULTANEOUS_ACCESS);
D3D12_RENDER_TARGET_VIEW_DESC rtv_desc = { };
rtv_desc.Format = DXGI_FORMAT_R16_UNORM;
rtv_desc.ViewDimension = D3D12_RTV_DIMENSION_TEXTURE1D;
rtv_desc.Texture1D.MipSlice = 0;
gst_d3d12_desc_heap_pool_acquire (priv->rtv_heap_pool, &desc_heap);
if (!desc_heap) {
GST_ERROR_OBJECT (device, "Couldn't acquire descriptor heap");
gst_d3d12_fence_data_unref (fence_data);
return E_FAIL;
}
gst_d3d12_fence_data_push (fence_data,
FENCE_NOTIFY_MINI_OBJECT (desc_heap));
auto desc_handle = gst_d3d12_desc_heap_get_handle (desc_heap);
auto cpu_handle = CD3DX12_CPU_DESCRIPTOR_HANDLE
(GetCPUDescriptorHandleForHeapStart (desc_handle));
cl->SetGraphicsRootSignature (priv->gamma_lut_rs.Get ());
cl->SetPipelineState (priv->gamma_lut_pso.Get ());
cl->IASetIndexBuffer (ibv);
cl->IASetVertexBuffers (0, 1, vbv);
cl->IASetPrimitiveTopology (D3D_PRIMITIVE_TOPOLOGY_TRIANGLELIST);
D3D12_VIEWPORT viewport = { };
viewport.Width = lut_size;
viewport.Height = 1;
viewport.MinDepth = 0;
viewport.MaxDepth = 1;
cl->RSSetViewports (1, &viewport);
D3D12_RECT scissor_rect = { };
scissor_rect.left = 0;
scissor_rect.top = 0;
scissor_rect.right = lut_size;
scissor_rect.bottom = 1;
cl->RSSetScissorRects (1, &scissor_rect);
if (!gamma_dec_resource) {
store_dec = true;
hr = priv->device->CreateCommittedResource (&heap_prop, heap_flags,
&resource_desc, D3D12_RESOURCE_STATE_COMMON, nullptr,
IID_PPV_ARGS (&gamma_dec_resource));
if (!gst_d3d12_result (hr, device)) {
GST_ERROR_OBJECT (device, "Couldn't create LUT texture");
return hr;
}
priv->device->CreateRenderTargetView (gamma_dec_resource.Get (),
&rtv_desc, cpu_handle);
cl->SetGraphicsRoot32BitConstant (0, 1, 0);
cl->SetGraphicsRoot32BitConstant (0, gamma_dec_func_d3d12, 1);
cl->OMSetRenderTargets (1, &cpu_handle, FALSE, nullptr);
cl->DrawIndexedInstanced (6, 1, 0, 0, 0);
cpu_handle.Offset (priv->rtv_inc_size);
}
if (!gamma_enc_resource) {
store_enc = true;
hr = priv->device->CreateCommittedResource (&heap_prop, heap_flags,
&resource_desc, D3D12_RESOURCE_STATE_COMMON, nullptr,
IID_PPV_ARGS (&gamma_enc_resource));
if (!gst_d3d12_result (hr, device)) {
GST_ERROR_OBJECT (device, "Couldn't create LUT texture");
return hr;
}
priv->device->CreateRenderTargetView (gamma_enc_resource.Get (),
&rtv_desc, cpu_handle);
cl->SetGraphicsRoot32BitConstant (0, 0, 0);
cl->SetGraphicsRoot32BitConstant (0, gamma_enc_func_d3d12, 1);
cl->OMSetRenderTargets (1, &cpu_handle, FALSE, nullptr);
cl->DrawIndexedInstanced (6, 1, 0, 0, 0);
}
}
}
hr = cl->Close ();
if (!gst_d3d12_result (hr, device)) {
GST_ERROR_OBJECT (device, "Couldn't close command list");
gst_d3d12_fence_data_unref (fence_data);
return hr;
}
ID3D12CommandList *cmd_list[] = { cl.Get () };
hr = gst_d3d12_cmd_queue_execute_command_lists (priv->direct_queue,
1, cmd_list, fence_val);
if (!gst_d3d12_result (hr, device)) {
GST_ERROR_OBJECT (device, "Couldn't execute command list");
gst_d3d12_fence_data_unref (fence_data);
return hr;
}
gst_d3d12_cmd_queue_set_notify (priv->direct_queue, *fence_val, fence_data,
(GDestroyNotify) gst_d3d12_fence_data_unref);
if (need_lut) {
if (store_dec)
priv->gamma_dec_lut[gamma_dec_func_d3d12] = gamma_dec_resource;
if (store_enc)
priv->gamma_enc_lut[gamma_enc_func_d3d12] = gamma_enc_resource;
*gamma_dec = gamma_dec_resource.Detach ();
*gamma_enc = gamma_enc_resource.Detach ();
}
*fence = gst_d3d12_cmd_queue_get_fence_handle (priv->direct_queue);
(*fence)->AddRef ();
return S_OK;
}