/* GStreamer * Copyright (C) 2002 Erik Walthinsen * 2002 Wim Taymans * * gstspideridentity.c: identity element for the spider autoplugger * * This library is free software; you can redistribute it and/or * modify it under the terms of the GNU Library General Public * License as published by the Free Software Foundation; either * version 2 of the License, or (at your option) any later version. * * This library is distributed in the hope that it will be useful, * but WITHOUT ANY WARRANTY; without even the implied warranty of * MERCHANTABILITY or FITNESS FOR A PARTICULAR PURPOSE. See the GNU * Library General Public License for more details. * * You should have received a copy of the GNU Library General Public * License along with this library; if not, write to the * Free Software Foundation, Inc., 59 Temple Place - Suite 330, * Boston, MA 02111-1307, USA. */ #include "gstspideridentity.h" #include "gstspider.h" GstElementDetails gst_spider_identity_details = { "SpiderIdentity", "Generic", "LGPL", "Link between spider and outside elements", VERSION, "Benjamin Otte ", "(C) 2002", }; /* generic templates * delete me when meging with spider.c */ GST_PAD_TEMPLATE_FACTORY (spider_src_factory, "src", GST_PAD_SRC, GST_PAD_REQUEST, NULL /* no caps */ ); GST_PAD_TEMPLATE_FACTORY (spider_sink_factory, "sink", GST_PAD_SINK, GST_PAD_REQUEST, NULL /* no caps */ ); /* SpiderIdentity signals and args */ enum { /* FILL ME */ LAST_SIGNAL }; enum { ARG_0, /* FILL ME */ }; /* GObject stuff */ static void gst_spider_identity_class_init (GstSpiderIdentityClass *klass); static void gst_spider_identity_init (GstSpiderIdentity *spider_identity); /* functions set in pads, elements and stuff */ static void gst_spider_identity_chain (GstPad *pad, GstBuffer *buf); static GstElementStateReturn gst_spider_identity_change_state (GstElement *element); static GstPadLinkReturn gst_spider_identity_link (GstPad *pad, GstCaps *caps); static GstCaps * gst_spider_identity_getcaps (GstPad *pad, GstCaps *caps); /* loop functions */ static void gst_spider_identity_dumb_loop (GstSpiderIdentity *ident); static void gst_spider_identity_src_loop (GstSpiderIdentity *ident); static void gst_spider_identity_sink_loop_type_finding (GstSpiderIdentity *ident); static gboolean gst_spider_identity_handle_src_event (GstPad *pad, GstEvent *event); /* other functions */ static void gst_spider_identity_start_type_finding (GstSpiderIdentity *ident); static GstElementClass * parent_class = NULL; /* no signals static guint gst_spider_identity_signals[LAST_SIGNAL] = { 0 }; */ GType gst_spider_identity_get_type (void) { static GType spider_identity_type = 0; if (!spider_identity_type) { static const GTypeInfo spider_identity_info = { sizeof(GstSpiderIdentityClass), NULL, NULL, (GClassInitFunc)gst_spider_identity_class_init, NULL, NULL, sizeof(GstSpiderIdentity), 0, (GInstanceInitFunc)gst_spider_identity_init, }; spider_identity_type = g_type_register_static (GST_TYPE_ELEMENT, "GstSpiderIdentity", &spider_identity_info, 0); } return spider_identity_type; } static void gst_spider_identity_class_init (GstSpiderIdentityClass *klass) { GstElementClass *gstelement_class = (GstElementClass *) klass; parent_class = g_type_class_ref (GST_TYPE_ELEMENT); /* add our two pad templates */ gst_element_class_add_pad_template (gstelement_class, GST_PAD_TEMPLATE_GET (spider_src_factory)); gst_element_class_add_pad_template (gstelement_class, GST_PAD_TEMPLATE_GET (spider_sink_factory)); gstelement_class->change_state = GST_DEBUG_FUNCPTR(gst_spider_identity_change_state); gstelement_class->request_new_pad = GST_DEBUG_FUNCPTR(gst_spider_identity_request_new_pad); } static GstBufferPool* gst_spider_identity_get_bufferpool (GstPad *pad) { GstSpiderIdentity *ident; ident = GST_SPIDER_IDENTITY (gst_pad_get_parent (pad)); return gst_pad_get_bufferpool (ident->src); } static void gst_spider_identity_init (GstSpiderIdentity *ident) { /* sink */ ident->sink = gst_pad_new_from_template (GST_PAD_TEMPLATE_GET (spider_sink_factory), "sink"); gst_element_add_pad (GST_ELEMENT (ident), ident->sink); gst_pad_set_link_function (ident->sink, GST_DEBUG_FUNCPTR (gst_spider_identity_link)); gst_pad_set_getcaps_function (ident->sink, GST_DEBUG_FUNCPTR (gst_spider_identity_getcaps)); gst_pad_set_bufferpool_function (ident->sink, GST_DEBUG_FUNCPTR (gst_spider_identity_get_bufferpool)); /* src */ ident->src = gst_pad_new_from_template (GST_PAD_TEMPLATE_GET (spider_src_factory), "src"); gst_element_add_pad (GST_ELEMENT (ident), ident->src); gst_pad_set_link_function (ident->src, GST_DEBUG_FUNCPTR (gst_spider_identity_link)); gst_pad_set_getcaps_function (ident->src, GST_DEBUG_FUNCPTR (gst_spider_identity_getcaps)); gst_pad_set_event_function (ident->src, GST_DEBUG_FUNCPTR (gst_spider_identity_handle_src_event)); /* variables */ ident->plugged = FALSE; } static void gst_spider_identity_chain (GstPad *pad, GstBuffer *buf) { GstSpiderIdentity *ident; /*g_print ("chaining on pad %s:%s with buffer %p\n", GST_DEBUG_PAD_NAME (pad), buf);*/ g_return_if_fail (pad != NULL); g_return_if_fail (GST_IS_PAD (pad)); if (buf == NULL) return; ident = GST_SPIDER_IDENTITY (gst_pad_get_parent (pad)); if (GST_IS_EVENT (buf)) { /* start hack for current event stuff here */ /* check for unlinked elements and send them the EOS event, too */ if (GST_EVENT_TYPE (GST_EVENT (buf)) == GST_EVENT_EOS) { GstSpider *spider = (GstSpider *) GST_OBJECT_PARENT (ident); GList *list = spider->links; while (list) { GstSpiderConnection *conn = (GstSpiderConnection *) list->data; list = g_list_next (list); if (conn->current != (GstElement *) conn->src) { GST_DEBUG (GST_CAT_AUTOPLUG, "sending EOS to unconnected element %s from %s", GST_ELEMENT_NAME (conn->src), GST_ELEMENT_NAME (ident)); gst_pad_push (conn->src->src, GST_BUFFER (gst_event_new (GST_EVENT_EOS))); gst_element_set_eos (GST_ELEMENT (conn->src)); } } } /* end hack for current event stuff here */ gst_pad_event_default (pad, GST_EVENT (buf)); return; } if ((ident->src != NULL) && (GST_PAD_PEER (ident->src) != NULL)) { /* g_print("pushing buffer %p (refcount %d - buffersize %d) to pad %s:%s\n", buf, GST_BUFFER_REFCOUNT (buf), GST_BUFFER_SIZE (buf), GST_DEBUG_PAD_NAME (ident->src)); */ GST_DEBUG (0, "push %p %" G_GINT64_FORMAT, buf, GST_BUFFER_OFFSET (buf)); gst_pad_push (ident->src, buf); } else if (GST_IS_BUFFER (buf)) { gst_buffer_unref (buf); } } GstSpiderIdentity* gst_spider_identity_new_src (gchar *name) { GstSpiderIdentity *ret = (GstSpiderIdentity *) gst_element_factory_make ("spideridentity", name); /* set the right functions */ gst_element_set_loop_function (GST_ELEMENT (ret), (GstElementLoopFunction) GST_DEBUG_FUNCPTR (gst_spider_identity_src_loop)); return ret; } GstSpiderIdentity* gst_spider_identity_new_sink (gchar *name) { GstSpiderIdentity *ret = (GstSpiderIdentity *) gst_element_factory_make ("spideridentity", name); /* set the right functions */ gst_element_set_loop_function (GST_ELEMENT (ret), (GstElementLoopFunction) GST_DEBUG_FUNCPTR (gst_spider_identity_dumb_loop)); return ret; } /* shamelessly stolen from gstqueue.c to get proxy links */ static GstPadLinkReturn gst_spider_identity_link (GstPad *pad, GstCaps *caps) { GstSpiderIdentity *spider_identity = GST_SPIDER_IDENTITY (gst_pad_get_parent (pad)); GstPad *otherpad; if (pad == spider_identity->src) otherpad = spider_identity->sink; else otherpad = spider_identity->src; if (otherpad != NULL) return gst_pad_proxy_link (otherpad, caps); return GST_PAD_LINK_OK; } static GstCaps* gst_spider_identity_getcaps (GstPad *pad, GstCaps *caps) { GstSpiderIdentity *spider_identity = GST_SPIDER_IDENTITY (gst_pad_get_parent (pad)); GstPad *otherpad; if (pad == spider_identity->src) otherpad = spider_identity->sink; else otherpad = spider_identity->src; if (otherpad != NULL) return gst_pad_get_allowed_caps (otherpad); return NULL; } GstPad* gst_spider_identity_request_new_pad (GstElement *element, GstPadTemplate *templ, const gchar *name) { GstSpiderIdentity *ident; /*checks */ g_return_val_if_fail (templ != NULL, NULL); g_return_val_if_fail (GST_IS_PAD_TEMPLATE (templ), NULL); ident = GST_SPIDER_IDENTITY (element); g_return_val_if_fail (ident != NULL, NULL); g_return_val_if_fail (GST_IS_SPIDER_IDENTITY (ident), NULL); switch (GST_PAD_TEMPLATE_DIRECTION (templ)) { case GST_PAD_SINK: if (ident->sink != NULL) break; /* sink */ GST_DEBUG(0, "element %s requests new sink pad", GST_ELEMENT_NAME(ident)); ident->sink = gst_pad_new ("sink", GST_PAD_SINK); gst_element_add_pad (GST_ELEMENT (ident), ident->sink); gst_pad_set_link_function (ident->sink, GST_DEBUG_FUNCPTR (gst_spider_identity_link)); gst_pad_set_getcaps_function (ident->sink, GST_DEBUG_FUNCPTR (gst_spider_identity_getcaps)); gst_pad_set_bufferpool_function (ident->sink, GST_DEBUG_FUNCPTR (gst_spider_identity_get_bufferpool)); return ident->sink; case GST_PAD_SRC: /* src */ if (ident->src != NULL) break; GST_DEBUG(0, "element %s requests new src pad", GST_ELEMENT_NAME(ident)); ident->src = gst_pad_new ("src", GST_PAD_SRC); gst_element_add_pad (GST_ELEMENT (ident), ident->src); gst_pad_set_link_function (ident->src, GST_DEBUG_FUNCPTR (gst_spider_identity_link)); gst_pad_set_getcaps_function (ident->src, GST_DEBUG_FUNCPTR (gst_spider_identity_getcaps)); gst_pad_set_event_function (ident->src, GST_DEBUG_FUNCPTR (gst_spider_identity_handle_src_event)); return ident->src; default: break; } GST_DEBUG(0, "element %s requested a new pad but none could be created", GST_ELEMENT_NAME(ident)); return NULL; } /* this function has to * - start the autoplugger * - start type finding * ... */ static GstElementStateReturn gst_spider_identity_change_state (GstElement *element) { GstSpiderIdentity *ident; GstSpider *spider; GstElementStateReturn ret = GST_STATE_SUCCESS; /* element check */ ident = GST_SPIDER_IDENTITY (element); g_return_val_if_fail (ident != NULL, GST_STATE_FAILURE); g_return_val_if_fail (GST_IS_SPIDER_IDENTITY (ident), GST_STATE_FAILURE); switch (GST_STATE_TRANSITION (element)) { case GST_STATE_PAUSED_TO_PLAYING: /* autoplugger check */ spider = GST_SPIDER (GST_ELEMENT_PARENT (ident)); g_return_val_if_fail (spider != NULL, GST_STATE_FAILURE); g_return_val_if_fail (GST_IS_SPIDER (spider), GST_STATE_FAILURE); /* start typefinding or plugging */ if ((GST_RPAD_PEER (ident->sink) != NULL) && (GST_RPAD_PEER (ident->src) == NULL)) { if (gst_pad_get_caps ((GstPad *) GST_PAD_PEER (ident->sink)) == NULL) { gst_spider_identity_start_type_finding (ident); break; } else { gst_spider_identity_plug (ident); } } /* autoplug on src */ if ((GST_RPAD_PEER (ident->src) != NULL) && (GST_RPAD_PEER (ident->sink) == NULL)) { gst_spider_identity_plug (ident); } default: break; } if ((ret != GST_STATE_FAILURE) && (GST_ELEMENT_CLASS (parent_class)->change_state)) ret = GST_ELEMENT_CLASS (parent_class)->change_state (element); return ret; } static void gst_spider_identity_start_type_finding (GstSpiderIdentity *ident) { /* GstElement* typefind; gchar *name;*/ gboolean restart = FALSE; GST_DEBUG (GST_CAT_AUTOPLUG, "element %s starts typefinding", GST_ELEMENT_NAME(ident)); if (GST_STATE (GST_ELEMENT_PARENT (ident)) == GST_STATE_PLAYING) { gst_element_set_state (GST_ELEMENT (GST_ELEMENT_PARENT (ident)), GST_STATE_PAUSED); restart = TRUE; } gst_element_set_loop_function (GST_ELEMENT (ident), (GstElementLoopFunction) GST_DEBUG_FUNCPTR (gst_spider_identity_sink_loop_type_finding)); if (restart) { gst_element_set_state (GST_ELEMENT (GST_ELEMENT_PARENT (ident)), GST_STATE_PLAYING); } } /* since we can't set the loop function to NULL if there's a cothread for us, * we have to use a dumb one */ static void gst_spider_identity_dumb_loop (GstSpiderIdentity *ident) { GstBuffer *buf; g_return_if_fail (ident != NULL); g_return_if_fail (GST_IS_SPIDER_IDENTITY (ident)); g_assert (ident->sink != NULL); buf = gst_pad_pull (ident->sink); gst_spider_identity_chain (ident->sink, buf); } /* do nothing until we're linked - then disable yourself */ static void gst_spider_identity_src_loop (GstSpiderIdentity *ident) { /* checks - disable for speed */ g_return_if_fail (ident != NULL); g_return_if_fail (GST_IS_SPIDER_IDENTITY (ident)); /* we don't want a loop function if we're plugged */ if (ident->sink && GST_PAD_PEER (ident->sink)) { gst_element_set_loop_function (GST_ELEMENT (ident), (GstElementLoopFunction) GST_DEBUG_FUNCPTR (gst_spider_identity_dumb_loop)); gst_spider_identity_dumb_loop (ident); return; } gst_element_interrupt (GST_ELEMENT (ident)); } /* This loop function is only needed when typefinding. */ static void gst_spider_identity_sink_loop_type_finding (GstSpiderIdentity *ident) { GstBuffer *buf=NULL; GstBuffer *typefindbuf = NULL; gboolean getmorebuf = TRUE; GList *type_list; GstCaps *caps; /* this should possibly be a property */ guint bufsizelimit = 4096; g_return_if_fail (GST_IS_SPIDER_IDENTITY (ident)); while (getmorebuf){ /* check if our buffer is big enough to do a typefind */ if (typefindbuf && GST_BUFFER_SIZE(typefindbuf) >= bufsizelimit){ getmorebuf = FALSE; break; } buf = gst_pad_pull (ident->sink); /* if it's an event... */ while (GST_IS_EVENT (buf)) { switch (GST_EVENT_TYPE (GST_EVENT (buf))){ case GST_EVENT_EOS: getmorebuf = FALSE; /* FIXME Notify the srcs that EOS has happened */ gst_pad_event_default (ident->sink, GST_EVENT (buf)); break; default: gst_pad_event_default (ident->sink, GST_EVENT (buf)); buf = gst_pad_pull (ident->sink); break; } /* handle DISCONT events, please */ } typefindbuf = buf; getmorebuf = FALSE; /* FIXME merging doesn't work for some reason so * we'll just typefind with the first element if (!typefindbuf){ typefindbuf = buf; gst_buffer_ref(buf); } else { GstBuffer *oldbuf = typefindbuf; typefindbuf = gst_buffer_merge(typefindbuf, buf); gst_buffer_unref(oldbuf); gst_buffer_unref(buf); } */ } if (!typefindbuf){ goto end; } /* maybe there are already valid caps now? */ if ((caps = gst_pad_get_caps (ident->sink)) != NULL) { goto plug; } /* now do the actual typefinding with the supplied buffer */ type_list = (GList *) gst_type_get_list (); while (type_list) { GstType *type = (GstType *) type_list->data; GSList *factories = type->factories; while (factories) { GstTypeFactory *factory = GST_TYPE_FACTORY (factories->data); GstTypeFindFunc typefindfunc = (GstTypeFindFunc)factory->typefindfunc; GST_DEBUG (GST_CAT_AUTOPLUG_ATTEMPT, "trying typefind function %s", GST_PLUGIN_FEATURE_NAME (factory)); if (typefindfunc && (caps = typefindfunc (buf, factory))) { GST_INFO (GST_CAT_AUTOPLUG_ATTEMPT, "typefind function %s found caps", GST_PLUGIN_FEATURE_NAME (factory)); if (gst_pad_try_set_caps (ident->src, caps) <= 0) { g_warning ("typefind: found type but peer didn't accept it"); gst_caps_sink (caps); } else { goto plug; } } factories = g_slist_next (factories); } type_list = g_list_next (type_list); } gst_element_error(GST_ELEMENT(ident), "Could not find media type", NULL); gst_buffer_unref(buf); buf = GST_BUFFER (gst_event_new (GST_EVENT_EOS)); end: /* remove loop function */ gst_element_set_loop_function (GST_ELEMENT (ident), (GstElementLoopFunction) GST_DEBUG_FUNCPTR (gst_spider_identity_dumb_loop)); /* push the buffer */ gst_spider_identity_chain (ident->sink, buf); return; plug: gst_caps_debug (caps, "spider starting caps"); gst_caps_sink (caps); gst_spider_identity_plug (ident); goto end; } static gboolean gst_spider_identity_handle_src_event (GstPad *pad, GstEvent *event) { gboolean res = TRUE; GstSpiderIdentity *ident; GST_DEBUG (0, "spider_identity src_event"); ident = GST_SPIDER_IDENTITY (gst_pad_get_parent (pad)); switch (GST_EVENT_TYPE (event)) { case GST_EVENT_FLUSH: case GST_EVENT_SEEK: default: res = gst_pad_event_default (pad, event); break; } return res; }