mirror of
https://gitlab.freedesktop.org/gstreamer/gstreamer.git
synced 2025-01-11 09:55:36 +00:00
d3d11converter: Introduce config to be extensible
Add a config argument like that of GstVideoConverter so that we can add more options without modifying existing methods Part-of: <https://gitlab.freedesktop.org/gstreamer/gst-plugins-bad/-/merge_requests/2276>
This commit is contained in:
parent
f3331652f2
commit
0d34a0233b
9 changed files with 278 additions and 66 deletions
|
@ -1200,11 +1200,20 @@ gst_d3d11_compositor_pad_setup_converter (GstVideoAggregatorPad * pad,
|
||||||
#endif
|
#endif
|
||||||
|
|
||||||
if (!cpad->convert || cpad->alpha_updated || self->reconfigured) {
|
if (!cpad->convert || cpad->alpha_updated || self->reconfigured) {
|
||||||
|
GstStructure *config;
|
||||||
|
|
||||||
if (cpad->convert)
|
if (cpad->convert)
|
||||||
gst_d3d11_converter_free (cpad->convert);
|
gst_d3d11_converter_free (cpad->convert);
|
||||||
|
|
||||||
|
config = gst_structure_new_empty ("config");
|
||||||
|
if (cpad->alpha <= 1.0) {
|
||||||
|
gst_structure_set (config, GST_D3D11_CONVERTER_OPT_ALPHA_VALUE,
|
||||||
|
G_TYPE_DOUBLE, cpad->alpha, nullptr);
|
||||||
|
}
|
||||||
|
|
||||||
cpad->convert =
|
cpad->convert =
|
||||||
gst_d3d11_converter_new_with_alpha (self->device,
|
gst_d3d11_converter_new (self->device, &pad->info, &vagg->info, config);
|
||||||
&pad->info, &vagg->info, cpad->alpha);
|
|
||||||
cpad->alpha_updated = FALSE;
|
cpad->alpha_updated = FALSE;
|
||||||
if (!cpad->convert) {
|
if (!cpad->convert) {
|
||||||
GST_ERROR_OBJECT (pad, "Couldn't create converter");
|
GST_ERROR_OBJECT (pad, "Couldn't create converter");
|
||||||
|
@ -2049,7 +2058,7 @@ gst_d3d11_compositor_create_checker_quad (GstD3D11Compositor * self)
|
||||||
context_handle->Unmap (index_buffer.Get (), 0);
|
context_handle->Unmap (index_buffer.Get (), 0);
|
||||||
|
|
||||||
quad = gst_d3d11_quad_new (self->device,
|
quad = gst_d3d11_quad_new (self->device,
|
||||||
ps.Get (), vs.Get (), layout.Get (), NULL,
|
ps.Get (), vs.Get (), layout.Get (), nullptr, 0,
|
||||||
vertex_buffer.Get (), sizeof (VertexData), index_buffer.Get (),
|
vertex_buffer.Get (), sizeof (VertexData), index_buffer.Get (),
|
||||||
DXGI_FORMAT_R16_UINT, 6);
|
DXGI_FORMAT_R16_UINT, 6);
|
||||||
if (!quad) {
|
if (!quad) {
|
||||||
|
|
|
@ -1783,7 +1783,8 @@ gst_d3d11_base_convert_set_info (GstD3D11BaseFilter * filter,
|
||||||
goto format_unknown;
|
goto format_unknown;
|
||||||
}
|
}
|
||||||
|
|
||||||
self->converter = gst_d3d11_converter_new (filter->device, in_info, out_info);
|
self->converter =
|
||||||
|
gst_d3d11_converter_new (filter->device, in_info, out_info, nullptr);
|
||||||
|
|
||||||
if (!self->converter) {
|
if (!self->converter) {
|
||||||
GST_ERROR_OBJECT (self, "couldn't set converter");
|
GST_ERROR_OBJECT (self, "couldn't set converter");
|
||||||
|
|
|
@ -48,6 +48,12 @@ typedef struct
|
||||||
FLOAT padding[4];
|
FLOAT padding[4];
|
||||||
} PixelShaderColorTransform;
|
} PixelShaderColorTransform;
|
||||||
|
|
||||||
|
typedef struct
|
||||||
|
{
|
||||||
|
FLOAT alpha_mul;
|
||||||
|
FLOAT padding[3];
|
||||||
|
} AlphaConstBuffer;
|
||||||
|
|
||||||
typedef struct
|
typedef struct
|
||||||
{
|
{
|
||||||
struct {
|
struct {
|
||||||
|
@ -63,16 +69,24 @@ typedef struct
|
||||||
|
|
||||||
typedef struct
|
typedef struct
|
||||||
{
|
{
|
||||||
const gchar *constant_buffer;
|
gboolean has_transform;
|
||||||
|
gboolean has_alpha;
|
||||||
const gchar *func;
|
const gchar *func;
|
||||||
} PixelShaderTemplate;
|
} PixelShaderTemplate;
|
||||||
|
|
||||||
#define COLOR_TRANSFORM_COEFF \
|
static const gchar templ_color_transform_const_buffer[] =
|
||||||
"cbuffer PixelShaderColorTransform : register(b0)\n" \
|
"cbuffer PixelShaderColorTransform : register(b%u)\n"
|
||||||
"{\n" \
|
"{\n"
|
||||||
" float3x4 trans_matrix;\n" \
|
" float3x4 trans_matrix;\n"
|
||||||
" float3 padding;\n" \
|
" float3 padding;\n"
|
||||||
"};\n"
|
"};";
|
||||||
|
|
||||||
|
static const gchar templ_alpha_const_buffer[] =
|
||||||
|
"cbuffer AlphaConstBuffer : register(b%u)\n"
|
||||||
|
"{\n"
|
||||||
|
" float alpha_mul;\n"
|
||||||
|
" float3 padding;\n"
|
||||||
|
"};";
|
||||||
|
|
||||||
#define HLSL_FUNC_YUV_TO_RGB \
|
#define HLSL_FUNC_YUV_TO_RGB \
|
||||||
"float3 yuv_to_rgb (float3 yuv)\n" \
|
"float3 yuv_to_rgb (float3 yuv)\n" \
|
||||||
|
@ -99,18 +113,18 @@ typedef struct
|
||||||
" float4 Plane_1: SV_TARGET1;"
|
" float4 Plane_1: SV_TARGET1;"
|
||||||
|
|
||||||
static const PixelShaderTemplate templ_REORDER =
|
static const PixelShaderTemplate templ_REORDER =
|
||||||
{ NULL, NULL };
|
{ FALSE, TRUE, NULL };
|
||||||
|
|
||||||
static const PixelShaderTemplate templ_YUV_to_RGB =
|
static const PixelShaderTemplate templ_YUV_to_RGB =
|
||||||
{ COLOR_TRANSFORM_COEFF, HLSL_FUNC_YUV_TO_RGB };
|
{ TRUE, FALSE, HLSL_FUNC_YUV_TO_RGB };
|
||||||
|
|
||||||
static const PixelShaderTemplate templ_RGB_to_YUV =
|
static const PixelShaderTemplate templ_RGB_to_YUV =
|
||||||
{ COLOR_TRANSFORM_COEFF, HLSL_FUNC_RGB_TO_YUV };
|
{ TRUE, FALSE, HLSL_FUNC_RGB_TO_YUV };
|
||||||
|
|
||||||
static const gchar templ_REORDER_BODY[] =
|
static const gchar templ_REORDER_BODY[] =
|
||||||
" float4 xyza;\n"
|
" float4 xyza;\n"
|
||||||
" xyza.xyz = shaderTexture[0].Sample(samplerState, input.Texture).xyz;\n"
|
" xyza.xyz = shaderTexture[0].Sample(samplerState, input.Texture).xyz;\n"
|
||||||
" xyza.a = shaderTexture[0].Sample(samplerState, input.Texture).a * %f;\n"
|
" xyza.a = shaderTexture[0].Sample(samplerState, input.Texture).a * alpha_mul;\n"
|
||||||
" output.Plane_0 = xyza;\n";
|
" output.Plane_0 = xyza;\n";
|
||||||
|
|
||||||
static const gchar templ_VUYA_to_RGB_BODY[] =
|
static const gchar templ_VUYA_to_RGB_BODY[] =
|
||||||
|
@ -275,6 +289,7 @@ static const gchar templ_PACKED_YUV_TO_SEMI_PLANAR_CHROMA_BODY[] =
|
||||||
static const gchar templ_pixel_shader[] =
|
static const gchar templ_pixel_shader[] =
|
||||||
/* constant buffer */
|
/* constant buffer */
|
||||||
"%s\n"
|
"%s\n"
|
||||||
|
"%s\n"
|
||||||
"Texture2D shaderTexture[4];\n"
|
"Texture2D shaderTexture[4];\n"
|
||||||
"SamplerState samplerState;\n"
|
"SamplerState samplerState;\n"
|
||||||
"\n"
|
"\n"
|
||||||
|
@ -331,7 +346,7 @@ struct _GstD3D11Converter
|
||||||
GstD3D11Device *device;
|
GstD3D11Device *device;
|
||||||
GstVideoInfo in_info;
|
GstVideoInfo in_info;
|
||||||
GstVideoInfo out_info;
|
GstVideoInfo out_info;
|
||||||
gfloat alpha;
|
gdouble alpha;
|
||||||
|
|
||||||
const GstD3D11Format *in_d3d11_format;
|
const GstD3D11Format *in_d3d11_format;
|
||||||
const GstD3D11Format *out_d3d11_format;
|
const GstD3D11Format *out_d3d11_format;
|
||||||
|
@ -348,13 +363,32 @@ struct _GstD3D11Converter
|
||||||
gint input_texture_width;
|
gint input_texture_width;
|
||||||
gint input_texture_height;
|
gint input_texture_height;
|
||||||
ID3D11Buffer *vertex_buffer;
|
ID3D11Buffer *vertex_buffer;
|
||||||
|
ID3D11Buffer *alpha_const_buffer;
|
||||||
gboolean update_vertex;
|
gboolean update_vertex;
|
||||||
|
gboolean update_alpha;
|
||||||
|
|
||||||
ID3D11SamplerState *linear_sampler;
|
ID3D11SamplerState *linear_sampler;
|
||||||
|
|
||||||
ConvertInfo convert_info;
|
ConvertInfo convert_info;
|
||||||
|
|
||||||
|
GstStructure *config;
|
||||||
};
|
};
|
||||||
|
|
||||||
|
static gdouble
|
||||||
|
get_opt_double (GstD3D11Converter * self, const gchar * opt, gdouble def)
|
||||||
|
{
|
||||||
|
gdouble res;
|
||||||
|
if (!gst_structure_get_double (self->config, opt, &res))
|
||||||
|
res = def;
|
||||||
|
|
||||||
|
return res;
|
||||||
|
}
|
||||||
|
|
||||||
|
#define DEFAULT_OPT_ALPHA_VALUE 1.0
|
||||||
|
|
||||||
|
#define GET_OPT_ALPHA_VALUE(c) get_opt_double(c, \
|
||||||
|
GST_D3D11_CONVERTER_OPT_ALPHA_VALUE, DEFAULT_OPT_ALPHA_VALUE);
|
||||||
|
|
||||||
/* from video-converter.c */
|
/* from video-converter.c */
|
||||||
typedef struct
|
typedef struct
|
||||||
{
|
{
|
||||||
|
@ -594,7 +628,7 @@ setup_convert_info_rgb_to_rgb (GstD3D11Converter * self,
|
||||||
ConvertInfo *convert_info = &self->convert_info;
|
ConvertInfo *convert_info = &self->convert_info;
|
||||||
|
|
||||||
convert_info->templ = &templ_REORDER;
|
convert_info->templ = &templ_REORDER;
|
||||||
convert_info->ps_body[0] = g_strdup_printf (templ_REORDER_BODY, self->alpha);
|
convert_info->ps_body[0] = g_strdup_printf (templ_REORDER_BODY);
|
||||||
convert_info->ps_output[0] = HLSL_PS_OUTPUT_ONE_PLANE_BODY;
|
convert_info->ps_output[0] = HLSL_PS_OUTPUT_ONE_PLANE_BODY;
|
||||||
|
|
||||||
return TRUE;
|
return TRUE;
|
||||||
|
@ -847,7 +881,7 @@ setup_convert_info_vuya_to_vuya (GstD3D11Converter * self,
|
||||||
info->templ = &templ_REORDER;
|
info->templ = &templ_REORDER;
|
||||||
info->ps_output[0] = HLSL_PS_OUTPUT_ONE_PLANE_BODY;
|
info->ps_output[0] = HLSL_PS_OUTPUT_ONE_PLANE_BODY;
|
||||||
|
|
||||||
info->ps_body[0] = g_strdup_printf (templ_REORDER_BODY, self->alpha);
|
info->ps_body[0] = g_strdup_printf (templ_REORDER_BODY);
|
||||||
|
|
||||||
return TRUE;
|
return TRUE;
|
||||||
}
|
}
|
||||||
|
@ -1071,13 +1105,16 @@ gst_d3d11_color_convert_setup_shader (GstD3D11Converter * self,
|
||||||
ComPtr<ID3D11VertexShader> vs;
|
ComPtr<ID3D11VertexShader> vs;
|
||||||
ComPtr<ID3D11InputLayout> layout;
|
ComPtr<ID3D11InputLayout> layout;
|
||||||
ComPtr<ID3D11SamplerState> linear_sampler;
|
ComPtr<ID3D11SamplerState> linear_sampler;
|
||||||
ComPtr<ID3D11Buffer> const_buffer;
|
ComPtr<ID3D11Buffer> transform_const_buffer;
|
||||||
|
ComPtr<ID3D11Buffer> alpha_const_buffer;
|
||||||
ComPtr<ID3D11Buffer> vertex_buffer;
|
ComPtr<ID3D11Buffer> vertex_buffer;
|
||||||
ComPtr<ID3D11Buffer> index_buffer;
|
ComPtr<ID3D11Buffer> index_buffer;
|
||||||
/* *INDENT-ON* */
|
/* *INDENT-ON* */
|
||||||
const guint index_count = 2 * 3;
|
const guint index_count = 2 * 3;
|
||||||
gint i;
|
gint i;
|
||||||
gboolean ret;
|
gboolean ret;
|
||||||
|
ID3D11Buffer *const_buffers[2] = { nullptr, nullptr };
|
||||||
|
guint num_const_buffers = 0;
|
||||||
|
|
||||||
memset (&sampler_desc, 0, sizeof (sampler_desc));
|
memset (&sampler_desc, 0, sizeof (sampler_desc));
|
||||||
memset (input_desc, 0, sizeof (input_desc));
|
memset (input_desc, 0, sizeof (input_desc));
|
||||||
|
@ -1105,14 +1142,32 @@ gst_d3d11_color_convert_setup_shader (GstD3D11Converter * self,
|
||||||
gchar *shader_code = NULL;
|
gchar *shader_code = NULL;
|
||||||
|
|
||||||
if (convert_info->ps_body[i]) {
|
if (convert_info->ps_body[i]) {
|
||||||
|
gchar *transform_const_buffer = nullptr;
|
||||||
|
gchar *alpha_const_buffer = nullptr;
|
||||||
|
guint register_idx = 0;
|
||||||
|
|
||||||
g_assert (convert_info->ps_output[i] != NULL);
|
g_assert (convert_info->ps_output[i] != NULL);
|
||||||
|
|
||||||
|
if (convert_info->templ->has_transform) {
|
||||||
|
transform_const_buffer =
|
||||||
|
g_strdup_printf (templ_color_transform_const_buffer, register_idx);
|
||||||
|
register_idx++;
|
||||||
|
}
|
||||||
|
|
||||||
|
if (convert_info->templ->has_alpha) {
|
||||||
|
alpha_const_buffer =
|
||||||
|
g_strdup_printf (templ_alpha_const_buffer, register_idx);
|
||||||
|
register_idx++;
|
||||||
|
}
|
||||||
|
|
||||||
shader_code = g_strdup_printf (templ_pixel_shader,
|
shader_code = g_strdup_printf (templ_pixel_shader,
|
||||||
convert_info->templ->constant_buffer ?
|
transform_const_buffer ? transform_const_buffer : "",
|
||||||
convert_info->templ->constant_buffer : "",
|
alpha_const_buffer ? alpha_const_buffer : "",
|
||||||
convert_info->ps_output[i],
|
convert_info->ps_output[i],
|
||||||
convert_info->templ->func ? convert_info->templ->func : "",
|
convert_info->templ->func ? convert_info->templ->func : "",
|
||||||
convert_info->ps_body[i]);
|
convert_info->ps_body[i]);
|
||||||
|
g_free (transform_const_buffer);
|
||||||
|
g_free (alpha_const_buffer);
|
||||||
|
|
||||||
ret = gst_d3d11_create_pixel_shader (device, shader_code, &ps[i]);
|
ret = gst_d3d11_create_pixel_shader (device, shader_code, &ps[i]);
|
||||||
g_free (shader_code);
|
g_free (shader_code);
|
||||||
|
@ -1122,9 +1177,11 @@ gst_d3d11_color_convert_setup_shader (GstD3D11Converter * self,
|
||||||
}
|
}
|
||||||
}
|
}
|
||||||
|
|
||||||
if (convert_info->templ->constant_buffer) {
|
if (convert_info->templ->has_transform) {
|
||||||
D3D11_BUFFER_DESC const_buffer_desc = { 0, };
|
D3D11_BUFFER_DESC const_buffer_desc = { 0, };
|
||||||
|
|
||||||
|
G_STATIC_ASSERT (sizeof (PixelShaderColorTransform) % 16 == 0);
|
||||||
|
|
||||||
const_buffer_desc.Usage = D3D11_USAGE_DYNAMIC;
|
const_buffer_desc.Usage = D3D11_USAGE_DYNAMIC;
|
||||||
const_buffer_desc.ByteWidth = sizeof (PixelShaderColorTransform);
|
const_buffer_desc.ByteWidth = sizeof (PixelShaderColorTransform);
|
||||||
const_buffer_desc.BindFlags = D3D11_BIND_CONSTANT_BUFFER;
|
const_buffer_desc.BindFlags = D3D11_BIND_CONSTANT_BUFFER;
|
||||||
|
@ -1132,14 +1189,15 @@ gst_d3d11_color_convert_setup_shader (GstD3D11Converter * self,
|
||||||
const_buffer_desc.MiscFlags = 0;
|
const_buffer_desc.MiscFlags = 0;
|
||||||
const_buffer_desc.StructureByteStride = 0;
|
const_buffer_desc.StructureByteStride = 0;
|
||||||
|
|
||||||
hr = device_handle->CreateBuffer (&const_buffer_desc, NULL, &const_buffer);
|
hr = device_handle->CreateBuffer (&const_buffer_desc, nullptr,
|
||||||
|
&transform_const_buffer);
|
||||||
if (!gst_d3d11_result (hr, device)) {
|
if (!gst_d3d11_result (hr, device)) {
|
||||||
GST_ERROR ("Couldn't create constant buffer, hr: 0x%x", (guint) hr);
|
GST_ERROR ("Couldn't create constant buffer, hr: 0x%x", (guint) hr);
|
||||||
return FALSE;
|
return FALSE;
|
||||||
}
|
}
|
||||||
|
|
||||||
gst_d3d11_device_lock (device);
|
gst_d3d11_device_lock (device);
|
||||||
hr = context_handle->Map (const_buffer.Get (),
|
hr = context_handle->Map (transform_const_buffer.Get (),
|
||||||
0, D3D11_MAP_WRITE_DISCARD, 0, &map);
|
0, D3D11_MAP_WRITE_DISCARD, 0, &map);
|
||||||
|
|
||||||
if (!gst_d3d11_result (hr, device)) {
|
if (!gst_d3d11_result (hr, device)) {
|
||||||
|
@ -1151,8 +1209,55 @@ gst_d3d11_color_convert_setup_shader (GstD3D11Converter * self,
|
||||||
memcpy (map.pData, &convert_info->transform,
|
memcpy (map.pData, &convert_info->transform,
|
||||||
sizeof (PixelShaderColorTransform));
|
sizeof (PixelShaderColorTransform));
|
||||||
|
|
||||||
context_handle->Unmap (const_buffer.Get (), 0);
|
context_handle->Unmap (transform_const_buffer.Get (), 0);
|
||||||
gst_d3d11_device_unlock (device);
|
gst_d3d11_device_unlock (device);
|
||||||
|
|
||||||
|
const_buffers[num_const_buffers] = transform_const_buffer.Get ();
|
||||||
|
num_const_buffers++;
|
||||||
|
}
|
||||||
|
|
||||||
|
if (convert_info->templ->has_alpha) {
|
||||||
|
D3D11_BUFFER_DESC const_buffer_desc = { 0, };
|
||||||
|
AlphaConstBuffer *alpha_const;
|
||||||
|
|
||||||
|
G_STATIC_ASSERT (sizeof (AlphaConstBuffer) % 16 == 0);
|
||||||
|
|
||||||
|
const_buffer_desc.Usage = D3D11_USAGE_DYNAMIC;
|
||||||
|
const_buffer_desc.ByteWidth = sizeof (AlphaConstBuffer);
|
||||||
|
const_buffer_desc.BindFlags = D3D11_BIND_CONSTANT_BUFFER;
|
||||||
|
const_buffer_desc.CPUAccessFlags = D3D11_CPU_ACCESS_WRITE;
|
||||||
|
const_buffer_desc.MiscFlags = 0;
|
||||||
|
const_buffer_desc.StructureByteStride = 0;
|
||||||
|
|
||||||
|
hr = device_handle->CreateBuffer (&const_buffer_desc,
|
||||||
|
nullptr, &alpha_const_buffer);
|
||||||
|
if (!gst_d3d11_result (hr, device)) {
|
||||||
|
GST_ERROR ("Couldn't create constant buffer, hr: 0x%x", (guint) hr);
|
||||||
|
return FALSE;
|
||||||
|
}
|
||||||
|
|
||||||
|
gst_d3d11_device_lock (device);
|
||||||
|
hr = context_handle->Map (alpha_const_buffer.Get (),
|
||||||
|
0, D3D11_MAP_WRITE_DISCARD, 0, &map);
|
||||||
|
|
||||||
|
if (!gst_d3d11_result (hr, device)) {
|
||||||
|
GST_ERROR ("Couldn't map constant buffer, hr: 0x%x", (guint) hr);
|
||||||
|
gst_d3d11_device_unlock (device);
|
||||||
|
return FALSE;
|
||||||
|
}
|
||||||
|
|
||||||
|
alpha_const = (AlphaConstBuffer *) map.pData;
|
||||||
|
memset (alpha_const, 0, sizeof (AlphaConstBuffer));
|
||||||
|
alpha_const->alpha_mul = (FLOAT) self->alpha;
|
||||||
|
|
||||||
|
context_handle->Unmap (alpha_const_buffer.Get (), 0);
|
||||||
|
gst_d3d11_device_unlock (device);
|
||||||
|
|
||||||
|
self->alpha_const_buffer = alpha_const_buffer.Get ();
|
||||||
|
/* We will hold this buffer and update later */
|
||||||
|
self->alpha_const_buffer->AddRef ();
|
||||||
|
const_buffers[num_const_buffers] = alpha_const_buffer.Get ();
|
||||||
|
num_const_buffers++;
|
||||||
}
|
}
|
||||||
|
|
||||||
input_desc[0].SemanticName = "POSITION";
|
input_desc[0].SemanticName = "POSITION";
|
||||||
|
@ -1265,13 +1370,15 @@ gst_d3d11_color_convert_setup_shader (GstD3D11Converter * self,
|
||||||
|
|
||||||
self->quad[0] = gst_d3d11_quad_new (device,
|
self->quad[0] = gst_d3d11_quad_new (device,
|
||||||
ps[0].Get (), vs.Get (), layout.Get (),
|
ps[0].Get (), vs.Get (), layout.Get (),
|
||||||
const_buffer.Get (), vertex_buffer.Get (), sizeof (VertexData),
|
const_buffers, num_const_buffers,
|
||||||
|
vertex_buffer.Get (), sizeof (VertexData),
|
||||||
index_buffer.Get (), DXGI_FORMAT_R16_UINT, index_count);
|
index_buffer.Get (), DXGI_FORMAT_R16_UINT, index_count);
|
||||||
|
|
||||||
if (ps[1]) {
|
if (ps[1]) {
|
||||||
self->quad[1] = gst_d3d11_quad_new (device,
|
self->quad[1] = gst_d3d11_quad_new (device,
|
||||||
ps[1].Get (), vs.Get (), layout.Get (),
|
ps[1].Get (), vs.Get (), layout.Get (),
|
||||||
const_buffer.Get (), vertex_buffer.Get (), sizeof (VertexData),
|
const_buffers, num_const_buffers,
|
||||||
|
vertex_buffer.Get (), sizeof (VertexData),
|
||||||
index_buffer.Get (), DXGI_FORMAT_R16_UINT, index_count);
|
index_buffer.Get (), DXGI_FORMAT_R16_UINT, index_count);
|
||||||
}
|
}
|
||||||
|
|
||||||
|
@ -1299,9 +1406,28 @@ gst_d3d11_color_convert_setup_shader (GstD3D11Converter * self,
|
||||||
return TRUE;
|
return TRUE;
|
||||||
}
|
}
|
||||||
|
|
||||||
static GstD3D11Converter *
|
static gboolean
|
||||||
gst_d3d11_converter_new_internal (GstD3D11Device * device,
|
copy_config (GQuark field_id, const GValue * value, GstD3D11Converter * self)
|
||||||
GstVideoInfo * in_info, GstVideoInfo * out_info, gfloat alpha)
|
{
|
||||||
|
gst_structure_id_set_value (self->config, field_id, value);
|
||||||
|
|
||||||
|
return TRUE;
|
||||||
|
}
|
||||||
|
|
||||||
|
static gboolean
|
||||||
|
gst_d3d11_converter_set_config (GstD3D11Converter * converter,
|
||||||
|
GstStructure * config)
|
||||||
|
{
|
||||||
|
gst_structure_foreach (config, (GstStructureForeachFunc) copy_config,
|
||||||
|
converter);
|
||||||
|
gst_structure_free (config);
|
||||||
|
|
||||||
|
return TRUE;
|
||||||
|
}
|
||||||
|
|
||||||
|
GstD3D11Converter *
|
||||||
|
gst_d3d11_converter_new (GstD3D11Device * device,
|
||||||
|
GstVideoInfo * in_info, GstVideoInfo * out_info, GstStructure * config)
|
||||||
{
|
{
|
||||||
const GstVideoInfo *unknown_info;
|
const GstVideoInfo *unknown_info;
|
||||||
const GstD3D11Format *in_d3d11_format;
|
const GstD3D11Format *in_d3d11_format;
|
||||||
|
@ -1338,7 +1464,11 @@ gst_d3d11_converter_new_internal (GstD3D11Device * device,
|
||||||
|
|
||||||
converter = g_new0 (GstD3D11Converter, 1);
|
converter = g_new0 (GstD3D11Converter, 1);
|
||||||
converter->device = (GstD3D11Device *) gst_object_ref (device);
|
converter->device = (GstD3D11Device *) gst_object_ref (device);
|
||||||
converter->alpha = alpha;
|
converter->config = gst_structure_new_empty ("GstD3D11Converter-Config");
|
||||||
|
if (config)
|
||||||
|
gst_d3d11_converter_set_config (converter, config);
|
||||||
|
|
||||||
|
converter->alpha = GET_OPT_ALPHA_VALUE (converter);
|
||||||
|
|
||||||
if (GST_VIDEO_INFO_IS_RGB (in_info)) {
|
if (GST_VIDEO_INFO_IS_RGB (in_info)) {
|
||||||
if (GST_VIDEO_INFO_IS_RGB (out_info)) {
|
if (GST_VIDEO_INFO_IS_RGB (out_info)) {
|
||||||
|
@ -1408,6 +1538,9 @@ format_unknown:
|
||||||
{
|
{
|
||||||
GST_ERROR ("%s couldn't be converted to d3d11 format",
|
GST_ERROR ("%s couldn't be converted to d3d11 format",
|
||||||
gst_video_format_to_string (GST_VIDEO_INFO_FORMAT (unknown_info)));
|
gst_video_format_to_string (GST_VIDEO_INFO_FORMAT (unknown_info)));
|
||||||
|
if (config)
|
||||||
|
gst_structure_free (config);
|
||||||
|
|
||||||
return NULL;
|
return NULL;
|
||||||
}
|
}
|
||||||
conversion_not_supported:
|
conversion_not_supported:
|
||||||
|
@ -1420,23 +1553,6 @@ conversion_not_supported:
|
||||||
}
|
}
|
||||||
}
|
}
|
||||||
|
|
||||||
GstD3D11Converter *
|
|
||||||
gst_d3d11_converter_new (GstD3D11Device * device,
|
|
||||||
GstVideoInfo * in_info, GstVideoInfo * out_info)
|
|
||||||
{
|
|
||||||
return gst_d3d11_converter_new_internal (device, in_info, out_info, 1.0f);
|
|
||||||
}
|
|
||||||
|
|
||||||
GstD3D11Converter *
|
|
||||||
gst_d3d11_converter_new_with_alpha (GstD3D11Device * device,
|
|
||||||
GstVideoInfo * in_info, GstVideoInfo * out_info, gfloat alpha)
|
|
||||||
{
|
|
||||||
g_return_val_if_fail (alpha >= 0.0f, NULL);
|
|
||||||
g_return_val_if_fail (alpha <= 1.0f, NULL);
|
|
||||||
|
|
||||||
return gst_d3d11_converter_new_internal (device, in_info, out_info, alpha);
|
|
||||||
}
|
|
||||||
|
|
||||||
void
|
void
|
||||||
gst_d3d11_converter_free (GstD3D11Converter * converter)
|
gst_d3d11_converter_free (GstD3D11Converter * converter)
|
||||||
{
|
{
|
||||||
|
@ -1453,8 +1569,13 @@ gst_d3d11_converter_free (GstD3D11Converter * converter)
|
||||||
|
|
||||||
GST_D3D11_CLEAR_COM (converter->vertex_buffer);
|
GST_D3D11_CLEAR_COM (converter->vertex_buffer);
|
||||||
GST_D3D11_CLEAR_COM (converter->linear_sampler);
|
GST_D3D11_CLEAR_COM (converter->linear_sampler);
|
||||||
|
GST_D3D11_CLEAR_COM (converter->alpha_const_buffer);
|
||||||
|
|
||||||
gst_clear_object (&converter->device);
|
gst_clear_object (&converter->device);
|
||||||
|
|
||||||
|
if (converter->config)
|
||||||
|
gst_structure_free (converter->config);
|
||||||
|
|
||||||
g_free (converter);
|
g_free (converter);
|
||||||
}
|
}
|
||||||
|
|
||||||
|
@ -1608,6 +1729,32 @@ gst_d3d11_converter_convert_unlocked (GstD3D11Converter * converter,
|
||||||
}
|
}
|
||||||
}
|
}
|
||||||
|
|
||||||
|
if (converter->update_alpha) {
|
||||||
|
D3D11_MAPPED_SUBRESOURCE map;
|
||||||
|
ID3D11DeviceContext *context_handle;
|
||||||
|
AlphaConstBuffer *alpha_const;
|
||||||
|
HRESULT hr;
|
||||||
|
|
||||||
|
g_assert (converter->alpha_const_buffer != nullptr);
|
||||||
|
|
||||||
|
context_handle =
|
||||||
|
gst_d3d11_device_get_device_context_handle (converter->device);
|
||||||
|
|
||||||
|
hr = context_handle->Map (converter->alpha_const_buffer,
|
||||||
|
0, D3D11_MAP_WRITE_DISCARD, 0, &map);
|
||||||
|
|
||||||
|
if (!gst_d3d11_result (hr, converter->device)) {
|
||||||
|
GST_ERROR ("Couldn't map constant buffer, hr: 0x%x", (guint) hr);
|
||||||
|
return FALSE;
|
||||||
|
}
|
||||||
|
|
||||||
|
alpha_const = (AlphaConstBuffer *) map.pData;
|
||||||
|
alpha_const->alpha_mul = (FLOAT) converter->alpha;
|
||||||
|
|
||||||
|
context_handle->Unmap (converter->alpha_const_buffer, 0);
|
||||||
|
converter->update_alpha = FALSE;
|
||||||
|
}
|
||||||
|
|
||||||
ret = gst_d3d11_draw_quad_unlocked (converter->quad[0], converter->viewport,
|
ret = gst_d3d11_draw_quad_unlocked (converter->quad[0], converter->viewport,
|
||||||
1, srv, converter->num_input_view, rtv, 1, blend, blend_factor,
|
1, srv, converter->num_input_view, rtv, 1, blend, blend_factor,
|
||||||
&converter->linear_sampler, 1);
|
&converter->linear_sampler, 1);
|
||||||
|
@ -1670,6 +1817,7 @@ gst_d3d11_converter_update_src_rect (GstD3D11Converter * converter,
|
||||||
g_return_val_if_fail (converter != NULL, FALSE);
|
g_return_val_if_fail (converter != NULL, FALSE);
|
||||||
g_return_val_if_fail (src_rect != NULL, FALSE);
|
g_return_val_if_fail (src_rect != NULL, FALSE);
|
||||||
|
|
||||||
|
gst_d3d11_device_lock (converter->device);
|
||||||
if (converter->src_rect.left != src_rect->left ||
|
if (converter->src_rect.left != src_rect->left ||
|
||||||
converter->src_rect.top != src_rect->top ||
|
converter->src_rect.top != src_rect->top ||
|
||||||
converter->src_rect.right != src_rect->right ||
|
converter->src_rect.right != src_rect->right ||
|
||||||
|
@ -1679,6 +1827,7 @@ gst_d3d11_converter_update_src_rect (GstD3D11Converter * converter,
|
||||||
/* vertex buffer will be updated on next convert() call */
|
/* vertex buffer will be updated on next convert() call */
|
||||||
converter->update_vertex = TRUE;
|
converter->update_vertex = TRUE;
|
||||||
}
|
}
|
||||||
|
gst_d3d11_device_unlock (converter->device);
|
||||||
|
|
||||||
return TRUE;
|
return TRUE;
|
||||||
}
|
}
|
||||||
|
@ -1690,6 +1839,7 @@ gst_d3d11_converter_update_dest_rect (GstD3D11Converter * converter,
|
||||||
g_return_val_if_fail (converter != NULL, FALSE);
|
g_return_val_if_fail (converter != NULL, FALSE);
|
||||||
g_return_val_if_fail (dest_rect != NULL, FALSE);
|
g_return_val_if_fail (dest_rect != NULL, FALSE);
|
||||||
|
|
||||||
|
gst_d3d11_device_lock (converter->device);
|
||||||
if (converter->dest_rect.left != dest_rect->left ||
|
if (converter->dest_rect.left != dest_rect->left ||
|
||||||
converter->dest_rect.top != dest_rect->top ||
|
converter->dest_rect.top != dest_rect->top ||
|
||||||
converter->dest_rect.right != dest_rect->right ||
|
converter->dest_rect.right != dest_rect->right ||
|
||||||
|
@ -1699,6 +1849,32 @@ gst_d3d11_converter_update_dest_rect (GstD3D11Converter * converter,
|
||||||
/* vertex buffer will be updated on next convert() call */
|
/* vertex buffer will be updated on next convert() call */
|
||||||
converter->update_vertex = TRUE;
|
converter->update_vertex = TRUE;
|
||||||
}
|
}
|
||||||
|
gst_d3d11_device_unlock (converter->device);
|
||||||
|
|
||||||
|
return TRUE;
|
||||||
|
}
|
||||||
|
|
||||||
|
gboolean
|
||||||
|
gst_d3d11_converter_update_config (GstD3D11Converter * converter,
|
||||||
|
GstStructure * config)
|
||||||
|
{
|
||||||
|
g_return_val_if_fail (converter != nullptr, FALSE);
|
||||||
|
g_return_val_if_fail (config != nullptr, FALSE);
|
||||||
|
|
||||||
|
gst_d3d11_device_lock (converter->device);
|
||||||
|
gst_d3d11_converter_set_config (converter, config);
|
||||||
|
|
||||||
|
/* Check whether options are updated or not */
|
||||||
|
if (converter->alpha_const_buffer) {
|
||||||
|
gdouble alpha = GET_OPT_ALPHA_VALUE (converter);
|
||||||
|
|
||||||
|
if (alpha != converter->alpha) {
|
||||||
|
GST_DEBUG ("Updating alpha %lf -> %lf", converter->alpha, alpha);
|
||||||
|
converter->alpha = alpha;
|
||||||
|
converter->update_alpha = TRUE;
|
||||||
|
}
|
||||||
|
}
|
||||||
|
gst_d3d11_device_unlock (converter->device);
|
||||||
|
|
||||||
return TRUE;
|
return TRUE;
|
||||||
}
|
}
|
||||||
|
|
|
@ -28,14 +28,18 @@ G_BEGIN_DECLS
|
||||||
|
|
||||||
typedef struct _GstD3D11Converter GstD3D11Converter;
|
typedef struct _GstD3D11Converter GstD3D11Converter;
|
||||||
|
|
||||||
|
/**
|
||||||
|
* GST_D3D11_CONVERTER_OPT_ALPHA_VALUE
|
||||||
|
*
|
||||||
|
* #G_TYPE_FLOAT, the alpha value color value to use.
|
||||||
|
* Default is 1.0
|
||||||
|
*/
|
||||||
|
#define GST_D3D11_CONVERTER_OPT_ALPHA_VALUE "GstD3D11Converter.alpha-value"
|
||||||
|
|
||||||
GstD3D11Converter * gst_d3d11_converter_new (GstD3D11Device * device,
|
GstD3D11Converter * gst_d3d11_converter_new (GstD3D11Device * device,
|
||||||
GstVideoInfo * in_info,
|
GstVideoInfo * in_info,
|
||||||
GstVideoInfo * out_info);
|
GstVideoInfo * out_info,
|
||||||
|
GstStructure * config);
|
||||||
GstD3D11Converter * gst_d3d11_converter_new_with_alpha (GstD3D11Device * device,
|
|
||||||
GstVideoInfo * in_info,
|
|
||||||
GstVideoInfo * out_info,
|
|
||||||
gfloat alpha);
|
|
||||||
|
|
||||||
void gst_d3d11_converter_free (GstD3D11Converter * converter);
|
void gst_d3d11_converter_free (GstD3D11Converter * converter);
|
||||||
|
|
||||||
|
@ -60,6 +64,9 @@ gboolean gst_d3d11_converter_update_src_rect (GstD3D11Converter * con
|
||||||
gboolean gst_d3d11_converter_update_dest_rect (GstD3D11Converter * converter,
|
gboolean gst_d3d11_converter_update_dest_rect (GstD3D11Converter * converter,
|
||||||
RECT * dest_rect);
|
RECT * dest_rect);
|
||||||
|
|
||||||
|
gboolean gst_d3d11_converter_update_config (GstD3D11Converter * converter,
|
||||||
|
GstStructure * config);
|
||||||
|
|
||||||
G_END_DECLS
|
G_END_DECLS
|
||||||
|
|
||||||
#endif /* __GST_D3D11_COLOR_CONVERTER_H__ */
|
#endif /* __GST_D3D11_COLOR_CONVERTER_H__ */
|
||||||
|
|
|
@ -285,7 +285,7 @@ gst_d3d11_composition_overlay_new (GstD3D11OverlayCompositor * self,
|
||||||
overlay->texture = texture.Detach ();
|
overlay->texture = texture.Detach ();
|
||||||
overlay->srv = srv.Detach ();
|
overlay->srv = srv.Detach ();
|
||||||
overlay->quad = gst_d3d11_quad_new (device,
|
overlay->quad = gst_d3d11_quad_new (device,
|
||||||
self->ps, self->vs, self->layout, NULL,
|
self->ps, self->vs, self->layout, nullptr, 0,
|
||||||
vertex_buffer.Get (), sizeof (VertexData),
|
vertex_buffer.Get (), sizeof (VertexData),
|
||||||
self->index_buffer, DXGI_FORMAT_R16_UINT, index_count);
|
self->index_buffer, DXGI_FORMAT_R16_UINT, index_count);
|
||||||
|
|
||||||
|
|
|
@ -37,6 +37,9 @@ GST_DEBUG_CATEGORY_EXTERN (gst_d3d11_shader_debug);
|
||||||
G_END_DECLS
|
G_END_DECLS
|
||||||
/* *INDENT-ON* */
|
/* *INDENT-ON* */
|
||||||
|
|
||||||
|
/* too many const buffers doesn't make sense */
|
||||||
|
#define MAX_CONST_BUFFERS 16
|
||||||
|
|
||||||
static GModule *d3d_compiler_module = NULL;
|
static GModule *d3d_compiler_module = NULL;
|
||||||
static pD3DCompile GstD3DCompileFunc = NULL;
|
static pD3DCompile GstD3DCompileFunc = NULL;
|
||||||
|
|
||||||
|
@ -237,7 +240,8 @@ struct _GstD3D11Quad
|
||||||
ID3D11PixelShader *ps;
|
ID3D11PixelShader *ps;
|
||||||
ID3D11VertexShader *vs;
|
ID3D11VertexShader *vs;
|
||||||
ID3D11InputLayout *layout;
|
ID3D11InputLayout *layout;
|
||||||
ID3D11Buffer *const_buffer;
|
ID3D11Buffer *const_buffer[MAX_CONST_BUFFERS];
|
||||||
|
guint num_const_buffers;
|
||||||
ID3D11Buffer *vertex_buffer;
|
ID3D11Buffer *vertex_buffer;
|
||||||
guint vertex_stride;
|
guint vertex_stride;
|
||||||
ID3D11Buffer *index_buffer;
|
ID3D11Buffer *index_buffer;
|
||||||
|
@ -253,7 +257,7 @@ struct _GstD3D11Quad
|
||||||
GstD3D11Quad *
|
GstD3D11Quad *
|
||||||
gst_d3d11_quad_new (GstD3D11Device * device, ID3D11PixelShader * pixel_shader,
|
gst_d3d11_quad_new (GstD3D11Device * device, ID3D11PixelShader * pixel_shader,
|
||||||
ID3D11VertexShader * vertex_shader, ID3D11InputLayout * layout,
|
ID3D11VertexShader * vertex_shader, ID3D11InputLayout * layout,
|
||||||
ID3D11Buffer * const_buffer,
|
ID3D11Buffer ** const_buffers, guint num_const_buffers,
|
||||||
ID3D11Buffer * vertex_buffer, guint vertex_stride,
|
ID3D11Buffer * vertex_buffer, guint vertex_stride,
|
||||||
ID3D11Buffer * index_buffer, DXGI_FORMAT index_format, guint index_count)
|
ID3D11Buffer * index_buffer, DXGI_FORMAT index_format, guint index_count)
|
||||||
{
|
{
|
||||||
|
@ -263,6 +267,7 @@ gst_d3d11_quad_new (GstD3D11Device * device, ID3D11PixelShader * pixel_shader,
|
||||||
g_return_val_if_fail (pixel_shader != NULL, NULL);
|
g_return_val_if_fail (pixel_shader != NULL, NULL);
|
||||||
g_return_val_if_fail (vertex_shader != NULL, NULL);
|
g_return_val_if_fail (vertex_shader != NULL, NULL);
|
||||||
g_return_val_if_fail (layout != NULL, NULL);
|
g_return_val_if_fail (layout != NULL, NULL);
|
||||||
|
g_return_val_if_fail (num_const_buffers <= MAX_CONST_BUFFERS, NULL);
|
||||||
g_return_val_if_fail (vertex_buffer != NULL, NULL);
|
g_return_val_if_fail (vertex_buffer != NULL, NULL);
|
||||||
g_return_val_if_fail (vertex_stride > 0, NULL);
|
g_return_val_if_fail (vertex_stride > 0, NULL);
|
||||||
g_return_val_if_fail (index_buffer != NULL, NULL);
|
g_return_val_if_fail (index_buffer != NULL, NULL);
|
||||||
|
@ -286,9 +291,17 @@ gst_d3d11_quad_new (GstD3D11Device * device, ID3D11PixelShader * pixel_shader,
|
||||||
vertex_buffer->AddRef ();
|
vertex_buffer->AddRef ();
|
||||||
index_buffer->AddRef ();
|
index_buffer->AddRef ();
|
||||||
|
|
||||||
if (const_buffer) {
|
if (num_const_buffers > 0) {
|
||||||
quad->const_buffer = const_buffer;
|
guint i;
|
||||||
const_buffer->AddRef ();
|
|
||||||
|
g_assert (const_buffers);
|
||||||
|
|
||||||
|
for (i = 0; i < num_const_buffers; i++) {
|
||||||
|
quad->const_buffer[i] = const_buffers[i];
|
||||||
|
quad->const_buffer[i]->AddRef ();
|
||||||
|
}
|
||||||
|
|
||||||
|
quad->num_const_buffers = num_const_buffers;
|
||||||
}
|
}
|
||||||
|
|
||||||
return quad;
|
return quad;
|
||||||
|
@ -297,12 +310,15 @@ gst_d3d11_quad_new (GstD3D11Device * device, ID3D11PixelShader * pixel_shader,
|
||||||
void
|
void
|
||||||
gst_d3d11_quad_free (GstD3D11Quad * quad)
|
gst_d3d11_quad_free (GstD3D11Quad * quad)
|
||||||
{
|
{
|
||||||
|
guint i;
|
||||||
|
|
||||||
g_return_if_fail (quad != NULL);
|
g_return_if_fail (quad != NULL);
|
||||||
|
|
||||||
GST_D3D11_CLEAR_COM (quad->ps);
|
GST_D3D11_CLEAR_COM (quad->ps);
|
||||||
GST_D3D11_CLEAR_COM (quad->vs);
|
GST_D3D11_CLEAR_COM (quad->vs);
|
||||||
GST_D3D11_CLEAR_COM (quad->layout);
|
GST_D3D11_CLEAR_COM (quad->layout);
|
||||||
GST_D3D11_CLEAR_COM (quad->const_buffer);
|
for (i = 0; i < quad->num_const_buffers; i++)
|
||||||
|
quad->const_buffer[i]->Release ();
|
||||||
GST_D3D11_CLEAR_COM (quad->vertex_buffer);
|
GST_D3D11_CLEAR_COM (quad->vertex_buffer);
|
||||||
GST_D3D11_CLEAR_COM (quad->index_buffer);
|
GST_D3D11_CLEAR_COM (quad->index_buffer);
|
||||||
|
|
||||||
|
@ -364,8 +380,10 @@ gst_d3d11_draw_quad_unlocked (GstD3D11Quad * quad,
|
||||||
context->PSSetShader (quad->ps, NULL, 0);
|
context->PSSetShader (quad->ps, NULL, 0);
|
||||||
context->RSSetViewports (num_viewport, viewport);
|
context->RSSetViewports (num_viewport, viewport);
|
||||||
|
|
||||||
if (quad->const_buffer)
|
if (quad->num_const_buffers) {
|
||||||
context->PSSetConstantBuffers (0, 1, &quad->const_buffer);
|
context->PSSetConstantBuffers (0, quad->num_const_buffers,
|
||||||
|
quad->const_buffer);
|
||||||
|
}
|
||||||
|
|
||||||
if (srv)
|
if (srv)
|
||||||
context->PSSetShaderResources (0, num_srv, srv);
|
context->PSSetShaderResources (0, num_srv, srv);
|
||||||
|
|
|
@ -47,7 +47,8 @@ GstD3D11Quad * gst_d3d11_quad_new (GstD3D11Device * device,
|
||||||
ID3D11PixelShader * pixel_shader,
|
ID3D11PixelShader * pixel_shader,
|
||||||
ID3D11VertexShader * vertex_shader,
|
ID3D11VertexShader * vertex_shader,
|
||||||
ID3D11InputLayout * layout,
|
ID3D11InputLayout * layout,
|
||||||
ID3D11Buffer * const_buffer,
|
ID3D11Buffer ** const_buffers,
|
||||||
|
guint num_const_buffers,
|
||||||
ID3D11Buffer * vertex_buffer,
|
ID3D11Buffer * vertex_buffer,
|
||||||
guint vertex_stride,
|
guint vertex_stride,
|
||||||
ID3D11Buffer * index_buffer,
|
ID3D11Buffer * index_buffer,
|
||||||
|
|
|
@ -734,7 +734,7 @@ gst_d3d11_window_prepare_default (GstD3D11Window * window, guint display_width,
|
||||||
/* configure shader even if video processor is available for fallback */
|
/* configure shader even if video processor is available for fallback */
|
||||||
window->converter =
|
window->converter =
|
||||||
gst_d3d11_converter_new (window->device, &window->info,
|
gst_d3d11_converter_new (window->device, &window->info,
|
||||||
&window->render_info);
|
&window->render_info, nullptr);
|
||||||
|
|
||||||
if (!window->converter) {
|
if (!window->converter) {
|
||||||
GST_ERROR_OBJECT (window, "Cannot create converter");
|
GST_ERROR_OBJECT (window, "Cannot create converter");
|
||||||
|
|
|
@ -176,7 +176,7 @@ gst_d3d11_window_dummy_prepare (GstD3D11Window * window,
|
||||||
|
|
||||||
window->converter =
|
window->converter =
|
||||||
gst_d3d11_converter_new (window->device, &window->info,
|
gst_d3d11_converter_new (window->device, &window->info,
|
||||||
&window->render_info);
|
&window->render_info, nullptr);
|
||||||
|
|
||||||
if (!window->converter) {
|
if (!window->converter) {
|
||||||
GST_ERROR_OBJECT (window, "Cannot create converter");
|
GST_ERROR_OBJECT (window, "Cannot create converter");
|
||||||
|
|
Loading…
Reference in a new issue