mirror of
https://gitlab.freedesktop.org/gstreamer/gstreamer.git
synced 2024-11-27 12:11:13 +00:00
d3d11decoder: Refactor device open step and negotiation
* Remove redundant method arguments * Don't allocate staging texture if downstream supports d3d11 memory Part-of: <https://gitlab.freedesktop.org/gstreamer/gst-plugins-bad/-/merge_requests/2079>
This commit is contained in:
parent
347d9ceb4e
commit
72345875b1
7 changed files with 246 additions and 219 deletions
|
@ -123,6 +123,7 @@ struct _GstD3D11Decoder
|
||||||
GstObject parent;
|
GstObject parent;
|
||||||
|
|
||||||
gboolean configured;
|
gboolean configured;
|
||||||
|
gboolean opened;
|
||||||
|
|
||||||
GstD3D11Device *device;
|
GstD3D11Device *device;
|
||||||
|
|
||||||
|
@ -132,6 +133,11 @@ struct _GstD3D11Decoder
|
||||||
ID3D11VideoDecoder *decoder_handle;
|
ID3D11VideoDecoder *decoder_handle;
|
||||||
|
|
||||||
GstVideoInfo info;
|
GstVideoInfo info;
|
||||||
|
GstD3D11Codec codec;
|
||||||
|
gint coded_width;
|
||||||
|
gint coded_height;
|
||||||
|
DXGI_FORMAT decoder_format;
|
||||||
|
gboolean downstream_supports_d3d11;
|
||||||
|
|
||||||
GstBufferPool *internal_pool;
|
GstBufferPool *internal_pool;
|
||||||
/* Internal pool params */
|
/* Internal pool params */
|
||||||
|
@ -252,39 +258,33 @@ gst_d3d11_decoder_get_property (GObject * object, guint prop_id,
|
||||||
}
|
}
|
||||||
}
|
}
|
||||||
|
|
||||||
static gboolean
|
static void
|
||||||
gst_d3d11_decoder_close (GstD3D11Decoder * self)
|
gst_d3d11_decoder_clear_resource (GstD3D11Decoder * self)
|
||||||
{
|
{
|
||||||
gst_d3d11_decoder_reset (self);
|
gst_clear_object (&self->internal_pool);
|
||||||
|
|
||||||
GST_D3D11_CLEAR_COM (self->video_device);
|
GST_D3D11_CLEAR_COM (self->decoder_handle);
|
||||||
GST_D3D11_CLEAR_COM (self->video_context);
|
GST_D3D11_CLEAR_COM (self->staging);
|
||||||
|
|
||||||
return TRUE;
|
memset (self->staging_texture_offset,
|
||||||
|
0, sizeof (self->staging_texture_offset));
|
||||||
|
memset (self->stating_texture_stride,
|
||||||
|
0, sizeof (self->stating_texture_stride));
|
||||||
}
|
}
|
||||||
|
|
||||||
static void
|
static void
|
||||||
gst_d3d11_decoder_reset_unlocked (GstD3D11Decoder * decoder)
|
gst_d3d11_decoder_reset (GstD3D11Decoder * self)
|
||||||
{
|
{
|
||||||
gst_clear_object (&decoder->internal_pool);
|
gst_d3d11_decoder_clear_resource (self);
|
||||||
|
|
||||||
GST_D3D11_CLEAR_COM (decoder->decoder_handle);
|
self->dpb_size = 0;
|
||||||
GST_D3D11_CLEAR_COM (decoder->staging);
|
self->downstream_min_buffers = 0;
|
||||||
|
|
||||||
decoder->dpb_size = 0;
|
self->configured = FALSE;
|
||||||
decoder->downstream_min_buffers = 0;
|
self->opened = FALSE;
|
||||||
|
|
||||||
decoder->configured = FALSE;
|
self->use_array_of_texture = FALSE;
|
||||||
}
|
self->downstream_supports_d3d11 = FALSE;
|
||||||
|
|
||||||
void
|
|
||||||
gst_d3d11_decoder_reset (GstD3D11Decoder * decoder)
|
|
||||||
{
|
|
||||||
g_return_if_fail (GST_IS_D3D11_DECODER (decoder));
|
|
||||||
|
|
||||||
gst_d3d11_device_lock (decoder->device);
|
|
||||||
gst_d3d11_decoder_reset_unlocked (decoder);
|
|
||||||
gst_d3d11_device_unlock (decoder->device);
|
|
||||||
}
|
}
|
||||||
|
|
||||||
static void
|
static void
|
||||||
|
@ -292,11 +292,12 @@ gst_d3d11_decoder_dispose (GObject * obj)
|
||||||
{
|
{
|
||||||
GstD3D11Decoder *self = GST_D3D11_DECODER (obj);
|
GstD3D11Decoder *self = GST_D3D11_DECODER (obj);
|
||||||
|
|
||||||
if (self->device) {
|
gst_d3d11_decoder_reset (self);
|
||||||
gst_d3d11_decoder_close (self);
|
|
||||||
gst_object_unref (self->device);
|
GST_D3D11_CLEAR_COM (self->video_device);
|
||||||
self->device = NULL;
|
GST_D3D11_CLEAR_COM (self->video_context);
|
||||||
}
|
|
||||||
|
gst_clear_object (&self->device);
|
||||||
|
|
||||||
G_OBJECT_CLASS (parent_class)->dispose (obj);
|
G_OBJECT_CLASS (parent_class)->dispose (obj);
|
||||||
}
|
}
|
||||||
|
@ -438,7 +439,7 @@ gst_d3d11_decoder_get_supported_decoder_profile (GstD3D11Decoder * decoder,
|
||||||
HRESULT hr;
|
HRESULT hr;
|
||||||
ID3D11VideoDevice *video_device;
|
ID3D11VideoDevice *video_device;
|
||||||
const GUID **profile_list = NULL;
|
const GUID **profile_list = NULL;
|
||||||
gint profile_size = 0;
|
guint profile_size = 0;
|
||||||
|
|
||||||
g_return_val_if_fail (GST_IS_D3D11_DECODER (decoder), FALSE);
|
g_return_val_if_fail (GST_IS_D3D11_DECODER (decoder), FALSE);
|
||||||
g_return_val_if_fail (selected_profile != NULL, FALSE);
|
g_return_val_if_fail (selected_profile != NULL, FALSE);
|
||||||
|
@ -563,25 +564,12 @@ gst_d3d11_decoder_get_supported_decoder_profile (GstD3D11Decoder * decoder,
|
||||||
return TRUE;
|
return TRUE;
|
||||||
}
|
}
|
||||||
|
|
||||||
|
|
||||||
gboolean
|
gboolean
|
||||||
gst_d3d11_decoder_configure (GstD3D11Decoder * decoder, GstD3D11Codec codec,
|
gst_d3d11_decoder_configure (GstD3D11Decoder * decoder, GstD3D11Codec codec,
|
||||||
GstVideoInfo * info, gint coded_width, gint coded_height, guint dpb_size)
|
GstVideoInfo * info, gint coded_width, gint coded_height, guint dpb_size)
|
||||||
{
|
{
|
||||||
const GstD3D11Format *d3d11_format;
|
const GstD3D11Format *d3d11_format;
|
||||||
HRESULT hr;
|
|
||||||
BOOL can_support = FALSE;
|
|
||||||
guint config_count;
|
|
||||||
D3D11_VIDEO_DECODER_CONFIG *config_list;
|
|
||||||
D3D11_VIDEO_DECODER_CONFIG *best_config = NULL;
|
|
||||||
D3D11_VIDEO_DECODER_DESC decoder_desc = { 0, };
|
|
||||||
D3D11_TEXTURE2D_DESC staging_desc = { 0, };
|
|
||||||
const GUID *selected_profile = NULL;
|
|
||||||
guint i;
|
|
||||||
gint aligned_width, aligned_height;
|
|
||||||
guint alignment;
|
|
||||||
GstD3D11DeviceVendor vendor;
|
|
||||||
ID3D11Device *device_handle;
|
|
||||||
ID3D11VideoDevice *video_device;
|
|
||||||
|
|
||||||
g_return_val_if_fail (GST_IS_D3D11_DECODER (decoder), FALSE);
|
g_return_val_if_fail (GST_IS_D3D11_DECODER (decoder), FALSE);
|
||||||
g_return_val_if_fail (codec > GST_D3D11_CODEC_NONE, FALSE);
|
g_return_val_if_fail (codec > GST_D3D11_CODEC_NONE, FALSE);
|
||||||
|
@ -591,11 +579,7 @@ gst_d3d11_decoder_configure (GstD3D11Decoder * decoder, GstD3D11Codec codec,
|
||||||
g_return_val_if_fail (coded_height >= GST_VIDEO_INFO_HEIGHT (info), FALSE);
|
g_return_val_if_fail (coded_height >= GST_VIDEO_INFO_HEIGHT (info), FALSE);
|
||||||
g_return_val_if_fail (dpb_size > 0, FALSE);
|
g_return_val_if_fail (dpb_size > 0, FALSE);
|
||||||
|
|
||||||
decoder->configured = FALSE;
|
gst_d3d11_decoder_reset (decoder);
|
||||||
decoder->use_array_of_texture = FALSE;
|
|
||||||
|
|
||||||
device_handle = gst_d3d11_device_get_device_handle (decoder->device);
|
|
||||||
video_device = decoder->video_device;
|
|
||||||
|
|
||||||
d3d11_format = gst_d3d11_device_format_from_gst (decoder->device,
|
d3d11_format = gst_d3d11_device_format_from_gst (decoder->device,
|
||||||
GST_VIDEO_INFO_FORMAT (info));
|
GST_VIDEO_INFO_FORMAT (info));
|
||||||
|
@ -605,26 +589,95 @@ gst_d3d11_decoder_configure (GstD3D11Decoder * decoder, GstD3D11Codec codec,
|
||||||
return FALSE;
|
return FALSE;
|
||||||
}
|
}
|
||||||
|
|
||||||
gst_d3d11_device_lock (decoder->device);
|
decoder->codec = codec;
|
||||||
if (!gst_d3d11_decoder_get_supported_decoder_profile (decoder,
|
decoder->info = *info;
|
||||||
codec, GST_VIDEO_INFO_FORMAT (info), &selected_profile)) {
|
decoder->coded_width = coded_width;
|
||||||
|
decoder->coded_height = coded_height;
|
||||||
|
decoder->dpb_size = dpb_size;
|
||||||
|
decoder->decoder_format = d3d11_format->dxgi_format;
|
||||||
|
|
||||||
|
decoder->configured = TRUE;
|
||||||
|
|
||||||
|
return TRUE;
|
||||||
|
}
|
||||||
|
|
||||||
|
static gboolean
|
||||||
|
gst_d3d11_decoder_ensure_staging_texture (GstD3D11Decoder * self)
|
||||||
|
{
|
||||||
|
ID3D11Device *device_handle;
|
||||||
|
D3D11_TEXTURE2D_DESC desc = { 0, };
|
||||||
|
HRESULT hr;
|
||||||
|
|
||||||
|
if (self->staging)
|
||||||
|
return TRUE;
|
||||||
|
|
||||||
|
device_handle = gst_d3d11_device_get_device_handle (self->device);
|
||||||
|
|
||||||
|
/* create stage texture to copy out */
|
||||||
|
desc.Width = self->aligned_width;
|
||||||
|
desc.Height = self->aligned_height;
|
||||||
|
desc.MipLevels = 1;
|
||||||
|
desc.Format = self->decoder_format;
|
||||||
|
desc.SampleDesc.Count = 1;
|
||||||
|
desc.ArraySize = 1;
|
||||||
|
desc.Usage = D3D11_USAGE_STAGING;
|
||||||
|
desc.CPUAccessFlags = D3D11_CPU_ACCESS_READ;
|
||||||
|
|
||||||
|
hr = device_handle->CreateTexture2D (&desc, NULL, &self->staging);
|
||||||
|
if (!gst_d3d11_result (hr, self->device)) {
|
||||||
|
GST_ERROR_OBJECT (self, "Couldn't create staging texture");
|
||||||
|
return FALSE;
|
||||||
|
}
|
||||||
|
|
||||||
|
return TRUE;
|
||||||
|
}
|
||||||
|
|
||||||
|
static gboolean
|
||||||
|
gst_d3d11_decoder_open (GstD3D11Decoder * self)
|
||||||
|
{
|
||||||
|
HRESULT hr;
|
||||||
|
BOOL can_support = FALSE;
|
||||||
|
guint config_count;
|
||||||
|
D3D11_VIDEO_DECODER_CONFIG *config_list;
|
||||||
|
D3D11_VIDEO_DECODER_CONFIG *best_config = NULL;
|
||||||
|
D3D11_VIDEO_DECODER_DESC decoder_desc = { 0, };
|
||||||
|
const GUID *selected_profile = NULL;
|
||||||
|
guint i;
|
||||||
|
gint aligned_width, aligned_height;
|
||||||
|
guint alignment;
|
||||||
|
GstD3D11DeviceVendor vendor;
|
||||||
|
ID3D11VideoDevice *video_device;
|
||||||
|
GstVideoInfo *info = &self->info;
|
||||||
|
|
||||||
|
if (self->opened)
|
||||||
|
return TRUE;
|
||||||
|
|
||||||
|
if (!self->configured) {
|
||||||
|
GST_ERROR_OBJECT (self, "Should configure first");
|
||||||
|
return FALSE;
|
||||||
|
}
|
||||||
|
|
||||||
|
video_device = self->video_device;
|
||||||
|
|
||||||
|
gst_d3d11_device_lock (self->device);
|
||||||
|
if (!gst_d3d11_decoder_get_supported_decoder_profile (self,
|
||||||
|
self->codec, GST_VIDEO_INFO_FORMAT (info), &selected_profile)) {
|
||||||
goto error;
|
goto error;
|
||||||
}
|
}
|
||||||
|
|
||||||
hr = video_device->CheckVideoDecoderFormat (selected_profile,
|
hr = video_device->CheckVideoDecoderFormat (selected_profile,
|
||||||
d3d11_format->dxgi_format, &can_support);
|
self->decoder_format, &can_support);
|
||||||
if (!gst_d3d11_result (hr, decoder->device) || !can_support) {
|
if (!gst_d3d11_result (hr, self->device) || !can_support) {
|
||||||
GST_ERROR_OBJECT (decoder,
|
GST_ERROR_OBJECT (self,
|
||||||
"VideoDevice could not support dxgi format %d, hr: 0x%x",
|
"VideoDevice could not support dxgi format %d, hr: 0x%x",
|
||||||
d3d11_format->dxgi_format, (guint) hr);
|
self->decoder_format, (guint) hr);
|
||||||
goto error;
|
goto error;
|
||||||
}
|
}
|
||||||
|
|
||||||
gst_d3d11_decoder_reset_unlocked (decoder);
|
gst_d3d11_decoder_clear_resource (self);
|
||||||
|
self->can_direct_rendering = TRUE;
|
||||||
|
|
||||||
decoder->can_direct_rendering = TRUE;
|
vendor = gst_d3d11_get_device_vendor (self->device);
|
||||||
|
|
||||||
vendor = gst_d3d11_get_device_vendor (decoder->device);
|
|
||||||
switch (vendor) {
|
switch (vendor) {
|
||||||
case GST_D3D11_DEVICE_VENDOR_XBOX:
|
case GST_D3D11_DEVICE_VENDOR_XBOX:
|
||||||
case GST_D3D11_DEVICE_VENDOR_QUALCOMM:
|
case GST_D3D11_DEVICE_VENDOR_QUALCOMM:
|
||||||
|
@ -633,7 +686,7 @@ gst_d3d11_decoder_configure (GstD3D11Decoder * decoder, GstD3D11Codec codec,
|
||||||
*
|
*
|
||||||
* Qualcomm driver seems to be buggy in zero-copy scenario
|
* Qualcomm driver seems to be buggy in zero-copy scenario
|
||||||
*/
|
*/
|
||||||
decoder->can_direct_rendering = FALSE;
|
self->can_direct_rendering = FALSE;
|
||||||
break;
|
break;
|
||||||
default:
|
default:
|
||||||
break;
|
break;
|
||||||
|
@ -644,7 +697,7 @@ gst_d3d11_decoder_configure (GstD3D11Decoder * decoder, GstD3D11Codec codec,
|
||||||
* See ff_dxva2_common_frame_params() in dxva.c of ffmpeg and
|
* See ff_dxva2_common_frame_params() in dxva.c of ffmpeg and
|
||||||
* directx_va_Setup() in directx_va.c of vlc.
|
* directx_va_Setup() in directx_va.c of vlc.
|
||||||
* But... where it is? */
|
* But... where it is? */
|
||||||
switch (codec) {
|
switch (self->codec) {
|
||||||
case GST_D3D11_CODEC_H265:
|
case GST_D3D11_CODEC_H265:
|
||||||
/* See directx_va_Setup() impl. in vlc */
|
/* See directx_va_Setup() impl. in vlc */
|
||||||
if (vendor != GST_D3D11_DEVICE_VENDOR_XBOX)
|
if (vendor != GST_D3D11_DEVICE_VENDOR_XBOX)
|
||||||
|
@ -661,27 +714,32 @@ gst_d3d11_decoder_configure (GstD3D11Decoder * decoder, GstD3D11Codec codec,
|
||||||
break;
|
break;
|
||||||
}
|
}
|
||||||
|
|
||||||
aligned_width = GST_ROUND_UP_N (coded_width, alignment);
|
aligned_width = GST_ROUND_UP_N (self->coded_width, alignment);
|
||||||
aligned_height = GST_ROUND_UP_N (coded_height, alignment);
|
aligned_height = GST_ROUND_UP_N (self->coded_height, alignment);
|
||||||
if (aligned_width != coded_width || aligned_height != coded_height) {
|
if (aligned_width != self->coded_width ||
|
||||||
GST_DEBUG_OBJECT (decoder,
|
aligned_height != self->coded_height) {
|
||||||
|
GST_DEBUG_OBJECT (self,
|
||||||
"coded resolution %dx%d is not aligned to %d, adjust to %dx%d",
|
"coded resolution %dx%d is not aligned to %d, adjust to %dx%d",
|
||||||
coded_width, coded_height, alignment, aligned_width, aligned_height);
|
self->coded_width, self->coded_height, alignment, aligned_width,
|
||||||
|
aligned_height);
|
||||||
}
|
}
|
||||||
|
|
||||||
|
self->aligned_width = aligned_width;
|
||||||
|
self->aligned_height = aligned_height;
|
||||||
|
|
||||||
decoder_desc.SampleWidth = aligned_width;
|
decoder_desc.SampleWidth = aligned_width;
|
||||||
decoder_desc.SampleHeight = aligned_height;
|
decoder_desc.SampleHeight = aligned_height;
|
||||||
decoder_desc.OutputFormat = d3d11_format->dxgi_format;
|
decoder_desc.OutputFormat = self->decoder_format;
|
||||||
decoder_desc.Guid = *selected_profile;
|
decoder_desc.Guid = *selected_profile;
|
||||||
|
|
||||||
hr = video_device->GetVideoDecoderConfigCount (&decoder_desc, &config_count);
|
hr = video_device->GetVideoDecoderConfigCount (&decoder_desc, &config_count);
|
||||||
if (!gst_d3d11_result (hr, decoder->device) || config_count == 0) {
|
if (!gst_d3d11_result (hr, self->device) || config_count == 0) {
|
||||||
GST_ERROR_OBJECT (decoder, "Could not get decoder config count, hr: 0x%x",
|
GST_ERROR_OBJECT (self, "Could not get decoder config count, hr: 0x%x",
|
||||||
(guint) hr);
|
(guint) hr);
|
||||||
goto error;
|
goto error;
|
||||||
}
|
}
|
||||||
|
|
||||||
GST_DEBUG_OBJECT (decoder, "Total %d config available", config_count);
|
GST_DEBUG_OBJECT (self, "Total %d config available", config_count);
|
||||||
|
|
||||||
config_list = (D3D11_VIDEO_DECODER_CONFIG *)
|
config_list = (D3D11_VIDEO_DECODER_CONFIG *)
|
||||||
g_alloca (sizeof (D3D11_VIDEO_DECODER_CONFIG) * config_count);
|
g_alloca (sizeof (D3D11_VIDEO_DECODER_CONFIG) * config_count);
|
||||||
|
@ -689,15 +747,15 @@ gst_d3d11_decoder_configure (GstD3D11Decoder * decoder, GstD3D11Codec codec,
|
||||||
for (i = 0; i < config_count; i++) {
|
for (i = 0; i < config_count; i++) {
|
||||||
hr = video_device->GetVideoDecoderConfig (&decoder_desc, i,
|
hr = video_device->GetVideoDecoderConfig (&decoder_desc, i,
|
||||||
&config_list[i]);
|
&config_list[i]);
|
||||||
if (!gst_d3d11_result (hr, decoder->device)) {
|
if (!gst_d3d11_result (hr, self->device)) {
|
||||||
GST_ERROR_OBJECT (decoder, "Could not get decoder %dth config, hr: 0x%x",
|
GST_ERROR_OBJECT (self, "Could not get decoder %dth config, hr: 0x%x",
|
||||||
i, (guint) hr);
|
i, (guint) hr);
|
||||||
goto error;
|
goto error;
|
||||||
}
|
}
|
||||||
|
|
||||||
/* FIXME: need support DXVA_Slice_H264_Long ?? */
|
/* FIXME: need support DXVA_Slice_H264_Long ?? */
|
||||||
/* this config uses DXVA_Slice_H264_Short */
|
/* this config uses DXVA_Slice_H264_Short */
|
||||||
switch (codec) {
|
switch (self->codec) {
|
||||||
case GST_D3D11_CODEC_H264:
|
case GST_D3D11_CODEC_H264:
|
||||||
if (config_list[i].ConfigBitstreamRaw == 2)
|
if (config_list[i].ConfigBitstreamRaw == 2)
|
||||||
best_config = &config_list[i];
|
best_config = &config_list[i];
|
||||||
|
@ -719,11 +777,11 @@ gst_d3d11_decoder_configure (GstD3D11Decoder * decoder, GstD3D11Codec codec,
|
||||||
}
|
}
|
||||||
|
|
||||||
if (best_config == NULL) {
|
if (best_config == NULL) {
|
||||||
GST_ERROR_OBJECT (decoder, "Could not determine decoder config");
|
GST_ERROR_OBJECT (self, "Could not determine decoder config");
|
||||||
goto error;
|
goto error;
|
||||||
}
|
}
|
||||||
|
|
||||||
GST_DEBUG_OBJECT (decoder, "ConfigDecoderSpecific 0x%x",
|
GST_DEBUG_OBJECT (self, "ConfigDecoderSpecific 0x%x",
|
||||||
best_config->ConfigDecoderSpecific);
|
best_config->ConfigDecoderSpecific);
|
||||||
|
|
||||||
/* FIXME: Revisit this at some point.
|
/* FIXME: Revisit this at some point.
|
||||||
|
@ -734,64 +792,43 @@ gst_d3d11_decoder_configure (GstD3D11Decoder * decoder, GstD3D11Codec codec,
|
||||||
/* bit 14 is equal to 1b means this config support array of texture and
|
/* bit 14 is equal to 1b means this config support array of texture and
|
||||||
* it's recommended type as per DXVA spec */
|
* it's recommended type as per DXVA spec */
|
||||||
if ((best_config->ConfigDecoderSpecific & 0x4000) == 0x4000) {
|
if ((best_config->ConfigDecoderSpecific & 0x4000) == 0x4000) {
|
||||||
GST_DEBUG_OBJECT (decoder, "Config support array of texture");
|
GST_DEBUG_OBJECT (self, "Config support array of texture");
|
||||||
decoder->use_array_of_texture = TRUE;
|
self->use_array_of_texture = TRUE;
|
||||||
}
|
}
|
||||||
#endif
|
#endif
|
||||||
|
|
||||||
hr = video_device->CreateVideoDecoder (&decoder_desc,
|
hr = video_device->CreateVideoDecoder (&decoder_desc,
|
||||||
best_config, &decoder->decoder_handle);
|
best_config, &self->decoder_handle);
|
||||||
if (!gst_d3d11_result (hr, decoder->device) || !decoder->decoder_handle) {
|
if (!gst_d3d11_result (hr, self->device) || !self->decoder_handle) {
|
||||||
GST_ERROR_OBJECT (decoder,
|
GST_ERROR_OBJECT (self,
|
||||||
"Could not create decoder object, hr: 0x%x", (guint) hr);
|
"Could not create decoder object, hr: 0x%x", (guint) hr);
|
||||||
goto error;
|
goto error;
|
||||||
}
|
}
|
||||||
|
|
||||||
GST_DEBUG_OBJECT (decoder,
|
GST_DEBUG_OBJECT (self, "Decoder object %p created", self->decoder_handle);
|
||||||
"Decoder object %p created", decoder->decoder_handle);
|
|
||||||
|
|
||||||
/* create stage texture to copy out */
|
if (!self->downstream_supports_d3d11 &&
|
||||||
staging_desc.Width = aligned_width;
|
!gst_d3d11_decoder_ensure_staging_texture (self)) {
|
||||||
staging_desc.Height = aligned_height;
|
GST_ERROR_OBJECT (self, "Couldn't prepare staging texture");
|
||||||
staging_desc.MipLevels = 1;
|
|
||||||
staging_desc.Format = d3d11_format->dxgi_format;
|
|
||||||
staging_desc.SampleDesc.Count = 1;
|
|
||||||
staging_desc.ArraySize = 1;
|
|
||||||
staging_desc.Usage = D3D11_USAGE_STAGING;
|
|
||||||
staging_desc.CPUAccessFlags = D3D11_CPU_ACCESS_READ;
|
|
||||||
|
|
||||||
hr = device_handle->CreateTexture2D (&staging_desc, NULL, &decoder->staging);
|
|
||||||
if (!gst_d3d11_result (hr, decoder->device)) {
|
|
||||||
GST_ERROR_OBJECT (decoder, "Couldn't create staging texture");
|
|
||||||
goto error;
|
goto error;
|
||||||
}
|
}
|
||||||
|
|
||||||
memset (decoder->staging_texture_offset,
|
self->decoder_profile = *selected_profile;
|
||||||
0, sizeof (decoder->staging_texture_offset));
|
|
||||||
memset (decoder->stating_texture_stride,
|
|
||||||
0, sizeof (decoder->stating_texture_stride));
|
|
||||||
|
|
||||||
decoder->decoder_profile = *selected_profile;
|
|
||||||
|
|
||||||
/* Store pool related information here, then we will setup internal pool
|
/* Store pool related information here, then we will setup internal pool
|
||||||
* later once the number of min buffer size required by downstream is known.
|
* later once the number of min buffer size required by downstream is known.
|
||||||
* Actual buffer pool size will be "dpb_size + downstream_min_buffers"
|
* Actual buffer pool size will be "dpb_size + downstream_min_buffers"
|
||||||
*/
|
*/
|
||||||
decoder->info = *info;
|
self->downstream_min_buffers = 0;
|
||||||
decoder->dpb_size = dpb_size;
|
|
||||||
decoder->aligned_width = aligned_width;
|
|
||||||
decoder->aligned_height = aligned_height;
|
|
||||||
decoder->downstream_min_buffers = 0;
|
|
||||||
|
|
||||||
decoder->configured = TRUE;
|
self->opened = TRUE;
|
||||||
|
gst_d3d11_device_unlock (self->device);
|
||||||
gst_d3d11_device_unlock (decoder->device);
|
|
||||||
|
|
||||||
return TRUE;
|
return TRUE;
|
||||||
|
|
||||||
error:
|
error:
|
||||||
gst_d3d11_decoder_reset_unlocked (decoder);
|
gst_d3d11_decoder_reset (self);
|
||||||
gst_d3d11_device_unlock (decoder->device);
|
gst_d3d11_device_unlock (self->device);
|
||||||
|
|
||||||
return FALSE;
|
return FALSE;
|
||||||
}
|
}
|
||||||
|
@ -1048,6 +1085,11 @@ copy_to_system (GstD3D11Decoder * self, GstVideoInfo * info, gint display_width,
|
||||||
ID3D11DeviceContext *device_context =
|
ID3D11DeviceContext *device_context =
|
||||||
gst_d3d11_device_get_device_context_handle (self->device);
|
gst_d3d11_device_get_device_context_handle (self->device);
|
||||||
|
|
||||||
|
if (!gst_d3d11_decoder_ensure_staging_texture (self)) {
|
||||||
|
GST_ERROR_OBJECT (self, "Staging texture is not available");
|
||||||
|
return FALSE;
|
||||||
|
}
|
||||||
|
|
||||||
if (!gst_video_frame_map (&out_frame, info, output, GST_MAP_WRITE)) {
|
if (!gst_video_frame_map (&out_frame, info, output, GST_MAP_WRITE)) {
|
||||||
GST_ERROR_OBJECT (self, "Couldn't map output buffer");
|
GST_ERROR_OBJECT (self, "Couldn't map output buffer");
|
||||||
return FALSE;
|
return FALSE;
|
||||||
|
@ -1199,32 +1241,33 @@ do_process:
|
||||||
}
|
}
|
||||||
|
|
||||||
gboolean
|
gboolean
|
||||||
gst_d3d11_decoder_negotiate (GstVideoDecoder * decoder,
|
gst_d3d11_decoder_negotiate (GstD3D11Decoder * decoder,
|
||||||
GstVideoCodecState * input_state, GstVideoFormat format,
|
GstVideoDecoder * videodec, GstVideoCodecState * input_state,
|
||||||
guint width, guint height, GstVideoInterlaceMode interlace_mode,
|
GstVideoCodecState ** output_state)
|
||||||
GstVideoCodecState ** output_state, gboolean * downstream_supports_d3d11)
|
|
||||||
{
|
{
|
||||||
GstCaps *peer_caps;
|
GstCaps *peer_caps;
|
||||||
GstVideoCodecState *state = NULL;
|
GstVideoCodecState *state = NULL;
|
||||||
gboolean alternate_interlaced;
|
gboolean alternate_interlaced;
|
||||||
gboolean alternate_supported = FALSE;
|
gboolean alternate_supported = FALSE;
|
||||||
gboolean d3d11_supported = FALSE;
|
gboolean d3d11_supported = FALSE;
|
||||||
|
GstVideoInfo *info;
|
||||||
|
|
||||||
g_return_val_if_fail (GST_IS_VIDEO_DECODER (decoder), FALSE);
|
g_return_val_if_fail (GST_IS_D3D11_DECODER (decoder), FALSE);
|
||||||
|
g_return_val_if_fail (GST_IS_VIDEO_DECODER (videodec), FALSE);
|
||||||
g_return_val_if_fail (input_state != NULL, FALSE);
|
g_return_val_if_fail (input_state != NULL, FALSE);
|
||||||
g_return_val_if_fail (format != GST_VIDEO_FORMAT_UNKNOWN, FALSE);
|
|
||||||
g_return_val_if_fail (width > 0, FALSE);
|
|
||||||
g_return_val_if_fail (height > 0, FALSE);
|
|
||||||
g_return_val_if_fail (output_state != NULL, FALSE);
|
g_return_val_if_fail (output_state != NULL, FALSE);
|
||||||
g_return_val_if_fail (downstream_supports_d3d11 != NULL, FALSE);
|
|
||||||
|
|
||||||
alternate_interlaced = (interlace_mode == GST_VIDEO_INTERLACE_MODE_ALTERNATE);
|
info = &decoder->info;
|
||||||
|
|
||||||
peer_caps = gst_pad_get_allowed_caps (GST_VIDEO_DECODER_SRC_PAD (decoder));
|
alternate_interlaced =
|
||||||
GST_DEBUG_OBJECT (decoder, "Allowed caps %" GST_PTR_FORMAT, peer_caps);
|
(GST_VIDEO_INFO_INTERLACE_MODE (info) ==
|
||||||
|
GST_VIDEO_INTERLACE_MODE_ALTERNATE);
|
||||||
|
|
||||||
|
peer_caps = gst_pad_get_allowed_caps (GST_VIDEO_DECODER_SRC_PAD (videodec));
|
||||||
|
GST_DEBUG_OBJECT (videodec, "Allowed caps %" GST_PTR_FORMAT, peer_caps);
|
||||||
|
|
||||||
if (!peer_caps || gst_caps_is_any (peer_caps)) {
|
if (!peer_caps || gst_caps_is_any (peer_caps)) {
|
||||||
GST_DEBUG_OBJECT (decoder,
|
GST_DEBUG_OBJECT (videodec,
|
||||||
"cannot determine output format, use system memory");
|
"cannot determine output format, use system memory");
|
||||||
} else {
|
} else {
|
||||||
GstCapsFeatures *features;
|
GstCapsFeatures *features;
|
||||||
|
@ -1252,27 +1295,31 @@ gst_d3d11_decoder_negotiate (GstVideoDecoder * decoder,
|
||||||
}
|
}
|
||||||
gst_clear_caps (&peer_caps);
|
gst_clear_caps (&peer_caps);
|
||||||
|
|
||||||
GST_DEBUG_OBJECT (decoder,
|
GST_DEBUG_OBJECT (videodec,
|
||||||
"Downstream feature support, D3D11 memory: %d, interlaced format %d",
|
"Downstream feature support, D3D11 memory: %d, interlaced format %d",
|
||||||
d3d11_supported, alternate_supported);
|
d3d11_supported, alternate_supported);
|
||||||
|
|
||||||
if (alternate_interlaced) {
|
if (alternate_interlaced) {
|
||||||
/* FIXME: D3D11 cannot support alternating interlaced stream yet */
|
/* FIXME: D3D11 cannot support alternating interlaced stream yet */
|
||||||
GST_FIXME_OBJECT (decoder,
|
GST_FIXME_OBJECT (videodec,
|
||||||
"Implement alternating interlaced stream for D3D11");
|
"Implement alternating interlaced stream for D3D11");
|
||||||
|
|
||||||
if (alternate_supported) {
|
if (alternate_supported) {
|
||||||
|
gint height = GST_VIDEO_INFO_HEIGHT (info);
|
||||||
|
|
||||||
/* Set caps resolution with display size, that's how we designed
|
/* Set caps resolution with display size, that's how we designed
|
||||||
* for alternating interlaced stream */
|
* for alternating interlaced stream */
|
||||||
height = 2 * height;
|
height = 2 * height;
|
||||||
state = gst_video_decoder_set_interlaced_output_state (decoder,
|
state = gst_video_decoder_set_interlaced_output_state (videodec,
|
||||||
format, interlace_mode, width, height, input_state);
|
GST_VIDEO_INFO_FORMAT (info), GST_VIDEO_INFO_INTERLACE_MODE (info),
|
||||||
|
GST_VIDEO_INFO_WIDTH (info), height, input_state);
|
||||||
} else {
|
} else {
|
||||||
GST_WARNING_OBJECT (decoder,
|
GST_WARNING_OBJECT (videodec,
|
||||||
"Downstream doesn't support alternating interlaced stream");
|
"Downstream doesn't support alternating interlaced stream");
|
||||||
|
|
||||||
state = gst_video_decoder_set_output_state (decoder,
|
state = gst_video_decoder_set_output_state (videodec,
|
||||||
format, width, height, input_state);
|
GST_VIDEO_INFO_FORMAT (info), GST_VIDEO_INFO_WIDTH (info),
|
||||||
|
GST_VIDEO_INFO_HEIGHT (info), input_state);
|
||||||
|
|
||||||
/* XXX: adjust PAR, this would produce output similar to that of
|
/* XXX: adjust PAR, this would produce output similar to that of
|
||||||
* "line doubling" (so called bob deinterlacing) processing.
|
* "line doubling" (so called bob deinterlacing) processing.
|
||||||
|
@ -1284,8 +1331,9 @@ gst_d3d11_decoder_negotiate (GstVideoDecoder * decoder,
|
||||||
state->info.fps_n *= 2;
|
state->info.fps_n *= 2;
|
||||||
}
|
}
|
||||||
} else {
|
} else {
|
||||||
state = gst_video_decoder_set_interlaced_output_state (decoder,
|
state = gst_video_decoder_set_interlaced_output_state (videodec,
|
||||||
format, interlace_mode, width, height, input_state);
|
GST_VIDEO_INFO_FORMAT (info), GST_VIDEO_INFO_INTERLACE_MODE (info),
|
||||||
|
GST_VIDEO_INFO_WIDTH (info), GST_VIDEO_INFO_HEIGHT (info), input_state);
|
||||||
}
|
}
|
||||||
|
|
||||||
if (!state) {
|
if (!state) {
|
||||||
|
@ -1304,15 +1352,14 @@ gst_d3d11_decoder_negotiate (GstVideoDecoder * decoder,
|
||||||
gst_caps_features_new (GST_CAPS_FEATURE_MEMORY_D3D11_MEMORY, NULL));
|
gst_caps_features_new (GST_CAPS_FEATURE_MEMORY_D3D11_MEMORY, NULL));
|
||||||
}
|
}
|
||||||
|
|
||||||
*downstream_supports_d3d11 = d3d11_supported;
|
decoder->downstream_supports_d3d11 = d3d11_supported;
|
||||||
|
|
||||||
return TRUE;
|
return gst_d3d11_decoder_open (decoder);
|
||||||
}
|
}
|
||||||
|
|
||||||
gboolean
|
gboolean
|
||||||
gst_d3d11_decoder_decide_allocation (GstVideoDecoder * decoder,
|
gst_d3d11_decoder_decide_allocation (GstD3D11Decoder * decoder,
|
||||||
GstQuery * query, GstD3D11Device * device, GstD3D11Codec codec,
|
GstVideoDecoder * videodec, GstQuery * query)
|
||||||
gboolean use_d3d11_pool, GstD3D11Decoder * d3d11_decoder)
|
|
||||||
{
|
{
|
||||||
GstCaps *outcaps;
|
GstCaps *outcaps;
|
||||||
GstBufferPool *pool = NULL;
|
GstBufferPool *pool = NULL;
|
||||||
|
@ -1320,12 +1367,16 @@ gst_d3d11_decoder_decide_allocation (GstVideoDecoder * decoder,
|
||||||
GstVideoInfo vinfo = { 0, };
|
GstVideoInfo vinfo = { 0, };
|
||||||
GstStructure *config;
|
GstStructure *config;
|
||||||
GstD3D11AllocationParams *d3d11_params;
|
GstD3D11AllocationParams *d3d11_params;
|
||||||
|
gboolean use_d3d11_pool;
|
||||||
|
|
||||||
g_return_val_if_fail (GST_IS_VIDEO_DECODER (decoder), FALSE);
|
g_return_val_if_fail (GST_IS_D3D11_DECODER (decoder), FALSE);
|
||||||
|
g_return_val_if_fail (GST_IS_VIDEO_DECODER (videodec), FALSE);
|
||||||
g_return_val_if_fail (query != NULL, FALSE);
|
g_return_val_if_fail (query != NULL, FALSE);
|
||||||
g_return_val_if_fail (GST_IS_D3D11_DEVICE (device), FALSE);
|
|
||||||
g_return_val_if_fail (codec > GST_D3D11_CODEC_NONE &&
|
if (!decoder->opened) {
|
||||||
codec < GST_D3D11_CODEC_LAST, FALSE);
|
GST_ERROR_OBJECT (videodec, "Should open decoder first");
|
||||||
|
return FALSE;
|
||||||
|
}
|
||||||
|
|
||||||
gst_query_parse_allocation (query, &outcaps, NULL);
|
gst_query_parse_allocation (query, &outcaps, NULL);
|
||||||
|
|
||||||
|
@ -1334,6 +1385,8 @@ gst_d3d11_decoder_decide_allocation (GstVideoDecoder * decoder,
|
||||||
return FALSE;
|
return FALSE;
|
||||||
}
|
}
|
||||||
|
|
||||||
|
use_d3d11_pool = decoder->downstream_supports_d3d11;
|
||||||
|
|
||||||
gst_video_info_from_caps (&vinfo, outcaps);
|
gst_video_info_from_caps (&vinfo, outcaps);
|
||||||
n = gst_query_get_n_allocation_pools (query);
|
n = gst_query_get_n_allocation_pools (query);
|
||||||
if (n > 0)
|
if (n > 0)
|
||||||
|
@ -1347,7 +1400,7 @@ gst_d3d11_decoder_decide_allocation (GstVideoDecoder * decoder,
|
||||||
|
|
||||||
if (!pool) {
|
if (!pool) {
|
||||||
if (use_d3d11_pool)
|
if (use_d3d11_pool)
|
||||||
pool = gst_d3d11_buffer_pool_new (device);
|
pool = gst_d3d11_buffer_pool_new (decoder->device);
|
||||||
else
|
else
|
||||||
pool = gst_video_buffer_pool_new ();
|
pool = gst_video_buffer_pool_new ();
|
||||||
|
|
||||||
|
@ -1367,7 +1420,7 @@ gst_d3d11_decoder_decide_allocation (GstVideoDecoder * decoder,
|
||||||
|
|
||||||
d3d11_params = gst_buffer_pool_config_get_d3d11_allocation_params (config);
|
d3d11_params = gst_buffer_pool_config_get_d3d11_allocation_params (config);
|
||||||
if (!d3d11_params)
|
if (!d3d11_params)
|
||||||
d3d11_params = gst_d3d11_allocation_params_new (device, &vinfo,
|
d3d11_params = gst_d3d11_allocation_params_new (decoder->device, &vinfo,
|
||||||
(GstD3D11AllocationFlags) 0, 0);
|
(GstD3D11AllocationFlags) 0, 0);
|
||||||
|
|
||||||
width = GST_VIDEO_INFO_WIDTH (&vinfo);
|
width = GST_VIDEO_INFO_WIDTH (&vinfo);
|
||||||
|
@ -1377,7 +1430,7 @@ gst_d3d11_decoder_decide_allocation (GstVideoDecoder * decoder,
|
||||||
align.padding_right = GST_ROUND_UP_16 (width) - width;
|
align.padding_right = GST_ROUND_UP_16 (width) - width;
|
||||||
align.padding_bottom = GST_ROUND_UP_16 (height) - height;
|
align.padding_bottom = GST_ROUND_UP_16 (height) - height;
|
||||||
if (!gst_d3d11_allocation_params_alignment (d3d11_params, &align)) {
|
if (!gst_d3d11_allocation_params_alignment (d3d11_params, &align)) {
|
||||||
GST_ERROR_OBJECT (decoder, "Cannot set alignment");
|
GST_ERROR_OBJECT (videodec, "Cannot set alignment");
|
||||||
return FALSE;
|
return FALSE;
|
||||||
}
|
}
|
||||||
|
|
||||||
|
@ -1393,9 +1446,9 @@ gst_d3d11_decoder_decide_allocation (GstVideoDecoder * decoder,
|
||||||
/* Store min buffer size. We need to take account of the amount of buffers
|
/* Store min buffer size. We need to take account of the amount of buffers
|
||||||
* which might be held by downstream in case of zero-copy playback */
|
* which might be held by downstream in case of zero-copy playback */
|
||||||
/* XXX: hardcoded bound 16, to avoid too large pool size */
|
/* XXX: hardcoded bound 16, to avoid too large pool size */
|
||||||
d3d11_decoder->downstream_min_buffers = MIN (min, 16);
|
decoder->downstream_min_buffers = MIN (min, 16);
|
||||||
|
|
||||||
GST_DEBUG_OBJECT (decoder, "Downstream min buffres: %d", min);
|
GST_DEBUG_OBJECT (videodec, "Downstream min buffres: %d", min);
|
||||||
}
|
}
|
||||||
|
|
||||||
gst_buffer_pool_set_config (pool, config);
|
gst_buffer_pool_set_config (pool, config);
|
||||||
|
@ -1423,7 +1476,7 @@ gst_d3d11_decoder_can_direct_render (GstD3D11Decoder * decoder,
|
||||||
g_return_val_if_fail (GST_IS_BUFFER (view_buffer), FALSE);
|
g_return_val_if_fail (GST_IS_BUFFER (view_buffer), FALSE);
|
||||||
g_return_val_if_fail (picture != NULL, FALSE);
|
g_return_val_if_fail (picture != NULL, FALSE);
|
||||||
|
|
||||||
if (!decoder->can_direct_rendering)
|
if (!decoder->can_direct_rendering || !decoder->downstream_supports_d3d11)
|
||||||
return FALSE;
|
return FALSE;
|
||||||
|
|
||||||
/* XXX: Not a thread-safe way, but should not be a problem.
|
/* XXX: Not a thread-safe way, but should not be a problem.
|
||||||
|
|
|
@ -66,8 +66,6 @@ gboolean gst_d3d11_decoder_configure (GstD3D11Decoder * decoder,
|
||||||
gint coded_height,
|
gint coded_height,
|
||||||
guint dpb_size);
|
guint dpb_size);
|
||||||
|
|
||||||
void gst_d3d11_decoder_reset (GstD3D11Decoder * decoder);
|
|
||||||
|
|
||||||
gboolean gst_d3d11_decoder_begin_frame (GstD3D11Decoder * decoder,
|
gboolean gst_d3d11_decoder_begin_frame (GstD3D11Decoder * decoder,
|
||||||
ID3D11VideoDecoderOutputView * output_view,
|
ID3D11VideoDecoderOutputView * output_view,
|
||||||
guint content_key_size,
|
guint content_key_size,
|
||||||
|
@ -102,21 +100,14 @@ gboolean gst_d3d11_decoder_process_output (GstD3D11Decoder * decod
|
||||||
GstBuffer * decoder_buffer,
|
GstBuffer * decoder_buffer,
|
||||||
GstBuffer * output);
|
GstBuffer * output);
|
||||||
|
|
||||||
gboolean gst_d3d11_decoder_negotiate (GstVideoDecoder * decoder,
|
gboolean gst_d3d11_decoder_negotiate (GstD3D11Decoder * decoder,
|
||||||
|
GstVideoDecoder * videodec,
|
||||||
GstVideoCodecState * input_state,
|
GstVideoCodecState * input_state,
|
||||||
GstVideoFormat format,
|
GstVideoCodecState ** output_state);
|
||||||
guint width,
|
|
||||||
guint height,
|
|
||||||
GstVideoInterlaceMode interlace_mode,
|
|
||||||
GstVideoCodecState ** output_state,
|
|
||||||
gboolean * downstream_supports_d3d11);
|
|
||||||
|
|
||||||
gboolean gst_d3d11_decoder_decide_allocation (GstVideoDecoder * decoder,
|
gboolean gst_d3d11_decoder_decide_allocation (GstD3D11Decoder * decoder,
|
||||||
GstQuery * query,
|
GstVideoDecoder * videodec,
|
||||||
GstD3D11Device * device,
|
GstQuery * query);
|
||||||
GstD3D11Codec codec,
|
|
||||||
gboolean use_d3d11_pool,
|
|
||||||
GstD3D11Decoder * d3d11_decoder);
|
|
||||||
|
|
||||||
gboolean gst_d3d11_decoder_can_direct_render (GstD3D11Decoder * decoder,
|
gboolean gst_d3d11_decoder_can_direct_render (GstD3D11Decoder * decoder,
|
||||||
GstBuffer * view_buffer,
|
GstBuffer * view_buffer,
|
||||||
|
|
|
@ -123,8 +123,6 @@ typedef struct _GstD3D11H264Dec
|
||||||
guint remaining_buffer_size;
|
guint remaining_buffer_size;
|
||||||
guint8 *bitstream_buffer_data;
|
guint8 *bitstream_buffer_data;
|
||||||
|
|
||||||
gboolean use_d3d11_output;
|
|
||||||
|
|
||||||
DXVA_PicEntry_H264 ref_frame_list[16];
|
DXVA_PicEntry_H264 ref_frame_list[16];
|
||||||
INT field_order_cnt_list[16][2];
|
INT field_order_cnt_list[16][2];
|
||||||
USHORT frame_num_list[16];
|
USHORT frame_num_list[16];
|
||||||
|
@ -368,12 +366,10 @@ gst_d3d11_h264_dec_negotiate (GstVideoDecoder * decoder)
|
||||||
GstD3D11H264Dec *self = GST_D3D11_H264_DEC (decoder);
|
GstD3D11H264Dec *self = GST_D3D11_H264_DEC (decoder);
|
||||||
GstH264Decoder *h264dec = GST_H264_DECODER (decoder);
|
GstH264Decoder *h264dec = GST_H264_DECODER (decoder);
|
||||||
|
|
||||||
if (!gst_d3d11_decoder_negotiate (decoder, h264dec->input_state,
|
if (!gst_d3d11_decoder_negotiate (self->d3d11_decoder, decoder,
|
||||||
self->out_format, self->width, self->height,
|
h264dec->input_state, &self->output_state)) {
|
||||||
self->interlaced ? GST_VIDEO_INTERLACE_MODE_MIXED :
|
|
||||||
GST_VIDEO_INTERLACE_MODE_PROGRESSIVE,
|
|
||||||
&self->output_state, &self->use_d3d11_output))
|
|
||||||
return FALSE;
|
return FALSE;
|
||||||
|
}
|
||||||
|
|
||||||
return GST_VIDEO_DECODER_CLASS (parent_class)->negotiate (decoder);
|
return GST_VIDEO_DECODER_CLASS (parent_class)->negotiate (decoder);
|
||||||
}
|
}
|
||||||
|
@ -384,9 +380,10 @@ gst_d3d11_h264_dec_decide_allocation (GstVideoDecoder * decoder,
|
||||||
{
|
{
|
||||||
GstD3D11H264Dec *self = GST_D3D11_H264_DEC (decoder);
|
GstD3D11H264Dec *self = GST_D3D11_H264_DEC (decoder);
|
||||||
|
|
||||||
if (!gst_d3d11_decoder_decide_allocation (decoder, query, self->device,
|
if (!gst_d3d11_decoder_decide_allocation (self->d3d11_decoder,
|
||||||
GST_D3D11_CODEC_H264, self->use_d3d11_output, self->d3d11_decoder))
|
decoder, query)) {
|
||||||
return FALSE;
|
return FALSE;
|
||||||
|
}
|
||||||
|
|
||||||
return GST_VIDEO_DECODER_CLASS (parent_class)->decide_allocation
|
return GST_VIDEO_DECODER_CLASS (parent_class)->decide_allocation
|
||||||
(decoder, query);
|
(decoder, query);
|
||||||
|
@ -486,6 +483,8 @@ gst_d3d11_h264_dec_new_sequence (GstH264Decoder * decoder,
|
||||||
|
|
||||||
gst_video_info_set_format (&info,
|
gst_video_info_set_format (&info,
|
||||||
self->out_format, self->width, self->height);
|
self->out_format, self->width, self->height);
|
||||||
|
if (self->interlaced)
|
||||||
|
GST_VIDEO_INFO_INTERLACE_MODE (&info) = GST_VIDEO_INTERLACE_MODE_MIXED;
|
||||||
|
|
||||||
/* Store configured DPB size here. Then, it will be referenced later
|
/* Store configured DPB size here. Then, it will be referenced later
|
||||||
* to decide whether we need to re-open decoder object or not.
|
* to decide whether we need to re-open decoder object or not.
|
||||||
|
@ -493,7 +492,6 @@ gst_d3d11_h264_dec_new_sequence (GstH264Decoder * decoder,
|
||||||
* new DPB size is decreased, we can reuse existing decoder object.
|
* new DPB size is decreased, we can reuse existing decoder object.
|
||||||
*/
|
*/
|
||||||
self->max_dpb_size = max_dpb_size;
|
self->max_dpb_size = max_dpb_size;
|
||||||
gst_d3d11_decoder_reset (self->d3d11_decoder);
|
|
||||||
if (!gst_d3d11_decoder_configure (self->d3d11_decoder, GST_D3D11_CODEC_H264,
|
if (!gst_d3d11_decoder_configure (self->d3d11_decoder, GST_D3D11_CODEC_H264,
|
||||||
&info, self->coded_width, self->coded_height,
|
&info, self->coded_width, self->coded_height,
|
||||||
/* Additional 4 views margin for zero-copy rendering */
|
/* Additional 4 views margin for zero-copy rendering */
|
||||||
|
@ -836,8 +834,7 @@ gst_d3d11_h264_dec_output_picture (GstH264Decoder * decoder,
|
||||||
* expose our decoder view without copy. In case of reverse playback, however,
|
* expose our decoder view without copy. In case of reverse playback, however,
|
||||||
* we cannot do that since baseclass will store the decoded buffer
|
* we cannot do that since baseclass will store the decoded buffer
|
||||||
* up to gop size but our dpb pool cannot be increased */
|
* up to gop size but our dpb pool cannot be increased */
|
||||||
if (self->use_d3d11_output
|
if (GST_VIDEO_DECODER (self)->input_segment.rate > 0
|
||||||
&& GST_VIDEO_DECODER (self)->input_segment.rate > 0
|
|
||||||
&& gst_d3d11_decoder_can_direct_render (self->d3d11_decoder, view_buffer,
|
&& gst_d3d11_decoder_can_direct_render (self->d3d11_decoder, view_buffer,
|
||||||
GST_MINI_OBJECT_CAST (picture))) {
|
GST_MINI_OBJECT_CAST (picture))) {
|
||||||
direct_rendering = TRUE;
|
direct_rendering = TRUE;
|
||||||
|
|
|
@ -93,8 +93,6 @@ typedef struct _GstD3D11H265Dec
|
||||||
guint remaining_buffer_size;
|
guint remaining_buffer_size;
|
||||||
guint8 *bitstream_buffer_data;
|
guint8 *bitstream_buffer_data;
|
||||||
|
|
||||||
gboolean use_d3d11_output;
|
|
||||||
|
|
||||||
DXVA_PicEntry_HEVC ref_pic_list[15];
|
DXVA_PicEntry_HEVC ref_pic_list[15];
|
||||||
INT pic_order_cnt_val_list[15];
|
INT pic_order_cnt_val_list[15];
|
||||||
UCHAR ref_pic_set_st_curr_before[8];
|
UCHAR ref_pic_set_st_curr_before[8];
|
||||||
|
@ -323,9 +321,8 @@ gst_d3d11_h265_dec_negotiate (GstVideoDecoder * decoder)
|
||||||
GstD3D11H265Dec *self = GST_D3D11_H265_DEC (decoder);
|
GstD3D11H265Dec *self = GST_D3D11_H265_DEC (decoder);
|
||||||
GstH265Decoder *h265dec = GST_H265_DECODER (decoder);
|
GstH265Decoder *h265dec = GST_H265_DECODER (decoder);
|
||||||
|
|
||||||
if (!gst_d3d11_decoder_negotiate (decoder, h265dec->input_state,
|
if (!gst_d3d11_decoder_negotiate (self->d3d11_decoder, decoder,
|
||||||
self->out_format, self->width, self->height,
|
h265dec->input_state, &self->output_state)) {
|
||||||
self->interlace_mode, &self->output_state, &self->use_d3d11_output)) {
|
|
||||||
return FALSE;
|
return FALSE;
|
||||||
}
|
}
|
||||||
|
|
||||||
|
@ -338,9 +335,10 @@ gst_d3d11_h265_dec_decide_allocation (GstVideoDecoder * decoder,
|
||||||
{
|
{
|
||||||
GstD3D11H265Dec *self = GST_D3D11_H265_DEC (decoder);
|
GstD3D11H265Dec *self = GST_D3D11_H265_DEC (decoder);
|
||||||
|
|
||||||
if (!gst_d3d11_decoder_decide_allocation (decoder, query, self->device,
|
if (!gst_d3d11_decoder_decide_allocation (self->d3d11_decoder,
|
||||||
GST_D3D11_CODEC_H265, self->use_d3d11_output, self->d3d11_decoder))
|
decoder, query)) {
|
||||||
return FALSE;
|
return FALSE;
|
||||||
|
}
|
||||||
|
|
||||||
return GST_VIDEO_DECODER_CLASS (parent_class)->decide_allocation
|
return GST_VIDEO_DECODER_CLASS (parent_class)->decide_allocation
|
||||||
(decoder, query);
|
(decoder, query);
|
||||||
|
@ -452,8 +450,8 @@ gst_d3d11_h265_dec_new_sequence (GstH265Decoder * decoder,
|
||||||
|
|
||||||
gst_video_info_set_format (&info,
|
gst_video_info_set_format (&info,
|
||||||
self->out_format, self->width, self->height);
|
self->out_format, self->width, self->height);
|
||||||
|
GST_VIDEO_INFO_INTERLACE_MODE (&info) = self->interlace_mode;
|
||||||
|
|
||||||
gst_d3d11_decoder_reset (self->d3d11_decoder);
|
|
||||||
if (!gst_d3d11_decoder_configure (self->d3d11_decoder, GST_D3D11_CODEC_H265,
|
if (!gst_d3d11_decoder_configure (self->d3d11_decoder, GST_D3D11_CODEC_H265,
|
||||||
&info, self->coded_width, self->coded_height,
|
&info, self->coded_width, self->coded_height,
|
||||||
/* Additional 4 views margin for zero-copy rendering */
|
/* Additional 4 views margin for zero-copy rendering */
|
||||||
|
@ -821,8 +819,7 @@ gst_d3d11_h265_dec_output_picture (GstH265Decoder * decoder,
|
||||||
* expose our decoder view without copy. In case of reverse playback, however,
|
* expose our decoder view without copy. In case of reverse playback, however,
|
||||||
* we cannot do that since baseclass will store the decoded buffer
|
* we cannot do that since baseclass will store the decoded buffer
|
||||||
* up to gop size but our dpb pool cannot be increased */
|
* up to gop size but our dpb pool cannot be increased */
|
||||||
if (self->use_d3d11_output
|
if (GST_VIDEO_DECODER (self)->input_segment.rate > 0
|
||||||
&& GST_VIDEO_DECODER (self)->input_segment.rate > 0
|
|
||||||
&& gst_d3d11_decoder_can_direct_render (self->d3d11_decoder, view_buffer,
|
&& gst_d3d11_decoder_can_direct_render (self->d3d11_decoder, view_buffer,
|
||||||
GST_MINI_OBJECT_CAST (picture))) {
|
GST_MINI_OBJECT_CAST (picture))) {
|
||||||
direct_rendering = TRUE;
|
direct_rendering = TRUE;
|
||||||
|
|
|
@ -92,8 +92,6 @@ typedef struct _GstD3D11Mpeg2Dec
|
||||||
guint written_buffer_size;
|
guint written_buffer_size;
|
||||||
guint remaining_buffer_size;
|
guint remaining_buffer_size;
|
||||||
guint8 *bitstream_buffer_data;
|
guint8 *bitstream_buffer_data;
|
||||||
|
|
||||||
gboolean use_d3d11_output;
|
|
||||||
} GstD3D11Mpeg2Dec;
|
} GstD3D11Mpeg2Dec;
|
||||||
|
|
||||||
typedef struct _GstD3D11Mpeg2DecClass
|
typedef struct _GstD3D11Mpeg2DecClass
|
||||||
|
@ -310,12 +308,10 @@ gst_d3d11_mpeg2_dec_negotiate (GstVideoDecoder * decoder)
|
||||||
GstD3D11Mpeg2Dec *self = GST_D3D11_MPEG2_DEC (decoder);
|
GstD3D11Mpeg2Dec *self = GST_D3D11_MPEG2_DEC (decoder);
|
||||||
GstMpeg2Decoder *mpeg2dec = GST_MPEG2_DECODER (decoder);
|
GstMpeg2Decoder *mpeg2dec = GST_MPEG2_DECODER (decoder);
|
||||||
|
|
||||||
if (!gst_d3d11_decoder_negotiate (decoder, mpeg2dec->input_state,
|
if (!gst_d3d11_decoder_negotiate (self->d3d11_decoder, decoder,
|
||||||
self->out_format, self->width, self->height,
|
mpeg2dec->input_state, &self->output_state)) {
|
||||||
self->interlaced ? GST_VIDEO_INTERLACE_MODE_MIXED :
|
|
||||||
GST_VIDEO_INTERLACE_MODE_PROGRESSIVE,
|
|
||||||
&self->output_state, &self->use_d3d11_output))
|
|
||||||
return FALSE;
|
return FALSE;
|
||||||
|
}
|
||||||
|
|
||||||
return GST_VIDEO_DECODER_CLASS (parent_class)->negotiate (decoder);
|
return GST_VIDEO_DECODER_CLASS (parent_class)->negotiate (decoder);
|
||||||
}
|
}
|
||||||
|
@ -326,9 +322,10 @@ gst_d3d11_mpeg2_dec_decide_allocation (GstVideoDecoder * decoder,
|
||||||
{
|
{
|
||||||
GstD3D11Mpeg2Dec *self = GST_D3D11_MPEG2_DEC (decoder);
|
GstD3D11Mpeg2Dec *self = GST_D3D11_MPEG2_DEC (decoder);
|
||||||
|
|
||||||
if (!gst_d3d11_decoder_decide_allocation (decoder, query, self->device,
|
if (!gst_d3d11_decoder_decide_allocation (self->d3d11_decoder,
|
||||||
GST_D3D11_CODEC_MPEG2, self->use_d3d11_output, self->d3d11_decoder))
|
decoder, query)) {
|
||||||
return FALSE;
|
return FALSE;
|
||||||
|
}
|
||||||
|
|
||||||
return GST_VIDEO_DECODER_CLASS (parent_class)->decide_allocation
|
return GST_VIDEO_DECODER_CLASS (parent_class)->decide_allocation
|
||||||
(decoder, query);
|
(decoder, query);
|
||||||
|
@ -417,8 +414,9 @@ gst_d3d11_mpeg2_dec_new_sequence (GstMpeg2Decoder * decoder,
|
||||||
|
|
||||||
gst_video_info_set_format (&info,
|
gst_video_info_set_format (&info,
|
||||||
self->out_format, self->width, self->height);
|
self->out_format, self->width, self->height);
|
||||||
|
if (self->interlaced)
|
||||||
|
GST_VIDEO_INFO_INTERLACE_MODE (&info) = GST_VIDEO_INTERLACE_MODE_MIXED;
|
||||||
|
|
||||||
gst_d3d11_decoder_reset (self->d3d11_decoder);
|
|
||||||
if (!gst_d3d11_decoder_configure (self->d3d11_decoder,
|
if (!gst_d3d11_decoder_configure (self->d3d11_decoder,
|
||||||
GST_D3D11_CODEC_MPEG2, &info, self->width, self->height,
|
GST_D3D11_CODEC_MPEG2, &info, self->width, self->height,
|
||||||
NUM_OUTPUT_VIEW)) {
|
NUM_OUTPUT_VIEW)) {
|
||||||
|
@ -892,8 +890,7 @@ gst_d3d11_mpeg2_dec_output_picture (GstMpeg2Decoder * decoder,
|
||||||
* expose our decoder view without copy. In case of reverse playback, however,
|
* expose our decoder view without copy. In case of reverse playback, however,
|
||||||
* we cannot do that since baseclass will store the decoded buffer
|
* we cannot do that since baseclass will store the decoded buffer
|
||||||
* up to gop size but our dpb pool cannot be increased */
|
* up to gop size but our dpb pool cannot be increased */
|
||||||
if (self->use_d3d11_output
|
if (GST_VIDEO_DECODER (self)->input_segment.rate > 0
|
||||||
&& GST_VIDEO_DECODER (self)->input_segment.rate > 0
|
|
||||||
&& gst_d3d11_decoder_can_direct_render (self->d3d11_decoder, view_buffer,
|
&& gst_d3d11_decoder_can_direct_render (self->d3d11_decoder, view_buffer,
|
||||||
GST_MINI_OBJECT_CAST (picture))) {
|
GST_MINI_OBJECT_CAST (picture))) {
|
||||||
direct_rendering = TRUE;
|
direct_rendering = TRUE;
|
||||||
|
|
|
@ -79,8 +79,6 @@ typedef struct _GstD3D11Vp8Dec
|
||||||
|
|
||||||
guint width, height;
|
guint width, height;
|
||||||
GstVideoFormat out_format;
|
GstVideoFormat out_format;
|
||||||
|
|
||||||
gboolean use_d3d11_output;
|
|
||||||
} GstD3D11Vp8Dec;
|
} GstD3D11Vp8Dec;
|
||||||
|
|
||||||
typedef struct _GstD3D11Vp8DecClass
|
typedef struct _GstD3D11Vp8DecClass
|
||||||
|
@ -274,11 +272,10 @@ gst_d3d11_vp8_dec_negotiate (GstVideoDecoder * decoder)
|
||||||
GstD3D11Vp8Dec *self = GST_D3D11_VP8_DEC (decoder);
|
GstD3D11Vp8Dec *self = GST_D3D11_VP8_DEC (decoder);
|
||||||
GstVp8Decoder *vp8dec = GST_VP8_DECODER (decoder);
|
GstVp8Decoder *vp8dec = GST_VP8_DECODER (decoder);
|
||||||
|
|
||||||
if (!gst_d3d11_decoder_negotiate (decoder, vp8dec->input_state,
|
if (!gst_d3d11_decoder_negotiate (self->d3d11_decoder,
|
||||||
self->out_format, self->width, self->height,
|
decoder, vp8dec->input_state, &self->output_state)) {
|
||||||
GST_VIDEO_INTERLACE_MODE_PROGRESSIVE,
|
|
||||||
&self->output_state, &self->use_d3d11_output))
|
|
||||||
return FALSE;
|
return FALSE;
|
||||||
|
}
|
||||||
|
|
||||||
return GST_VIDEO_DECODER_CLASS (parent_class)->negotiate (decoder);
|
return GST_VIDEO_DECODER_CLASS (parent_class)->negotiate (decoder);
|
||||||
}
|
}
|
||||||
|
@ -289,9 +286,10 @@ gst_d3d11_vp8_dec_decide_allocation (GstVideoDecoder * decoder,
|
||||||
{
|
{
|
||||||
GstD3D11Vp8Dec *self = GST_D3D11_VP8_DEC (decoder);
|
GstD3D11Vp8Dec *self = GST_D3D11_VP8_DEC (decoder);
|
||||||
|
|
||||||
if (!gst_d3d11_decoder_decide_allocation (decoder, query, self->device,
|
if (!gst_d3d11_decoder_decide_allocation (self->d3d11_decoder, decoder,
|
||||||
GST_D3D11_CODEC_VP8, self->use_d3d11_output, self->d3d11_decoder))
|
query)) {
|
||||||
return FALSE;
|
return FALSE;
|
||||||
|
}
|
||||||
|
|
||||||
return GST_VIDEO_DECODER_CLASS (parent_class)->decide_allocation
|
return GST_VIDEO_DECODER_CLASS (parent_class)->decide_allocation
|
||||||
(decoder, query);
|
(decoder, query);
|
||||||
|
@ -333,7 +331,6 @@ gst_d3d11_vp8_dec_new_sequence (GstVp8Decoder * decoder,
|
||||||
gst_video_info_set_format (&info,
|
gst_video_info_set_format (&info,
|
||||||
self->out_format, self->width, self->height);
|
self->out_format, self->width, self->height);
|
||||||
|
|
||||||
gst_d3d11_decoder_reset (self->d3d11_decoder);
|
|
||||||
if (!gst_d3d11_decoder_configure (self->d3d11_decoder, GST_D3D11_CODEC_VP8,
|
if (!gst_d3d11_decoder_configure (self->d3d11_decoder, GST_D3D11_CODEC_VP8,
|
||||||
&info, self->width, self->height, NUM_OUTPUT_VIEW)) {
|
&info, self->width, self->height, NUM_OUTPUT_VIEW)) {
|
||||||
GST_ERROR_OBJECT (self, "Failed to create decoder");
|
GST_ERROR_OBJECT (self, "Failed to create decoder");
|
||||||
|
@ -397,8 +394,7 @@ gst_d3d11_vp8_dec_output_picture (GstVp8Decoder * decoder,
|
||||||
* expose our decoder view without copy. In case of reverse playback, however,
|
* expose our decoder view without copy. In case of reverse playback, however,
|
||||||
* we cannot do that since baseclass will store the decoded buffer
|
* we cannot do that since baseclass will store the decoded buffer
|
||||||
* up to gop size but our dpb pool cannot be increased */
|
* up to gop size but our dpb pool cannot be increased */
|
||||||
if (self->use_d3d11_output
|
if (GST_VIDEO_DECODER (self)->input_segment.rate > 0
|
||||||
&& GST_VIDEO_DECODER (self)->input_segment.rate > 0
|
|
||||||
&& gst_d3d11_decoder_can_direct_render (self->d3d11_decoder, view_buffer,
|
&& gst_d3d11_decoder_can_direct_render (self->d3d11_decoder, view_buffer,
|
||||||
GST_MINI_OBJECT_CAST (picture))) {
|
GST_MINI_OBJECT_CAST (picture))) {
|
||||||
direct_rendering = TRUE;
|
direct_rendering = TRUE;
|
||||||
|
|
|
@ -113,8 +113,6 @@ typedef struct _GstD3D11Vp9Dec
|
||||||
GstVP9Profile profile;
|
GstVP9Profile profile;
|
||||||
|
|
||||||
GstVideoFormat out_format;
|
GstVideoFormat out_format;
|
||||||
|
|
||||||
gboolean use_d3d11_output;
|
|
||||||
} GstD3D11Vp9Dec;
|
} GstD3D11Vp9Dec;
|
||||||
|
|
||||||
typedef struct _GstD3D11Vp9DecClass
|
typedef struct _GstD3D11Vp9DecClass
|
||||||
|
@ -312,11 +310,10 @@ gst_d3d11_vp9_dec_negotiate (GstVideoDecoder * decoder)
|
||||||
GstD3D11Vp9Dec *self = GST_D3D11_VP9_DEC (decoder);
|
GstD3D11Vp9Dec *self = GST_D3D11_VP9_DEC (decoder);
|
||||||
GstVp9Decoder *vp9dec = GST_VP9_DECODER (decoder);
|
GstVp9Decoder *vp9dec = GST_VP9_DECODER (decoder);
|
||||||
|
|
||||||
if (!gst_d3d11_decoder_negotiate (decoder, vp9dec->input_state,
|
if (!gst_d3d11_decoder_negotiate (self->d3d11_decoder,
|
||||||
self->out_format, self->width, self->height,
|
decoder, vp9dec->input_state, &self->output_state)) {
|
||||||
GST_VIDEO_INTERLACE_MODE_PROGRESSIVE,
|
|
||||||
&self->output_state, &self->use_d3d11_output))
|
|
||||||
return FALSE;
|
return FALSE;
|
||||||
|
}
|
||||||
|
|
||||||
return GST_VIDEO_DECODER_CLASS (parent_class)->negotiate (decoder);
|
return GST_VIDEO_DECODER_CLASS (parent_class)->negotiate (decoder);
|
||||||
}
|
}
|
||||||
|
@ -327,9 +324,10 @@ gst_d3d11_vp9_dec_decide_allocation (GstVideoDecoder * decoder,
|
||||||
{
|
{
|
||||||
GstD3D11Vp9Dec *self = GST_D3D11_VP9_DEC (decoder);
|
GstD3D11Vp9Dec *self = GST_D3D11_VP9_DEC (decoder);
|
||||||
|
|
||||||
if (!gst_d3d11_decoder_decide_allocation (decoder, query, self->device,
|
if (!gst_d3d11_decoder_decide_allocation (self->d3d11_decoder,
|
||||||
GST_D3D11_CODEC_VP9, self->use_d3d11_output, self->d3d11_decoder))
|
decoder, query)) {
|
||||||
return FALSE;
|
return FALSE;
|
||||||
|
}
|
||||||
|
|
||||||
return GST_VIDEO_DECODER_CLASS (parent_class)->decide_allocation
|
return GST_VIDEO_DECODER_CLASS (parent_class)->decide_allocation
|
||||||
(decoder, query);
|
(decoder, query);
|
||||||
|
@ -396,7 +394,6 @@ gst_d3d11_vp9_dec_new_sequence (GstVp9Decoder * decoder,
|
||||||
gst_video_info_set_format (&info,
|
gst_video_info_set_format (&info,
|
||||||
self->out_format, self->width, self->height);
|
self->out_format, self->width, self->height);
|
||||||
|
|
||||||
gst_d3d11_decoder_reset (self->d3d11_decoder);
|
|
||||||
if (!gst_d3d11_decoder_configure (self->d3d11_decoder, GST_D3D11_CODEC_VP9,
|
if (!gst_d3d11_decoder_configure (self->d3d11_decoder, GST_D3D11_CODEC_VP9,
|
||||||
&info, self->width, self->height, NUM_OUTPUT_VIEW)) {
|
&info, self->width, self->height, NUM_OUTPUT_VIEW)) {
|
||||||
GST_ERROR_OBJECT (self, "Failed to create decoder");
|
GST_ERROR_OBJECT (self, "Failed to create decoder");
|
||||||
|
@ -488,8 +485,7 @@ gst_d3d11_vp9_dec_output_picture (GstVp9Decoder * decoder,
|
||||||
* expose our decoder view without copy. In case of reverse playback, however,
|
* expose our decoder view without copy. In case of reverse playback, however,
|
||||||
* we cannot do that since baseclass will store the decoded buffer
|
* we cannot do that since baseclass will store the decoded buffer
|
||||||
* up to gop size but our dpb pool cannot be increased */
|
* up to gop size but our dpb pool cannot be increased */
|
||||||
if (self->use_d3d11_output
|
if (GST_VIDEO_DECODER (self)->input_segment.rate > 0
|
||||||
&& GST_VIDEO_DECODER (self)->input_segment.rate > 0
|
|
||||||
&& gst_d3d11_decoder_can_direct_render (self->d3d11_decoder, view_buffer,
|
&& gst_d3d11_decoder_can_direct_render (self->d3d11_decoder, view_buffer,
|
||||||
GST_MINI_OBJECT_CAST (picture))) {
|
GST_MINI_OBJECT_CAST (picture))) {
|
||||||
direct_rendering = TRUE;
|
direct_rendering = TRUE;
|
||||||
|
|
Loading…
Reference in a new issue