d3d11convert: Add support for transform

Adding 3D rotation and scale transform support to d3d11convert element

Part-of: <https://gitlab.freedesktop.org/gstreamer/gstreamer/-/merge_requests/5572>
This commit is contained in:
Seungha Yang 2023-10-28 22:08:31 +09:00
parent 1813b4e3d9
commit af31b46992
2 changed files with 608 additions and 12 deletions

View file

@ -9076,6 +9076,20 @@
"type": "GstD3D11AlphaMode", "type": "GstD3D11AlphaMode",
"writable": true "writable": true
}, },
"fov": {
"blurb": "Field of view angle in degrees",
"conditionally-available": false,
"construct": false,
"construct-only": false,
"controllable": false,
"default": "90",
"max": "3.40282e+38",
"min": "0",
"mutable": "playing",
"readable": true,
"type": "gfloat",
"writable": true
},
"gamma-mode": { "gamma-mode": {
"blurb": "Gamma conversion mode", "blurb": "Gamma conversion mode",
"conditionally-available": false, "conditionally-available": false,
@ -9088,6 +9102,30 @@
"type": "GstVideoGammaMode", "type": "GstVideoGammaMode",
"writable": true "writable": true
}, },
"msaa": {
"blurb": "MSAA (Multi-Sampling Anti-Aliasing) level. This value will be ignored if device does not support the specified level for negotiated output format",
"conditionally-available": false,
"construct": false,
"construct-only": false,
"controllable": false,
"default": "disabled (0)",
"mutable": "playing",
"readable": true,
"type": "GstD3D11MSAAMode",
"writable": true
},
"ortho": {
"blurb": "Use orthographic projection",
"conditionally-available": false,
"construct": false,
"construct-only": false,
"controllable": false,
"default": "false",
"mutable": "playing",
"readable": true,
"type": "gboolean",
"writable": true
},
"primaries-mode": { "primaries-mode": {
"blurb": "Primaries conversion mode", "blurb": "Primaries conversion mode",
"conditionally-available": false, "conditionally-available": false,
@ -9100,6 +9138,76 @@
"type": "GstVideoPrimariesMode", "type": "GstVideoPrimariesMode",
"writable": true "writable": true
}, },
"rotation-x": {
"blurb": "x-axis rotation angle in degrees",
"conditionally-available": false,
"construct": false,
"construct-only": false,
"controllable": false,
"default": "0",
"max": "3.40282e+38",
"min": "-3.40282e+38",
"mutable": "playing",
"readable": true,
"type": "gfloat",
"writable": true
},
"rotation-y": {
"blurb": "y-axis rotation angle in degrees",
"conditionally-available": false,
"construct": false,
"construct-only": false,
"controllable": false,
"default": "0",
"max": "3.40282e+38",
"min": "-3.40282e+38",
"mutable": "playing",
"readable": true,
"type": "gfloat",
"writable": true
},
"rotation-z": {
"blurb": "z-axis rotation angle in degrees",
"conditionally-available": false,
"construct": false,
"construct-only": false,
"controllable": false,
"default": "0",
"max": "3.40282e+38",
"min": "-3.40282e+38",
"mutable": "playing",
"readable": true,
"type": "gfloat",
"writable": true
},
"scale-x": {
"blurb": "Scale multiplier for x-axis",
"conditionally-available": false,
"construct": false,
"construct-only": false,
"controllable": false,
"default": "1",
"max": "3.40282e+38",
"min": "-3.40282e+38",
"mutable": "playing",
"readable": true,
"type": "gfloat",
"writable": true
},
"scale-y": {
"blurb": "Scale multiplier for y-axis",
"conditionally-available": false,
"construct": false,
"construct-only": false,
"controllable": false,
"default": "1",
"max": "3.40282e+38",
"min": "-3.40282e+38",
"mutable": "playing",
"readable": true,
"type": "gfloat",
"writable": true
},
"src-alpha-mode": { "src-alpha-mode": {
"blurb": "Applied input alpha mode", "blurb": "Applied input alpha mode",
"conditionally-available": false, "conditionally-available": false,

View file

@ -124,6 +124,11 @@ gst_d3d11_base_convert_sampling_method_to_filter (GstD3D11SamplingMethod method)
#define DEFAULT_PRIMARIES_MODE GST_VIDEO_PRIMARIES_MODE_NONE #define DEFAULT_PRIMARIES_MODE GST_VIDEO_PRIMARIES_MODE_NONE
#define DEFAULT_SAMPLING_METHOD GST_D3D11_SAMPLING_METHOD_BILINEAR #define DEFAULT_SAMPLING_METHOD GST_D3D11_SAMPLING_METHOD_BILINEAR
#define DEFAULT_ALPHA_MODE GST_D3D11_ALPHA_MODE_UNSPECIFIED #define DEFAULT_ALPHA_MODE GST_D3D11_ALPHA_MODE_UNSPECIFIED
#define DEFAULT_ROTATION 0.0f
#define DEFAULT_SCALE 1.0f
#define DEFAULT_FOV 90.0f
#define DEFAULT_ORTHO FALSE
#define DEFAULT_MSAA GST_D3D11_MSAA_DISABLED
struct _GstD3D11BaseConvert struct _GstD3D11BaseConvert
{ {
@ -141,6 +146,9 @@ struct _GstD3D11BaseConvert
gint borders_h; gint borders_h;
gint borders_w; gint borders_w;
GstBuffer *msaa_buf;
GstBufferPool *msaa_pool;
/* Updated by subclass */ /* Updated by subclass */
gboolean add_borders; gboolean add_borders;
gboolean active_add_borders; gboolean active_add_borders;
@ -170,6 +178,19 @@ struct _GstD3D11BaseConvert
GstD3D11AlphaMode src_alpha_mode; GstD3D11AlphaMode src_alpha_mode;
GstD3D11AlphaMode dst_alpha_mode; GstD3D11AlphaMode dst_alpha_mode;
/* transform */
gfloat fov;
gboolean ortho;
gfloat rotation_x;
gfloat rotation_y;
gfloat rotation_z;
gfloat scale_x;
gfloat scale_y;
gboolean transform_updated;
GstD3D11MSAAMode msaa;
gboolean msaa_updated;
SRWLOCK lock; SRWLOCK lock;
}; };
@ -414,6 +435,7 @@ gst_d3d11_base_convert_class_init (GstD3D11BaseConvertClass * klass)
(GstPluginAPIFlags) 0); (GstPluginAPIFlags) 0);
gst_type_mark_as_plugin_api (GST_TYPE_D3D11_ALPHA_MODE, gst_type_mark_as_plugin_api (GST_TYPE_D3D11_ALPHA_MODE,
(GstPluginAPIFlags) 0); (GstPluginAPIFlags) 0);
gst_type_mark_as_plugin_api (GST_TYPE_D3D11_MSAA_MODE, (GstPluginAPIFlags) 0);
} }
static void static void
@ -426,6 +448,14 @@ gst_d3d11_base_convert_init (GstD3D11BaseConvert * self)
self->sampling_method = self->active_sampling_method = self->sampling_method = self->active_sampling_method =
DEFAULT_SAMPLING_METHOD; DEFAULT_SAMPLING_METHOD;
self->src_alpha_mode = self->dst_alpha_mode = DEFAULT_ALPHA_MODE; self->src_alpha_mode = self->dst_alpha_mode = DEFAULT_ALPHA_MODE;
self->fov = DEFAULT_FOV;
self->ortho = DEFAULT_ORTHO;
self->rotation_x = DEFAULT_ROTATION;
self->rotation_y = DEFAULT_ROTATION;
self->rotation_z = DEFAULT_ROTATION;
self->scale_x = DEFAULT_SCALE;
self->scale_y = DEFAULT_SCALE;
self->msaa = DEFAULT_MSAA;
} }
static void static void
@ -434,6 +464,11 @@ gst_d3d11_base_convert_dispose (GObject * object)
GstD3D11BaseConvert *self = GST_D3D11_BASE_CONVERT (object); GstD3D11BaseConvert *self = GST_D3D11_BASE_CONVERT (object);
gst_clear_object (&self->converter); gst_clear_object (&self->converter);
gst_clear_buffer (&self->msaa_buf);
if (self->msaa_pool) {
gst_buffer_pool_set_active (self->msaa_pool, FALSE);
gst_clear_object (&self->msaa_pool);
}
G_OBJECT_CLASS (parent_class)->dispose (object); G_OBJECT_CLASS (parent_class)->dispose (object);
} }
@ -1607,6 +1642,103 @@ gst_d3d11_base_convert_needs_color_convert (GstD3D11BaseConvert * self,
return FALSE; return FALSE;
} }
static void
gst_d3d11_base_convert_setup_msaa_texture (GstD3D11BaseConvert * self,
GstD3D11Device * device, const GstVideoInfo * info, GstCaps * caps)
{
ID3D11Device *device_handle;
UINT quality_levels = 0;
UINT sample_count = 1;
GstD3D11Format device_format;
DXGI_FORMAT dxgi_format;
GstD3D11AllocationParams *params;
GstStructure *config;
HRESULT hr;
gst_clear_buffer (&self->msaa_buf);
if (self->msaa_pool) {
gst_buffer_pool_set_active (self->msaa_pool, FALSE);
gst_clear_object (&self->msaa_pool);
}
switch (self->msaa) {
case GST_D3D11_MSAA_2X:
sample_count = 2;
break;
case GST_D3D11_MSAA_4X:
sample_count = 4;
break;
case GST_D3D11_MSAA_8X:
sample_count = 8;
break;
default:
return;
}
if (!gst_d3d11_device_get_format (device, GST_VIDEO_INFO_FORMAT (info),
&device_format)) {
GST_WARNING_OBJECT (self, "Couldn't get device format");
return;
}
if (device_format.dxgi_format != DXGI_FORMAT_UNKNOWN)
dxgi_format = device_format.dxgi_format;
else
dxgi_format = device_format.resource_format[0];
device_handle = gst_d3d11_device_get_device_handle (device);
while (sample_count > 1) {
hr = device_handle->CheckMultisampleQualityLevels (dxgi_format,
sample_count, &quality_levels);
if (gst_d3d11_result (hr, device) && quality_levels > 0)
break;
sample_count = sample_count / 2;
};
if (sample_count <= 1 || quality_levels == 0) {
GST_DEBUG_OBJECT (self, "Device does not support MSAA for DXGI format %d",
dxgi_format);
return;
}
self->msaa_pool = gst_d3d11_buffer_pool_new (device);
config = gst_buffer_pool_get_config (self->msaa_pool);
params = gst_d3d11_allocation_params_new (device, info,
GST_D3D11_ALLOCATION_FLAG_DEFAULT, D3D11_BIND_RENDER_TARGET, 0);
for (guint i = 0; i < GST_VIDEO_INFO_N_PLANES (info); i++) {
params->desc[i].SampleDesc.Count = sample_count;
params->desc[i].SampleDesc.Quality = quality_levels - 1;
}
gst_buffer_pool_config_set_d3d11_allocation_params (config, params);
gst_d3d11_allocation_params_free (params);
gst_buffer_pool_config_set_params (config, caps, info->size, 0, 0);
if (!gst_buffer_pool_set_config (self->msaa_pool, config)) {
GST_ERROR_OBJECT (self, "Couldn't set pool config");
gst_clear_object (&self->msaa_pool);
return;
}
if (!gst_buffer_pool_set_active (self->msaa_pool, TRUE)) {
GST_ERROR_OBJECT (self, "Pool active failed");
gst_clear_object (&self->msaa_pool);
return;
}
gst_buffer_pool_acquire_buffer (self->msaa_pool, &self->msaa_buf, nullptr);
if (!self->msaa_buf) {
GST_ERROR_OBJECT (self, "Couldn't acquire MSAA buffer");
gst_buffer_pool_set_active (self->msaa_pool, FALSE);
gst_clear_object (&self->msaa_pool);
return;
}
GST_DEBUG_OBJECT (self, "MSAA %dx buffer is configured", sample_count);
}
static gboolean static gboolean
gst_d3d11_base_convert_set_info (GstD3D11BaseFilter * filter, gst_d3d11_base_convert_set_info (GstD3D11BaseFilter * filter,
GstCaps * incaps, GstVideoInfo * in_info, GstCaps * outcaps, GstCaps * incaps, GstVideoInfo * in_info, GstCaps * outcaps,
@ -1617,7 +1749,9 @@ gst_d3d11_base_convert_set_info (GstD3D11BaseFilter * filter,
gint border_offset_x = 0; gint border_offset_x = 0;
gint border_offset_y = 0; gint border_offset_y = 0;
gboolean need_flip = FALSE; gboolean need_flip = FALSE;
gboolean need_transform = FALSE;
gint in_width, in_height, in_par_n, in_par_d; gint in_width, in_height, in_par_n, in_par_d;
gint out_width, out_height;
GstStructure *config; GstStructure *config;
GstD3D11SRWLockGuard lk (&self->lock); GstD3D11SRWLockGuard lk (&self->lock);
@ -1627,6 +1761,9 @@ gst_d3d11_base_convert_set_info (GstD3D11BaseFilter * filter,
self->active_primaries_mode = self->primaries_mode; self->active_primaries_mode = self->primaries_mode;
self->active_sampling_method = self->sampling_method; self->active_sampling_method = self->sampling_method;
self->transform_updated = FALSE;
self->msaa_updated = FALSE;
GST_DEBUG_OBJECT (self, "method %d, add-borders %d, gamma-mode %d, " GST_DEBUG_OBJECT (self, "method %d, add-borders %d, gamma-mode %d, "
"primaries-mode %d, sampling %d", self->active_method, "primaries-mode %d, sampling %d", self->active_method,
self->active_add_borders, self->active_gamma_mode, self->active_add_borders, self->active_gamma_mode,
@ -1635,7 +1772,10 @@ gst_d3d11_base_convert_set_info (GstD3D11BaseFilter * filter,
if (self->active_method != GST_VIDEO_ORIENTATION_IDENTITY) if (self->active_method != GST_VIDEO_ORIENTATION_IDENTITY)
need_flip = TRUE; need_flip = TRUE;
if (!need_flip && gst_caps_is_equal (incaps, outcaps)) { need_transform = gst_d3d11_need_transform (self->rotation_x, self->rotation_y,
self->rotation_z, self->scale_x, self->scale_y);
if (!need_flip && !need_transform && gst_caps_is_equal (incaps, outcaps)) {
self->same_caps = TRUE; self->same_caps = TRUE;
} else { } else {
self->same_caps = FALSE; self->same_caps = FALSE;
@ -1711,7 +1851,7 @@ gst_d3d11_base_convert_set_info (GstD3D11BaseFilter * filter,
if (in_width == out_info->width && in_height == out_info->height if (in_width == out_info->width && in_height == out_info->height
&& in_info->finfo == out_info->finfo && self->borders_w == 0 && && in_info->finfo == out_info->finfo && self->borders_w == 0 &&
self->borders_h == 0 && !need_flip && self->borders_h == 0 && !need_flip && !need_transform &&
!gst_d3d11_base_convert_needs_color_convert (self, in_info, out_info)) { !gst_d3d11_base_convert_needs_color_convert (self, in_info, out_info)) {
self->same_caps = TRUE; self->same_caps = TRUE;
} }
@ -1770,17 +1910,39 @@ gst_d3d11_base_convert_set_info (GstD3D11BaseFilter * filter,
self->out_rect.bottom = GST_VIDEO_INFO_HEIGHT (out_info); self->out_rect.bottom = GST_VIDEO_INFO_HEIGHT (out_info);
} }
out_width = self->out_rect.right - self->out_rect.left;
out_height = self->out_rect.bottom - self->out_rect.top;
g_object_set (self->converter, "dest-x", (gint) self->out_rect.left, g_object_set (self->converter, "dest-x", (gint) self->out_rect.left,
"dest-y", (gint) self->out_rect.top, "dest-y", (gint) self->out_rect.top,
"dest-width", (gint) (self->out_rect.right - self->out_rect.left), "dest-width", out_width, "dest-height", out_height, nullptr);
"dest-height", (gint) (self->out_rect.bottom - self->out_rect.top),
"video-direction", self->active_method, nullptr);
if (self->borders_w > 0 || self->borders_h > 0) { if (need_transform) {
gfloat transform_matrix[16];
GST_DEBUG_OBJECT (self, "Applying custom transform");
gst_d3d11_calculate_transform_matrix (self->active_method,
(gfloat) out_width, (gfloat) out_height, self->fov,
self->ortho, self->rotation_x, self->rotation_y,
self->rotation_z, self->scale_x, self->scale_y, transform_matrix);
g_object_set (self->converter,
"video-direction", GST_VIDEO_ORIENTATION_CUSTOM, nullptr);
gst_d3d11_converter_set_transform_matrix (self->converter,
transform_matrix);
} else {
g_object_set (self->converter,
"video-direction", self->active_method, nullptr);
}
if (self->borders_w > 0 || self->borders_h > 0 || need_transform) {
g_object_set (self->converter, "fill-border", TRUE, "border-color", g_object_set (self->converter, "fill-border", TRUE, "border-color",
self->border_color, nullptr); self->border_color, nullptr);
} }
gst_d3d11_base_convert_setup_msaa_texture (self, filter->device,
out_info, outcaps);
return TRUE; return TRUE;
} }
@ -2063,7 +2225,8 @@ gst_d3d11_base_convert_before_transform (GstBaseTransform * trans,
self->add_borders != self->active_add_borders || self->add_borders != self->active_add_borders ||
self->gamma_mode != self->active_gamma_mode || self->gamma_mode != self->active_gamma_mode ||
self->primaries_mode != self->active_primaries_mode || self->primaries_mode != self->active_primaries_mode ||
self->sampling_method != self->active_sampling_method) { self->sampling_method != self->active_sampling_method ||
self->transform_updated || self->msaa_updated) {
update = TRUE; update = TRUE;
} }
ReleaseSRWLockExclusive (&self->lock); ReleaseSRWLockExclusive (&self->lock);
@ -2099,8 +2262,12 @@ gst_d3d11_base_convert_transform (GstBaseTransform * trans,
GstBuffer * inbuf, GstBuffer * outbuf) GstBuffer * inbuf, GstBuffer * outbuf)
{ {
GstD3D11BaseConvert *self = GST_D3D11_BASE_CONVERT (trans); GstD3D11BaseConvert *self = GST_D3D11_BASE_CONVERT (trans);
GstD3D11BaseFilter *filter = GST_D3D11_BASE_FILTER (trans);
RECT in_rect; RECT in_rect;
GstVideoCropMeta *crop_meta; GstVideoCropMeta *crop_meta;
GstD3D11Device *device = filter->device;
GstBuffer *target_buf;
GstFlowReturn ret = GST_FLOW_OK;
crop_meta = gst_buffer_get_video_crop_meta (inbuf); crop_meta = gst_buffer_get_video_crop_meta (inbuf);
if (crop_meta) { if (crop_meta) {
@ -2126,13 +2293,64 @@ gst_d3d11_base_convert_transform (GstBaseTransform * trans,
"src-height", (gint) in_rect.bottom - in_rect.top, nullptr); "src-height", (gint) in_rect.bottom - in_rect.top, nullptr);
} }
if (!gst_d3d11_converter_convert_buffer (self->converter, inbuf, outbuf)) { target_buf = self->msaa_buf ? self->msaa_buf : outbuf;
GST_ELEMENT_ERROR (self, CORE, FAILED, (nullptr),
("Couldn't convert texture")); gst_d3d11_device_lock (device);
return GST_FLOW_ERROR; if (!gst_d3d11_converter_convert_buffer_unlocked (self->converter,
inbuf, target_buf)) {
ret = GST_FLOW_ERROR;
} }
return GST_FLOW_OK; if (ret == GST_FLOW_OK && target_buf == self->msaa_buf) {
guint num_memory = gst_buffer_n_memory (self->msaa_buf);
ID3D11DeviceContext *context =
gst_d3d11_device_get_device_context_handle (device);
for (guint i = 0; i < num_memory; i++) {
GstMemory *src_mem;
GstMemory *dst_mem;
GstD3D11Memory *dst_dmem;
GstMapInfo src_map, dst_map;
guint subresource_idx;
D3D11_TEXTURE2D_DESC desc;
src_mem = gst_buffer_peek_memory (self->msaa_buf, i);
dst_mem = gst_buffer_peek_memory (outbuf, i);
dst_dmem = GST_D3D11_MEMORY_CAST (dst_mem);
subresource_idx = gst_d3d11_memory_get_subresource_index (dst_dmem);
gst_d3d11_memory_get_texture_desc (dst_dmem, &desc);
if (!gst_memory_map (src_mem, &src_map,
(GstMapFlags) (GST_MAP_D3D11 | GST_MAP_READ))) {
GST_ERROR_OBJECT (self, "Couldn't map src mem");
ret = GST_FLOW_ERROR;
break;
}
if (!gst_memory_map (dst_mem, &dst_map,
(GstMapFlags) (GST_MAP_D3D11 | GST_MAP_WRITE))) {
GST_ERROR_OBJECT (self, "Couldn't map dst mem");
gst_memory_unmap (src_mem, &src_map);
ret = GST_FLOW_ERROR;
break;
}
context->ResolveSubresource ((ID3D11Resource *) dst_map.data,
subresource_idx, (ID3D11Resource *) src_map.data, 0, desc.Format);
gst_memory_unmap (dst_mem, &dst_map);
gst_memory_unmap (src_mem, &src_map);
}
}
gst_d3d11_device_unlock (device);
if (ret != GST_FLOW_OK) {
GST_ELEMENT_ERROR (self, CORE, FAILED, (nullptr),
("Couldn't convert texture"));
}
return ret;
} }
static void static void
@ -2266,6 +2484,106 @@ gst_d3d11_base_convert_set_dst_alpha_mode (GstD3D11BaseConvert * self,
} }
} }
static void
gst_d3d11_base_convert_set_fov (GstD3D11BaseConvert * self, gfloat fov)
{
GstD3D11SRWLockGuard lk (&self->lock);
if (self->fov != fov) {
self->fov = fov;
self->transform_updated = TRUE;
gst_base_transform_reconfigure_src (GST_BASE_TRANSFORM_CAST (self));
}
}
static void
gst_d3d11_base_convert_set_ortho (GstD3D11BaseConvert * self, gboolean ortho)
{
GstD3D11SRWLockGuard lk (&self->lock);
if (self->ortho != ortho) {
self->ortho = ortho;
self->transform_updated = TRUE;
gst_base_transform_reconfigure_src (GST_BASE_TRANSFORM_CAST (self));
}
}
static void
gst_d3d11_base_convert_set_rotation_x (GstD3D11BaseConvert * self,
gfloat rotation_x)
{
GstD3D11SRWLockGuard lk (&self->lock);
if (self->rotation_x != rotation_x) {
self->rotation_x = rotation_x;
self->transform_updated = TRUE;
gst_base_transform_reconfigure_src (GST_BASE_TRANSFORM_CAST (self));
}
}
static void
gst_d3d11_base_convert_set_rotation_y (GstD3D11BaseConvert * self,
gfloat rotation_y)
{
GstD3D11SRWLockGuard lk (&self->lock);
if (self->rotation_y != rotation_y) {
self->rotation_y = rotation_y;
self->transform_updated = TRUE;
gst_base_transform_reconfigure_src (GST_BASE_TRANSFORM_CAST (self));
}
}
static void
gst_d3d11_base_convert_set_rotation_z (GstD3D11BaseConvert * self,
gfloat rotation_z)
{
GstD3D11SRWLockGuard lk (&self->lock);
if (self->rotation_z != rotation_z) {
self->rotation_z = rotation_z;
self->transform_updated = TRUE;
gst_base_transform_reconfigure_src (GST_BASE_TRANSFORM_CAST (self));
}
}
static void
gst_d3d11_base_convert_set_scale_x (GstD3D11BaseConvert * self, gfloat scale_x)
{
GstD3D11SRWLockGuard lk (&self->lock);
if (self->scale_x != scale_x) {
self->scale_x = scale_x;
self->transform_updated = TRUE;
gst_base_transform_reconfigure_src (GST_BASE_TRANSFORM_CAST (self));
}
}
static void
gst_d3d11_base_convert_set_scale_y (GstD3D11BaseConvert * self, gfloat scale_y)
{
GstD3D11SRWLockGuard lk (&self->lock);
if (self->scale_y != scale_y) {
self->scale_y = scale_y;
self->transform_updated = TRUE;
gst_base_transform_reconfigure_src (GST_BASE_TRANSFORM_CAST (self));
}
}
static void
gst_d3d11_base_convert_set_msaa (GstD3D11BaseConvert * self,
GstD3D11MSAAMode msaa)
{
GstD3D11SRWLockGuard lk (&self->lock);
if (self->msaa != msaa) {
self->msaa = msaa;
self->msaa_updated = TRUE;
gst_base_transform_reconfigure_src (GST_BASE_TRANSFORM_CAST (self));
}
}
/** /**
* SECTION:element-d3d11convert * SECTION:element-d3d11convert
* @title: d3d11convert * @title: d3d11convert
@ -2300,6 +2618,14 @@ enum
PROP_CONVERT_PRIMARIES_MODE, PROP_CONVERT_PRIMARIES_MODE,
PROP_CONVERT_SRC_ALPHA_MODE, PROP_CONVERT_SRC_ALPHA_MODE,
PROP_CONVERT_DEST_ALPHA_MODE, PROP_CONVERT_DEST_ALPHA_MODE,
PROP_CONVERT_FOV,
PROP_CONVERT_ORTHO,
PROP_CONVERT_ROTATION_X,
PROP_CONVERT_ROTATION_Y,
PROP_CONVERT_ROTATION_Z,
PROP_CONVERT_SCALE_X,
PROP_CONVERT_SCALE_Y,
PROP_CONVERT_MSAA,
}; };
struct _GstD3D11Convert struct _GstD3D11Convert
@ -2427,6 +2753,119 @@ gst_d3d11_convert_class_init (GstD3D11ConvertClass * klass)
(GParamFlags) (GST_PARAM_MUTABLE_PLAYING | (GParamFlags) (GST_PARAM_MUTABLE_PLAYING |
G_PARAM_READWRITE | G_PARAM_STATIC_STRINGS))); G_PARAM_READWRITE | G_PARAM_STATIC_STRINGS)));
/**
* GstD3D11Convert:fov:
*
* Field of view angle in degrees
*
* Since: 1.24
*/
g_object_class_install_property (gobject_class, PROP_CONVERT_FOV,
g_param_spec_float ("fov", "Fov",
"Field of view angle in degrees",
0, G_MAXFLOAT, DEFAULT_FOV,
(GParamFlags) (GST_PARAM_MUTABLE_PLAYING |
G_PARAM_READWRITE | G_PARAM_STATIC_STRINGS)));
/**
* GstD3D11Convert:ortho:
*
* Use orthographic projection
*
* Since: 1.24
*/
g_object_class_install_property (gobject_class, PROP_CONVERT_ORTHO,
g_param_spec_boolean ("ortho", "Orthographic",
"Use orthographic projection", DEFAULT_ORTHO,
(GParamFlags) (GST_PARAM_MUTABLE_PLAYING |
G_PARAM_READWRITE | G_PARAM_STATIC_STRINGS)));
/**
* GstD3D11Convert:rotation-x:
*
* x-axis rotation angle to be applied prior to "video-direction"
*
* Since: 1.24
*/
g_object_class_install_property (gobject_class, PROP_CONVERT_ROTATION_X,
g_param_spec_float ("rotation-x", "Rotation X",
"x-axis rotation angle in degrees",
-G_MAXFLOAT, G_MAXFLOAT, DEFAULT_ROTATION,
(GParamFlags) (GST_PARAM_MUTABLE_PLAYING |
G_PARAM_READWRITE | G_PARAM_STATIC_STRINGS)));
/**
* GstD3D11Convert:rotation-y:
*
* y-axis rotation angle to be applied prior to "video-direction"
*
* Since: 1.24
*/
g_object_class_install_property (gobject_class, PROP_CONVERT_ROTATION_Y,
g_param_spec_float ("rotation-y", "Rotation Y",
"y-axis rotation angle in degrees",
-G_MAXFLOAT, G_MAXFLOAT, DEFAULT_ROTATION,
(GParamFlags) (GST_PARAM_MUTABLE_PLAYING |
G_PARAM_READWRITE | G_PARAM_STATIC_STRINGS)));
/**
* GstD3D11Convert:rotation-z:
*
* z-axis rotation angle to be applied prior to "video-direction"
*
* Since: 1.24
*/
g_object_class_install_property (gobject_class, PROP_CONVERT_ROTATION_Z,
g_param_spec_float ("rotation-z", "Rotation Z",
"z-axis rotation angle in degrees",
-G_MAXFLOAT, G_MAXFLOAT, DEFAULT_ROTATION,
(GParamFlags) (GST_PARAM_MUTABLE_PLAYING |
G_PARAM_READWRITE | G_PARAM_STATIC_STRINGS)));
/**
* GstD3D11Convert:scale-x:
*
* Scale multiplier for x-axis
*
* Since: 1.24
*/
g_object_class_install_property (gobject_class, PROP_CONVERT_SCALE_X,
g_param_spec_float ("scale-x", "Scale X",
"Scale multiplier for x-axis",
-G_MAXFLOAT, G_MAXFLOAT, DEFAULT_SCALE,
(GParamFlags) (GST_PARAM_MUTABLE_PLAYING |
G_PARAM_READWRITE | G_PARAM_STATIC_STRINGS)));
/**
* GstD3D11Convert:scale-y:
*
* Scale multiplier for y-axis
*
* Since: 1.24
*/
g_object_class_install_property (gobject_class, PROP_CONVERT_SCALE_Y,
g_param_spec_float ("scale-y", "Scale Y",
"Scale multiplier for y-axis",
-G_MAXFLOAT, G_MAXFLOAT, DEFAULT_SCALE,
(GParamFlags) (GST_PARAM_MUTABLE_PLAYING |
G_PARAM_READWRITE | G_PARAM_STATIC_STRINGS)));
/**
* GstD3D11Convert:msaa:
*
* MSAA (Multi-Sampling Anti-Aliasing) level
*
* Since: 1.24
*/
g_object_class_install_property (gobject_class, PROP_CONVERT_MSAA,
g_param_spec_enum ("msaa", "MSAA",
"MSAA (Multi-Sampling Anti-Aliasing) level. "
"This value will be ignored if device does not support the specified "
"level for negotiated output format",
GST_TYPE_D3D11_MSAA_MODE, DEFAULT_MSAA,
(GParamFlags) (GST_PARAM_MUTABLE_PLAYING |
G_PARAM_READWRITE | G_PARAM_STATIC_STRINGS)));
gst_element_class_set_static_metadata (element_class, gst_element_class_set_static_metadata (element_class,
"Direct3D11 Converter", "Direct3D11 Converter",
"Filter/Converter/Scaler/Effect/Video/Hardware", "Filter/Converter/Scaler/Effect/Video/Hardware",
@ -2476,6 +2915,31 @@ gst_d3d11_convert_set_property (GObject * object, guint prop_id,
gst_d3d11_base_convert_set_dst_alpha_mode (base, gst_d3d11_base_convert_set_dst_alpha_mode (base,
(GstD3D11AlphaMode) g_value_get_enum (value)); (GstD3D11AlphaMode) g_value_get_enum (value));
break; break;
case PROP_CONVERT_FOV:
gst_d3d11_base_convert_set_fov (base, g_value_get_float (value));
break;
case PROP_CONVERT_ORTHO:
gst_d3d11_base_convert_set_ortho (base, g_value_get_boolean (value));
break;
case PROP_CONVERT_ROTATION_X:
gst_d3d11_base_convert_set_rotation_x (base, g_value_get_float (value));
break;
case PROP_CONVERT_ROTATION_Y:
gst_d3d11_base_convert_set_rotation_y (base, g_value_get_float (value));
break;
case PROP_CONVERT_ROTATION_Z:
gst_d3d11_base_convert_set_rotation_z (base, g_value_get_float (value));
break;
case PROP_CONVERT_SCALE_X:
gst_d3d11_base_convert_set_scale_x (base, g_value_get_float (value));
break;
case PROP_CONVERT_SCALE_Y:
gst_d3d11_base_convert_set_scale_y (base, g_value_get_float (value));
break;
case PROP_CONVERT_MSAA:
gst_d3d11_base_convert_set_msaa (base,
(GstD3D11MSAAMode) g_value_get_enum (value));
break;
default: default:
G_OBJECT_WARN_INVALID_PROPERTY_ID (object, prop_id, pspec); G_OBJECT_WARN_INVALID_PROPERTY_ID (object, prop_id, pspec);
break; break;
@ -2510,6 +2974,30 @@ gst_d3d11_convert_get_property (GObject * object, guint prop_id,
case PROP_CONVERT_DEST_ALPHA_MODE: case PROP_CONVERT_DEST_ALPHA_MODE:
g_value_set_enum (value, base->dst_alpha_mode); g_value_set_enum (value, base->dst_alpha_mode);
break; break;
case PROP_CONVERT_FOV:
g_value_set_float (value, base->fov);
break;
case PROP_CONVERT_ORTHO:
g_value_set_boolean (value, base->ortho);
break;
case PROP_CONVERT_ROTATION_X:
g_value_set_float (value, base->rotation_x);
break;
case PROP_CONVERT_ROTATION_Y:
g_value_set_float (value, base->rotation_y);
break;
case PROP_CONVERT_ROTATION_Z:
g_value_set_float (value, base->rotation_z);
break;
case PROP_CONVERT_SCALE_X:
g_value_set_float (value, base->scale_x);
break;
case PROP_CONVERT_SCALE_Y:
g_value_set_float (value, base->scale_y);
break;
case PROP_CONVERT_MSAA:
g_value_set_enum (value, base->msaa);
break;
default: default:
G_OBJECT_WARN_INVALID_PROPERTY_ID (object, prop_id, pspec); G_OBJECT_WARN_INVALID_PROPERTY_ID (object, prop_id, pspec);
break; break;