glupload/download/convert: provide transform_caps functions

Allows finer grain decisions about formats and features at each
stage of the pipeline.
Also provide propose_allocation for glupload besed on the supported
methods.
This commit is contained in:
Matthew Waters 2015-02-03 00:11:06 +11:00
parent 7d46357627
commit c06715bde6
12 changed files with 747 additions and 317 deletions

View file

@ -139,6 +139,8 @@ gst_glimage_sink_change_state (GstElement * element, GstStateChange transition);
static void gst_glimage_sink_get_times (GstBaseSink * bsink, GstBuffer * buf, static void gst_glimage_sink_get_times (GstBaseSink * bsink, GstBuffer * buf,
GstClockTime * start, GstClockTime * end); GstClockTime * start, GstClockTime * end);
static gboolean gst_glimage_sink_set_caps (GstBaseSink * bsink, GstCaps * caps); static gboolean gst_glimage_sink_set_caps (GstBaseSink * bsink, GstCaps * caps);
static GstCaps *gst_glimage_sink_get_caps (GstBaseSink * bsink,
GstCaps * filter);
static GstFlowReturn gst_glimage_sink_prepare (GstBaseSink * bsink, static GstFlowReturn gst_glimage_sink_prepare (GstBaseSink * bsink,
GstBuffer * buf); GstBuffer * buf);
static GstFlowReturn gst_glimage_sink_show_frame (GstVideoSink * bsink, static GstFlowReturn gst_glimage_sink_show_frame (GstVideoSink * bsink,
@ -353,6 +355,7 @@ gst_glimage_sink_class_init (GstGLImageSinkClass * klass)
gstelement_class->set_context = gst_glimage_sink_set_context; gstelement_class->set_context = gst_glimage_sink_set_context;
gstbasesink_class->query = GST_DEBUG_FUNCPTR (gst_glimage_sink_query); gstbasesink_class->query = GST_DEBUG_FUNCPTR (gst_glimage_sink_query);
gstbasesink_class->set_caps = gst_glimage_sink_set_caps; gstbasesink_class->set_caps = gst_glimage_sink_set_caps;
gstbasesink_class->get_caps = gst_glimage_sink_get_caps;
gstbasesink_class->get_times = gst_glimage_sink_get_times; gstbasesink_class->get_times = gst_glimage_sink_get_times;
gstbasesink_class->prepare = gst_glimage_sink_prepare; gstbasesink_class->prepare = gst_glimage_sink_prepare;
gstbasesink_class->propose_allocation = gst_glimage_sink_propose_allocation; gstbasesink_class->propose_allocation = gst_glimage_sink_propose_allocation;
@ -764,6 +767,80 @@ gst_glimage_sink_get_times (GstBaseSink * bsink, GstBuffer * buf,
} }
} }
/* copies the given caps */
static GstCaps *
gst_glimage_sink_caps_remove_format_info (GstCaps * caps)
{
GstStructure *st;
GstCapsFeatures *f;
gint i, n;
GstCaps *res;
res = gst_caps_new_empty ();
n = gst_caps_get_size (caps);
for (i = 0; i < n; i++) {
st = gst_caps_get_structure (caps, i);
f = gst_caps_get_features (caps, i);
/* If this is already expressed by the existing caps
* skip this structure */
if (i > 0 && gst_caps_is_subset_structure_full (res, st, f))
continue;
st = gst_structure_copy (st);
/* Only remove format info for the cases when we can actually convert */
if (!gst_caps_features_is_any (f)
&& gst_caps_features_is_equal (f,
GST_CAPS_FEATURES_MEMORY_SYSTEM_MEMORY))
gst_structure_remove_fields (st, "format", "colorimetry", "chroma-site",
NULL);
gst_structure_remove_fields (st, "width", "height", NULL);
gst_caps_append_structure_full (res, st, gst_caps_features_copy (f));
}
return res;
}
static GstCaps *
gst_glimage_sink_get_caps (GstBaseSink * bsink, GstCaps * filter)
{
GstGLImageSink *gl_sink = GST_GLIMAGE_SINK (bsink);
GstCaps *tmp = NULL;
GstCaps *result = NULL;
tmp = gst_caps_from_string ("video/x-raw(memory:GLMemory),format=RGBA");
result = gst_glimage_sink_caps_remove_format_info (tmp);
gst_caps_unref (tmp);
tmp = result;
GST_DEBUG_OBJECT (bsink, "remove format returned caps %" GST_PTR_FORMAT, tmp);
result =
gst_gl_color_convert_transform_caps (gl_sink->context, GST_PAD_SRC, tmp,
NULL);
gst_caps_unref (tmp);
tmp = result;
result =
gst_gl_upload_transform_caps (gl_sink->context, GST_PAD_SRC, tmp, NULL);
gst_caps_unref (tmp);
tmp = result;
GST_DEBUG_OBJECT (bsink, "transfer returned caps %" GST_PTR_FORMAT, tmp);
if (filter) {
result = gst_caps_intersect_full (filter, tmp, GST_CAPS_INTERSECT_FIRST);
gst_caps_unref (tmp);
} else {
result = tmp;
}
GST_DEBUG_OBJECT (bsink, "returning caps: %" GST_PTR_FORMAT, result);
return result;
}
static gboolean static gboolean
gst_glimage_sink_set_caps (GstBaseSink * bsink, GstCaps * caps) gst_glimage_sink_set_caps (GstBaseSink * bsink, GstCaps * caps)
{ {
@ -1083,11 +1160,6 @@ gst_glimage_sink_propose_allocation (GstBaseSink * bsink, GstQuery * query)
GstCaps *caps; GstCaps *caps;
guint size; guint size;
gboolean need_pool; gboolean need_pool;
GstStructure *gl_context;
gchar *platform, *gl_apis;
gpointer handle;
GstAllocator *allocator = NULL;
GstAllocationParams params;
if (!_ensure_gl_setup (glimage_sink)) if (!_ensure_gl_setup (glimage_sink))
return FALSE; return FALSE;
@ -1140,45 +1212,11 @@ gst_glimage_sink_propose_allocation (GstBaseSink * bsink, GstQuery * query)
gst_object_unref (pool); gst_object_unref (pool);
} }
/* we also support various metadata */ gst_gl_upload_propose_allocation (glimage_sink->upload, NULL, query);
gst_query_add_allocation_meta (query, GST_VIDEO_META_API_TYPE, 0);
if (glimage_sink->context->gl_vtable->FenceSync) if (glimage_sink->context->gl_vtable->FenceSync)
gst_query_add_allocation_meta (query, GST_GL_SYNC_META_API_TYPE, 0); gst_query_add_allocation_meta (query, GST_GL_SYNC_META_API_TYPE, 0);
gl_apis =
gst_gl_api_to_string (gst_gl_context_get_gl_api (glimage_sink->context));
platform =
gst_gl_platform_to_string (gst_gl_context_get_gl_platform
(glimage_sink->context));
handle = (gpointer) gst_gl_context_get_gl_context (glimage_sink->context);
gl_context =
gst_structure_new ("GstVideoGLTextureUploadMeta", "gst.gl.GstGLContext",
GST_GL_TYPE_CONTEXT, glimage_sink->context, "gst.gl.context.handle",
G_TYPE_POINTER, handle, "gst.gl.context.type", G_TYPE_STRING, platform,
"gst.gl.context.apis", G_TYPE_STRING, gl_apis, NULL);
gst_query_add_allocation_meta (query,
GST_VIDEO_GL_TEXTURE_UPLOAD_META_API_TYPE, gl_context);
g_free (gl_apis);
g_free (platform);
gst_structure_free (gl_context);
gst_allocation_params_init (&params);
allocator = gst_allocator_find (GST_GL_MEMORY_ALLOCATOR);
gst_query_add_allocation_param (query, allocator, &params);
gst_object_unref (allocator);
#if GST_GL_HAVE_PLATFORM_EGL
if (gst_gl_context_check_feature (glimage_sink->context,
"EGL_KHR_image_base")) {
allocator = gst_allocator_find (GST_EGL_IMAGE_MEMORY_TYPE);
gst_query_add_allocation_param (query, allocator, &params);
gst_object_unref (allocator);
}
#endif
return TRUE; return TRUE;
/* ERRORS */ /* ERRORS */

View file

