/** * GMyth Library * * @file gmyth/mmyth_tvplayer.c * * @brief

This component provides playback of the remote A/V using * GStreamer. * * Copyright (C) 2006 INdT - Instituto Nokia de Tecnologia. * @author Hallyson Luiz de Morais Melo * *//* * * This program is free software; you can redistribute it and/or modify * it under the terms of the GNU Lesser General Public License as published by * the Free Software Foundation; either version 2 of the License, or * (at your option) any later version. * * This program 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 General Public License for more details. * * You should have received a copy of the GNU Lesser General Public License * along with this program; if not, write to the Free Software * Foundation, Inc., 59 Temple Place, Suite 330, Boston, MA 02111-1307 USA */ #include "mmyth_tvplayer.h" #include #include #define MYTHTV_VERSION_DEFAULT 30 typedef struct _GstPlayerWindowStateChange { GstElement *play; GstState old_state, new_state; MMythTVPlayer *tvplayer; } GstPlayerWindowStateChange; typedef struct _GstPlayerWindowTagFound { GstElement *play; GstTagList *taglist; MMythTVPlayer *tvplayer; } GstPlayerWindowTagFound; /* * static gboolean idle_state (gpointer data); */ static gboolean bus_call(GstBus * bus, GstMessage * msg, gpointer data); static void mmyth_tvplayer_class_init(MMythTVPlayerClass * klass); static void mmyth_tvplayer_init(MMythTVPlayer * object); static void mmyth_tvplayer_dispose(GObject * object); static void mmyth_tvplayer_finalize(GObject * object); G_DEFINE_TYPE(MMythTVPlayer, mmyth_tvplayer, G_TYPE_OBJECT) static gboolean mmyth_tvplayer_create_pipeline(MMythTVPlayer * tvplayer); static void new_pad_cb(GstElement * element, GstPad * pad, gpointer data); static gboolean expose_cb(GtkWidget * widget, GdkEventExpose * event, gpointer user_data); static void mmyth_tvplayer_class_init(MMythTVPlayerClass * klass) { GObjectClass *gobject_class; gobject_class = (GObjectClass *) klass; gobject_class->dispose = mmyth_tvplayer_dispose; gobject_class->finalize = mmyth_tvplayer_finalize; } static void new_pad_cb(GstElement * element, GstPad * pad, gpointer data) { MMythTVPlayer *tvplayer = MMYTH_TVPLAYER(data); GstPadLinkReturn ret; char *s; s = gst_caps_to_string(pad->caps); if (s[0] == 'a') { ret = gst_pad_link(pad, gst_element_get_pad(tvplayer->audioqueue1, "sink")); } else { ret = gst_pad_link(pad, gst_element_get_pad(tvplayer->videoqueue1, "sink")); } g_free(s); } static gboolean expose_cb(GtkWidget * widget, GdkEventExpose * event, gpointer user_data) { MMythTVPlayer *tvplayer = MMYTH_TVPLAYER(user_data); if (tvplayer && tvplayer->videow) { gst_x_overlay_set_xwindow_id(GST_X_OVERLAY (tvplayer->gst_videosink), GDK_WINDOW_XWINDOW(widget->window)); return TRUE; } g_warning("MMythTVPlayer expose called before setting video window\n"); return FALSE; } static void mmyth_tvplayer_init(MMythTVPlayer * tvplayer) { tvplayer->gst_pipeline = NULL; tvplayer->gst_source = NULL; tvplayer->gst_videodec = NULL; tvplayer->gst_videosink = NULL; tvplayer->gst_videocolortrs = NULL; tvplayer->videoqueue1 = NULL; tvplayer->videoqueue2 = NULL; tvplayer->audioqueue1 = NULL; tvplayer->audioqueue2 = NULL; /* * GTKWidget for rendering the video */ tvplayer->videow = NULL; tvplayer->expose_handler = 0; tvplayer->backend_hostname = NULL; tvplayer->backend_port = 0; tvplayer->local_hostname = NULL; tvplayer->recorder = NULL; tvplayer->tvchain = NULL; tvplayer->proginfo = NULL; } static void mmyth_tvplayer_dispose(GObject * object) { G_OBJECT_CLASS(mmyth_tvplayer_parent_class)->dispose(object); } static void mmyth_tvplayer_finalize(GObject * object) { g_signal_handlers_destroy(object); MMythTVPlayer *tvplayer = MMYTH_TVPLAYER(object); g_debug("[%s] Finalizing tvplayer", __FUNCTION__); if (tvplayer->videow != NULL) { if (g_signal_handler_is_connected(tvplayer->videow, tvplayer->expose_handler)) { g_signal_handler_disconnect(tvplayer->videow, tvplayer->expose_handler); } g_object_unref(tvplayer->videow); } if (tvplayer->recorder != NULL) g_object_unref(tvplayer->recorder); if (tvplayer->tvchain != NULL) g_object_unref(tvplayer->tvchain); if (tvplayer->proginfo != NULL) g_object_unref(tvplayer->proginfo); // Release Gstreamer elements if (tvplayer->gst_pipeline != NULL) g_object_unref(tvplayer->gst_pipeline); if (tvplayer->gst_source != NULL) g_object_unref(tvplayer->gst_source); if (tvplayer->gst_videodec != NULL) g_object_unref(tvplayer->gst_videodec); if (tvplayer->gst_videocolortrs != NULL) g_object_unref(tvplayer->gst_videocolortrs); if (tvplayer->gst_videosink != NULL) g_object_unref(tvplayer->gst_videosink); if (tvplayer->videoqueue1 != NULL) g_object_unref(tvplayer->videoqueue1); if (tvplayer->videoqueue2 != NULL) g_object_unref(tvplayer->videoqueue2); if (tvplayer->audioqueue1 != NULL) g_object_unref(tvplayer->audioqueue1); if (tvplayer->audioqueue2 != NULL) g_object_unref(tvplayer->audioqueue2); G_OBJECT_CLASS(mmyth_tvplayer_parent_class)->finalize(object); } /** Creates a new instance of MMythTVPlayer. * * @return a new instance of MMythTVPlayer. */ MMythTVPlayer * mmyth_tvplayer_new() { MMythTVPlayer *tvplayer = MMYTH_TVPLAYER(g_object_new(MMYTH_TVPLAYER_TYPE, NULL)); return tvplayer; } /** Initializes the tv player. * * @param tvplayer the object instance. * @return gboolean TRUE if the pipeline was created * successfully, FALSE otherwise. */ gboolean mmyth_tvplayer_initialize(MMythTVPlayer * tvplayer, GMythBackendInfo * backend_info) { tvplayer->backend_info = backend_info; if (!mmyth_tvplayer_create_pipeline(tvplayer)) { g_warning ("[%s] Error while creating pipeline. TV Player not initialized", __FUNCTION__); return FALSE; } else { g_debug("[%s] GStreamer pipeline created", __FUNCTION__); } return TRUE; } /** Creates the GStreamer pipeline used by the player. * * @param tvplayer the object instance. * @return gboolean TRUE if the pipeline was created * successfully, FALSE otherwise. */ static gboolean mmyth_tvplayer_create_pipeline(MMythTVPlayer * tvplayer) { GstElement *pipeline; GstElement *source, *parser; GstElement *videodec, *videosink; GstElement *videocolortrs; #ifndef MAEMO_PLATFORM GstElement *audiodec, *audioconv, *audioqueue2; #endif GstElement *audiosink; GstElement *videoqueue1, *videoqueue2, *audioqueue1; g_debug("MMythTVPlayer: Setting the Gstreamer pipeline\n"); pipeline = gst_pipeline_new("video-player"); source = gst_element_factory_make("mythtvsrc", "myth-source"); parser = gst_element_factory_make("nuvdemux", "nuv-demux"); /* * Gstreamer Video elements */ videoqueue1 = gst_element_factory_make("queue", "video-queue1"); videodec = gst_element_factory_make("ffdec_mpeg4", "video-decoder"); videoqueue2 = gst_element_factory_make("queue", "video-queue2"); videocolortrs = gst_element_factory_make("ffmpegcolorspace", "image-color-transforms"); #ifdef MAEMO_PLATFORM videosink = gst_element_factory_make("sdlvideosink", "image-output"); #else videosink = gst_element_factory_make("xvimagesink", "image-output"); #endif /* * Gstreamer Audio elements */ audioqueue1 = gst_element_factory_make("queue", "audio-queue1"); #ifdef MAEMO_PLATFORM audiosink = gst_element_factory_make("dspmp3sink", "audio-output"); #else audioqueue2 = gst_element_factory_make("queue", "audio-queue2"); audiodec = gst_element_factory_make("mad", "audio-decoder"); audioconv = gst_element_factory_make("audioconvert", "audio-converter"); audiosink = gst_element_factory_make("alsasink", "audio-output"); #endif if (!(pipeline && source && parser && videodec && videosink) || !(videoqueue1 && videoqueue2 && audioqueue1 && audiosink)) { /* * FIXME: hanlde the error correctly */ /* * video_alignment is not being created (below) and is causing * problems to the ui */ tvplayer->gst_pipeline = NULL; tvplayer->gst_videodec = NULL; tvplayer->gst_videosink = NULL; tvplayer->gst_videocolortrs = NULL; g_warning("GstElement creation error!\n"); return FALSE; } #ifndef MAEMO_PLATFORM if (!(audiodec && audioconv)) { g_warning("GstElement for audio stream creation error!"); return FALSE; } #endif tvplayer->gst_pipeline = pipeline; tvplayer->gst_source = source; tvplayer->gst_videodec = videodec; tvplayer->gst_videosink = videosink; tvplayer->gst_videocolortrs = videocolortrs; g_object_ref(tvplayer->gst_pipeline); g_object_ref(tvplayer->gst_source); g_object_ref(tvplayer->gst_videodec); g_object_ref(tvplayer->gst_videosink); g_object_ref(tvplayer->gst_videocolortrs); tvplayer->videoqueue1 = videoqueue1; tvplayer->videoqueue2 = videoqueue2; tvplayer->audioqueue1 = audioqueue1; g_object_ref(tvplayer->videoqueue1); g_object_ref(tvplayer->videoqueue2); g_object_ref(tvplayer->audioqueue1); #ifndef MAEMO_PLATFORM tvplayer->audioqueue2 = audioqueue2; g_object_ref(tvplayer->audioqueue2); #endif // g_object_set (G_OBJECT (videosink), "sync", TRUE, NULL); g_object_set(G_OBJECT(audiosink), "sync", FALSE, NULL); gst_bus_add_watch(gst_pipeline_get_bus (GST_PIPELINE(tvplayer->gst_pipeline)), bus_call, tvplayer); gst_bin_add_many(GST_BIN(pipeline), source, parser, videoqueue1, videodec, videoqueue2, videocolortrs, videosink, NULL); #ifndef MAEMO_PLATFORM gst_bin_add_many(GST_BIN(pipeline), audioqueue1, audiodec, audioconv, audioqueue2, audiosink, NULL); #else gst_bin_add_many(GST_BIN(pipeline), audioqueue1, audiosink, NULL); #endif { // GstCaps *rtpcaps = gst_caps_new_simple ("application/x-rtp", // NULL); // gst_element_link_filtered(source, parser, rtpcaps); } gst_element_link(source, parser); gst_element_link_many(videoqueue1, videodec, videoqueue2, videocolortrs, videosink, NULL); #ifndef MAEMO_PLATFORM gst_element_link_many(videosink, audioqueue1, audiodec, audioconv, audioqueue2, audiosink, NULL); #else gst_element_link_many(videosink, audioqueue1, audiosink, NULL); #endif g_signal_connect(parser, "pad-added", G_CALLBACK(new_pad_cb), tvplayer); return TRUE; } /** Configures the backend and the tv player * for playing the recorded content A/V. * * FIXME: Change filename to program info or other structure about the recorded * * @param tvplayer the object instance. * @param filename the file uri of the recorded content to be played. * @return TRUE if successfull, FALSE if any error happens. */ gboolean mmyth_tvplayer_record_setup(MMythTVPlayer * tvplayer, const gchar * filename) { // FIXME: we should receive the uri instead of filename const gchar *hostname = gmyth_backend_info_get_hostname(tvplayer->backend_info); const gint port = gmyth_backend_info_get_port(tvplayer->backend_info); GString *fullpath = g_string_new("myth://"); g_string_append_printf(fullpath, "%s:%d/%s", hostname, port, filename); tvplayer->is_livetv = FALSE; g_debug("[%s] Setting record uri to gstreamer pipeline to %s", __FUNCTION__, fullpath->str); g_object_set(G_OBJECT(tvplayer->gst_source), "location", fullpath->str, NULL); return TRUE; } /** Configures the backend and the tv player * for playing the live tv. * * @param tvplayer the object instance. * @return TRUE if successfull, FALSE if any error happens. */ gboolean mmyth_tvplayer_livetv_setup(MMythTVPlayer * tvplayer) { gboolean res = TRUE; tvplayer->livetv = gmyth_livetv_new(); if (!gmyth_livetv_setup(tvplayer->livetv, tvplayer->backend_info)) goto error; return res; error: res = FALSE; if (tvplayer->livetv != NULL) { g_object_unref(tvplayer->livetv); } if (tvplayer->local_hostname != NULL) { g_string_free(tvplayer->local_hostname, TRUE); } if (tvplayer->recorder != NULL) { g_object_unref(tvplayer->recorder); tvplayer->recorder = NULL; } if (tvplayer->tvchain != NULL) { g_object_unref(tvplayer->tvchain); tvplayer->tvchain = NULL; } if (tvplayer->proginfo != NULL) { g_object_unref(tvplayer->proginfo); tvplayer->proginfo = NULL; } return res; } /** Sets the GTK video widget for the tv player. * * @param tvplayer the object instance. * @param videow the GTK video window. * @return TRUE if successfull, FALSE if any error happens. */ gboolean mmyth_tvplayer_set_widget(MMythTVPlayer * tvplayer, GtkWidget * videow) { tvplayer->videow = videow; g_object_ref(videow); g_debug("[%s] Setting widget for tv player render", __FUNCTION__); tvplayer->expose_handler = g_signal_connect(tvplayer->videow, "expose-event", G_CALLBACK(expose_cb), tvplayer); // g_signal_connect(miptv_ui->videow, "size_request", // G_CALLBACK(cb_preferred_video_size), miptv_ui); return TRUE; } static gboolean bus_call(GstBus * bus, GstMessage * msg, gpointer data) { // MMythTVPlayer *tvplayer = MMYTH_TVPLAYER ( data ); // GMainLoop *loop = tvplayer->loop; switch (GST_MESSAGE_TYPE(msg)) { case GST_MESSAGE_EOS: printf("End of stream\n"); // g_idle_add ((GSourceFunc) idle_eos, data); gst_element_set_state(GST_ELEMENT(GST_MESSAGE_SRC(msg)), GST_STATE_NULL); gst_element_set_locked_state(GST_ELEMENT(GST_MESSAGE_SRC(msg)), TRUE); break; case GST_MESSAGE_ERROR: { gchar *debug; GError *err; gst_message_parse_error(msg, &err, &debug); g_free(debug); printf("Error: %s\n", err->message); g_error_free(err); // g_main_loop_quit (loop); } break; default: printf(gst_message_type_get_name(GST_MESSAGE_TYPE(msg))); printf("\n"); break; } return TRUE; } #if 0 static gboolean idle_state(gpointer data) { GstPlayerWindowStateChange *st = data; if (st->old_state == GST_STATE_PLAYING) { if (st->miptv_ui->idle_id != 0) { g_source_remove(st->miptv_ui->idle_id); st->miptv_ui->idle_id = 0; } } else if (st->new_state == GST_STATE_PLAYING) { if (st->miptv_ui->idle_id != 0) g_source_remove(st->miptv_ui->idle_id); st->miptv_ui->idle_id = g_idle_add(cb_iterate, st->miptv_ui); } /* * new movie loaded? */ if (st->old_state == GST_STATE_READY && st->new_state > GST_STATE_READY) { /* * gboolean have_video = FALSE; */ gtk_widget_show(st->miptv_ui->videow); gtk_window_resize(GTK_WINDOW(st->miptv_ui->main_window), 1, 1); } /* * discarded movie? */ if (st->old_state > GST_STATE_READY && st->new_state == GST_STATE_READY) { if (st->miptv_ui->tagcache) { gst_tag_list_free(st->miptv_ui->tagcache); st->miptv_ui->tagcache = NULL; } } gst_object_unref(GST_OBJECT(st->play)); // g_object_unref (G_OBJECT (st->win)); g_free(st); /* * once */ return FALSE; } #endif /** Stops playing the current A/V. * * FIXME: How to proceed differently between livetv * and recorded content? * * @param tvplayer the object instance. * @return void */ void mmyth_tvplayer_stop_playing(MMythTVPlayer * tvplayer) { g_debug("[%s] Setting gstreamer pipeline state to NULL", __FUNCTION__); gst_element_set_state(tvplayer->gst_pipeline, GST_STATE_NULL); if (tvplayer->is_livetv) { if (!gmyth_recorder_stop_livetv(tvplayer->recorder)) { g_warning("[%s] Error while stoping remote encoder", __FUNCTION__); } } } /** Queries if the tvplayer is active playing A/V content. * * @param tvplayer the object instance. * @return TRUE if the tvplayer is active, FALSE otherwise. */ gboolean mmyth_tvplayer_is_playing(MMythTVPlayer * tvplayer) { return (GST_STATE(tvplayer->gst_pipeline) == GST_STATE_PLAYING); } /** Static function that sets the tvplayer state to PLAYING. * * @param tvplayer the object instance. * @return TRUE if the tvplayer is active, FALSE otherwise. */ static gboolean idle_play(gpointer data) { MMythTVPlayer *tvplayer = MMYTH_TVPLAYER(data); g_debug("MMythTVPlayer: Setting pipeline state to PLAYING\n"); gst_element_set_state(tvplayer->gst_pipeline, GST_STATE_PLAYING); return FALSE; } /** Start playing A/V with the tvplayer attributes. * * @param tvplayer the object instance. */ void mmyth_tvplayer_start_playing(MMythTVPlayer * tvplayer) { // FIXME: Move this to livetv_setup?? if (tvplayer->is_livetv) { #if 0 if (!tvplayer || !(tvplayer->proginfo) || !(tvplayer->local_hostname) || !(tvplayer->gst_source)) { g_warning("GMythtvPlayer not ready to start playing\n"); } if (!(tvplayer->proginfo->pathname)) { g_warning ("[%s] Playback url is null, could not play the myth content", __FUNCTION__); return; } g_debug("MMythTVPlayer: Start playing %s", tvplayer->proginfo->pathname->str); #endif g_object_set(G_OBJECT(tvplayer->gst_source), "mythtv-live", TRUE, NULL); #if 0 if (tvplayer->tvchain != NULL) { GString *str_chainid = gmyth_tvchain_get_id(tvplayer->tvchain); g_print("[%s]\tCHAIN ID: %s\n", __FUNCTION__, str_chainid->str); g_object_set(G_OBJECT(tvplayer->gst_source), "mythtv-live-chainid", g_strdup(str_chainid->str), NULL); if (str_chainid != NULL) g_string_free(str_chainid, FALSE); } if (tvplayer->recorder != NULL) g_object_set(G_OBJECT(tvplayer->gst_source), "mythtv-live-id", tvplayer->recorder->recorder_num, NULL); g_debug("[%s] Setting location to %s", __FUNCTION__, tvplayer->proginfo->pathname->str); /* * Sets the gstreamer properties acording to the service access * address */ g_object_set(G_OBJECT(tvplayer->gst_source), "location", tvplayer->proginfo->pathname->str, NULL); #endif } g_object_set(G_OBJECT(tvplayer->gst_source), "mythtv-version", MYTHTV_VERSION_DEFAULT, NULL); g_object_set(G_OBJECT(tvplayer->gst_source), "mythtv-debug", TRUE, NULL); g_idle_add(idle_play, tvplayer); }