2 * vim:ts=8:sw=3:sts=8:noexpandtab:cino=>5n-3f0^-2{2
4 #include "emotion_gstreamer.h"
5 #include "emotion_gstreamer_pipeline.h"
8 static void dvd_pad_added_cb (GstElement *dvddemuxer,
12 static void dvd_no_more_pads_cb (GstElement *dvddemuxer,
15 static int no_more_pads = 0;
19 emotion_pipeline_dvd_build(void *video, const char *device)
21 GstElement *dvdreadsrc;
23 Emotion_Gstreamer_Video *ev;
27 ev = (Emotion_Gstreamer_Video *)video;
30 dvdreadsrc = gst_element_factory_make("dvdreadsrc", "src");
32 goto failure_dvdreadsrc;
34 g_object_set(G_OBJECT(dvdreadsrc), "device", device, NULL);
36 dvddemux = gst_element_factory_make("dvddemux", "dvddemux");
38 goto failure_dvddemux;
39 g_signal_connect(dvddemux, "pad-added",
40 G_CALLBACK(dvd_pad_added_cb), ev);
41 g_signal_connect(dvddemux, "no-more-pads",
42 G_CALLBACK(dvd_no_more_pads_cb), ev);
44 gst_bin_add_many(GST_BIN(ev->pipeline), dvdreadsrc, dvddemux, NULL);
45 if (!gst_element_link(dvdreadsrc, dvddemux))
48 if (!emotion_pipeline_pause(ev->pipeline))
49 goto failure_gstreamer_pause;
51 while (no_more_pads == 0)
57 /* We get the informations of streams */
58 alist = ev->audio_sinks;
59 vlist = ev->video_sinks;
65 it = gst_element_iterate_src_pads(dvddemux);
66 while (gst_iterator_next(it, &data) == GST_ITERATOR_OK)
74 caps = gst_pad_get_caps(pad);
75 str = gst_caps_to_string(caps);
76 g_print("caps !! %s\n", str);
78 if (g_str_has_prefix(str, "video/mpeg"))
80 Emotion_Video_Sink *vsink;
84 vsink = (Emotion_Video_Sink *)eina_list_data_get(vlist);
85 vlist = eina_list_next(vlist);
86 sink_pad = gst_element_get_pad(gst_bin_get_by_name(GST_BIN(ev->pipeline), "mpeg2dec"), "src");
87 sink_caps = gst_pad_get_caps(sink_pad);
88 str = gst_caps_to_string(sink_caps);
89 g_print(" ** caps v !! %s\n", str);
91 emotion_video_sink_fill(vsink, sink_pad, sink_caps);
93 gst_caps_unref(sink_caps);
94 gst_object_unref(sink_pad);
97 else if (g_str_has_prefix(str, "audio/"))
99 Emotion_Audio_Sink *asink;
103 asink = (Emotion_Audio_Sink *)eina_list_data_get(alist);
104 sink_pad = gst_element_get_pad(gst_bin_get_by_name(GST_BIN(ev->pipeline), "a52dec"), "src");
105 sink_caps = gst_pad_get_caps(sink_pad);
107 emotion_audio_sink_fill(asink, sink_pad, sink_caps);
109 gst_caps_unref(caps);
111 gst_object_unref(pad);
113 gst_iterator_free(it);
116 /* The first vsink is a valid Emotion_Video_Sink * */
117 /* If no video stream is found, it's a visualisation sink */
119 Emotion_Video_Sink *vsink;
121 vsink = (Emotion_Video_Sink *)eina_list_data_get(ev->video_sinks);
122 if (vsink && vsink->sink)
124 g_object_set(G_OBJECT(vsink->sink), "sync", TRUE, NULL);
125 g_object_set(G_OBJECT(vsink->sink), "signal-handoffs", TRUE, NULL);
126 g_signal_connect(G_OBJECT(vsink->sink),
128 G_CALLBACK(cb_handoff), ev);
134 failure_gstreamer_pause:
136 gst_element_set_state(ev->pipeline, GST_STATE_NULL);
137 gst_bin_remove(GST_BIN(ev->pipeline), dvddemux);
139 gst_bin_remove(GST_BIN(ev->pipeline), dvdreadsrc);
146 dvd_pad_added_cb(GstElement *dvddemuxer,
150 Emotion_Gstreamer_Video *ev;
154 ev = (Emotion_Gstreamer_Video *)user_data;
155 caps = gst_pad_get_caps(GST_PAD(new_pad));
156 str = gst_caps_to_string(caps);
158 if (g_str_has_prefix(str, "video/mpeg"))
160 Emotion_Video_Sink *vsink;
165 vsink = (Emotion_Video_Sink *)malloc(sizeof(Emotion_Video_Sink));
167 ev->video_sinks = eina_list_append(ev->video_sinks, vsink);
168 if (!eina_list_data_find(ev->video_sinks, vsink))
174 queue = gst_element_factory_make("queue", NULL);
175 decoder = gst_element_factory_make("mpeg2dec", "mpeg2dec");
176 vsink->sink = gst_element_factory_make("fakesink", "videosink");
177 gst_bin_add_many(GST_BIN(ev->pipeline), queue, decoder, vsink->sink, NULL);
178 gst_element_link(queue, decoder);
179 gst_element_link(decoder, vsink->sink);
180 videopad = gst_element_get_pad(queue, "sink");
181 gst_pad_link(GST_PAD(new_pad), videopad);
182 gst_object_unref(videopad);
183 if (eina_list_count(ev->video_sinks) == 1)
185 ev->ratio = (double)vsink->width / (double)vsink->height;
187 gst_element_set_state(queue, GST_STATE_PAUSED);
188 gst_element_set_state(decoder, GST_STATE_PAUSED);
189 gst_element_set_state(vsink->sink, GST_STATE_PAUSED);
192 else if (g_str_has_prefix(str, "audio/"))
194 Emotion_Audio_Sink *asink;
198 GstElement *resample;
203 asink = (Emotion_Audio_Sink *)malloc(sizeof(Emotion_Audio_Sink));
205 ev->audio_sinks = eina_list_append(ev->audio_sinks, asink);
206 if (!eina_list_data_find(ev->audio_sinks, asink))
212 queue = gst_element_factory_make("queue", NULL);
213 decoder = gst_element_factory_make("a52dec", "a52dec");
214 conv = gst_element_factory_make("audioconvert", NULL);
215 resample = gst_element_factory_make("audioresample", NULL);
216 volume = gst_element_factory_make("volume", "volume");
217 g_object_get(G_OBJECT(volume), "volume", &vol, NULL);
218 ev->volume = vol / 10.0;
220 /* FIXME: must manage several audio streams */
221 asink->sink = gst_element_factory_make("fakesink", NULL);
223 gst_bin_add_many(GST_BIN(ev->pipeline),
224 queue, decoder, conv, resample, volume, asink->sink, NULL);
225 gst_element_link_many(queue, decoder, conv, resample, volume, asink->sink, NULL);
227 audiopad = gst_element_get_pad(queue, "sink");
228 gst_pad_link(GST_PAD(new_pad), audiopad);
229 gst_object_unref(audiopad);
231 gst_element_set_state(queue, GST_STATE_PAUSED);
232 gst_element_set_state(decoder, GST_STATE_PAUSED);
233 gst_element_set_state(conv, GST_STATE_PAUSED);
234 gst_element_set_state(resample, GST_STATE_PAUSED);
235 gst_element_set_state(volume, GST_STATE_PAUSED);
236 gst_element_set_state(asink->sink, GST_STATE_PAUSED);
241 dvd_no_more_pads_cb(GstElement *dvddemuxer,