@ -117,6 +117,46 @@ gst_gl_mixer_pad_finalize (GObject * object)
G_OBJECT_CLASS (gst_gl_mixer_pad_parent_class)->finalize (object); G_OBJECT_CLASS (gst_gl_mixer_pad_parent_class)->finalize (object);
} }
static void
_init_upload (GstGLMixer * mix, GstGLMixerPad * pad)
{
GstVideoAggregatorPad *vaggpad = GST_VIDEO_AGGREGATOR_PAD (pad);
if (!pad->upload) {
GstCaps *in_caps = gst_pad_get_current_caps (GST_PAD (pad));
GstCaps *upload_caps = gst_caps_copy (in_caps);
GstCapsFeatures *gl_features =
gst_caps_features_from_string (GST_CAPS_FEATURE_MEMORY_GL_MEMORY);
pad->upload = gst_gl_upload_new (mix->context);
gst_caps_set_features (upload_caps, 0,
gst_caps_features_copy (gl_features));
gst_gl_upload_set_caps (pad->upload, in_caps, upload_caps);
gst_caps_unref (in_caps);
if (!pad->convert) {
GstVideoInfo gl_info;
GstCaps *gl_caps;
gst_video_info_set_format (&gl_info,
GST_VIDEO_FORMAT_RGBA,
GST_VIDEO_INFO_WIDTH (&vaggpad->info),
GST_VIDEO_INFO_HEIGHT (&vaggpad->info));
gl_caps = gst_video_info_to_caps (&gl_info);
gst_caps_set_features (gl_caps, 0, gst_caps_features_copy (gl_features));
pad->convert = gst_gl_color_convert_new (mix->context);
gst_gl_color_convert_set_caps (pad->convert, upload_caps, gl_caps);
gst_caps_unref (gl_caps);
}
gst_caps_unref (upload_caps);
gst_caps_features_free (gl_features);
}
}
static void static void
gst_gl_mixer_pad_get_property (GObject * object, guint prop_id, gst_gl_mixer_pad_get_property (GObject * object, guint prop_id,
GValue * value, GParamSpec * pspec) GValue * value, GParamSpec * pspec)
@ -153,7 +193,7 @@ _negotiated_caps (GstVideoAggregator * vagg, GstCaps * caps)
} }
static gboolean static gboolean
gst_gl_mixer_propose_allocation (GstGLMixer * mix, gst_gl_mixer_propose_allocation (GstGLMixer * mix, GstGLMixerPad * pad,
GstQuery * decide_query, GstQuery * query) GstQuery * decide_query, GstQuery * query)
{ {
GstGLMixerClass *mix_class = GST_GL_MIXER_GET_CLASS (mix); GstGLMixerClass *mix_class = GST_GL_MIXER_GET_CLASS (mix);
@ -163,11 +203,6 @@ gst_gl_mixer_propose_allocation (GstGLMixer * mix,
guint size = 0; guint size = 0;
gboolean need_pool; gboolean need_pool;
GError *error = NULL; GError *error = NULL;
GstStructure *gl_context;
gchar *platform, *gl_apis;
gpointer handle;
GstAllocator *allocator = NULL;
GstAllocationParams params;
gst_query_parse_allocation (query, &caps, &need_pool); gst_query_parse_allocation (query, &caps, &need_pool);
@ -229,32 +264,12 @@ gst_gl_mixer_propose_allocation (GstGLMixer * mix,
} }
/* we also support various metadata */ /* we also support various metadata */
gst_query_add_allocation_meta (query, GST_VIDEO_META_API_TYPE, 0);
if (mix->context->gl_vtable->FenceSync) if (mix->context->gl_vtable->FenceSync)
gst_query_add_allocation_meta (query, GST_GL_SYNC_META_API_TYPE, 0); gst_query_add_allocation_meta (query, GST_GL_SYNC_META_API_TYPE, 0);
gl_apis = gst_gl_api_to_string (gst_gl_context_get_gl_api (mix->context)); _init_upload (mix, pad);
platform =
gst_gl_platform_to_string (gst_gl_context_get_gl_platform (mix->context));
handle = (gpointer) gst_gl_context_get_gl_context (mix->context);
gl_context = gst_gl_upload_propose_allocation (pad->upload, decide_query, query);
gst_structure_new ("GstVideoGLTextureUploadMeta", "gst.gl.GstGLContext",
GST_GL_TYPE_CONTEXT, mix->context, "gst.gl.context.handle",
G_TYPE_POINTER, handle, "gst.gl.context.type", G_TYPE_STRING, platform,
"gst.gl.context.apis", G_TYPE_STRING, gl_apis, NULL);
gst_query_add_allocation_meta (query,
GST_VIDEO_GL_TEXTURE_UPLOAD_META_API_TYPE, gl_context);
g_free (gl_apis);
g_free (platform);
gst_structure_free (gl_context);
gst_allocation_params_init (&params);
allocator = gst_allocator_find (GST_GL_MEMORY_ALLOCATOR);
gst_query_add_allocation_param (query, allocator, &params);
gst_object_unref (allocator);
return TRUE; return TRUE;
@ -306,26 +321,9 @@ static GstCaps *
gst_gl_mixer_set_caps_features (const GstCaps * caps, gst_gl_mixer_set_caps_features (const GstCaps * caps,
const gchar * feature_name) const gchar * feature_name)
{ {
GstCaps *tmp = gst_caps_copy (caps); GstCaps *ret = gst_gl_caps_replace_all_caps_features (caps, feature_name);
guint n = gst_caps_get_size (tmp); gst_caps_set_simple (ret, "format", G_TYPE_STRING, "RGBA", NULL);
guint i = 0; return ret;
for (i = 0; i < n; i++) {
GstCapsFeatures *features = gst_caps_get_features (tmp, i);
if (features) {
guint n_f = gst_caps_features_get_size (features);
guint j = 0;
for (j = 0; j < n_f; j++) {
gst_caps_features_remove_id (features,
gst_caps_features_get_nth_id (features, j));
}
}
gst_caps_features_add (features, feature_name);
gst_caps_set_simple (tmp, "format", G_TYPE_STRING, "RGBA", NULL);
}
return tmp;
} }
/* copies the given caps */ /* copies the given caps */
@ -367,30 +365,25 @@ gst_gl_mixer_caps_remove_format_info (GstCaps * caps)
GstCaps * GstCaps *
gst_gl_mixer_update_caps (GstGLMixer * mix, GstCaps * caps) gst_gl_mixer_update_caps (GstGLMixer * mix, GstCaps * caps)
{ {
GstCaps *result = NULL; GstCaps *result, *tmp, *gl_caps;
GstCaps *glcaps = gst_gl_mixer_set_caps_features (caps,
GST_CAPS_FEATURE_MEMORY_GL_MEMORY);
#if GST_GL_HAVE_PLATFORM_EGL
GstCaps *eglcaps = gst_gl_mixer_set_caps_features (caps,
GST_CAPS_FEATURE_MEMORY_EGL_IMAGE);
#endif
GstCaps *uploadcaps = gst_gl_mixer_set_caps_features (caps,
GST_CAPS_FEATURE_META_GST_VIDEO_GL_TEXTURE_UPLOAD_META);
GstCaps *raw_caps =
gst_caps_from_string (GST_VIDEO_CAPS_MAKE (GST_GL_COLOR_CONVERT_FORMATS));
result = gst_caps_new_empty (); gl_caps = gst_caps_from_string ("video/x-raw(memory:GLMemory),format=RGBA");
result = gst_caps_merge (result, glcaps); result =
#if GST_GL_HAVE_PLATFORM_EGL gst_gl_color_convert_transform_caps (mix->context, GST_PAD_SRC, gl_caps,
result = gst_caps_merge (result, eglcaps); NULL);
#endif tmp = result;
result = gst_caps_merge (result, uploadcaps); GST_DEBUG_OBJECT (mix, "convert returned caps %" GST_PTR_FORMAT, tmp);
result = gst_caps_merge (result, raw_caps);
result = gst_caps_merge (result, gst_gl_mixer_caps_remove_format_info (caps)); result = gst_gl_mixer_caps_remove_format_info (tmp);
gst_caps_unref (tmp);
tmp = result;
GST_DEBUG_OBJECT (mix, "remove format returned caps %" GST_PTR_FORMAT, tmp);
GST_DEBUG_OBJECT (mix, "returning %" GST_PTR_FORMAT, result); result = gst_gl_upload_transform_caps (mix->context, GST_PAD_SRC, tmp, NULL);
gst_caps_unref (tmp);
tmp = result;
GST_DEBUG_OBJECT (mix, "transfer returned caps %" GST_PTR_FORMAT, tmp);
return result; return result;
} }
@ -398,7 +391,7 @@ gst_gl_mixer_update_caps (GstGLMixer * mix, GstCaps * caps)
static GstCaps * static GstCaps *
gst_gl_mixer_pad_sink_getcaps (GstPad * pad, GstGLMixer * mix, GstCaps * filter) gst_gl_mixer_pad_sink_getcaps (GstPad * pad, GstGLMixer * mix, GstCaps * filter)
{ {
GstCaps *srccaps; GstCaps *sinkcaps;
GstCaps *template_caps; GstCaps *template_caps;
GstCaps *filtered_caps; GstCaps *filtered_caps;
GstCaps *returned_caps; GstCaps *returned_caps;
@ -406,17 +399,18 @@ gst_gl_mixer_pad_sink_getcaps (GstPad * pad, GstGLMixer * mix, GstCaps * filter)
template_caps = gst_pad_get_pad_template_caps (pad); template_caps = gst_pad_get_pad_template_caps (pad);
srccaps = gst_pad_get_current_caps (pad); sinkcaps = gst_pad_get_current_caps (pad);
if (srccaps == NULL) { if (sinkcaps == NULL) {
had_current_caps = FALSE; had_current_caps = FALSE;
srccaps = template_caps; sinkcaps = template_caps;
} else { } else {
srccaps = gst_caps_merge (srccaps, gst_gl_mixer_update_caps (mix, srccaps)); sinkcaps =
gst_caps_merge (sinkcaps, gst_gl_mixer_update_caps (mix, sinkcaps));
} }
filtered_caps = srccaps; filtered_caps = sinkcaps;
if (filter) if (filter)
filtered_caps = gst_caps_intersect (srccaps, filter); filtered_caps = gst_caps_intersect (sinkcaps, filter);
returned_caps = gst_caps_intersect (filtered_caps, template_caps); returned_caps = gst_caps_intersect (filtered_caps, template_caps);
if (filter) if (filter)
@ -436,6 +430,7 @@ gst_gl_mixer_sink_query (GstAggregator * agg, GstAggregatorPad * bpad,
gboolean ret = FALSE; gboolean ret = FALSE;
GstGLMixer *mix = GST_GL_MIXER (agg); GstGLMixer *mix = GST_GL_MIXER (agg);
GstGLMixerClass *mix_class = GST_GL_MIXER_GET_CLASS (mix); GstGLMixerClass *mix_class = GST_GL_MIXER_GET_CLASS (mix);
GstGLMixerPad *pad = GST_GL_MIXER_PAD (bpad);
GST_TRACE ("QUERY %" GST_PTR_FORMAT, query); GST_TRACE ("QUERY %" GST_PTR_FORMAT, query);
@ -481,7 +476,7 @@ gst_gl_mixer_sink_query (GstAggregator * agg, GstAggregatorPad * bpad,
decide_query); decide_query);
/* pass the query to the propose_allocation vmethod if any */ /* pass the query to the propose_allocation vmethod if any */
ret = gst_gl_mixer_propose_allocation (mix, decide_query, query); ret = gst_gl_mixer_propose_allocation (mix, pad, decide_query, query);
if (decide_query) if (decide_query)
gst_query_unref (decide_query); gst_query_unref (decide_query);
@ -721,7 +716,8 @@ gst_gl_mixer_src_activate_mode (GstAggregator * aggregator, GstPadMode mode,
static gboolean static gboolean
gst_gl_mixer_query_caps (GstPad * pad, GstAggregator * agg, GstQuery * query) gst_gl_mixer_query_caps (GstPad * pad, GstAggregator * agg, GstQuery * query)
{ {
GstCaps *filter, *current_caps, *retcaps; GstGLMixer *mix = GST_GL_MIXER (agg);
GstCaps *filter, *current_caps, *retcaps, *gl_caps;
gst_query_parse_caps (query, &filter); gst_query_parse_caps (query, &filter);
@ -729,12 +725,30 @@ gst_gl_mixer_query_caps (GstPad * pad, GstAggregator * agg, GstQuery * query)
if (current_caps == NULL) if (current_caps == NULL)
current_caps = gst_pad_get_pad_template_caps (agg->srcpad); current_caps = gst_pad_get_pad_template_caps (agg->srcpad);
/* convert from current caps to GLMemory caps */
gl_caps =
gst_caps_merge (gst_caps_merge (gst_gl_mixer_set_caps_features
(current_caps, GST_CAPS_FEATURE_MEMORY_GL_MEMORY),
gst_gl_mixer_set_caps_features (current_caps,
GST_CAPS_FEATURE_MEMORY_EGL_IMAGE)),
gst_gl_mixer_set_caps_features (current_caps,
GST_CAPS_FEATURE_META_GST_VIDEO_GL_TEXTURE_UPLOAD_META));
retcaps =
gst_gl_download_transform_caps (mix->context, GST_PAD_SINK, current_caps,
NULL);
retcaps = gst_caps_merge (gl_caps, retcaps);
gst_caps_unref (current_caps);
current_caps = retcaps;
retcaps = gst_gl_mixer_caps_remove_format_info (current_caps); retcaps = gst_gl_mixer_caps_remove_format_info (current_caps);
gst_caps_unref (current_caps); gst_caps_unref (current_caps);
if (filter) if (filter) {
retcaps = current_caps =
gst_caps_intersect_full (filter, retcaps, GST_CAPS_INTERSECT_FIRST); gst_caps_intersect_full (filter, retcaps, GST_CAPS_INTERSECT_FIRST);
gst_caps_unref (retcaps);
retcaps = current_caps;
}
gst_query_set_caps_result (query, retcaps); gst_query_set_caps_result (query, retcaps);
gst_caps_unref (retcaps); gst_caps_unref (retcaps);
@ -1054,8 +1068,6 @@ _default_pad_upload_buffer (GstGLMixer * mix, GstGLMixerFrameData * frame,
GstVideoAggregatorPad *vaggpad = GST_VIDEO_AGGREGATOR_PAD (frame->pad); GstVideoAggregatorPad *vaggpad = GST_VIDEO_AGGREGATOR_PAD (frame->pad);
GstGLMixerPad *pad = frame->pad; GstGLMixerPad *pad = frame->pad;
GstBuffer *uploaded_buf, *gl_buffer; GstBuffer *uploaded_buf, *gl_buffer;
GstCaps *gl_caps;
GstCapsFeatures *gl_features;
GstVideoInfo gl_info; GstVideoInfo gl_info;
GstVideoFrame gl_frame; GstVideoFrame gl_frame;
GstGLSyncMeta *sync_meta; GstGLSyncMeta *sync_meta;
@ -1064,34 +1076,8 @@ _default_pad_upload_buffer (GstGLMixer * mix, GstGLMixerFrameData * frame,
GST_VIDEO_FORMAT_RGBA, GST_VIDEO_FORMAT_RGBA,
GST_VIDEO_INFO_WIDTH (&vaggpad->info), GST_VIDEO_INFO_WIDTH (&vaggpad->info),
GST_VIDEO_INFO_HEIGHT (&vaggpad->info)); GST_VIDEO_INFO_HEIGHT (&vaggpad->info));
gl_features =
gst_caps_features_from_string (GST_CAPS_FEATURE_MEMORY_GL_MEMORY);
gl_caps = gst_video_info_to_caps (&gl_info); _init_upload (mix, pad);
gst_caps_set_features (gl_caps, 0, gst_caps_features_copy (gl_features));
if (!pad->upload) {
GstCaps *in_caps = gst_pad_get_current_caps (GST_PAD (pad));
GstCaps *upload_caps = gst_caps_copy (in_caps);
pad->upload = gst_gl_upload_new (mix->context);
gst_caps_set_features (upload_caps, 0,
gst_caps_features_copy (gl_features));
gst_gl_upload_set_caps (pad->upload, in_caps, upload_caps);
if (!pad->convert) {
pad->convert = gst_gl_color_convert_new (mix->context);
gst_gl_color_convert_set_caps (pad->convert, upload_caps, gl_caps);
}
gst_caps_unref (upload_caps);
gst_caps_unref (in_caps);
}
gst_caps_features_free (gl_features);
gst_caps_unref (gl_caps);
sync_meta = gst_buffer_get_gl_sync_meta (vaggpad->buffer); sync_meta = gst_buffer_get_gl_sync_meta (vaggpad->buffer);
if (sync_meta) if (sync_meta)

View file

@ -93,6 +93,7 @@ static void gst_gl_test_src_get_property (GObject * object, guint prop_id,
static void gst_gl_test_src_dispose (GObject * object); static void gst_gl_test_src_dispose (GObject * object);
static gboolean gst_gl_test_src_setcaps (GstBaseSrc * bsrc, GstCaps * caps); static gboolean gst_gl_test_src_setcaps (GstBaseSrc * bsrc, GstCaps * caps);
static GstCaps *gst_gl_test_src_getcaps (GstBaseSrc * bsrc, GstCaps * filter);
static GstCaps *gst_gl_test_src_fixate (GstBaseSrc * bsrc, GstCaps * caps); static GstCaps *gst_gl_test_src_fixate (GstBaseSrc * bsrc, GstCaps * caps);
static gboolean gst_gl_test_src_is_seekable (GstBaseSrc * psrc); static gboolean gst_gl_test_src_is_seekable (GstBaseSrc * psrc);
@ -189,6 +190,7 @@ gst_gl_test_src_class_init (GstGLTestSrcClass * klass)
element_class->set_context = gst_gl_test_src_set_context; element_class->set_context = gst_gl_test_src_set_context;
gstbasesrc_class->set_caps = gst_gl_test_src_setcaps; gstbasesrc_class->set_caps = gst_gl_test_src_setcaps;
gstbasesrc_class->get_caps = gst_gl_test_src_getcaps;
gstbasesrc_class->is_seekable = gst_gl_test_src_is_seekable; gstbasesrc_class->is_seekable = gst_gl_test_src_is_seekable;
gstbasesrc_class->do_seek = gst_gl_test_src_do_seek; gstbasesrc_class->do_seek = gst_gl_test_src_do_seek;
gstbasesrc_class->query = gst_gl_test_src_query; gstbasesrc_class->query = gst_gl_test_src_query;
@ -470,6 +472,91 @@ wrong_caps:
} }
} }
/* copies the given caps */
static GstCaps *
gst_gl_test_src_caps_remove_format_info (GstCaps * caps)
{
GstStructure *st;
GstCapsFeatures *f;
gint i, n;
GstCaps *res;
res = gst_caps_new_empty ();
n = gst_caps_get_size (caps);
for (i = 0; i < n; i++) {
st = gst_caps_get_structure (caps, i);
f = gst_caps_get_features (caps, i);
/* If this is already expressed by the existing caps
* skip this structure */
if (i > 0 && gst_caps_is_subset_structure_full (res, st, f))
continue;
st = gst_structure_copy (st);
/* Only remove format info for the cases when we can actually convert */
if (!gst_caps_features_is_any (f)
&& gst_caps_features_is_equal (f,
GST_CAPS_FEATURES_MEMORY_SYSTEM_MEMORY))
gst_structure_remove_fields (st, "format", "colorimetry", "chroma-site",
NULL);
gst_structure_remove_fields (st, "width", "height", NULL);
gst_caps_append_structure_full (res, st, gst_caps_features_copy (f));
}
return res;
}
static GstCaps *
gst_gl_test_src_set_caps_features (const GstCaps * caps,
const gchar * feature_name)
{
GstCaps *ret = gst_gl_caps_replace_all_caps_features (caps, feature_name);
gst_caps_set_simple (ret, "format", G_TYPE_STRING, "RGBA", NULL);
return ret;
}
static GstCaps *
gst_gl_test_src_getcaps (GstBaseSrc * bsrc, GstCaps * filter)
{
GstGLTestSrc *src = GST_GL_TEST_SRC (bsrc);
GstCaps *tmp = NULL;
GstCaps *result = NULL;
GstCaps *gl_caps;
GstCaps *caps =
gst_caps_from_string ("video/x-raw(memory:GLMemory),format=RGBA");
tmp = gst_gl_test_src_caps_remove_format_info (caps);
GST_DEBUG_OBJECT (bsrc, "remove format returned caps %" GST_PTR_FORMAT, tmp);
gl_caps =
gst_caps_merge (gst_caps_merge (gst_gl_test_src_set_caps_features (tmp,
GST_CAPS_FEATURE_MEMORY_GL_MEMORY),
gst_gl_test_src_set_caps_features (tmp,
GST_CAPS_FEATURE_MEMORY_EGL_IMAGE)),
gst_gl_test_src_set_caps_features (tmp,
GST_CAPS_FEATURE_META_GST_VIDEO_GL_TEXTURE_UPLOAD_META));
result =
gst_gl_download_transform_caps (src->context, GST_PAD_SINK, tmp, NULL);
result = gst_caps_merge (gl_caps, result);
gst_caps_unref (tmp);
tmp = result;
GST_DEBUG_OBJECT (bsrc, "transfer returned caps %" GST_PTR_FORMAT, tmp);
if (filter) {
result = gst_caps_intersect_full (filter, tmp, GST_CAPS_INTERSECT_FIRST);
gst_caps_unref (tmp);
} else {
result = tmp;
}
GST_DEBUG_OBJECT (bsrc, "returning caps: %" GST_PTR_FORMAT, result);
return result;
}
static void static void
gst_gl_test_src_set_context (GstElement * element, GstContext * context) gst_gl_test_src_set_context (GstElement * element, GstContext * context)
{ {

View file

@ -631,6 +631,29 @@ gst_gl_color_convert_set_caps (GstGLColorConvert * convert,
return ret; return ret;
} }
GstCaps *
gst_gl_color_convert_transform_caps (GstGLContext * convert,
GstPadDirection direction, GstCaps * caps, GstCaps * filter)
{
GstCaps *templ, *result, *tmp;
templ =
gst_caps_from_string (GST_VIDEO_CAPS_MAKE_WITH_FEATURES
(GST_CAPS_FEATURE_MEMORY_GL_MEMORY, GST_GL_COLOR_CONVERT_FORMATS));
tmp = gst_caps_intersect_full (caps, templ, GST_CAPS_INTERSECT_FIRST);
gst_caps_unref (templ);
if (filter) {
result = gst_caps_intersect_full (filter, tmp, GST_CAPS_INTERSECT_FIRST);
gst_caps_unref (tmp);
} else {
result = tmp;
}
return result;
}
/** /**
* gst_gl_color_convert_perform: * gst_gl_color_convert_perform:
* @convert: a #GstGLColorConvert * @convert: a #GstGLColorConvert

View file

@ -99,6 +99,10 @@ struct _GstGLColorConvertClass
GstGLColorConvert * gst_gl_color_convert_new (GstGLContext * context); GstGLColorConvert * gst_gl_color_convert_new (GstGLContext * context);
GstCaps * gst_gl_color_convert_transform_caps (GstGLContext * convert,
GstPadDirection direction,
GstCaps * caps,
GstCaps * filter);
gboolean gst_gl_color_convert_set_caps (GstGLColorConvert * convert, gboolean gst_gl_color_convert_set_caps (GstGLColorConvert * convert,
GstCaps * in_caps, GstCaps * in_caps,
GstCaps * out_caps); GstCaps * out_caps);

View file

@ -180,6 +180,77 @@ gst_gl_download_set_format (GstGLDownload * download, GstVideoInfo * out_info)
GST_OBJECT_UNLOCK (download); GST_OBJECT_UNLOCK (download);
} }
static GstCaps *
_set_caps_features (const GstCaps * caps, const gchar * feature_name)
{
GstCaps *tmp = gst_caps_copy (caps);
guint n = gst_caps_get_size (tmp);
guint i = 0;
for (i = 0; i < n; i++) {
GstCapsFeatures *features = gst_caps_get_features (tmp, i);
if (features) {
guint n_f = gst_caps_features_get_size (features);
guint j = 0;
for (j = 0; j < n_f; j++) {
gst_caps_features_remove_id (features,
gst_caps_features_get_nth_id (features, j));
}
}
gst_caps_features_add (features, feature_name);
}
return tmp;
}
GstCaps *
gst_gl_download_transform_caps (GstGLContext * context,
GstPadDirection direction, GstCaps * caps, GstCaps * filter)
{
GstCaps *gl_templ, *templ, *result, *tmp;
templ =
gst_caps_from_string (GST_VIDEO_CAPS_MAKE (GST_GL_COLOR_CONVERT_FORMATS));
gl_templ =
gst_caps_from_string (GST_VIDEO_CAPS_MAKE_WITH_FEATURES
(GST_CAPS_FEATURE_MEMORY_GL_MEMORY, GST_GL_COLOR_CONVERT_FORMATS));
if (direction == GST_PAD_SRC) {
tmp = gst_caps_intersect_full (caps, templ, GST_CAPS_INTERSECT_FIRST);
result = _set_caps_features (tmp, GST_CAPS_FEATURE_MEMORY_GL_MEMORY);
gst_caps_unref (tmp);
tmp = result;
} else {
tmp = gst_caps_ref (caps);
}
result =
gst_gl_color_convert_transform_caps (context, direction, tmp, filter);
gst_caps_unref (tmp);
tmp = result;
if (direction == GST_PAD_SINK) {
result = _set_caps_features (tmp, GST_CAPS_FEATURE_MEMORY_SYSTEM_MEMORY);
gst_caps_unref (tmp);
tmp = result;
result = gst_caps_intersect_full (tmp, templ, GST_CAPS_INTERSECT_FIRST);
gst_caps_unref (tmp);
tmp = result;
}
if (filter) {
result = gst_caps_intersect_full (filter, tmp, GST_CAPS_INTERSECT_FIRST);
gst_caps_unref (tmp);
} else {
result = tmp;
}
gst_caps_unref (templ);
gst_caps_unref (gl_templ);
return result;
}
/** /**
* gst_gl_download_perform_with_data: * gst_gl_download_perform_with_data:
* @download: a #GstGLDownload * @download: a #GstGLDownload

View file

@ -73,6 +73,10 @@ struct _GstGLDownloadClass
GstGLDownload * gst_gl_download_new (GstGLContext * context); GstGLDownload * gst_gl_download_new (GstGLContext * context);
void gst_gl_download_set_format (GstGLDownload * download, GstVideoInfo * out_info); void gst_gl_download_set_format (GstGLDownload * download, GstVideoInfo * out_info);
GstCaps * gst_gl_download_transform_caps (GstGLContext * convert,
GstPadDirection direction,
GstCaps * caps,
GstCaps * filter);
gboolean gst_gl_download_perform_with_data (GstGLDownload * download, gboolean gst_gl_download_perform_with_data (GstGLDownload * download,
GLuint texture_id, GLuint texture_target, GLuint texture_id, GLuint texture_target,

View file

@ -705,31 +705,13 @@ done:
return othercaps; return othercaps;
} }
static GstCaps * static GstCaps *
gst_gl_filter_set_caps_features (const GstCaps * caps, gst_gl_filter_set_caps_features (const GstCaps * caps,
const gchar * feature_name) const gchar * feature_name)
{ {
GstCaps *tmp = gst_caps_copy (caps); GstCaps *ret = gst_gl_caps_replace_all_caps_features (caps, feature_name);
guint n = gst_caps_get_size (tmp); gst_caps_set_simple (ret, "format", G_TYPE_STRING, "RGBA", NULL);
guint i = 0; return ret;
for (i = 0; i < n; i++) {
GstCapsFeatures *features = gst_caps_get_features (tmp, i);
if (features) {
guint n_f = gst_caps_features_get_size (features);
guint j = 0;
for (j = 0; j < n_f; j++) {
gst_caps_features_remove_id (features,
gst_caps_features_get_nth_id (features, j));
}
}
gst_caps_features_add (features, feature_name);
gst_caps_set_simple (tmp, "format", G_TYPE_STRING, "RGBA", NULL);
}
return tmp;
} }
/* copies the given caps */ /* copies the given caps */
@ -770,35 +752,79 @@ gst_gl_filter_caps_remove_format_info (GstCaps * caps)
static GstCaps * static GstCaps *
gst_gl_filter_transform_caps (GstBaseTransform * bt, gst_gl_filter_transform_caps (GstBaseTransform * bt,
GstPadDirection direction, GstCaps * caps, GstCaps * filter) GstPadDirection direction, GstCaps * caps, GstCaps * filter_caps)
{ {
GstGLFilter *filter = GST_GL_FILTER (bt);
GstCaps *tmp = NULL; GstCaps *tmp = NULL;
GstCaps *result = NULL; GstCaps *result = NULL;
GstCaps *glcaps = gst_gl_filter_set_caps_features (caps,
GST_CAPS_FEATURE_MEMORY_GL_MEMORY);
#if GST_GL_HAVE_PLATFORM_EGL
GstCaps *eglcaps = gst_gl_filter_set_caps_features (caps,
GST_CAPS_FEATURE_MEMORY_EGL_IMAGE);
#endif
GstCaps *uploadcaps = gst_gl_filter_set_caps_features (caps,
GST_CAPS_FEATURE_META_GST_VIDEO_GL_TEXTURE_UPLOAD_META);
GstCaps *raw_caps =
gst_caps_from_string (GST_VIDEO_CAPS_MAKE (GST_GL_COLOR_CONVERT_FORMATS));
tmp = gst_caps_new_empty (); /* The following is the list of caps transformations performed. When
* direction == GST_PAD_SINK we start at the sinkpad and work toward the src
* pad and vice versa for direction == GST_PAD_SRC.
*
* sinkpad <-> (upload <-> convert) <-> filter (possible resize) <->
* (convert <-> download/output) <-> srcpad
*/
if (direction == GST_PAD_SINK) {
result =
gst_gl_upload_transform_caps (filter->context, direction, caps, NULL);
tmp = result;
tmp = gst_caps_merge (tmp, gst_gl_filter_caps_remove_format_info (glcaps)); result =
#if GST_GL_HAVE_PLATFORM_EGL gst_gl_color_convert_transform_caps (filter->context, direction, tmp,
tmp = gst_caps_merge (tmp, gst_gl_filter_caps_remove_format_info (eglcaps)); NULL);
#endif gst_caps_unref (tmp);
tmp = } else {
gst_caps_merge (tmp, gst_gl_filter_caps_remove_format_info (uploadcaps)); GstCaps *gl_caps =
tmp = gst_caps_merge (tmp, gst_gl_filter_caps_remove_format_info (raw_caps)); gst_caps_merge (gst_caps_merge (gst_gl_filter_set_caps_features (caps,
GST_CAPS_FEATURE_MEMORY_GL_MEMORY),
gst_gl_filter_set_caps_features (caps,
GST_CAPS_FEATURE_MEMORY_EGL_IMAGE)),
gst_gl_filter_set_caps_features (caps,
GST_CAPS_FEATURE_META_GST_VIDEO_GL_TEXTURE_UPLOAD_META));
tmp = gst_caps_merge (tmp, gst_gl_filter_caps_remove_format_info (caps)); result =
gst_gl_download_transform_caps (filter->context, direction, caps, NULL);
if (filter) { result = gst_caps_merge (gl_caps, result);
result = gst_caps_intersect_full (filter, tmp, GST_CAPS_INTERSECT_FIRST); }
tmp = result;
GST_DEBUG_OBJECT (bt, "transfer returned caps %" GST_PTR_FORMAT, tmp);
result = gst_gl_filter_caps_remove_format_info (tmp);
gst_caps_unref (tmp);
tmp = result;
GST_DEBUG_OBJECT (bt, "remove format returned caps %" GST_PTR_FORMAT, tmp);
if (direction == GST_PAD_SRC) {
result =
gst_gl_color_convert_transform_caps (filter->context, direction, tmp,
NULL);
gst_caps_unref (tmp);
tmp = result;
result =
gst_gl_upload_transform_caps (filter->context, direction, tmp, NULL);
} else {
GstCaps *gl_caps =
gst_caps_merge (gst_caps_merge (gst_gl_filter_set_caps_features (tmp,
GST_CAPS_FEATURE_MEMORY_GL_MEMORY),
gst_gl_filter_set_caps_features (tmp,
GST_CAPS_FEATURE_MEMORY_EGL_IMAGE)),
gst_gl_filter_set_caps_features (tmp,
GST_CAPS_FEATURE_META_GST_VIDEO_GL_TEXTURE_UPLOAD_META));
result =
gst_gl_download_transform_caps (filter->context, direction, tmp, NULL);
result = gst_caps_merge (gl_caps, result);
}
gst_caps_unref (tmp);
tmp = result;
GST_DEBUG_OBJECT (bt, "transfer returned caps %" GST_PTR_FORMAT, tmp);
if (filter_caps) {
result =
gst_caps_intersect_full (filter_caps, tmp, GST_CAPS_INTERSECT_FIRST);
gst_caps_unref (tmp); gst_caps_unref (tmp);
} else { } else {
result = tmp; result = tmp;
@ -868,18 +894,11 @@ gst_gl_filter_propose_allocation (GstBaseTransform * trans,
GstQuery * decide_query, GstQuery * query) GstQuery * decide_query, GstQuery * query)
{ {
GstGLFilter *filter = GST_GL_FILTER (trans); GstGLFilter *filter = GST_GL_FILTER (trans);
GstGLFilterClass *filter_class = GST_GL_FILTER_GET_CLASS (filter);
GstBufferPool *pool; GstBufferPool *pool;
GstStructure *config; GstStructure *config;
GstCaps *caps, *decide_caps; GstCaps *caps, *decide_caps;
guint size; guint size;
gboolean need_pool; gboolean need_pool;
GError *error = NULL;
GstStructure *gl_context;
gchar *platform, *gl_apis;
gpointer handle;
GstAllocator *allocator = NULL;
GstAllocationParams params;
gst_query_parse_allocation (query, &caps, &need_pool); gst_query_parse_allocation (query, &caps, &need_pool);
@ -906,19 +925,6 @@ gst_gl_filter_propose_allocation (GstBaseTransform * trans,
gst_structure_free (config); gst_structure_free (config);
} }
if (!gst_gl_ensure_element_data (filter, &filter->display,
&filter->other_context))
return FALSE;
gst_gl_display_filter_gl_api (filter->display,
filter_class->supported_gl_api);
if (!filter->context) {
filter->context = gst_gl_context_new (filter->display);
if (!gst_gl_context_create (filter->context, filter->other_context, &error))
goto context_error;
}
if (pool == NULL && need_pool) { if (pool == NULL && need_pool) {
GstVideoInfo info; GstVideoInfo info;
GstBufferPool *decide_pool = NULL; GstBufferPool *decide_pool = NULL;
@ -959,43 +965,11 @@ gst_gl_filter_propose_allocation (GstBaseTransform * trans,
gst_object_unref (pool); gst_object_unref (pool);
} }
/* we also support various metadata */ gst_gl_upload_propose_allocation (filter->upload, decide_query, query);
gst_query_add_allocation_meta (query, GST_VIDEO_META_API_TYPE, 0);
if (filter->context->gl_vtable->FenceSync) if (filter->context->gl_vtable->FenceSync)
gst_query_add_allocation_meta (query, GST_GL_SYNC_META_API_TYPE, 0); gst_query_add_allocation_meta (query, GST_GL_SYNC_META_API_TYPE, 0);
gl_apis = gst_gl_api_to_string (gst_gl_context_get_gl_api (filter->context));
platform =
gst_gl_platform_to_string (gst_gl_context_get_gl_platform
(filter->context));
handle = (gpointer) gst_gl_context_get_gl_context (filter->context);
gl_context =
gst_structure_new ("GstVideoGLTextureUploadMeta", "gst.gl.GstGLContext",
GST_GL_TYPE_CONTEXT, filter->context, "gst.gl.context.handle",
G_TYPE_POINTER, handle, "gst.gl.context.type", G_TYPE_STRING, platform,
"gst.gl.context.apis", G_TYPE_STRING, gl_apis, NULL);
gst_query_add_allocation_meta (query,
GST_VIDEO_GL_TEXTURE_UPLOAD_META_API_TYPE, gl_context);
g_free (gl_apis);
g_free (platform);
gst_structure_free (gl_context);
gst_allocation_params_init (&params);
allocator = gst_allocator_find (GST_GL_MEMORY_ALLOCATOR);
gst_query_add_allocation_param (query, allocator, &params);
gst_object_unref (allocator);
#if GST_GL_HAVE_PLATFORM_EGL
if (gst_gl_context_check_feature (filter->context, "EGL_KHR_image_base")) {
allocator = gst_allocator_find (GST_EGL_IMAGE_MEMORY_TYPE);
gst_query_add_allocation_param (query, allocator, &params);
gst_object_unref (allocator);
}
#endif
return TRUE; return TRUE;
/* ERRORS */ /* ERRORS */
@ -1014,12 +988,6 @@ config_failed:
GST_DEBUG_OBJECT (trans, "failed setting config"); GST_DEBUG_OBJECT (trans, "failed setting config");
return FALSE; return FALSE;
} }
context_error:
{
GST_ELEMENT_ERROR (trans, RESOURCE, NOT_FOUND, ("%s", error->message),
(NULL));
return FALSE;
}
} }
static gboolean static gboolean
@ -1037,10 +1005,19 @@ gst_gl_filter_decide_allocation (GstBaseTransform * trans, GstQuery * query)
guint in_width, in_height, out_width, out_height; guint in_width, in_height, out_width, out_height;
GstGLContext *other_context = NULL; GstGLContext *other_context = NULL;
gboolean same_downstream_gl_context = FALSE; gboolean same_downstream_gl_context = FALSE;
GstCapsFeatures *uploaded_features;
GstCaps *uploaded_caps;
GstCapsFeatures *converted_features;
GstVideoInfo converted_info;
gst_query_parse_allocation (query, &caps, NULL);
if (!caps)
return FALSE;
if (!gst_gl_ensure_element_data (filter, &filter->display, if (!gst_gl_ensure_element_data (filter, &filter->display,
&filter->other_context)) &filter->other_context)) {
return FALSE; return FALSE;
}
gst_gl_display_filter_gl_api (filter->display, gst_gl_display_filter_gl_api (filter->display,
filter_class->supported_gl_api); filter_class->supported_gl_api);
@ -1105,6 +1082,41 @@ gst_gl_filter_decide_allocation (GstBaseTransform * trans, GstQuery * query)
out_width = GST_VIDEO_INFO_WIDTH (&filter->out_info); out_width = GST_VIDEO_INFO_WIDTH (&filter->out_info);
out_height = GST_VIDEO_INFO_HEIGHT (&filter->out_info); out_height = GST_VIDEO_INFO_HEIGHT (&filter->out_info);
if (!filter->upload) {
filter->upload = gst_gl_upload_new (filter->context);
}
uploaded_caps = gst_caps_copy (caps);
uploaded_features =
gst_caps_features_from_string (GST_CAPS_FEATURE_MEMORY_GL_MEMORY);
gst_caps_set_features (uploaded_caps, 0, uploaded_features);
if (!gst_gl_upload_set_caps (filter->upload, caps, uploaded_caps)) {
gst_caps_unref (uploaded_caps);
return FALSE;
}
if (!filter->in_convert) {
filter->in_convert = gst_gl_color_convert_new (filter->context);
}
gst_video_info_set_format (&converted_info, GST_VIDEO_FORMAT_RGBA,
in_width, in_height);
converted_features =
gst_caps_features_from_string (GST_CAPS_FEATURE_MEMORY_GL_MEMORY);
if (filter->in_converted_caps)
gst_caps_unref (filter->in_converted_caps);
filter->in_converted_caps = gst_video_info_to_caps (&converted_info);
gst_caps_set_features (filter->in_converted_caps, 0, converted_features);
if (!gst_gl_color_convert_set_caps (filter->in_convert, uploaded_caps,
filter->in_converted_caps)) {
gst_caps_unref (uploaded_caps);
return FALSE;
}
gst_caps_unref (uploaded_caps);
if (filter->fbo) { if (filter->fbo) {
gst_gl_context_del_fbo (filter->context, filter->fbo, filter->depthbuffer); gst_gl_context_del_fbo (filter->context, filter->fbo, filter->depthbuffer);
filter->fbo = 0; filter->fbo = 0;
@ -1140,8 +1152,6 @@ gst_gl_filter_decide_allocation (GstBaseTransform * trans, GstQuery * query)
goto error; goto error;
} }
gst_query_parse_allocation (query, &caps, NULL);
if (gst_query_get_n_allocation_pools (query) > 0) { if (gst_query_get_n_allocation_pools (query) > 0) {
gst_query_parse_nth_allocation_pool (query, 0, &pool, &size, &min, &max); gst_query_parse_nth_allocation_pool (query, 0, &pool, &size, &min, &max);
@ -1178,11 +1188,6 @@ gst_gl_filter_decide_allocation (GstBaseTransform * trans, GstQuery * query)
gst_buffer_pool_set_config (pool, config); gst_buffer_pool_set_config (pool, config);
if (filter->upload) {
gst_object_unref (filter->upload);
filter->upload = NULL;
}
if (update_pool) if (update_pool)
gst_query_set_nth_allocation_pool (query, 0, pool, size, min, max); gst_query_set_nth_allocation_pool (query, 0, pool, size, min, max);
else else
@ -1328,58 +1333,6 @@ gst_gl_filter_transform (GstBaseTransform * bt, GstBuffer * inbuf,
if (!filter->display) if (!filter->display)
return GST_FLOW_NOT_NEGOTIATED; return GST_FLOW_NOT_NEGOTIATED;
if (!filter->upload) {
GstCaps *in_caps =
gst_pad_get_current_caps (GST_BASE_TRANSFORM_SINK_PAD (bt));
GstCaps *uploaded_caps;
GstCapsFeatures *out_features;
GstVideoInfo out_info;
gst_video_info_set_format (&out_info,
GST_VIDEO_FORMAT_RGBA,
GST_VIDEO_INFO_WIDTH (&filter->in_info),
GST_VIDEO_INFO_HEIGHT (&filter->in_info));
out_features =
gst_caps_features_from_string (GST_CAPS_FEATURE_MEMORY_GL_MEMORY);
uploaded_caps = gst_caps_copy (in_caps);
gst_caps_set_features (uploaded_caps, 0, out_features);
filter->upload = gst_gl_upload_new (filter->context);
if (!gst_gl_upload_set_caps (filter->upload, in_caps, uploaded_caps)) {
gst_caps_unref (in_caps);
return GST_FLOW_ERROR;
}
gst_caps_unref (in_caps);
if (!filter->in_convert) {
GstCapsFeatures *converted_features;
GstVideoInfo converted_info;
gst_video_info_set_format (&converted_info,
GST_VIDEO_FORMAT_RGBA,
GST_VIDEO_INFO_WIDTH (&filter->in_info),
GST_VIDEO_INFO_HEIGHT (&filter->in_info));
converted_features =
gst_caps_features_from_string (GST_CAPS_FEATURE_MEMORY_GL_MEMORY);
if (filter->in_converted_caps)
gst_caps_unref (filter->in_converted_caps);
filter->in_converted_caps = gst_video_info_to_caps (&converted_info);
gst_caps_set_features (filter->in_converted_caps, 0, converted_features);
filter->in_convert = gst_gl_color_convert_new (filter->context);
if (!gst_gl_color_convert_set_caps (filter->in_convert, uploaded_caps,
filter->in_converted_caps)) {
gst_caps_unref (uploaded_caps);
return GST_FLOW_ERROR;
}
}
gst_caps_unref (uploaded_caps);
}
g_assert (filter_class->filter || filter_class->filter_texture); g_assert (filter_class->filter || filter_class->filter_texture);
in_sync_meta = gst_buffer_get_gl_sync_meta (inbuf); in_sync_meta = gst_buffer_get_gl_sync_meta (inbuf);

View file

@ -63,11 +63,39 @@ struct _GstGLUploadPrivate
GstBuffer *outbuf; GstBuffer *outbuf;
/* all method impl pointers */
gpointer *upload_impl;
/* current method */
const UploadMethod *method; const UploadMethod *method;
gpointer method_impl; gpointer method_impl;
int method_i; int method_i;
}; };
static GstCaps *
_set_caps_features (const GstCaps * caps, const gchar * feature_name)
{
GstCaps *tmp = gst_caps_copy (caps);
guint n = gst_caps_get_size (tmp);
guint i = 0;
for (i = 0; i < n; i++) {
GstCapsFeatures *features = gst_caps_get_features (tmp, i);
if (features) {
guint n_f = gst_caps_features_get_size (features);
guint j = 0;
for (j = 0; j < n_f; j++) {
gst_caps_features_remove_id (features,
gst_caps_features_get_nth_id (features, j));
}
}
gst_caps_features_add (features, feature_name);
}
return tmp;
}
typedef enum typedef enum
{ {
METHOD_FLAG_CAN_SHARE_CONTEXT = 1, METHOD_FLAG_CAN_SHARE_CONTEXT = 1,
@ -79,8 +107,12 @@ struct _UploadMethod
GstGLUploadMethodFlags flags; GstGLUploadMethodFlags flags;
gpointer (*new) (GstGLUpload * upload); gpointer (*new) (GstGLUpload * upload);
GstCaps *(*transform_caps) (GstGLContext * context,
GstPadDirection direction, GstCaps * caps);
gboolean (*accept) (gpointer impl, GstBuffer * buffer, GstCaps * in_caps, gboolean (*accept) (gpointer impl, GstBuffer * buffer, GstCaps * in_caps,
GstCaps * out_caps); GstCaps * out_caps);
void (*propose_allocation) (gpointer impl, GstQuery * decide_query,
GstQuery * query);
GstGLUploadReturn (*perform) (gpointer impl, GstBuffer * buffer, GstGLUploadReturn (*perform) (gpointer impl, GstBuffer * buffer,
GstBuffer ** outbuf); GstBuffer ** outbuf);
void (*release) (gpointer impl, GstBuffer * buffer); void (*release) (gpointer impl, GstBuffer * buffer);
@ -102,6 +134,13 @@ _gl_memory_upload_new (GstGLUpload * upload)
return mem; return mem;
} }
static GstCaps *
_gl_memory_upload_transform_caps (GstGLContext * context,
GstPadDirection direction, GstCaps * caps)
{
return _set_caps_features (caps, GST_CAPS_FEATURE_MEMORY_GL_MEMORY);
}
static gboolean static gboolean
_gl_memory_upload_accept (gpointer impl, GstBuffer * buffer, GstCaps * in_caps, _gl_memory_upload_accept (gpointer impl, GstBuffer * buffer, GstCaps * in_caps,
GstCaps * out_caps) GstCaps * out_caps)
@ -137,6 +176,20 @@ _gl_memory_upload_accept (gpointer impl, GstBuffer * buffer, GstCaps * in_caps,
return TRUE; return TRUE;
} }
static void
_gl_memory_upload_propose_allocation (gpointer impl, GstQuery * decide_query,
GstQuery * query)
{
GstAllocationParams params;
GstAllocator *allocator;
gst_allocation_params_init (&params);
allocator = gst_allocator_find (GST_GL_MEMORY_ALLOCATOR);
gst_query_add_allocation_param (query, allocator, &params);
gst_object_unref (allocator);
}
static GstGLUploadReturn static GstGLUploadReturn
_gl_memory_upload_perform (gpointer impl, GstBuffer * buffer, _gl_memory_upload_perform (gpointer impl, GstBuffer * buffer,
GstBuffer ** outbuf) GstBuffer ** outbuf)
@ -173,7 +226,9 @@ static const UploadMethod _gl_memory_upload = {
"GLMemory", "GLMemory",
METHOD_FLAG_CAN_SHARE_CONTEXT, METHOD_FLAG_CAN_SHARE_CONTEXT,
&_gl_memory_upload_new, &_gl_memory_upload_new,
&_gl_memory_upload_transform_caps,
&_gl_memory_upload_accept, &_gl_memory_upload_accept,
&_gl_memory_upload_propose_allocation,
&_gl_memory_upload_perform, &_gl_memory_upload_perform,
&_gl_memory_upload_release, &_gl_memory_upload_release,
&_gl_memory_upload_free &_gl_memory_upload_free
@ -195,6 +250,21 @@ _egl_image_upload_new (GstGLUpload * upload)
return image; return image;
} }
static GstCaps *
_egl_image_upload_transform_caps (GstGLContext * context,
GstPadDirection direction, GstCaps * caps)
{
GstCaps *ret;
if (direction == GST_PAD_SINK) {
ret = _set_caps_features (caps, GST_CAPS_FEATURE_MEMORY_GL_MEMORY);
} else {
ret = _set_caps_features (caps, GST_CAPS_FEATURE_MEMORY_EGL_IMAGE);
}
return ret;
}
static gboolean static gboolean
_egl_image_upload_accept (gpointer impl, GstBuffer * buffer, GstCaps * in_caps, _egl_image_upload_accept (gpointer impl, GstBuffer * buffer, GstCaps * in_caps,
GstCaps * out_caps) GstCaps * out_caps)
@ -237,6 +307,24 @@ _egl_image_upload_accept (gpointer impl, GstBuffer * buffer, GstCaps * in_caps,
return TRUE; return TRUE;
} }
static void
_egl_image_upload_propose_allocation (gpointer impl, GstQuery * decide_query,
GstQuery * query)
{
struct EGLImageUpload *image = impl;
GstAllocationParams params;
GstAllocator *allocator;
gst_allocation_params_init (&params);
if (gst_gl_context_check_feature (image->upload->context,
"EGL_KHR_image_base")) {
allocator = gst_allocator_find (GST_EGL_IMAGE_MEMORY_TYPE);
gst_query_add_allocation_param (query, allocator, &params);
gst_object_unref (allocator);
}
}
static GstGLUploadReturn static GstGLUploadReturn
_egl_image_upload_perform (gpointer impl, GstBuffer * buffer, _egl_image_upload_perform (gpointer impl, GstBuffer * buffer,
GstBuffer ** outbuf) GstBuffer ** outbuf)
@ -285,7 +373,9 @@ static const UploadMethod _egl_image_upload = {
"EGLImage", "EGLImage",
0, 0,
&_egl_image_upload_new, &_egl_image_upload_new,
&_egl_image_upload_transform_caps,
&_egl_image_upload_accept, &_egl_image_upload_accept,
&_egl_image_upload_propose_allocation,
&_egl_image_upload_perform, &_egl_image_upload_perform,
&_egl_image_upload_release, &_egl_image_upload_release,
&_egl_image_upload_free &_egl_image_upload_free
@ -311,6 +401,23 @@ _upload_meta_upload_new (GstGLUpload * upload)
return meta; return meta;
} }
static GstCaps *
_upload_meta_upload_transform_caps (GstGLContext * context,
GstPadDirection direction, GstCaps * caps)
{
GstCaps *ret;
if (direction == GST_PAD_SINK) {
ret = _set_caps_features (caps, GST_CAPS_FEATURE_MEMORY_GL_MEMORY);
} else {
ret =
_set_caps_features (caps,
GST_CAPS_FEATURE_META_GST_VIDEO_GL_TEXTURE_UPLOAD_META);
}
return ret;
}
static gboolean static gboolean
_upload_meta_upload_accept (gpointer impl, GstBuffer * buffer, _upload_meta_upload_accept (gpointer impl, GstBuffer * buffer,
GstCaps * in_caps, GstCaps * out_caps) GstCaps * in_caps, GstCaps * out_caps)
@ -358,6 +465,36 @@ _upload_meta_upload_accept (gpointer impl, GstBuffer * buffer,
return TRUE; return TRUE;
} }
static void
_upload_meta_upload_propose_allocation (gpointer impl, GstQuery * decide_query,
GstQuery * query)
{
struct GLUploadMeta *upload = impl;
GstStructure *gl_context;
gchar *platform, *gl_apis;
gpointer handle;
gl_apis =
gst_gl_api_to_string (gst_gl_context_get_gl_api (upload->upload->
context));
platform =
gst_gl_platform_to_string (gst_gl_context_get_gl_platform (upload->
upload->context));
handle = (gpointer) gst_gl_context_get_gl_context (upload->upload->context);
gl_context =
gst_structure_new ("GstVideoGLTextureUploadMeta", "gst.gl.GstGLContext",
GST_GL_TYPE_CONTEXT, upload->upload->context, "gst.gl.context.handle",
G_TYPE_POINTER, handle, "gst.gl.context.type", G_TYPE_STRING, platform,
"gst.gl.context.apis", G_TYPE_STRING, gl_apis, NULL);
gst_query_add_allocation_meta (query,
GST_VIDEO_GL_TEXTURE_UPLOAD_META_API_TYPE, gl_context);
g_free (gl_apis);
g_free (platform);
gst_structure_free (gl_context);
}
/* /*
* Uploads using gst_video_gl_texture_upload_meta_upload(). * Uploads using gst_video_gl_texture_upload_meta_upload().
* i.e. consumer of GstVideoGLTextureUploadMeta * i.e. consumer of GstVideoGLTextureUploadMeta
@ -437,7 +574,9 @@ static const UploadMethod _upload_meta_upload = {
"UploadMeta", "UploadMeta",
METHOD_FLAG_CAN_SHARE_CONTEXT, METHOD_FLAG_CAN_SHARE_CONTEXT,
&_upload_meta_upload_new, &_upload_meta_upload_new,
&_upload_meta_upload_transform_caps,
&_upload_meta_upload_accept, &_upload_meta_upload_accept,
&_upload_meta_upload_propose_allocation,
&_upload_meta_upload_perform, &_upload_meta_upload_perform,
&_upload_meta_upload_release, &_upload_meta_upload_release,
&_upload_meta_upload_free &_upload_meta_upload_free
@ -460,6 +599,21 @@ _raw_data_upload_new (GstGLUpload * upload)
return raw; return raw;
} }
static GstCaps *
_raw_data_upload_transform_caps (GstGLContext * context,
GstPadDirection direction, GstCaps * caps)
{
GstCaps *ret;
if (direction == GST_PAD_SINK) {
ret = _set_caps_features (caps, GST_CAPS_FEATURE_MEMORY_GL_MEMORY);
} else {
ret = _set_caps_features (caps, GST_CAPS_FEATURE_MEMORY_SYSTEM_MEMORY);
}
return ret;
}
static gboolean static gboolean
_raw_data_upload_accept (gpointer impl, GstBuffer * buffer, GstCaps * in_caps, _raw_data_upload_accept (gpointer impl, GstBuffer * buffer, GstCaps * in_caps,
GstCaps * out_caps) GstCaps * out_caps)
@ -488,6 +642,13 @@ _raw_data_upload_accept (gpointer impl, GstBuffer * buffer, GstCaps * in_caps,
return TRUE; return TRUE;
} }
static void
_raw_data_upload_propose_allocation (gpointer impl, GstQuery * decide_query,
GstQuery * query)
{
gst_query_add_allocation_meta (query, GST_VIDEO_META_API_TYPE, 0);
}
static GstGLUploadReturn static GstGLUploadReturn
_raw_data_upload_perform (gpointer impl, GstBuffer * buffer, _raw_data_upload_perform (gpointer impl, GstBuffer * buffer,
GstBuffer ** outbuf) GstBuffer ** outbuf)
@ -540,7 +701,9 @@ static const UploadMethod _raw_data_upload = {
"Raw Data", "Raw Data",
0, 0,
&_raw_data_upload_new, &_raw_data_upload_new,
&_raw_data_upload_transform_caps,
&_raw_data_upload_accept, &_raw_data_upload_accept,
&_raw_data_upload_propose_allocation,
&_raw_data_upload_perform, &_raw_data_upload_perform,
&_raw_data_upload_release, &_raw_data_upload_release,
&_raw_data_upload_free &_raw_data_upload_free
@ -588,9 +751,16 @@ GstGLUpload *
gst_gl_upload_new (GstGLContext * context) gst_gl_upload_new (GstGLContext * context)
{ {
GstGLUpload *upload = g_object_new (GST_TYPE_GL_UPLOAD, NULL); GstGLUpload *upload = g_object_new (GST_TYPE_GL_UPLOAD, NULL);
gint i, n;
upload->context = gst_object_ref (context); upload->context = gst_object_ref (context);
n = G_N_ELEMENTS (upload_methods);
upload->priv->upload_impl = g_malloc (sizeof (gpointer) * n);
for (i = 0; i < n; i++) {
upload->priv->upload_impl[i] = upload_methods[i]->new (upload);
}
return upload; return upload;
} }
@ -598,6 +768,7 @@ static void
gst_gl_upload_finalize (GObject * object) gst_gl_upload_finalize (GObject * object)
{ {
GstGLUpload *upload; GstGLUpload *upload;
gint i, n;
upload = GST_GL_UPLOAD (object); upload = GST_GL_UPLOAD (object);
@ -627,9 +798,62 @@ gst_gl_upload_finalize (GObject * object)
upload->priv->out_caps = NULL; upload->priv->out_caps = NULL;
} }
n = G_N_ELEMENTS (upload_methods);
for (i = 0; i < n; i++) {
if (upload->priv->upload_impl[i])
upload_methods[i]->free (upload->priv->upload_impl[i]);
}
g_free (upload->priv->upload_impl);
G_OBJECT_CLASS (gst_gl_upload_parent_class)->finalize (object); G_OBJECT_CLASS (gst_gl_upload_parent_class)->finalize (object);
} }
GstCaps *
gst_gl_upload_transform_caps (GstGLContext * context, GstPadDirection direction,
GstCaps * caps, GstCaps * filter)
{
GstCaps *result, *tmp;
gint i;
tmp = gst_caps_new_empty ();
for (i = 0; i < G_N_ELEMENTS (upload_methods); i++) {
GstCaps *tmp2 =
upload_methods[i]->transform_caps (context, direction, caps);
if (tmp2)
tmp = gst_caps_merge (tmp, tmp2);
}
if (filter) {
result = gst_caps_intersect_full (filter, tmp, GST_CAPS_INTERSECT_FIRST);
gst_caps_unref (tmp);
} else {
result = tmp;
}
return result;
}
/**
* gst_gl_upload_propose_allocation:
* @upload: a #GstGLUpload
* @decide_query: (allow-none): a #GstQuery from a decide allocation
* @query: the proposed allocation query
*
* Adds the required allocation parameters to support uploading.
*/
void
gst_gl_upload_propose_allocation (GstGLUpload * upload, GstQuery * decide_query,
GstQuery * query)
{
gint i;
for (i = 0; i < G_N_ELEMENTS (upload_methods); i++)
upload_methods[i]->propose_allocation (upload->priv->upload_impl[i],
decide_query, query);
}
static gboolean static gboolean
_gst_gl_upload_set_caps_unlocked (GstGLUpload * upload, GstCaps * in_caps, _gst_gl_upload_set_caps_unlocked (GstGLUpload * upload, GstCaps * in_caps,
GstCaps * out_caps) GstCaps * out_caps)

