mirror of
https://gitlab.freedesktop.org/gstreamer/gstreamer.git
synced 2025-01-11 01:45:33 +00:00
d3d11converter: Fix YUY2 conversion error
Always configure shader conversion path, then fallback to the shader path if processor is not available Part-of: <https://gitlab.freedesktop.org/gstreamer/gstreamer/-/merge_requests/3392>
This commit is contained in:
parent
d0572622fa
commit
837881ad29
1 changed files with 355 additions and 207 deletions
|
@ -679,9 +679,14 @@ struct _GstD3D11ConverterPrivate
|
||||||
gchar *in_cll_str;
|
gchar *in_cll_str;
|
||||||
gchar *out_cll_str;
|
gchar *out_cll_str;
|
||||||
|
|
||||||
|
/* Fallback buffer and info, for shader */
|
||||||
GstVideoInfo fallback_info;
|
GstVideoInfo fallback_info;
|
||||||
GstBuffer *fallback_inbuf;
|
GstBuffer *fallback_inbuf;
|
||||||
|
|
||||||
|
/* Fallback buffer used for processor */
|
||||||
|
GstVideoInfo piv_info;
|
||||||
|
GstBuffer *piv_inbuf;
|
||||||
|
|
||||||
GstVideoOrientationMethod video_direction;
|
GstVideoOrientationMethod video_direction;
|
||||||
|
|
||||||
SRWLOCK prop_lock;
|
SRWLOCK prop_lock;
|
||||||
|
@ -849,6 +854,7 @@ gst_d3d11_converter_dispose (GObject * object)
|
||||||
GST_D3D11_CLEAR_COM (priv->ps[i]);
|
GST_D3D11_CLEAR_COM (priv->ps[i]);
|
||||||
|
|
||||||
gst_clear_buffer (&priv->fallback_inbuf);
|
gst_clear_buffer (&priv->fallback_inbuf);
|
||||||
|
gst_clear_buffer (&priv->piv_inbuf);
|
||||||
gst_clear_object (&self->device);
|
gst_clear_object (&self->device);
|
||||||
|
|
||||||
G_OBJECT_CLASS (parent_class)->dispose (object);
|
G_OBJECT_CLASS (parent_class)->dispose (object);
|
||||||
|
@ -1641,6 +1647,9 @@ gst_d3d11_converter_update_src_rect (GstD3D11Converter * self)
|
||||||
gint texture_width = priv->input_texture_width;
|
gint texture_width = priv->input_texture_width;
|
||||||
gint texture_height = priv->input_texture_height;
|
gint texture_height = priv->input_texture_height;
|
||||||
|
|
||||||
|
if (!priv->update_src_rect)
|
||||||
|
return TRUE;
|
||||||
|
|
||||||
priv->update_src_rect = FALSE;
|
priv->update_src_rect = FALSE;
|
||||||
|
|
||||||
priv->src_rect.left = priv->src_x;
|
priv->src_rect.left = priv->src_x;
|
||||||
|
@ -1788,6 +1797,9 @@ gst_d3d11_converter_update_dest_rect (GstD3D11Converter * self)
|
||||||
GstD3D11ConverterPrivate *priv = self->priv;
|
GstD3D11ConverterPrivate *priv = self->priv;
|
||||||
const GstVideoInfo *out_info = &priv->out_info;
|
const GstVideoInfo *out_info = &priv->out_info;
|
||||||
|
|
||||||
|
if (!priv->update_dest_rect)
|
||||||
|
return TRUE;
|
||||||
|
|
||||||
priv->viewport[0].TopLeftX = priv->dest_x;
|
priv->viewport[0].TopLeftX = priv->dest_x;
|
||||||
priv->viewport[0].TopLeftY = priv->dest_y;
|
priv->viewport[0].TopLeftY = priv->dest_y;
|
||||||
priv->viewport[0].Width = priv->dest_width;
|
priv->viewport[0].Width = priv->dest_width;
|
||||||
|
@ -3164,6 +3176,7 @@ gst_d3d11_converter_new (GstD3D11Device * device, const GstVideoInfo * in_info,
|
||||||
priv->const_data.alpha = 1.0;
|
priv->const_data.alpha = 1.0;
|
||||||
priv->in_info = *in_info;
|
priv->in_info = *in_info;
|
||||||
priv->fallback_info = *in_info;
|
priv->fallback_info = *in_info;
|
||||||
|
priv->piv_info = *in_info;
|
||||||
priv->out_info = *out_info;
|
priv->out_info = *out_info;
|
||||||
priv->in_d3d11_format = in_d3d11_format;
|
priv->in_d3d11_format = in_d3d11_format;
|
||||||
priv->out_d3d11_format = out_d3d11_format;
|
priv->out_d3d11_format = out_d3d11_format;
|
||||||
|
@ -3209,12 +3222,6 @@ gst_d3d11_converter_new (GstD3D11Device * device, const GstVideoInfo * in_info,
|
||||||
GST_DEBUG_OBJECT (self, "Video processor is available");
|
GST_DEBUG_OBJECT (self, "Video processor is available");
|
||||||
priv->supported_backend |= GST_D3D11_CONVERTER_BACKEND_VIDEO_PROCESSOR;
|
priv->supported_backend |= GST_D3D11_CONVERTER_BACKEND_VIDEO_PROCESSOR;
|
||||||
}
|
}
|
||||||
|
|
||||||
/* Always use converter */
|
|
||||||
if (GST_VIDEO_INFO_FORMAT (in_info) == GST_VIDEO_FORMAT_YUY2) {
|
|
||||||
GST_DEBUG_OBJECT (self, "Use video processor only");
|
|
||||||
goto out;
|
|
||||||
}
|
|
||||||
}
|
}
|
||||||
|
|
||||||
if ((wanted_backend & GST_D3D11_CONVERTER_BACKEND_SHADER) == 0)
|
if ((wanted_backend & GST_D3D11_CONVERTER_BACKEND_SHADER) == 0)
|
||||||
|
@ -3275,6 +3282,11 @@ gst_d3d11_converter_new (GstD3D11Device * device, const GstVideoInfo * in_info,
|
||||||
|
|
||||||
priv->unpack_convert =
|
priv->unpack_convert =
|
||||||
gst_video_converter_new (in_info, &tmp_info, nullptr);
|
gst_video_converter_new (in_info, &tmp_info, nullptr);
|
||||||
|
if (!priv->unpack_convert) {
|
||||||
|
GST_ERROR_OBJECT (self, "Couldn't create unpack convert");
|
||||||
|
priv->supported_backend = (GstD3D11ConverterBackend) 0;
|
||||||
|
goto out;
|
||||||
|
}
|
||||||
|
|
||||||
priv->fallback_info = tmp_info;
|
priv->fallback_info = tmp_info;
|
||||||
in_info = &priv->fallback_info;
|
in_info = &priv->fallback_info;
|
||||||
|
@ -3335,19 +3347,14 @@ gst_d3d11_converter_convert_internal (GstD3D11Converter * self,
|
||||||
resource.As (&texture);
|
resource.As (&texture);
|
||||||
texture->GetDesc (&desc);
|
texture->GetDesc (&desc);
|
||||||
|
|
||||||
if (priv->update_dest_rect && !gst_d3d11_converter_update_dest_rect (self)) {
|
if (desc.Width != (guint) priv->input_texture_width ||
|
||||||
GST_ERROR_OBJECT (self, "Failed to update dest rect");
|
|
||||||
return FALSE;
|
|
||||||
}
|
|
||||||
|
|
||||||
if (priv->update_src_rect ||
|
|
||||||
desc.Width != (guint) priv->input_texture_width ||
|
|
||||||
desc.Height != (guint) priv->input_texture_height) {
|
desc.Height != (guint) priv->input_texture_height) {
|
||||||
GST_DEBUG_OBJECT (self, "Update vertext buffer, texture resolution: %dx%d",
|
GST_DEBUG_OBJECT (self, "Update vertext buffer, texture resolution: %dx%d",
|
||||||
desc.Width, desc.Height);
|
desc.Width, desc.Height);
|
||||||
|
|
||||||
priv->input_texture_width = desc.Width;
|
priv->input_texture_width = desc.Width;
|
||||||
priv->input_texture_height = desc.Height;
|
priv->input_texture_height = desc.Height;
|
||||||
|
priv->update_src_rect = TRUE;
|
||||||
|
|
||||||
if (!gst_d3d11_converter_update_src_rect (self)) {
|
if (!gst_d3d11_converter_update_src_rect (self)) {
|
||||||
GST_ERROR_OBJECT (self, "Cannot update src rect");
|
GST_ERROR_OBJECT (self, "Cannot update src rect");
|
||||||
|
@ -3441,7 +3448,7 @@ gst_d3d11_converter_check_bind_flags_for_piv (guint bind_flags)
|
||||||
}
|
}
|
||||||
|
|
||||||
static gboolean
|
static gboolean
|
||||||
gst_d3d11_converter_ensure_d3d11_buffer (GstD3D11Converter * self,
|
gst_d3d11_converter_is_d3d11_buffer (GstD3D11Converter * self,
|
||||||
GstBuffer * buffer)
|
GstBuffer * buffer)
|
||||||
{
|
{
|
||||||
if (gst_buffer_n_memory (buffer) == 0) {
|
if (gst_buffer_n_memory (buffer) == 0) {
|
||||||
|
@ -3480,9 +3487,6 @@ gst_d3d11_converter_create_fallback_buffer (GstD3D11Converter * self)
|
||||||
|
|
||||||
gst_clear_buffer (&priv->fallback_inbuf);
|
gst_clear_buffer (&priv->fallback_inbuf);
|
||||||
|
|
||||||
if (priv->processor)
|
|
||||||
bind_flags |= D3D11_BIND_RENDER_TARGET;
|
|
||||||
|
|
||||||
params = gst_d3d11_allocation_params_new (self->device, &priv->fallback_info,
|
params = gst_d3d11_allocation_params_new (self->device, &priv->fallback_info,
|
||||||
GST_D3D11_ALLOCATION_FLAG_DEFAULT, bind_flags, 0);
|
GST_D3D11_ALLOCATION_FLAG_DEFAULT, bind_flags, 0);
|
||||||
|
|
||||||
|
@ -3521,7 +3525,8 @@ gst_d3d11_converter_create_fallback_buffer (GstD3D11Converter * self)
|
||||||
}
|
}
|
||||||
|
|
||||||
static gboolean
|
static gboolean
|
||||||
gst_d3d11_converter_copy_buffer (GstD3D11Converter * self, GstBuffer * in_buf)
|
gst_d3d11_converter_upload_for_shader (GstD3D11Converter * self,
|
||||||
|
GstBuffer * in_buf)
|
||||||
{
|
{
|
||||||
GstD3D11ConverterPrivate *priv = self->priv;
|
GstD3D11ConverterPrivate *priv = self->priv;
|
||||||
GstVideoFrame frame, fallback_frame;
|
GstVideoFrame frame, fallback_frame;
|
||||||
|
@ -3829,182 +3834,202 @@ gst_d3d11_converter_update_hdr10_meta (GstD3D11Converter * self)
|
||||||
}
|
}
|
||||||
|
|
||||||
static gboolean
|
static gboolean
|
||||||
gst_d3d11_converter_convert_buffer_internal (GstD3D11Converter * self,
|
gst_d3d11_converter_need_blend (GstD3D11Converter * self)
|
||||||
GstBuffer * in_buf, GstBuffer * out_buf)
|
|
||||||
{
|
{
|
||||||
GstD3D11ConverterPrivate *priv = self->priv;
|
GstD3D11ConverterPrivate *priv = self->priv;
|
||||||
ID3D11ShaderResourceView *srv[GST_VIDEO_MAX_PLANES] = { nullptr, };
|
|
||||||
ID3D11RenderTargetView *rtv[GST_VIDEO_MAX_PLANES] = { nullptr, };
|
|
||||||
ID3D11VideoProcessorInputView *piv = nullptr;
|
|
||||||
ID3D11VideoProcessorOutputView *pov = nullptr;
|
|
||||||
gboolean use_processor = FALSE;
|
|
||||||
GstD3D11Memory *in_dmem;
|
|
||||||
GstD3D11Memory *out_dmem;
|
|
||||||
GstMapInfo in_info[GST_VIDEO_MAX_PLANES];
|
|
||||||
GstMapInfo out_info[GST_VIDEO_MAX_PLANES];
|
|
||||||
D3D11_TEXTURE2D_DESC in_desc, out_desc;
|
|
||||||
guint num_srv, num_rtv;
|
|
||||||
gboolean ret = FALSE;
|
|
||||||
gboolean need_blend = FALSE;
|
|
||||||
|
|
||||||
/* Output buffer must be valid D3D11 buffer */
|
|
||||||
if (!gst_d3d11_converter_ensure_d3d11_buffer (self, out_buf)) {
|
|
||||||
GST_ERROR_OBJECT (self, "Output is not d3d11 buffer");
|
|
||||||
return FALSE;
|
|
||||||
}
|
|
||||||
|
|
||||||
if (gst_buffer_n_memory (in_buf) == 0) {
|
|
||||||
GST_ERROR_OBJECT (self, "Empty buffer");
|
|
||||||
return FALSE;
|
|
||||||
}
|
|
||||||
|
|
||||||
/* But allows input copying */
|
|
||||||
if (!gst_d3d11_converter_ensure_d3d11_buffer (self, in_buf) ||
|
|
||||||
(GST_VIDEO_INFO_FORMAT (&priv->in_info) == GST_VIDEO_FORMAT_YUY2 &&
|
|
||||||
priv->unpack_convert)) {
|
|
||||||
if (!gst_d3d11_converter_copy_buffer (self, in_buf)) {
|
|
||||||
GST_ERROR_OBJECT (self, "Couldn't copy into fallback buffer");
|
|
||||||
return FALSE;
|
|
||||||
}
|
|
||||||
|
|
||||||
in_buf = priv->fallback_inbuf;
|
|
||||||
}
|
|
||||||
|
|
||||||
in_dmem = (GstD3D11Memory *) gst_buffer_peek_memory (in_buf, 0);
|
|
||||||
out_dmem = (GstD3D11Memory *) gst_buffer_peek_memory (out_buf, 0);
|
|
||||||
|
|
||||||
if (!gst_d3d11_memory_get_texture_desc (in_dmem, &in_desc)) {
|
|
||||||
GST_ERROR_OBJECT (self, "Failed to get input desc");
|
|
||||||
return FALSE;
|
|
||||||
}
|
|
||||||
|
|
||||||
if (!gst_d3d11_memory_get_texture_desc (out_dmem, &out_desc)) {
|
|
||||||
GST_ERROR_OBJECT (self, "Failed to get output desc");
|
|
||||||
return FALSE;
|
|
||||||
}
|
|
||||||
|
|
||||||
if ((out_desc.BindFlags & D3D11_BIND_RENDER_TARGET) == 0) {
|
|
||||||
GST_ERROR_OBJECT (self, "Output is not bound to render target");
|
|
||||||
return FALSE;
|
|
||||||
}
|
|
||||||
|
|
||||||
if (!gst_d3d11_converter_map_buffer (self, in_buf, in_info,
|
|
||||||
(GstMapFlags) (GST_MAP_READ | GST_MAP_D3D11))) {
|
|
||||||
GST_ERROR_OBJECT (self, "Couldn't map input buffer");
|
|
||||||
return FALSE;
|
|
||||||
}
|
|
||||||
|
|
||||||
if (!gst_d3d11_converter_map_buffer (self, out_buf, out_info,
|
|
||||||
(GstMapFlags) (GST_MAP_WRITE | GST_MAP_D3D11))) {
|
|
||||||
GST_ERROR_OBJECT (self, "Couldn't map output buffer");
|
|
||||||
gst_d3d11_converter_unmap_buffer (self, in_buf, in_info);
|
|
||||||
return FALSE;
|
|
||||||
}
|
|
||||||
|
|
||||||
GstD3D11SRWLockGuard (&priv->prop_lock);
|
|
||||||
gst_d3d11_converter_update_hdr10_meta (self);
|
|
||||||
|
|
||||||
if (priv->blend && priv->blend_desc.RenderTarget[0].BlendEnable) {
|
if (priv->blend && priv->blend_desc.RenderTarget[0].BlendEnable) {
|
||||||
if (priv->alpha != 1.0) {
|
if (priv->alpha != 1.0) {
|
||||||
need_blend = TRUE;
|
return TRUE;
|
||||||
} else if ((priv->blend_desc.RenderTarget[0].SrcBlend ==
|
} else if ((priv->blend_desc.RenderTarget[0].SrcBlend ==
|
||||||
D3D11_BLEND_BLEND_FACTOR
|
D3D11_BLEND_BLEND_FACTOR
|
||||||
|| priv->blend_desc.RenderTarget[0].SrcBlend ==
|
|| priv->blend_desc.RenderTarget[0].SrcBlend ==
|
||||||
D3D11_BLEND_INV_BLEND_FACTOR)
|
D3D11_BLEND_INV_BLEND_FACTOR)
|
||||||
&& (priv->blend_factor[0] != 1.0 || priv->blend_factor[1] != 1.0
|
&& (priv->blend_factor[0] != 1.0 || priv->blend_factor[1] != 1.0
|
||||||
|| priv->blend_factor[2] != 1.0 || priv->blend_factor[3] != 1.0)) {
|
|| priv->blend_factor[2] != 1.0 || priv->blend_factor[3] != 1.0)) {
|
||||||
need_blend = TRUE;
|
return TRUE;
|
||||||
}
|
}
|
||||||
}
|
}
|
||||||
|
|
||||||
if (priv->supported_backend == GST_D3D11_CONVERTER_BACKEND_VIDEO_PROCESSOR) {
|
return FALSE;
|
||||||
piv =
|
}
|
||||||
gst_d3d11_memory_get_processor_input_view (in_dmem,
|
|
||||||
priv->video_device, priv->enumerator);
|
|
||||||
pov =
|
|
||||||
gst_d3d11_memory_get_processor_output_view (out_dmem,
|
|
||||||
priv->video_device, priv->enumerator);
|
|
||||||
|
|
||||||
use_processor = TRUE;
|
static gboolean
|
||||||
} else if ((priv->supported_backend &
|
gst_d3d11_converter_processor_available (GstD3D11Converter * self)
|
||||||
GST_D3D11_CONVERTER_BACKEND_VIDEO_PROCESSOR) != 0 && !need_blend
|
{
|
||||||
&& gst_d3d11_converter_check_bind_flags_for_piv (in_desc.BindFlags)) {
|
GstD3D11ConverterPrivate *priv = self->priv;
|
||||||
/* TODO: processor supports alpha blending */
|
|
||||||
|
|
||||||
/* Try processor when:
|
if ((priv->supported_backend &
|
||||||
* - We were using processor already
|
GST_D3D11_CONVERTER_BACKEND_VIDEO_PROCESSOR) == 0)
|
||||||
* - or SRV is unavailable (likely from decoder output)
|
return FALSE;
|
||||||
* - or HDR10, as we don't support tone-mapping via shader yet
|
|
||||||
*/
|
/* TODO: processor may be able to blend textures */
|
||||||
if (priv->processor_in_use ||
|
if (gst_d3d11_converter_need_blend (self))
|
||||||
(in_desc.BindFlags & D3D11_BIND_SHADER_RESOURCE) == 0 ||
|
return FALSE;
|
||||||
(priv->video_context2 && (priv->have_in_hdr10
|
|
||||||
|| priv->have_out_hdr10))) {
|
/* flip/rotate is not supported by processor */
|
||||||
piv =
|
if (priv->processor_direction_not_supported)
|
||||||
gst_d3d11_memory_get_processor_input_view (in_dmem,
|
return FALSE;
|
||||||
priv->video_device, priv->enumerator);
|
|
||||||
pov =
|
return TRUE;
|
||||||
gst_d3d11_memory_get_processor_output_view (out_dmem,
|
}
|
||||||
priv->video_device, priv->enumerator);
|
|
||||||
|
static gboolean
|
||||||
|
gst_d3d11_converter_piv_available (GstD3D11Converter * self, GstBuffer * in_buf)
|
||||||
|
{
|
||||||
|
GstD3D11Memory *mem;
|
||||||
|
D3D11_TEXTURE2D_DESC desc;
|
||||||
|
|
||||||
|
mem = (GstD3D11Memory *) gst_buffer_peek_memory (in_buf, 0);
|
||||||
|
gst_d3d11_memory_get_texture_desc (mem, &desc);
|
||||||
|
return gst_d3d11_converter_check_bind_flags_for_piv (desc.BindFlags);
|
||||||
|
}
|
||||||
|
|
||||||
|
static gboolean
|
||||||
|
gst_d3d11_converter_create_piv_buffer (GstD3D11Converter * self)
|
||||||
|
{
|
||||||
|
GstD3D11ConverterPrivate *priv = self->priv;
|
||||||
|
GstD3D11AllocationParams *params;
|
||||||
|
GstBufferPool *pool;
|
||||||
|
GstCaps *caps;
|
||||||
|
GstStructure *config;
|
||||||
|
|
||||||
|
gst_clear_buffer (&priv->piv_inbuf);
|
||||||
|
|
||||||
|
params = gst_d3d11_allocation_params_new (self->device, &priv->piv_info,
|
||||||
|
GST_D3D11_ALLOCATION_FLAG_DEFAULT, 0, 0);
|
||||||
|
|
||||||
|
caps = gst_video_info_to_caps (&priv->piv_info);
|
||||||
|
pool = gst_d3d11_buffer_pool_new (self->device);
|
||||||
|
|
||||||
|
config = gst_buffer_pool_get_config (pool);
|
||||||
|
gst_buffer_pool_config_set_params (config, caps, priv->piv_info.size, 0, 0);
|
||||||
|
gst_buffer_pool_config_set_d3d11_allocation_params (config, params);
|
||||||
|
gst_caps_unref (caps);
|
||||||
|
gst_d3d11_allocation_params_free (params);
|
||||||
|
|
||||||
|
if (!gst_buffer_pool_set_config (pool, config)) {
|
||||||
|
GST_ERROR_OBJECT (self, "Failed to set pool config");
|
||||||
|
gst_object_unref (pool);
|
||||||
|
return FALSE;
|
||||||
}
|
}
|
||||||
|
|
||||||
if (piv != nullptr && pov != nullptr)
|
if (!gst_buffer_pool_set_active (pool, TRUE)) {
|
||||||
use_processor = TRUE;
|
GST_ERROR_OBJECT (self, "Failed to set active");
|
||||||
|
gst_object_unref (pool);
|
||||||
|
return FALSE;
|
||||||
}
|
}
|
||||||
|
|
||||||
if (use_processor) {
|
gst_buffer_pool_acquire_buffer (pool, &priv->piv_inbuf, nullptr);
|
||||||
|
gst_buffer_pool_set_active (pool, FALSE);
|
||||||
|
gst_object_unref (pool);
|
||||||
|
|
||||||
|
if (!priv->piv_inbuf) {
|
||||||
|
GST_ERROR_OBJECT (self, "Failed to create PIV buffer");
|
||||||
|
return FALSE;
|
||||||
|
}
|
||||||
|
|
||||||
|
return TRUE;
|
||||||
|
}
|
||||||
|
|
||||||
|
static gboolean
|
||||||
|
gst_d3d11_converter_upload_for_processor (GstD3D11Converter * self,
|
||||||
|
GstBuffer * in_buf)
|
||||||
|
{
|
||||||
|
GstD3D11ConverterPrivate *priv = self->priv;
|
||||||
|
GstVideoFrame frame, fallback_frame;
|
||||||
|
GstVideoInfo *piv_info = &priv->piv_info;
|
||||||
|
gboolean ret = TRUE;
|
||||||
|
|
||||||
|
if (!gst_video_frame_map (&frame, &priv->in_info, in_buf, GST_MAP_READ)) {
|
||||||
|
GST_ERROR_OBJECT (self, "Failed to map input buffer");
|
||||||
|
return FALSE;
|
||||||
|
}
|
||||||
|
|
||||||
|
/* Probably cropped buffer */
|
||||||
|
if (piv_info->width != GST_VIDEO_FRAME_WIDTH (&frame) ||
|
||||||
|
piv_info->height != GST_VIDEO_FRAME_HEIGHT (&frame)) {
|
||||||
|
gst_clear_buffer (&priv->piv_inbuf);
|
||||||
|
|
||||||
|
*piv_info = frame.info;
|
||||||
|
}
|
||||||
|
|
||||||
|
if (!priv->piv_inbuf && !gst_d3d11_converter_create_piv_buffer (self)) {
|
||||||
|
goto error;
|
||||||
|
}
|
||||||
|
|
||||||
|
if (!gst_video_frame_map (&fallback_frame,
|
||||||
|
&priv->piv_info, priv->piv_inbuf, GST_MAP_WRITE)) {
|
||||||
|
GST_ERROR_OBJECT (self, "Couldn't map fallback buffer");
|
||||||
|
goto error;
|
||||||
|
}
|
||||||
|
|
||||||
|
ret = gst_video_frame_copy (&fallback_frame, &frame);
|
||||||
|
gst_video_frame_unmap (&fallback_frame);
|
||||||
|
gst_video_frame_unmap (&frame);
|
||||||
|
|
||||||
|
return ret;
|
||||||
|
|
||||||
|
error:
|
||||||
|
gst_video_frame_unmap (&frame);
|
||||||
|
return FALSE;
|
||||||
|
}
|
||||||
|
|
||||||
|
static gboolean
|
||||||
|
gst_d3d11_converter_do_processor_blt (GstD3D11Converter * self,
|
||||||
|
GstBuffer * in_buf, GstBuffer * out_buf)
|
||||||
|
{
|
||||||
|
GstD3D11ConverterPrivate *priv = self->priv;
|
||||||
|
ID3D11VideoProcessorInputView *piv = nullptr;
|
||||||
|
ID3D11VideoProcessorOutputView *pov = nullptr;
|
||||||
|
ID3D11VideoContext1 *video_ctx = priv->video_context;
|
||||||
|
ID3D11VideoProcessor *proc = priv->processor;
|
||||||
|
D3D11_VIDEO_PROCESSOR_STREAM stream = { 0, };
|
||||||
|
HRESULT hr;
|
||||||
|
GstMemory *in_mem, *out_mem;
|
||||||
|
GstD3D11Memory *in_dmem;
|
||||||
|
GstD3D11Memory *out_dmem;
|
||||||
|
GstMapInfo in_info, out_info;
|
||||||
|
gboolean ret = FALSE;
|
||||||
|
|
||||||
|
g_assert (gst_buffer_n_memory (in_buf) == 1);
|
||||||
|
g_assert (gst_buffer_n_memory (out_buf) == 1);
|
||||||
|
|
||||||
|
in_mem = gst_buffer_peek_memory (in_buf, 0);
|
||||||
|
out_mem = gst_buffer_peek_memory (out_buf, 0);
|
||||||
|
|
||||||
|
if (!gst_memory_map (in_mem, &in_info,
|
||||||
|
(GstMapFlags) (GST_MAP_READ | GST_MAP_D3D11))) {
|
||||||
|
GST_ERROR_OBJECT (self, "Couldn't map input buffer");
|
||||||
|
return FALSE;
|
||||||
|
}
|
||||||
|
|
||||||
|
if (!gst_memory_map (out_mem, &out_info,
|
||||||
|
(GstMapFlags) (GST_MAP_WRITE | GST_MAP_D3D11))) {
|
||||||
|
GST_ERROR_OBJECT (self, "Couldn't map output buffer");
|
||||||
|
gst_memory_unmap (in_mem, &in_info);
|
||||||
|
return FALSE;
|
||||||
|
}
|
||||||
|
|
||||||
|
in_dmem = GST_D3D11_MEMORY_CAST (in_mem);
|
||||||
|
out_dmem = GST_D3D11_MEMORY_CAST (out_mem);
|
||||||
|
|
||||||
|
piv = gst_d3d11_memory_get_processor_input_view (in_dmem,
|
||||||
|
priv->video_device, priv->enumerator);
|
||||||
|
if (!piv) {
|
||||||
|
GST_ERROR_OBJECT (self, "PIV is unavailable");
|
||||||
|
goto out;
|
||||||
|
}
|
||||||
|
|
||||||
|
pov = gst_d3d11_memory_get_processor_output_view (out_dmem,
|
||||||
|
priv->video_device, priv->enumerator);
|
||||||
if (!pov) {
|
if (!pov) {
|
||||||
GST_ERROR_OBJECT (self, "POV is unavailable");
|
GST_ERROR_OBJECT (self, "POV is unavailable");
|
||||||
goto out;
|
goto out;
|
||||||
}
|
}
|
||||||
|
|
||||||
if (!piv) {
|
video_ctx->VideoProcessorSetStreamSourceRect (proc, 0, TRUE, &priv->src_rect);
|
||||||
if (!gst_d3d11_converter_ensure_fallback_inbuf (self, in_buf, in_info)) {
|
video_ctx->VideoProcessorSetStreamDestRect (proc, 0, TRUE, &priv->dest_rect);
|
||||||
GST_ERROR_OBJECT (self, "Couldn't copy into fallback texture");
|
|
||||||
goto out;
|
|
||||||
}
|
|
||||||
|
|
||||||
gst_d3d11_converter_unmap_buffer (self, in_buf, in_info);
|
|
||||||
in_buf = priv->fallback_inbuf;
|
|
||||||
|
|
||||||
if (!gst_d3d11_converter_map_buffer (self,
|
|
||||||
in_buf, in_info, (GstMapFlags) (GST_MAP_READ | GST_MAP_D3D11))) {
|
|
||||||
GST_ERROR_OBJECT (self, "Couldn't map fallback buffer");
|
|
||||||
in_buf = nullptr;
|
|
||||||
goto out;
|
|
||||||
}
|
|
||||||
|
|
||||||
in_dmem = (GstD3D11Memory *) gst_buffer_peek_memory (in_buf, 0);
|
|
||||||
piv = gst_d3d11_memory_get_processor_input_view (in_dmem,
|
|
||||||
priv->video_device, priv->enumerator);
|
|
||||||
if (!piv) {
|
|
||||||
GST_ERROR_OBJECT (self, "Couldn't get POV from fallback buffer");
|
|
||||||
goto out;
|
|
||||||
}
|
|
||||||
}
|
|
||||||
}
|
|
||||||
|
|
||||||
priv->processor_in_use = use_processor;
|
|
||||||
if (use_processor && !priv->processor_direction_not_supported) {
|
|
||||||
ID3D11VideoContext1 *video_ctx = priv->video_context;
|
|
||||||
ID3D11VideoProcessor *proc = priv->processor;
|
|
||||||
D3D11_VIDEO_PROCESSOR_STREAM stream = { 0, };
|
|
||||||
HRESULT hr;
|
|
||||||
|
|
||||||
if (priv->update_dest_rect && !gst_d3d11_converter_update_dest_rect (self)) {
|
|
||||||
GST_ERROR_OBJECT (self, "Failed to update dest rect");
|
|
||||||
goto out;
|
|
||||||
}
|
|
||||||
|
|
||||||
if (priv->update_src_rect && !gst_d3d11_converter_update_src_rect (self)) {
|
|
||||||
GST_ERROR_OBJECT (self, "Cannot update src rect");
|
|
||||||
goto out;
|
|
||||||
}
|
|
||||||
|
|
||||||
video_ctx->VideoProcessorSetStreamSourceRect (proc,
|
|
||||||
0, TRUE, &priv->src_rect);
|
|
||||||
video_ctx->VideoProcessorSetStreamDestRect (proc,
|
|
||||||
0, TRUE, &priv->dest_rect);
|
|
||||||
|
|
||||||
if (priv->clear_background) {
|
if (priv->clear_background) {
|
||||||
video_ctx->VideoProcessorSetOutputTargetRect (proc,
|
video_ctx->VideoProcessorSetOutputTargetRect (proc,
|
||||||
|
@ -4012,8 +4037,7 @@ gst_d3d11_converter_convert_buffer_internal (GstD3D11Converter * self,
|
||||||
video_ctx->VideoProcessorSetOutputBackgroundColor (proc,
|
video_ctx->VideoProcessorSetOutputBackgroundColor (proc,
|
||||||
GST_VIDEO_INFO_IS_YUV (&priv->out_info), &priv->background_color);
|
GST_VIDEO_INFO_IS_YUV (&priv->out_info), &priv->background_color);
|
||||||
} else {
|
} else {
|
||||||
video_ctx->VideoProcessorSetOutputTargetRect (proc, TRUE,
|
video_ctx->VideoProcessorSetOutputTargetRect (proc, TRUE, &priv->dest_rect);
|
||||||
&priv->dest_rect);
|
|
||||||
}
|
}
|
||||||
|
|
||||||
if (priv->video_context2 &&
|
if (priv->video_context2 &&
|
||||||
|
@ -4050,9 +4074,110 @@ gst_d3d11_converter_convert_buffer_internal (GstD3D11Converter * self,
|
||||||
GST_TRACE_OBJECT (self, "Converting using processor");
|
GST_TRACE_OBJECT (self, "Converting using processor");
|
||||||
|
|
||||||
hr = video_ctx->VideoProcessorBlt (proc, pov, 0, 1, &stream);
|
hr = video_ctx->VideoProcessorBlt (proc, pov, 0, 1, &stream);
|
||||||
|
|
||||||
ret = gst_d3d11_result (hr, self->device);
|
ret = gst_d3d11_result (hr, self->device);
|
||||||
goto out;
|
|
||||||
|
priv->processor_in_use = ret;
|
||||||
|
|
||||||
|
out:
|
||||||
|
gst_memory_unmap (out_mem, &out_info);
|
||||||
|
gst_memory_unmap (in_mem, &in_info);
|
||||||
|
|
||||||
|
return ret;
|
||||||
|
}
|
||||||
|
|
||||||
|
static gboolean
|
||||||
|
gst_d3d11_converter_convert_buffer_internal (GstD3D11Converter * self,
|
||||||
|
GstBuffer * in_buf, GstBuffer * out_buf)
|
||||||
|
{
|
||||||
|
GstD3D11ConverterPrivate *priv = self->priv;
|
||||||
|
ID3D11ShaderResourceView *srv[GST_VIDEO_MAX_PLANES] = { nullptr, };
|
||||||
|
ID3D11RenderTargetView *rtv[GST_VIDEO_MAX_PLANES] = { nullptr, };
|
||||||
|
GstD3D11Memory *in_dmem;
|
||||||
|
GstD3D11Memory *out_dmem;
|
||||||
|
GstMapInfo in_info[GST_VIDEO_MAX_PLANES];
|
||||||
|
GstMapInfo out_info[GST_VIDEO_MAX_PLANES];
|
||||||
|
D3D11_TEXTURE2D_DESC desc;
|
||||||
|
guint num_srv, num_rtv;
|
||||||
|
gboolean ret = FALSE;
|
||||||
|
gboolean in_d3d11;
|
||||||
|
|
||||||
|
GstD3D11SRWLockGuard (&priv->prop_lock);
|
||||||
|
|
||||||
|
/* Output buffer must be valid D3D11 buffer */
|
||||||
|
if (!gst_d3d11_converter_is_d3d11_buffer (self, out_buf)) {
|
||||||
|
GST_ERROR_OBJECT (self, "Output is not d3d11 buffer");
|
||||||
|
return FALSE;
|
||||||
|
}
|
||||||
|
|
||||||
|
if (gst_buffer_n_memory (in_buf) == 0) {
|
||||||
|
GST_ERROR_OBJECT (self, "Empty input buffer");
|
||||||
|
return FALSE;
|
||||||
|
}
|
||||||
|
|
||||||
|
out_dmem = (GstD3D11Memory *) gst_buffer_peek_memory (out_buf, 0);
|
||||||
|
if (!gst_d3d11_memory_get_texture_desc (out_dmem, &desc)) {
|
||||||
|
GST_ERROR_OBJECT (self, "Failed to get output desc");
|
||||||
|
return FALSE;
|
||||||
|
}
|
||||||
|
|
||||||
|
if ((desc.BindFlags & D3D11_BIND_RENDER_TARGET) == 0) {
|
||||||
|
GST_ERROR_OBJECT (self, "Output is not bound to render target");
|
||||||
|
return FALSE;
|
||||||
|
}
|
||||||
|
|
||||||
|
gst_d3d11_converter_update_hdr10_meta (self);
|
||||||
|
/* Update in/out rect */
|
||||||
|
if (!gst_d3d11_converter_update_dest_rect (self)) {
|
||||||
|
GST_ERROR_OBJECT (self, "Failed to update dest rect");
|
||||||
|
return FALSE;
|
||||||
|
}
|
||||||
|
|
||||||
|
if (!gst_d3d11_converter_update_src_rect (self)) {
|
||||||
|
GST_ERROR_OBJECT (self, "Failed to update src rect");
|
||||||
|
return FALSE;
|
||||||
|
}
|
||||||
|
|
||||||
|
in_d3d11 = gst_d3d11_converter_is_d3d11_buffer (self, in_buf);
|
||||||
|
if (gst_d3d11_converter_processor_available (self)) {
|
||||||
|
gboolean use_processor = FALSE;
|
||||||
|
gboolean piv_available = FALSE;
|
||||||
|
|
||||||
|
if (in_d3d11)
|
||||||
|
piv_available = gst_d3d11_converter_piv_available (self, in_buf);
|
||||||
|
|
||||||
|
if ((priv->supported_backend & GST_D3D11_CONVERTER_BACKEND_SHADER) != 0) {
|
||||||
|
/* processor only */
|
||||||
|
use_processor = TRUE;
|
||||||
|
} else if (piv_available) {
|
||||||
|
in_dmem = (GstD3D11Memory *) gst_buffer_peek_memory (in_buf, 0);
|
||||||
|
|
||||||
|
if (GST_VIDEO_INFO_FORMAT (&priv->in_info) == GST_VIDEO_FORMAT_YUY2) {
|
||||||
|
/* Always use processor for packed YUV */
|
||||||
|
use_processor = TRUE;
|
||||||
|
} else if (!gst_d3d11_memory_get_shader_resource_view_size (in_dmem)) {
|
||||||
|
/* SRV is unavailable, use processor */
|
||||||
|
use_processor = TRUE;
|
||||||
|
} else if (priv->video_context2 &&
|
||||||
|
(priv->have_in_hdr10 || priv->have_out_hdr10)) {
|
||||||
|
/* HDR10 tonemap is needed */
|
||||||
|
use_processor = TRUE;
|
||||||
|
} else if (priv->processor_in_use) {
|
||||||
|
use_processor = TRUE;
|
||||||
|
}
|
||||||
|
}
|
||||||
|
|
||||||
|
if (use_processor) {
|
||||||
|
if (!piv_available) {
|
||||||
|
if (!gst_d3d11_converter_upload_for_processor (self, in_buf)) {
|
||||||
|
GST_ERROR_OBJECT (self, "Couldn't upload buffer");
|
||||||
|
return FALSE;
|
||||||
|
}
|
||||||
|
|
||||||
|
in_buf = priv->piv_inbuf;
|
||||||
|
}
|
||||||
|
|
||||||
|
return gst_d3d11_converter_do_processor_blt (self, in_buf, out_buf);
|
||||||
|
}
|
||||||
}
|
}
|
||||||
|
|
||||||
if ((priv->supported_backend & GST_D3D11_CONVERTER_BACKEND_SHADER) == 0) {
|
if ((priv->supported_backend & GST_D3D11_CONVERTER_BACKEND_SHADER) == 0) {
|
||||||
|
@ -4060,6 +4185,29 @@ gst_d3d11_converter_convert_buffer_internal (GstD3D11Converter * self,
|
||||||
goto out;
|
goto out;
|
||||||
}
|
}
|
||||||
|
|
||||||
|
if (!in_d3d11 ||
|
||||||
|
GST_VIDEO_INFO_FORMAT (&priv->in_info) == GST_VIDEO_FORMAT_YUY2) {
|
||||||
|
if (!gst_d3d11_converter_upload_for_shader (self, in_buf)) {
|
||||||
|
GST_ERROR_OBJECT (self, "Couldn't copy into fallback buffer");
|
||||||
|
return FALSE;
|
||||||
|
}
|
||||||
|
|
||||||
|
in_buf = priv->fallback_inbuf;
|
||||||
|
}
|
||||||
|
|
||||||
|
if (!gst_d3d11_converter_map_buffer (self, in_buf, in_info,
|
||||||
|
(GstMapFlags) (GST_MAP_READ | GST_MAP_D3D11))) {
|
||||||
|
GST_ERROR_OBJECT (self, "Couldn't map input buffer");
|
||||||
|
return FALSE;
|
||||||
|
}
|
||||||
|
|
||||||
|
if (!gst_d3d11_converter_map_buffer (self, out_buf, out_info,
|
||||||
|
(GstMapFlags) (GST_MAP_WRITE | GST_MAP_D3D11))) {
|
||||||
|
GST_ERROR_OBJECT (self, "Couldn't map output buffer");
|
||||||
|
gst_d3d11_converter_unmap_buffer (self, in_buf, in_info);
|
||||||
|
return FALSE;
|
||||||
|
}
|
||||||
|
|
||||||
num_rtv = gst_d3d11_converter_get_rtv (self, out_buf, rtv);
|
num_rtv = gst_d3d11_converter_get_rtv (self, out_buf, rtv);
|
||||||
if (!num_rtv) {
|
if (!num_rtv) {
|
||||||
GST_ERROR_OBJECT (self, "RTV is unavailable");
|
GST_ERROR_OBJECT (self, "RTV is unavailable");
|
||||||
|
|
Loading…
Reference in a new issue