summaryrefslogtreecommitdiffstats
path: root/gst/autoconvert/gstautoconvert.c
diff options
context:
space:
mode:
Diffstat (limited to 'gst/autoconvert/gstautoconvert.c')
-rw-r--r--gst/autoconvert/gstautoconvert.c140
1 files changed, 97 insertions, 43 deletions
diff --git a/gst/autoconvert/gstautoconvert.c b/gst/autoconvert/gstautoconvert.c
index 43110030..39b9c784 100644
--- a/gst/autoconvert/gstautoconvert.c
+++ b/gst/autoconvert/gstautoconvert.c
@@ -46,13 +46,6 @@ GST_DEBUG_CATEGORY (autoconvert_debug);
#define GST_CAT_DEFAULT (autoconvert_debug)
/* elementfactory information */
-static const GstElementDetails gst_auto_convert_details =
-GST_ELEMENT_DETAILS ("Select convertor based on caps",
- "Generic/Bin",
- "Selects the right transform element based on the caps",
- "Olivier Crete <olivier.crete@collabora.co.uk>");
-
-
static GstStaticPadTemplate sinktemplate = GST_STATIC_PAD_TEMPLATE ("sink",
GST_PAD_SINK,
GST_PAD_ALWAYS,
@@ -63,7 +56,6 @@ static GstStaticPadTemplate srctemplate = GST_STATIC_PAD_TEMPLATE ("src",
GST_PAD_ALWAYS,
GST_STATIC_CAPS_ANY);
-
static GstStaticPadTemplate sink_internal_template =
GST_STATIC_PAD_TEMPLATE ("sink_internal",
GST_PAD_SINK,
@@ -86,16 +78,18 @@ enum
enum
{
PROP_0,
- PROP_FACTORIES,
+ PROP_FACTORIES
};
-
static void gst_auto_convert_set_property (GObject * object,
guint prop_id, const GValue * value, GParamSpec * pspec);
static void gst_auto_convert_get_property (GObject * object,
guint prop_id, GValue * value, GParamSpec * pspec);
static void gst_auto_convert_dispose (GObject * object);
+static GstStateChangeReturn gst_auto_convert_change_state (GstElement * element,
+ GstStateChange transition);
+
static GstElement *gst_auto_convert_get_subelement (GstAutoConvert *
autoconvert);
static GstPad *gst_auto_convert_get_internal_sinkpad (GstAutoConvert *
@@ -118,7 +112,6 @@ static gboolean gst_auto_convert_src_event (GstPad * pad, GstEvent * event);
static gboolean gst_auto_convert_src_query (GstPad * pad, GstQuery * query);
static const GstQueryType *gst_auto_convert_src_query_type (GstPad * pad);
-
static GstFlowReturn gst_auto_convert_internal_sink_chain (GstPad * pad,
GstBuffer * buffer);
static gboolean gst_auto_convert_internal_sink_event (GstPad * pad,
@@ -140,18 +133,21 @@ static gboolean gst_auto_convert_internal_src_query (GstPad * pad,
static const GstQueryType *gst_auto_convert_internal_src_query_type (GstPad *
pad);
-
static void gst_auto_convert_load_factories (GstAutoConvert * autoconvert);
-GQuark internal_srcpad_quark = 0;
-GQuark internal_sinkpad_quark = 0;
-GQuark parent_quark = 0;
+static GQuark internal_srcpad_quark = 0;
+static GQuark internal_sinkpad_quark = 0;
+static GQuark parent_quark = 0;
static void
gst_auto_convert_do_init (GType type)
{
GST_DEBUG_CATEGORY_INIT (autoconvert_debug, "autoconvert", 0,
"Auto convert based on caps");
+
+ internal_srcpad_quark = g_quark_from_static_string ("internal_srcpad");
+ internal_sinkpad_quark = g_quark_from_static_string ("internal_sinkpad");
+ parent_quark = g_quark_from_static_string ("parent");
}
GST_BOILERPLATE_FULL (GstAutoConvert, gst_auto_convert, GstBin,
@@ -167,53 +163,43 @@ gst_auto_convert_base_init (gpointer klass)
gst_element_class_add_pad_template (element_class,
gst_static_pad_template_get (&sinktemplate));
- gst_element_class_set_details (element_class, &gst_auto_convert_details);
+ gst_element_class_set_details_simple (element_class,
+ "Select convertor based on caps", "Generic/Bin",
+ "Selects the right transform element based on the caps",
+ "Olivier Crete <olivier.crete@collabora.co.uk>");
}
static void
gst_auto_convert_class_init (GstAutoConvertClass * klass)
{
- GObjectClass *gobject_class;
- GstElementClass *gstelement_class;
- GstBinClass *gstbin_class;
-
- gobject_class = (GObjectClass *) klass;
- gstelement_class = (GstElementClass *) klass;
- gstbin_class = (GstBinClass *) klass;
+ GObjectClass *gobject_class = (GObjectClass *) klass;
+ GstElementClass *gstelement_class = (GstElementClass *) klass;
- gobject_class->dispose = GST_DEBUG_FUNCPTR (gst_auto_convert_dispose);
+ gobject_class->dispose = gst_auto_convert_dispose;
- gobject_class->set_property =
- GST_DEBUG_FUNCPTR (gst_auto_convert_set_property);
- gobject_class->get_property =
- GST_DEBUG_FUNCPTR (gst_auto_convert_get_property);
+ gobject_class->set_property = gst_auto_convert_set_property;
+ gobject_class->get_property = gst_auto_convert_get_property;
g_object_class_install_property (gobject_class, PROP_FACTORIES,
g_param_spec_pointer ("factories",
"GList of GstElementFactory",
"GList of GstElementFactory objects to pick from (the element takes"
" ownership of the list (NULL means it will go through all possible"
- " elements), can only be set once", G_PARAM_READWRITE));
-
- parent_class = g_type_class_peek_parent (klass);
+ " elements), can only be set once",
+ G_PARAM_READWRITE | G_PARAM_STATIC_STRINGS));
- internal_srcpad_quark = g_quark_from_static_string ("internal_srcpad");
- internal_sinkpad_quark = g_quark_from_static_string ("internal_sinkpad");
- parent_quark = g_quark_from_static_string ("parent");
+ gstelement_class->change_state =
+ GST_DEBUG_FUNCPTR (gst_auto_convert_change_state);
}
static void
gst_auto_convert_init (GstAutoConvert * autoconvert,
GstAutoConvertClass * klass)
{
-
autoconvert->sinkpad =
gst_pad_new_from_static_template (&sinktemplate, "sink");
autoconvert->srcpad = gst_pad_new_from_static_template (&srctemplate, "src");
- gst_element_add_pad (GST_ELEMENT (autoconvert), autoconvert->sinkpad);
- gst_element_add_pad (GST_ELEMENT (autoconvert), autoconvert->srcpad);
-
gst_pad_set_setcaps_function (autoconvert->sinkpad,
GST_DEBUG_FUNCPTR (gst_auto_convert_sink_setcaps));
gst_pad_set_getcaps_function (autoconvert->sinkpad,
@@ -235,6 +221,9 @@ gst_auto_convert_init (GstAutoConvert * autoconvert,
GST_DEBUG_FUNCPTR (gst_auto_convert_src_query));
gst_pad_set_query_type_function (autoconvert->srcpad,
GST_DEBUG_FUNCPTR (gst_auto_convert_src_query_type));
+
+ gst_element_add_pad (GST_ELEMENT (autoconvert), autoconvert->sinkpad);
+ gst_element_add_pad (GST_ELEMENT (autoconvert), autoconvert->srcpad);
}
static void
@@ -251,6 +240,11 @@ gst_auto_convert_dispose (GObject * object)
autoconvert->current_internal_sinkpad = NULL;
autoconvert->current_internal_srcpad = NULL;
}
+
+ g_list_foreach (autoconvert->cached_events, (GFunc) gst_mini_object_unref,
+ NULL);
+ g_list_free (autoconvert->cached_events);
+ autoconvert->cached_events = NULL;
GST_OBJECT_UNLOCK (object);
G_OBJECT_CLASS (parent_class)->dispose (object);
@@ -297,6 +291,35 @@ gst_auto_convert_get_property (GObject * object,
}
}
+static GstStateChangeReturn
+gst_auto_convert_change_state (GstElement * element, GstStateChange transition)
+{
+ GstAutoConvert *autoconvert = GST_AUTO_CONVERT (element);
+ GstStateChangeReturn ret;
+
+ switch (transition) {
+ default:
+ break;
+ }
+
+ ret = GST_ELEMENT_CLASS (parent_class)->change_state (element, transition);
+ if (ret == GST_STATE_CHANGE_FAILURE)
+ return ret;
+
+ switch (transition) {
+ case GST_STATE_CHANGE_PAUSED_TO_READY:
+ g_list_foreach (autoconvert->cached_events, (GFunc) gst_mini_object_unref,
+ NULL);
+ g_list_free (autoconvert->cached_events);
+ autoconvert->cached_events = NULL;
+ break;
+ default:
+ break;
+ }
+
+ return ret;
+}
+
static GstElement *
gst_auto_convert_get_element_by_type (GstAutoConvert * autoconvert, GType type)
{
@@ -908,6 +931,20 @@ gst_auto_convert_sink_chain (GstPad * pad, GstBuffer * buffer)
internal_srcpad = gst_auto_convert_get_internal_srcpad (autoconvert);
if (internal_srcpad) {
+ GST_OBJECT_LOCK (autoconvert);
+ if (autoconvert->cached_events) {
+ GList *l;
+
+ GST_DEBUG_OBJECT (autoconvert, "Sending cached events downstream");
+
+ autoconvert->cached_events = g_list_reverse (autoconvert->cached_events);
+
+ for (l = autoconvert->cached_events; l; l = l->next)
+ gst_pad_push_event (internal_srcpad, l->data);
+ g_list_free (autoconvert->cached_events);
+ autoconvert->cached_events = NULL;
+ }
+ GST_OBJECT_UNLOCK (autoconvert);
ret = gst_pad_push (internal_srcpad, buffer);
gst_object_unref (internal_srcpad);
} else {
@@ -932,9 +969,26 @@ gst_auto_convert_sink_event (GstPad * pad, GstEvent * event)
ret = gst_pad_push_event (internal_srcpad, event);
gst_object_unref (internal_srcpad);
} else {
- GST_WARNING_OBJECT (autoconvert, "Got event while no element was selected,"
- "letting through");
- ret = gst_pad_push_event (autoconvert->srcpad, event);
+ switch (GST_EVENT_TYPE (event)) {
+ case GST_EVENT_FLUSH_STOP:
+ GST_OBJECT_LOCK (autoconvert);
+ g_list_foreach (autoconvert->cached_events,
+ (GFunc) gst_mini_object_unref, NULL);
+ g_list_free (autoconvert->cached_events);
+ autoconvert->cached_events = NULL;
+ GST_OBJECT_UNLOCK (autoconvert);
+ /* fall through */
+ case GST_EVENT_FLUSH_START:
+ ret = gst_pad_push_event (autoconvert->srcpad, event);
+ break;
+ default:
+ GST_OBJECT_LOCK (autoconvert);
+ autoconvert->cached_events =
+ g_list_prepend (autoconvert->cached_events, event);
+ ret = TRUE;
+ GST_OBJECT_UNLOCK (autoconvert);
+ break;
+ }
}
gst_object_unref (autoconvert);
@@ -961,7 +1015,7 @@ gst_auto_convert_sink_query (GstPad * pad, GstQuery * query)
} else {
GST_WARNING_OBJECT (autoconvert, "Got query while no element was selected,"
"letting through");
- ret = gst_pad_query_default (pad, query);
+ ret = gst_pad_peer_query (autoconvert->srcpad, query);
}
gst_object_unref (autoconvert);
@@ -1199,7 +1253,7 @@ gst_auto_convert_src_query (GstPad * pad, GstQuery * query)
} else {
GST_WARNING_OBJECT (autoconvert, "Got query while not element was selected,"
"letting through");
- ret = gst_pad_query_default (pad, query);
+ ret = gst_pad_peer_query (autoconvert->sinkpad, query);
}
gst_object_unref (autoconvert);