View file

@ -78,13 +78,26 @@ struct _GstGLUploadClass
GstObjectClass object_class; GstObjectClass object_class;
}; };
GstGLUpload * gst_gl_upload_new (GstGLContext * context); GstGLUpload * gst_gl_upload_new (GstGLContext * context);
gboolean gst_gl_upload_set_caps (GstGLUpload * upload, GstCaps * in_caps, GstCaps * out_caps); GstCaps * gst_gl_upload_transform_caps (GstGLContext * context,
void gst_gl_upload_get_caps (GstGLUpload * upload, GstCaps ** in_caps, GstCaps ** out_caps); GstPadDirection direction,
GstCaps * caps,
GstCaps * filter);
gboolean gst_gl_upload_set_caps (GstGLUpload * upload,
GstCaps * in_caps,
GstCaps * out_caps);
void gst_gl_upload_get_caps (GstGLUpload * upload,
GstCaps ** in_caps,
GstCaps ** out_caps);
void gst_gl_upload_propose_allocation (GstGLUpload * upload,
GstQuery * decide_query,
GstQuery * query);
GstGLUploadReturn gst_gl_upload_perform_with_buffer (GstGLUpload * upload, GstBuffer * buffer, GstBuffer ** outbuf); GstGLUploadReturn gst_gl_upload_perform_with_buffer (GstGLUpload * upload,
void gst_gl_upload_release_buffer (GstGLUpload * upload); GstBuffer * buffer,
GstBuffer ** outbuf);
void gst_gl_upload_release_buffer (GstGLUpload * upload);
G_END_DECLS G_END_DECLS

