#include #include #include #include #include #include #include static int launch_argc; static char **launch_argv; static guint64 iterations = 0; static guint64 sum = 0; static guint64 min = G_MAXINT; static guint64 max = 0; gboolean idle_func (gpointer data) { gboolean busy; struct timeval tfthen, tfnow; guint64 diff; gettimeofday (&tfthen, (struct timezone *)NULL); busy = gst_bin_iterate (GST_BIN (data)); iterations++; gettimeofday (&tfnow, (struct timezone *)NULL); diff = ((guint64)tfnow.tv_sec*1000000LL+tfnow.tv_usec) - ((guint64)tfthen.tv_sec*1000000LL+tfthen.tv_usec); sum += diff; min = MIN (min, diff); max = MAX (max, diff); if (!busy) { gst_main_quit (); g_print ("execution ended after %llu iterations (sum %llu us, average %llu us, min %llu us, max %llu us)\n", iterations, sum, sum/iterations, min, max); } return busy; } /* TODO: write more outputs for ParamSpecs*/ static void property_change_callback (GObject *object, GstObject *orig, GParamSpec *pspec) { #ifdef USE_GLIB2 if (G_IS_PARAM_SPEC_STRING (pspec)) { gchar *str; g_object_get (orig, pspec->name, &str, NULL); g_print ("%s: %s = \"%s\"\n", GST_OBJECT_NAME (orig), pspec->name, str); } else if (G_IS_PARAM_SPEC_CHAR (pspec)) { gchar str; g_object_get (orig, pspec->name, &str, NULL); g_print ("%s: %s = \"%c\"\n", GST_OBJECT_NAME (orig), pspec->name, str); } else if (G_IS_PARAM_SPEC_INT (pspec)) { gint i; g_object_get (orig, pspec->name, &i, NULL); g_print ("%s: %s = %d\n", GST_OBJECT_NAME (orig), pspec->name, i); } else if (G_IS_PARAM_SPEC_INT64 (pspec)) { gint64 i; g_object_get (orig, pspec->name, &i, NULL); g_print ("%s: %s = %lld\n", GST_OBJECT_NAME (orig), pspec->name, i); } else if (G_IS_PARAM_SPEC_UINT (pspec)) { guint i; g_object_get (orig, pspec->name, &i, NULL); g_print ("%s: %s = %u\n", GST_OBJECT_NAME (orig), pspec->name, i); } else if (G_IS_PARAM_SPEC_UINT (pspec)) { guint64 i; g_object_get (orig, pspec->name, &i, NULL); g_print ("%s: %s = %llu\n", GST_OBJECT_NAME (orig), pspec->name, i); } else if (G_IS_PARAM_SPEC_ENUM (pspec)) { guint64 i; g_object_get (orig, pspec->name, &i, NULL); g_print ("%s: %s = \"%llu\"\n", GST_OBJECT_NAME (orig), pspec->name, i); } else if (G_IS_PARAM_SPEC_FLOAT (pspec)) { gfloat i; g_object_get (orig, pspec->name, &i, NULL); g_print ("%s: %s = %f\n", GST_OBJECT_NAME (orig), pspec->name, i); } else if (G_IS_PARAM_SPEC_DOUBLE (pspec)) { gdouble i; g_object_get (orig, pspec->name, &i, NULL); g_print ("%s: %s = %f\n", GST_OBJECT_NAME (orig), pspec->name, i); } else { g_print ("%s: changed \"%s\"\n", GST_OBJECT_NAME (orig), pspec->name); } #endif /* USE_GLIB2 */ } static void print_props (gpointer data, gpointer user_data) { GstPropsEntry *entry = (GstPropsEntry *)data; GstElement *element = GST_ELEMENT (user_data); g_print ("deprecated: %s: %s: ", gst_element_get_name (element), g_quark_to_string (entry->propid)); switch (entry->propstype) { case GST_PROPS_INT_ID: g_print ("%d\n", entry->data.int_data); break; case GST_PROPS_STRING_ID: g_print ("%s\n", entry->data.string_data.string); break; case GST_PROPS_FLOAT_ID: g_print ("%f\n", entry->data.float_data); break; default: g_print ("unknown\n"); } } static void event_func (GstElement *element, GstEvent *event) { GstProps *props; if (event == NULL) return; if (GST_EVENT_TYPE (event) == GST_EVENT_INFO) { props = GST_EVENT_INFO_PROPS (event); g_list_foreach (props->properties, print_props, GST_EVENT_SRC (event)); } } static GstElement* xmllaunch_parse_cmdline (const gchar *argv[]) { GstElement *pipeline = NULL, *e; GstXML *xml; gboolean err; const gchar *arg; gchar *element, *property, *value; GList *l; gint i = 0; if (!(arg = argv[0])) { g_print ("usage: gst-xmllaunch [ element.property=value ... ]\n", arg[0]); exit (1); } xml = gst_xml_new (); err = gst_xml_parse_file(xml, arg, NULL); if (err != TRUE) { fprintf (stderr, "ERROR: parse of xml file '%s' failed\n", arg); exit (1); } l = gst_xml_get_topelements (xml); if (!l) { fprintf (stderr, "ERROR: no toplevel pipeline element in file '%s'\n", arg); exit (1); } if (l->next) g_warning ("only one toplevel element is supported at this time"); pipeline = GST_ELEMENT (l->data); while ((arg = argv[++i])) { element = g_strdup (arg); property = strchr (element, '.'); value = strchr (element, '='); if (!(element < property && property < value)) { fprintf (stderr, "ERROR: could not parse command line argument %d: %s", i, element); g_free (element); exit (1); } *property++ = '\0'; *value++ = '\0'; e = gst_bin_get_by_name (GST_BIN (pipeline), element); if (!e) { g_warning ("element named '%s' not found", element); } else { gst_util_set_object_arg (G_OBJECT (e), property, value); } g_free (element); } if (!l) return NULL; else return l->data; } int main(int argc, char *argv[]) { GstElement *pipeline; gchar **argvn; gchar *cmdline; int i; gboolean save_pipeline = FALSE; gboolean run_pipeline = TRUE; gchar *savefile = ""; free (malloc (8)); /* -lefence */ gst_init (&argc, &argv); if (argc >= 3 && !strcmp(argv[1], "-o")) { save_pipeline = TRUE; run_pipeline = FALSE; savefile = argv[2]; argv[2] = argv[0]; argv+=2; argc-=2; } launch_argc = argc; launch_argv = argv; /* make a null-terminated version of argv */ argvn = g_new0 (char *,argc); memcpy (argvn, argv+1, sizeof (char*) * (argc-1)); if (strstr (argv[0], "gst-xmllaunch")) { pipeline = xmllaunch_parse_cmdline (argvn); } else { pipeline = (GstElement*) gst_parse_launchv (argvn); } if (!pipeline) { fprintf(stderr, "ERROR: pipeline could not be constructed\n"); exit(1); } g_signal_connect (G_OBJECT (pipeline), "event", G_CALLBACK (event_func), NULL); g_signal_connect (pipeline, "deep_notify", G_CALLBACK (property_change_callback), NULL); #ifndef GST_DISABLE_LOADSAVE if (save_pipeline) { gst_xml_write_file (GST_ELEMENT (pipeline), fopen (savefile, "w")); } #endif if (run_pipeline) { gst_buffer_print_stats(); fprintf(stderr,"RUNNING pipeline\n"); if (gst_element_set_state (pipeline, GST_STATE_PLAYING) != GST_STATE_SUCCESS) { fprintf(stderr,"pipeline doesn't want to play\n"); exit (-1); } g_idle_add (idle_func, pipeline); gst_main (); gst_element_set_state (pipeline, GST_STATE_NULL); gst_buffer_print_stats(); } gst_object_unref (GST_OBJECT (pipeline)); return 0; }