renatofilho@588: #ifdef HAVE_CONFIG_H renatofilho@588: #include "config.h" renatofilho@588: #endif renatofilho@588: renatofilho@588: #include renatofilho@588: #include renatofilho@588: #include renatofilho@588: renatofilho@588: #include "gmencoder.h" renatofilho@588: renatofilho@588: #define G_MENCODER_GET_PRIVATE(obj) \ renatofilho@588: (G_TYPE_INSTANCE_GET_PRIVATE ((obj), G_TYPE_MENCODER, GMencoderPrivate)) renatofilho@588: renatofilho@588: renatofilho@588: typedef struct _GMencoderPrivate GMencoderPrivate; renatofilho@588: renatofilho@588: struct _GMencoderPrivate renatofilho@588: { renatofilho@588: GstElement *pipe; renatofilho@588: GstElement *abin; renatofilho@588: GstElement *vbin; renatofilho@588: GstElement *sink; renatofilho@588: gboolean ready; renatofilho@588: }; renatofilho@588: renatofilho@588: enum { renatofilho@588: READY, renatofilho@588: PAUSED, renatofilho@588: PLAYING, renatofilho@588: STOPED, renatofilho@588: EOS, renatofilho@588: ERROR, renatofilho@588: LAST_SIGNAL renatofilho@588: }; renatofilho@588: renatofilho@588: static void g_mencoder_class_init (GMencoderClass *klass); renatofilho@588: static void g_mencoder_init (GMencoder *object); renatofilho@588: static void g_mencoder_dispose (GObject *object); renatofilho@588: static void g_mencoder_finalize (GObject *object); renatofilho@588: static GstElement* renatofilho@588: _create_audio_bin (GMencoder *self, renatofilho@588: const gchar* encode, renatofilho@588: gchar** encode_prop, renatofilho@588: gint rate); renatofilho@588: static GstElement* renatofilho@588: _create_video_bin (GMencoder* self, renatofilho@588: const gchar* encode, renatofilho@588: gchar** encode_prop, renatofilho@588: gdouble fps, renatofilho@588: gint rate, renatofilho@588: guint width, renatofilho@588: guint height); renatofilho@588: renatofilho@588: static gboolean renatofilho@588: _pipeline_bus_cb (GstBus *bus, renatofilho@588: GstMessage *msg, renatofilho@588: gpointer user_data); renatofilho@588: static void _decodebin_new_pad_cb (GstElement* object, renatofilho@588: GstPad* pad, renatofilho@588: gboolean flag, renatofilho@588: gpointer user_data); renatofilho@588: static void _decodebin_unknown_type_cb (GstElement* object, renatofilho@588: GstPad* pad, renatofilho@588: GstCaps* caps, renatofilho@588: gpointer user_data); renatofilho@588: renatofilho@588: renatofilho@588: static guint g_mencoder_signals[LAST_SIGNAL] = { 0 }; renatofilho@588: renatofilho@588: G_DEFINE_TYPE(GMencoder, g_mencoder, G_TYPE_OBJECT) renatofilho@588: renatofilho@588: static void renatofilho@588: g_mencoder_class_init (GMencoderClass *klass) renatofilho@588: { renatofilho@588: GObjectClass *object_class; renatofilho@588: renatofilho@588: object_class = (GObjectClass *) klass; renatofilho@588: renatofilho@588: g_type_class_add_private (klass, sizeof (GMencoderPrivate)); renatofilho@588: renatofilho@588: object_class->dispose = g_mencoder_dispose; renatofilho@588: object_class->finalize = g_mencoder_finalize; renatofilho@588: renatofilho@588: g_mencoder_signals[READY] = renatofilho@588: g_signal_new ("ready", renatofilho@588: G_OBJECT_CLASS_TYPE (object_class), renatofilho@588: G_SIGNAL_RUN_FIRST, renatofilho@588: 0, NULL, NULL, renatofilho@588: g_cclosure_marshal_VOID__VOID, renatofilho@588: G_TYPE_NONE, 0); renatofilho@588: renatofilho@588: g_mencoder_signals[PAUSED] = renatofilho@588: g_signal_new ("paused", renatofilho@588: G_OBJECT_CLASS_TYPE (object_class), renatofilho@588: G_SIGNAL_RUN_FIRST, renatofilho@588: 0, NULL, NULL, renatofilho@588: g_cclosure_marshal_VOID__VOID, renatofilho@588: G_TYPE_NONE, 0); renatofilho@588: renatofilho@588: g_mencoder_signals[PLAYING] = renatofilho@588: g_signal_new ("playing", renatofilho@588: G_OBJECT_CLASS_TYPE (object_class), renatofilho@588: G_SIGNAL_RUN_FIRST, renatofilho@588: 0, NULL, NULL, renatofilho@588: g_cclosure_marshal_VOID__VOID, renatofilho@588: G_TYPE_NONE, 0); renatofilho@588: renatofilho@588: g_mencoder_signals[STOPED] = renatofilho@588: g_signal_new ("stoped", renatofilho@588: G_OBJECT_CLASS_TYPE (object_class), renatofilho@588: G_SIGNAL_RUN_FIRST, renatofilho@588: 0, NULL, NULL, renatofilho@588: g_cclosure_marshal_VOID__VOID, renatofilho@588: G_TYPE_NONE, 0); renatofilho@588: renatofilho@588: g_mencoder_signals[EOS] = renatofilho@588: g_signal_new ("eos", renatofilho@588: G_OBJECT_CLASS_TYPE (object_class), renatofilho@588: G_SIGNAL_RUN_FIRST, renatofilho@588: 0, NULL, NULL, renatofilho@588: g_cclosure_marshal_VOID__VOID, renatofilho@588: G_TYPE_NONE, 0); renatofilho@588: renatofilho@588: renatofilho@588: g_mencoder_signals[ERROR] = renatofilho@588: g_signal_new ("error", renatofilho@588: G_OBJECT_CLASS_TYPE (object_class), renatofilho@588: G_SIGNAL_RUN_FIRST, renatofilho@588: 0, NULL, NULL, renatofilho@588: g_cclosure_marshal_VOID__STRING, renatofilho@588: G_TYPE_NONE, 1, G_TYPE_STRING); renatofilho@588: } renatofilho@588: renatofilho@588: static void renatofilho@588: g_mencoder_init (GMencoder *self) renatofilho@588: { renatofilho@588: } renatofilho@588: renatofilho@588: static void renatofilho@588: g_mencoder_dispose (GObject *object) renatofilho@588: { renatofilho@588: } renatofilho@588: renatofilho@588: static void renatofilho@588: g_mencoder_finalize (GObject *object) renatofilho@588: { renatofilho@588: g_mencoder_close_stream (G_MENCODER (object)); renatofilho@588: } renatofilho@588: renatofilho@588: GMencoder* renatofilho@588: g_mencoder_new (void) renatofilho@588: { renatofilho@588: return g_object_new (G_TYPE_MENCODER, NULL); renatofilho@588: } renatofilho@588: renatofilho@588: static void renatofilho@588: _obj_set_prop (GObject *obj, renatofilho@588: const gchar *prop_name, renatofilho@588: const gchar *prop_val) renatofilho@588: { renatofilho@588: GValue p = {0}; renatofilho@588: GValue v = {0}; renatofilho@588: GParamSpec *s = NULL; renatofilho@588: GObjectClass *k = G_OBJECT_GET_CLASS (obj); renatofilho@588: renatofilho@588: renatofilho@588: g_value_init (&v, G_TYPE_STRING); renatofilho@588: g_value_set_string (&v, prop_val); renatofilho@588: renatofilho@588: g_debug ("PROP [%s] VAL [%s]", prop_name, prop_val); renatofilho@588: renatofilho@588: s = g_object_class_find_property (k, prop_name); renatofilho@588: if (s == NULL) { renatofilho@588: g_print ("Invalid property name: %s\n", prop_name); renatofilho@588: return; renatofilho@588: } renatofilho@588: renatofilho@588: g_value_init (&p, s->value_type); renatofilho@588: switch (s->value_type) renatofilho@588: { renatofilho@588: case G_TYPE_INT: renatofilho@588: g_value_set_int (&p, atoi (prop_val)); renatofilho@588: break; renatofilho@588: case G_TYPE_STRING: renatofilho@588: g_value_set_string (&p, prop_val); renatofilho@588: break; renatofilho@588: default: renatofilho@588: return; renatofilho@588: } renatofilho@588: renatofilho@588: g_object_set_property (obj, prop_name, &p); renatofilho@588: g_value_unset (&v); renatofilho@588: g_value_unset (&p); renatofilho@588: } renatofilho@588: renatofilho@588: static GstElement* renatofilho@588: _create_element_with_prop (const gchar* factory_name, renatofilho@588: const gchar* element_name, renatofilho@588: gchar** prop) renatofilho@588: { renatofilho@588: GstElement *ret; renatofilho@588: int i; renatofilho@588: renatofilho@588: renatofilho@588: g_debug ("SET OBJ [%s]", factory_name); renatofilho@588: ret = gst_element_factory_make (factory_name, element_name); renatofilho@588: if (ret == NULL) renatofilho@588: return NULL; renatofilho@588: renatofilho@588: if (prop != NULL) { renatofilho@588: for (i=0; i < g_strv_length (prop); i++) { renatofilho@588: char** v = g_strsplit(prop[i], "=", 2); renatofilho@588: if (g_strv_length (v) == 2) { renatofilho@588: _obj_set_prop (G_OBJECT (ret), v[0], v[1]); renatofilho@588: } renatofilho@588: g_strfreev (v); renatofilho@588: } renatofilho@588: } renatofilho@588: renatofilho@588: return ret; renatofilho@588: renatofilho@588: } renatofilho@588: renatofilho@588: static GstElement* renatofilho@588: _create_audio_bin (GMencoder* self, renatofilho@588: const gchar* encode, renatofilho@588: gchar** encode_prop, renatofilho@588: gint rate) renatofilho@588: { renatofilho@588: GstElement *abin = NULL; renatofilho@588: GstElement *aqueue = NULL; renatofilho@588: GstElement *aconvert = NULL; renatofilho@588: GstElement *aencode = NULL; renatofilho@588: GstElement *aqueue_src = NULL; renatofilho@588: GstPad *apad = NULL; renatofilho@588: renatofilho@588: //audio/x-raw-int ! queue ! audioconvert ! faac ! rtpmp4gpay ! udpsink name=upd_audio host=224.0.0.1 port=5002 renatofilho@588: abin = gst_bin_new ("abin"); renatofilho@588: aqueue = gst_element_factory_make ("queue", "aqueue"); renatofilho@588: aconvert= gst_element_factory_make ("audioconvert", "aconvert"); renatofilho@588: aencode = _create_element_with_prop ((encode ? encode : "lame"), "aencode", encode_prop); renatofilho@588: aqueue_src= gst_element_factory_make ("queue", "aqueue_src"); renatofilho@588: renatofilho@588: if ((abin == NULL) || (aqueue == NULL) || (aconvert == NULL) renatofilho@588: || (aencode == NULL) || (aqueue_src == NULL)) { renatofilho@588: g_warning ("Audio elements not found"); renatofilho@588: goto error; renatofilho@588: } renatofilho@588: renatofilho@588: gst_bin_add_many (GST_BIN (abin), aqueue, aconvert, aencode, aqueue_src, NULL); renatofilho@588: gst_element_link_many (aqueue, aconvert, aencode, aqueue_src, NULL); renatofilho@588: renatofilho@588: //TODO: apply audio rate renatofilho@588: renatofilho@588: // ghost pad the audio bin renatofilho@588: apad = gst_element_get_pad (aqueue, "sink"); renatofilho@588: gst_element_add_pad (abin, gst_ghost_pad_new("sink", apad)); renatofilho@588: gst_object_unref (apad); renatofilho@588: renatofilho@588: apad = gst_element_get_pad (aqueue_src, "src"); renatofilho@588: gst_element_add_pad (abin, gst_ghost_pad_new("src", apad)); renatofilho@588: gst_object_unref (apad); renatofilho@588: renatofilho@588: return abin; renatofilho@588: error: renatofilho@588: if (abin != NULL) renatofilho@588: gst_object_unref (abin); renatofilho@588: renatofilho@588: if (aqueue != NULL) renatofilho@588: gst_object_unref (aqueue); renatofilho@588: renatofilho@588: if (aconvert != NULL) renatofilho@588: gst_object_unref (aconvert); renatofilho@588: renatofilho@588: if (aencode != NULL) renatofilho@588: gst_object_unref (aencode); renatofilho@588: renatofilho@588: if (aqueue_src != NULL) renatofilho@588: gst_object_unref (aqueue_src); renatofilho@588: renatofilho@588: if (apad != NULL) renatofilho@588: gst_object_unref (apad); renatofilho@588: renatofilho@588: return NULL; renatofilho@588: } renatofilho@588: renatofilho@588: renatofilho@588: renatofilho@588: renatofilho@588: //queue ! videoscale ! video/x-raw-yuv,width=240,height=144 ! colorspace ! rate ! encode ! queue renatofilho@588: static GstElement* renatofilho@588: _create_video_bin (GMencoder* self, renatofilho@588: const gchar* encode, renatofilho@588: gchar** encode_prop, renatofilho@588: gdouble fps, renatofilho@588: gint rate, renatofilho@588: guint width, renatofilho@588: guint height) renatofilho@588: { renatofilho@588: GstElement *vbin = NULL; renatofilho@588: GstElement *vqueue = NULL; renatofilho@588: GstElement* vqueue_src = NULL; renatofilho@588: GstElement *vcolorspace = NULL; renatofilho@588: GstElement *vencode = NULL; renatofilho@588: GstElement *vrate = NULL; renatofilho@588: GstPad *vpad = NULL; renatofilho@588: renatofilho@588: vbin = gst_bin_new ("vbin"); renatofilho@588: vqueue = gst_element_factory_make ("queue", "vqueue"); renatofilho@588: vcolorspace = gst_element_factory_make ("ffmpegcolorspace", "colorspace"); renatofilho@588: vencode = _create_element_with_prop ((encode ? encode : "ffenc_mpeg1video"), "vencode", encode_prop); renatofilho@588: vrate = gst_element_factory_make ("videorate", "vrate"); renatofilho@588: vqueue_src = gst_element_factory_make ("queue", "queue_src"); renatofilho@588: renatofilho@588: if ((vbin == NULL) || (vqueue == NULL) || (vcolorspace == NULL) renatofilho@588: || (vencode == NULL) || (vqueue_src == NULL)) { renatofilho@588: g_warning ("Video elements not found"); renatofilho@588: goto error; renatofilho@588: } renatofilho@588: renatofilho@588: gst_bin_add_many (GST_BIN (vbin), vqueue, vcolorspace, vencode, vqueue_src, NULL); renatofilho@588: renatofilho@588: renatofilho@588: if ((width > 0) && (height > 0)) { renatofilho@588: //Scalling video renatofilho@588: GstCaps *vcaps; renatofilho@588: GstElement *vscale = gst_element_factory_make ("videoscale", "vscale"); renatofilho@588: renatofilho@588: gst_bin_add (GST_BIN (vbin), vscale); renatofilho@588: renatofilho@588: vcaps = gst_caps_new_simple ("video/x-raw-yuv", renatofilho@588: "width", G_TYPE_INT, width, renatofilho@588: "height", G_TYPE_INT, height, renatofilho@588: NULL); renatofilho@588: renatofilho@588: gst_element_link (vqueue, vscale); renatofilho@588: renatofilho@588: if (gst_element_link_filtered (vscale, vcolorspace, vcaps) == FALSE) { renatofilho@588: g_warning ("Fail to resize video"); renatofilho@588: gst_object_unref (vcaps); renatofilho@588: gst_object_unref (vscale); renatofilho@588: goto error; renatofilho@588: } renatofilho@588: gst_caps_unref (vcaps); renatofilho@588: } else { renatofilho@588: gst_element_link (vqueue, vcolorspace); renatofilho@588: } renatofilho@588: renatofilho@588: if (fps > 0) { renatofilho@588: //Changing the video fps renatofilho@588: GstCaps *vcaps; renatofilho@588: renatofilho@588: gst_bin_add (GST_BIN (vbin), vrate); renatofilho@588: renatofilho@588: if (gst_element_link (vcolorspace, vrate) == FALSE) { renatofilho@588: g_warning ("Fail to link video elements"); renatofilho@588: goto error; renatofilho@588: } renatofilho@588: renatofilho@588: vcaps = gst_caps_new_simple ("video/x-raw-yuv", renatofilho@588: "framerate", GST_TYPE_FRACTION, (int) (fps * 1000), 1000, NULL); renatofilho@588: renatofilho@588: if (gst_element_link_filtered (vrate, vencode, vcaps) == FALSE) { renatofilho@588: g_warning ("Fail to link vrate with vencode."); renatofilho@588: goto error; renatofilho@588: } renatofilho@588: gst_caps_unref (vcaps); renatofilho@588: } else { renatofilho@588: if (gst_element_link (vcolorspace, vencode) == FALSE) { renatofilho@588: g_warning ("Fail to link colorspace and video encode element."); renatofilho@588: goto error; renatofilho@588: } renatofilho@588: } renatofilho@588: renatofilho@588: renatofilho@588: gst_element_link (vencode, vqueue_src); renatofilho@588: renatofilho@588: // ghost pad the video bin renatofilho@588: vpad = gst_element_get_pad (vqueue, "sink"); renatofilho@588: gst_element_add_pad (vbin, gst_ghost_pad_new ("sink", vpad)); renatofilho@588: gst_object_unref (vpad); renatofilho@588: renatofilho@588: vpad = gst_element_get_pad (vqueue_src, "src"); renatofilho@588: gst_element_add_pad (vbin, gst_ghost_pad_new ("src", vpad)); renatofilho@588: gst_object_unref (vpad); renatofilho@588: renatofilho@588: return vbin; renatofilho@588: renatofilho@588: error: renatofilho@588: if (vpad != NULL) renatofilho@588: gst_object_unref (vpad); renatofilho@588: renatofilho@588: if (vbin != NULL) renatofilho@588: gst_object_unref (vbin); renatofilho@588: renatofilho@588: if (vqueue != NULL) renatofilho@588: gst_object_unref (vqueue); renatofilho@588: renatofilho@588: if (vencode != NULL) renatofilho@588: gst_object_unref (vencode); renatofilho@588: renatofilho@588: if (vqueue_src != NULL) renatofilho@588: gst_object_unref (vqueue_src); renatofilho@588: renatofilho@588: if (vcolorspace != NULL) renatofilho@588: gst_object_unref (vcolorspace); renatofilho@588: renatofilho@588: return NULL; renatofilho@588: } renatofilho@588: renatofilho@588: renatofilho@588: gboolean renatofilho@588: g_mencoder_setup_stream (GMencoder *self, renatofilho@588: const gchar* uri, renatofilho@588: const gchar* video_encode, renatofilho@588: gchar** video_encode_prop, renatofilho@588: gdouble video_fps, renatofilho@588: gdouble video_rate, renatofilho@588: guint video_width, renatofilho@588: guint video_height, renatofilho@588: const gchar* audio_encode, renatofilho@588: gchar** audio_encode_prop, renatofilho@588: guint audio_rate, renatofilho@588: const gchar* sink_name, renatofilho@588: gchar** sink_prop) renatofilho@588: { renatofilho@588: GstBus *bus = NULL; renatofilho@588: GstElement *pipe = NULL; renatofilho@588: GstElement *fdsink = NULL; renatofilho@588: GstElement *mux = NULL; renatofilho@588: GstElement *decode = NULL; renatofilho@588: GstElement *src = NULL; renatofilho@588: GstElement *abin = NULL; renatofilho@588: GstElement *vbin = NULL; renatofilho@588: GstPad *aux_pad = NULL; renatofilho@588: GstPad *mux_pad = NULL; renatofilho@588: renatofilho@588: GMencoderPrivate *priv = G_MENCODER_GET_PRIVATE (self); renatofilho@588: renatofilho@588: pipe = gst_pipeline_new ("pipe"); renatofilho@588: src = gst_element_make_from_uri (GST_URI_SRC, uri, "src"); renatofilho@588: if (src == NULL) renatofilho@588: goto error; renatofilho@588: renatofilho@588: decode = gst_element_factory_make ("decodebin2", "decode"); renatofilho@588: if (decode == NULL) renatofilho@588: goto error; renatofilho@588: renatofilho@588: mux = gst_element_factory_make ("ffmux_mpeg", "mux"); renatofilho@588: if (mux == NULL) renatofilho@588: goto error; renatofilho@588: renatofilho@588: fdsink = _create_element_with_prop (sink_name, "sink", sink_prop); renatofilho@588: if (fdsink == NULL) renatofilho@588: goto error; renatofilho@588: renatofilho@588: abin = _create_audio_bin (self, audio_encode, audio_encode_prop, audio_rate); renatofilho@588: if (abin == NULL) renatofilho@588: goto error; renatofilho@588: renatofilho@588: vbin = _create_video_bin (self, video_encode, video_encode_prop, video_fps, video_rate, video_width, video_height); renatofilho@588: if (vbin == NULL) renatofilho@588: goto error; renatofilho@588: renatofilho@588: // Finish Pipe renatofilho@588: gst_bin_add_many (GST_BIN (pipe), src, decode, abin, vbin, mux, fdsink, NULL); renatofilho@588: gst_element_link (src, decode); renatofilho@588: renatofilho@588: //Link bins with mux renatofilho@588: aux_pad = gst_element_get_pad (abin, "src"); renatofilho@588: mux_pad = gst_element_get_pad (mux, "audio_0"); renatofilho@588: GstPadLinkReturn ret = gst_pad_link (aux_pad, mux_pad); renatofilho@588: if (ret != GST_PAD_LINK_OK) { renatofilho@588: g_warning ("Fail link audio and mux: %d", ret); renatofilho@588: goto error; renatofilho@588: renatofilho@588: } renatofilho@588: gst_object_unref (aux_pad); renatofilho@588: gst_object_unref (mux_pad); renatofilho@588: renatofilho@588: aux_pad = gst_element_get_pad (vbin, "src"); renatofilho@588: mux_pad = gst_element_get_pad (mux, "video_0"); renatofilho@588: ret = gst_pad_link (aux_pad, mux_pad); renatofilho@588: if (ret != GST_PAD_LINK_OK) { renatofilho@588: g_warning ("Fail link video and mux: %d", ret); renatofilho@588: goto error; renatofilho@588: } renatofilho@588: gst_object_unref (aux_pad); renatofilho@588: gst_object_unref (mux_pad); renatofilho@588: aux_pad = NULL; renatofilho@588: mux_pad = NULL; renatofilho@588: renatofilho@588: //Link mux with sink renatofilho@588: gst_element_link (mux, fdsink); renatofilho@588: renatofilho@588: g_signal_connect (G_OBJECT (decode), renatofilho@588: "new-decoded-pad", renatofilho@588: G_CALLBACK (_decodebin_new_pad_cb), renatofilho@588: self); renatofilho@588: renatofilho@588: g_signal_connect (G_OBJECT (decode), renatofilho@588: "unknown-type", renatofilho@588: G_CALLBACK (_decodebin_unknown_type_cb), renatofilho@588: self); renatofilho@588: renatofilho@588: renatofilho@588: bus = gst_pipeline_get_bus (GST_PIPELINE (pipe)); renatofilho@588: gst_bus_add_watch (bus, _pipeline_bus_cb, self); renatofilho@588: gst_object_unref (bus); renatofilho@588: renatofilho@588: renatofilho@588: priv->pipe = pipe; renatofilho@588: priv->abin = abin; renatofilho@588: priv->vbin = vbin; renatofilho@588: priv->sink = fdsink; renatofilho@588: priv->ready = FALSE; renatofilho@588: renatofilho@588: gst_element_set_state (pipe, GST_STATE_PAUSED); renatofilho@588: return TRUE; renatofilho@588: renatofilho@588: error: renatofilho@588: g_warning ("Invalid uri"); renatofilho@588: renatofilho@588: if (pipe != NULL) { renatofilho@588: gst_object_unref (pipe); renatofilho@588: } renatofilho@588: renatofilho@588: if (src != NULL) { renatofilho@588: gst_object_unref (src); renatofilho@588: } renatofilho@588: renatofilho@588: if (mux != NULL) { renatofilho@588: gst_object_unref (mux); renatofilho@588: } renatofilho@588: renatofilho@588: if (mux_pad != NULL) { renatofilho@588: gst_object_unref (mux_pad); renatofilho@588: } renatofilho@588: renatofilho@588: if (aux_pad != NULL) { renatofilho@588: gst_object_unref (mux_pad); renatofilho@588: } renatofilho@588: renatofilho@588: if (fdsink != NULL) { renatofilho@588: gst_object_unref (fdsink); renatofilho@588: } renatofilho@588: renatofilho@588: if (abin != NULL) { renatofilho@588: gst_object_unref (abin); renatofilho@588: } renatofilho@588: renatofilho@588: if (vbin != NULL) { renatofilho@588: gst_object_unref (vbin); renatofilho@588: } renatofilho@588: renatofilho@588: return FALSE; renatofilho@588: } renatofilho@588: renatofilho@588: gboolean renatofilho@588: g_mencoder_play_stream (GMencoder *self) renatofilho@588: { renatofilho@588: GMencoderPrivate *priv = G_MENCODER_GET_PRIVATE (self); renatofilho@588: renatofilho@588: g_return_val_if_fail (priv->ready == TRUE, FALSE); renatofilho@588: renatofilho@588: if (gst_element_set_state (priv->pipe, GST_STATE_PLAYING) != GST_STATE_CHANGE_FAILURE) { renatofilho@588: g_debug ("PLAYING"); renatofilho@588: return TRUE; renatofilho@588: } renatofilho@588: return FALSE; renatofilho@588: } renatofilho@588: renatofilho@588: gboolean renatofilho@588: g_mencoder_pause_stream (GMencoder *self) renatofilho@588: { renatofilho@588: GMencoderPrivate *priv = G_MENCODER_GET_PRIVATE (self); renatofilho@588: renatofilho@588: g_return_val_if_fail (priv->ready == TRUE, FALSE); renatofilho@588: renatofilho@588: if (gst_element_set_state (priv->pipe, GST_STATE_PAUSED) != GST_STATE_CHANGE_FAILURE) { renatofilho@588: g_debug ("PAUSED"); renatofilho@588: return TRUE; renatofilho@588: } renatofilho@588: return FALSE; renatofilho@588: } renatofilho@588: renatofilho@588: void renatofilho@588: g_mencoder_close_stream (GMencoder *self) renatofilho@588: { renatofilho@588: renatofilho@588: GMencoderPrivate *priv = G_MENCODER_GET_PRIVATE (self); renatofilho@588: renatofilho@588: g_return_if_fail (priv->ready == TRUE); renatofilho@588: renatofilho@588: gst_element_set_state (priv->pipe, GST_STATE_NULL); renatofilho@588: gst_object_unref (priv->pipe); renatofilho@588: priv->pipe = NULL; renatofilho@588: priv->abin = NULL; renatofilho@588: priv->vbin = NULL; renatofilho@588: priv->sink = NULL; renatofilho@588: priv->ready = FALSE; renatofilho@588: } renatofilho@588: renatofilho@588: static gboolean renatofilho@588: _pipeline_bus_cb (GstBus *bus, renatofilho@588: GstMessage *msg, renatofilho@588: gpointer user_data) renatofilho@588: { renatofilho@588: switch (GST_MESSAGE_TYPE (msg)) renatofilho@588: { renatofilho@588: case GST_MESSAGE_STATE_CHANGED: renatofilho@588: { renatofilho@588: GstState oldstate; renatofilho@588: GstState newstate; renatofilho@588: GstState pendingstate; renatofilho@588: renatofilho@588: GMencoderPrivate *priv = G_MENCODER_GET_PRIVATE (user_data); renatofilho@588: renatofilho@588: gst_message_parse_state_changed (msg, &oldstate, renatofilho@588: &newstate, &pendingstate); renatofilho@588: renatofilho@588: if (pendingstate != GST_STATE_VOID_PENDING) renatofilho@588: break; renatofilho@588: renatofilho@588: if ((oldstate == GST_STATE_READY) && renatofilho@588: (newstate == GST_STATE_PAUSED)) { renatofilho@588: if (priv->ready) renatofilho@588: g_signal_emit (user_data, g_mencoder_signals[PAUSED], 0); renatofilho@588: else { renatofilho@588: priv->ready = TRUE; renatofilho@588: g_signal_emit (user_data, g_mencoder_signals[READY], 0); renatofilho@588: } renatofilho@588: } else if ((oldstate == GST_STATE_PAUSED) && renatofilho@588: (newstate == GST_STATE_PLAYING)) { renatofilho@588: g_signal_emit (user_data, g_mencoder_signals[PLAYING], 0); renatofilho@588: } else if ((oldstate == GST_STATE_READY) && renatofilho@588: (newstate == GST_STATE_NULL)) { renatofilho@588: g_signal_emit (user_data, g_mencoder_signals[STOPED], 0); renatofilho@588: } renatofilho@588: break; renatofilho@588: } renatofilho@588: case GST_MESSAGE_ERROR: renatofilho@588: { renatofilho@588: GError *error; renatofilho@588: gchar *debug; renatofilho@588: gchar *err_str; renatofilho@588: renatofilho@588: gst_message_parse_error (msg, &error, &debug); renatofilho@588: err_str = g_strdup_printf ("Error [%d] %s (%s)", error->code, renatofilho@588: error->message, renatofilho@588: debug); renatofilho@588: g_signal_emit (user_data, g_mencoder_signals[ERROR], 0, err_str); renatofilho@588: g_free (err_str); renatofilho@588: g_clear_error (&error); renatofilho@588: g_free (debug); renatofilho@588: break; renatofilho@588: } renatofilho@588: renatofilho@588: case GST_MESSAGE_EOS: renatofilho@588: g_signal_emit (user_data, g_mencoder_signals[EOS], 0); renatofilho@588: break; renatofilho@588: default: renatofilho@588: break; renatofilho@588: } renatofilho@588: return TRUE; renatofilho@588: } renatofilho@588: renatofilho@588: static void renatofilho@588: _decodebin_new_pad_cb (GstElement* object, renatofilho@588: GstPad* pad, renatofilho@588: gboolean flag, renatofilho@588: gpointer user_data) renatofilho@588: { renatofilho@588: GstCaps *caps; renatofilho@588: gchar *str_caps = NULL; renatofilho@588: GMencoderPrivate *priv = G_MENCODER_GET_PRIVATE (user_data); renatofilho@588: renatofilho@588: caps = gst_pad_get_caps (pad); renatofilho@588: str_caps = gst_caps_to_string (caps); renatofilho@588: g_debug ("CAPS : %s", str_caps); renatofilho@588: renatofilho@588: if (strstr (str_caps, "audio") != NULL) { renatofilho@588: GstPad *apad = gst_element_get_pad (priv->abin, "sink"); renatofilho@588: gst_pad_link (pad, apad); renatofilho@588: gst_object_unref (apad); renatofilho@588: } else if (strstr (str_caps, "video") != NULL) { renatofilho@588: GstPad *vpad = gst_element_get_pad (priv->vbin, "sink"); renatofilho@588: gst_pad_link (pad, vpad); renatofilho@588: gst_object_unref (vpad); renatofilho@588: } else { renatofilho@588: g_warning ("invalid caps %s", str_caps); renatofilho@588: } renatofilho@588: renatofilho@588: g_free (str_caps); renatofilho@588: gst_caps_unref (caps); renatofilho@588: } renatofilho@588: renatofilho@588: static void renatofilho@588: _decodebin_unknown_type_cb (GstElement* object, renatofilho@588: GstPad* pad, renatofilho@588: GstCaps* caps, renatofilho@588: gpointer user_data) renatofilho@588: { renatofilho@588: g_warning ("Unknown Type"); renatofilho@588: }