View file

@ -872,3 +872,28 @@ gst_gl_get_plane_data_size (GstVideoInfo * info, GstVideoAlignment * align,
return plane_size; return plane_size;
} }
GstCaps *
gst_gl_caps_replace_all_caps_features (const GstCaps * caps,
const gchar * feature_name)
{
GstCaps *tmp = gst_caps_copy (caps);
guint n = gst_caps_get_size (tmp);
guint i = 0;
for (i = 0; i < n; i++) {
GstCapsFeatures *features = gst_caps_get_features (tmp, i);
if (features) {
guint n_f = gst_caps_features_get_size (features);
guint j = 0;
for (j = 0; j < n_f; j++) {
gst_caps_features_remove_id (features,
gst_caps_features_get_nth_id (features, j));
}
}
gst_caps_features_add (features, feature_name);
}
return tmp;
}

View file

@ -101,6 +101,8 @@ gboolean gst_gl_handle_context_query (GstElement * element, GstQuery * query,
gsize gst_gl_get_plane_data_size (GstVideoInfo * info, GstVideoAlignment * align, gsize gst_gl_get_plane_data_size (GstVideoInfo * info, GstVideoAlignment * align,
guint plane); guint plane);
GstCaps * gst_gl_caps_replace_all_caps_features (const GstCaps * caps,
const gchar * feature_name);
G_END_DECLS G_END_DECLS