#include <sys/stat.h> #include <fcntl.h> #include <unistd.h> #include <string.h> #include <gst/gst.h> #include <glib.h> static GMainLoop *mainloop = NULL; static gint64 d = 0; static gint64 gap = 10; typedef enum { MY_STREAM_TYPE_AUDIO = 0, MY_STREAM_TYPE_VIDEO = 1 } MyStreamType; typedef struct _StreamData StreamData; struct _StreamData { GstElement *bin; MyStreamType type; }; static void _stream_decode_pad_added_cb (GstElement *decode, GstPad *pad, gboolean arg1, gpointer user_data) { StreamData *data = (StreamData *) user_data; GstElement *queue; GstPad* sink_pad; GstCaps* caps = gst_pad_get_caps (pad); gchar *str_caps = gst_caps_to_string (caps); g_debug ("decode caps: [%d] [%s]", data->type, str_caps); switch (data->type) { case MY_STREAM_TYPE_AUDIO: g_debug ("Audio"); if (strstr (str_caps, "audio") == NULL) goto done; break; case MY_STREAM_TYPE_VIDEO: g_debug ("Video"); if (strstr (str_caps, "video") == NULL) goto done; break; } queue = gst_bin_get_by_name (GST_BIN (data->bin), "queue"); sink_pad = gst_element_get_pad (queue, "sink"); if (gst_pad_link (pad, sink_pad) != GST_PAD_LINK_OK) { g_warning ("Failed to link decode"); } gst_object_unref (queue); gst_object_unref (sink_pad); //g_free (data); g_debug ("Linked"); done: gst_caps_unref (caps); g_free (str_caps); } static GstElement* _create_src_element (const gchar* name, const gchar* uri, MyStreamType type, guint priority) { StreamData *data; GstElement *bin; GstElement *src; GstElement *decode; GstElement *queue; GstPad *src_pad; GstElement *gnl_src; g_debug ("element from uri: %s", uri); bin = gst_bin_new ("bin"); src = gst_element_make_from_uri (GST_URI_SRC, uri, "src"); g_return_val_if_fail (src != NULL, NULL); decode = gst_element_factory_make ("decodebin", NULL); g_return_val_if_fail (decode != NULL, NULL); queue = gst_element_factory_make ("queue", "queue"); g_return_val_if_fail (queue != NULL, NULL); gst_bin_add_many (GST_BIN (bin), src, decode, queue, NULL); gst_element_link (src, decode); data = g_new0 (StreamData, 1); data->bin = bin; data->type = type; g_debug ("Type : %d = %d", type, data->type); g_signal_connect (G_OBJECT (decode), "new-decoded-pad", G_CALLBACK (_stream_decode_pad_added_cb), data); src_pad = gst_element_get_pad (queue, "src"); g_return_val_if_fail (src_pad != NULL, NULL); gst_element_add_pad (bin, gst_ghost_pad_new ("src", src_pad)); gst_object_unref (src_pad); gnl_src = gst_element_factory_make ("gnlsource", name); g_return_val_if_fail (gnl_src != NULL, NULL); gst_bin_add (GST_BIN (gnl_src), bin); g_debug ("ADDING WITH: START [%lli] DUR [%lli]", d, gap); if (d == 0) { g_object_set (G_OBJECT (gnl_src), //"start", 0L, "duration", 10 * GST_SECOND, //"media-start", 0L, //"media-duration", 10 * GST_SECOND, "priority", priority, NULL); } else { g_object_set (G_OBJECT (gnl_src), "start", 10 * GST_SECOND, "duration", 10 * GST_SECOND, ///"media-start", 10 * GST_SECOND, //"media-duration", 10 * GST_SECOND, "priority", priority, NULL); } d++; return gnl_src; } static void _composition_pad_added_cb (GstElement *composition, GstPad *pad, gpointer data) { GstPad *sink_pad = gst_element_get_pad (GST_ELEMENT (data), "sink"); g_debug ("compose pad added"); if (gst_pad_link (pad, sink_pad) != GST_PAD_LINK_OK) { g_warning ("Failed to link decode"); } g_debug ("Linked ok"); } static void _compose_add_file (GstElement *compose, const gchar* e_name, const gchar* uri, MyStreamType type, guint priority) { GstElement *src; src = _create_src_element (e_name, uri, type, priority); gst_bin_add (GST_BIN (compose), src); } int main (int argc, char** argv) { GstElement* pipe; GstElement* gnl_compose_a; GstElement* gnl_compose_v; GstElement* asink; GstElement* vsink; GstElement* aqueue; GstElement* vqueue; g_type_init (); gst_init (&argc, &argv); mainloop = g_main_loop_new (NULL, FALSE); pipe = gst_pipeline_new ("test_pipeline"); gnl_compose_a = gst_element_factory_make ("gnlcomposition", "acompose"); g_return_val_if_fail (gnl_compose_a != NULL, 1); gnl_compose_v = gst_element_factory_make ("gnlcomposition", "vcompose"); g_return_val_if_fail (gnl_compose_v != NULL, 1); //_compose_add_file (gnl_compose_a, "src0", argv[1], MY_STREAM_TYPE_AUDIO, 1); //_compose_add_file (gnl_compose_a, "src1", argv[2], MY_STREAM_TYPE_AUDIO, 1); d = 0; _compose_add_file (gnl_compose_v, "src2", argv[1], MY_STREAM_TYPE_VIDEO, 1); _compose_add_file (gnl_compose_v, "src3", argv[2], MY_STREAM_TYPE_VIDEO, 1); //aqueue = gst_element_factory_make ("queue", "aqueue"); //asink = gst_element_factory_make ("alsasink", "asink"); vqueue = gst_element_factory_make ("queue", "vqueue"); vsink = gst_element_factory_make ("xvimagesink", "vsink"); gst_bin_add_many (GST_BIN (pipe), gnl_compose_a, gnl_compose_v, vqueue, vsink, //aqueue, asink, NULL); gst_element_link (vqueue, vsink); //gst_element_link (aqueue, asink); //g_signal_connect (G_OBJECT (gnl_compose_a), "pad-added", // G_CALLBACK (_composition_pad_added_cb), aqueue); g_signal_connect (G_OBJECT (gnl_compose_v), "pad-added", G_CALLBACK (_composition_pad_added_cb), vqueue); //g_idle_add (_play, pipe); gst_element_set_state (GST_ELEMENT (pipe), GST_STATE_PLAYING); g_main_loop_run (mainloop); return 0; }