mirror of
https://gitlab.freedesktop.org/gstreamer/gstreamer.git
synced 2025-01-03 05:59:10 +00:00
90beb712a0
When printing a GstStructure property (e.g. the "stats" property in rtpsession) the first field is printed on the same line of the type description, and this is both inconsistent compared to how Enum values are printed and confusing as the reader might miss the first field. To fix this, add a newline before printing GstStructure fields in properties. NOTE: this does not change the existing inconsistent behavior of an extra newline *after* a GstStructure property, but the latter is not as annoying and it would take more effort to fix because GstStructure fields are printed in CAPS descriptions too.
2163 lines
66 KiB
C
2163 lines
66 KiB
C
/* GStreamer
|
|
* Copyright (C) 1999,2000 Erik Walthinsen <omega@cse.ogi.edu>
|
|
* 2000 Wim Taymans <wtay@chello.be>
|
|
* 2004 Thomas Vander Stichele <thomas@apestaart.org>
|
|
* 2018 Collabora Ltd.
|
|
*
|
|
* gst-inspect.c: tool to inspect the GStreamer registry
|
|
*
|
|
* This library is free software; you can redistribute it and/or
|
|
* modify it under the terms of the GNU Library General Public
|
|
* License as published by the Free Software Foundation; either
|
|
* version 2 of the License, or (at your option) any later version.
|
|
*
|
|
* This library 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
|
|
* Library General Public License for more details.
|
|
*
|
|
* You should have received a copy of the GNU Library General Public
|
|
* License along with this library; if not, write to the
|
|
* Free Software Foundation, Inc., 51 Franklin St, Fifth Floor,
|
|
* Boston, MA 02110-1301, USA.
|
|
*/
|
|
|
|
#ifdef HAVE_CONFIG_H
|
|
# include "config.h"
|
|
#endif
|
|
|
|
/* FIXME 2.0: suppress warnings for deprecated API such as GValueArray
|
|
* with newer GLib versions (>= 2.31.0) */
|
|
#define GLIB_DISABLE_DEPRECATION_WARNINGS
|
|
|
|
#include "tools.h"
|
|
#include <gst/gst_private.h> /* for internal Factories */
|
|
|
|
#include <string.h>
|
|
#include <locale.h>
|
|
#include <glib/gprintf.h>
|
|
#ifdef G_OS_UNIX
|
|
# include <unistd.h>
|
|
# include <sys/wait.h>
|
|
#endif
|
|
|
|
#define DEFAULT_PAGER "less"
|
|
|
|
/* "R" : support color
|
|
* "X" : do not clear the screen when leaving the pager
|
|
* "F" : skip the pager if content fit into the screen
|
|
*/
|
|
#define DEFAULT_LESS_OPTS "RXF"
|
|
|
|
gboolean colored_output = TRUE;
|
|
|
|
#ifdef G_OS_UNIX
|
|
GPid child_pid = -1;
|
|
#endif
|
|
GMainLoop *loop = NULL;
|
|
|
|
/* Console colors */
|
|
|
|
/* Escape values for colors */
|
|
#define BLUE "\033[34m"
|
|
#define BRBLUE "\033[94m"
|
|
#define BRCYAN "\033[96m"
|
|
#define BRMAGENTA "\033[95m"
|
|
#define BRYELLOW "\033[33m"
|
|
#define CYAN "\033[36m"
|
|
#define GREEN "\033[32m"
|
|
#define MAGENTA "\033[35m"
|
|
#define YELLOW "\033[33m"
|
|
|
|
/* General colors */
|
|
#define RESET_COLOR (colored_output? "\033[0m": "")
|
|
#define HEADING_COLOR (colored_output? BRYELLOW : "")
|
|
#define PROP_NAME_COLOR (colored_output? BRBLUE : "")
|
|
#define PROP_VALUE_COLOR (colored_output? RESET_COLOR: "")
|
|
#define PROP_ATTR_NAME_COLOR (colored_output? BRYELLOW : "")
|
|
#define PROP_ATTR_VALUE_COLOR (colored_output? CYAN: "")
|
|
/* FIXME: find a good color that works on both dark & light bg. */
|
|
#define DESC_COLOR (colored_output? RESET_COLOR: "")
|
|
|
|
/* Datatype-related colors */
|
|
#define DATATYPE_COLOR (colored_output? GREEN : "")
|
|
#define CHILD_LINK_COLOR (colored_output? BRMAGENTA : "")
|
|
|
|
/* Caps colors */
|
|
#define FIELD_NAME_COLOR (colored_output? CYAN: "")
|
|
#define FIELD_VALUE_COLOR (colored_output? BRBLUE : "")
|
|
#define CAPS_TYPE_COLOR (colored_output? YELLOW : "")
|
|
#define STRUCT_NAME_COLOR (colored_output? YELLOW : "")
|
|
#define CAPS_FEATURE_COLOR (colored_output? GREEN : "")
|
|
|
|
/* Plugin listing colors */
|
|
#define PLUGIN_NAME_COLOR (colored_output? BRBLUE : "")
|
|
#define ELEMENT_NAME_COLOR (colored_output? GREEN : "")
|
|
/* FIXME: find a good color that works on both dark & light bg. */
|
|
#define ELEMENT_DETAIL_COLOR (colored_output? RESET_COLOR : "")
|
|
#define PLUGIN_FEATURE_COLOR (colored_output? BRBLUE: "")
|
|
|
|
/* Feature listing colors */
|
|
#define FEATURE_NAME_COLOR (colored_output? GREEN : "")
|
|
#define FEATURE_DIR_COLOR (colored_output? BRMAGENTA : "")
|
|
#define FEATURE_RANK_COLOR (colored_output? CYAN : "")
|
|
#define FEATURE_PROTO_COLOR (colored_output? BRYELLOW : "")
|
|
|
|
static char *_name = NULL;
|
|
static int indent = 0;
|
|
|
|
static int print_element_info (GstPluginFeature * feature,
|
|
gboolean print_names);
|
|
static int print_typefind_info (GstPluginFeature * feature,
|
|
gboolean print_names);
|
|
static int print_tracer_info (GstPluginFeature * feature, gboolean print_names);
|
|
|
|
#define push_indent() push_indent_n(1)
|
|
#define pop_indent() push_indent_n(-1)
|
|
#define pop_indent_n(n) push_indent_n(-n)
|
|
|
|
static void
|
|
push_indent_n (int n)
|
|
{
|
|
g_assert (n > 0 || indent > 0);
|
|
indent += n;
|
|
}
|
|
|
|
/* *INDENT-OFF* */
|
|
G_GNUC_PRINTF (1, 2)
|
|
/* *INDENT-ON* */
|
|
|
|
static void
|
|
n_print (const char *format, ...)
|
|
{
|
|
va_list args;
|
|
int i;
|
|
|
|
if (_name)
|
|
g_print ("%s", _name);
|
|
|
|
for (i = 0; i < indent; ++i)
|
|
g_print (" ");
|
|
|
|
va_start (args, format);
|
|
g_vprintf (format, args);
|
|
va_end (args);
|
|
}
|
|
|
|
static gboolean
|
|
print_field (GQuark field, const GValue * value, gpointer pfx)
|
|
{
|
|
gchar *str = gst_value_serialize (value);
|
|
|
|
n_print ("%s %s%15s%s: %s%s%s\n",
|
|
(gchar *) pfx, FIELD_NAME_COLOR, g_quark_to_string (field), RESET_COLOR,
|
|
FIELD_VALUE_COLOR, str, RESET_COLOR);
|
|
g_free (str);
|
|
return TRUE;
|
|
}
|
|
|
|
static void
|
|
print_caps (const GstCaps * caps, const gchar * pfx)
|
|
{
|
|
guint i;
|
|
|
|
g_return_if_fail (caps != NULL);
|
|
|
|
if (gst_caps_is_any (caps)) {
|
|
n_print ("%s%sANY%s\n", CAPS_TYPE_COLOR, pfx, RESET_COLOR);
|
|
return;
|
|
}
|
|
if (gst_caps_is_empty (caps)) {
|
|
n_print ("%s%sEMPTY%s\n", CAPS_TYPE_COLOR, pfx, RESET_COLOR);
|
|
return;
|
|
}
|
|
|
|
for (i = 0; i < gst_caps_get_size (caps); i++) {
|
|
GstStructure *structure = gst_caps_get_structure (caps, i);
|
|
GstCapsFeatures *features = gst_caps_get_features (caps, i);
|
|
|
|
if (features && (gst_caps_features_is_any (features) ||
|
|
!gst_caps_features_is_equal (features,
|
|
GST_CAPS_FEATURES_MEMORY_SYSTEM_MEMORY))) {
|
|
gchar *features_string = gst_caps_features_to_string (features);
|
|
|
|
n_print ("%s%s%s%s(%s%s%s)\n", pfx, STRUCT_NAME_COLOR,
|
|
gst_structure_get_name (structure), RESET_COLOR,
|
|
CAPS_FEATURE_COLOR, features_string, RESET_COLOR);
|
|
g_free (features_string);
|
|
} else {
|
|
n_print ("%s%s%s%s\n", pfx, STRUCT_NAME_COLOR,
|
|
gst_structure_get_name (structure), RESET_COLOR);
|
|
}
|
|
gst_structure_foreach (structure, print_field, (gpointer) pfx);
|
|
}
|
|
}
|
|
|
|
static const char *
|
|
get_rank_name (char *s, gint rank)
|
|
{
|
|
static const int ranks[4] = {
|
|
GST_RANK_NONE, GST_RANK_MARGINAL, GST_RANK_SECONDARY, GST_RANK_PRIMARY
|
|
};
|
|
static const char *rank_names[4] = { "none", "marginal", "secondary",
|
|
"primary"
|
|
};
|
|
int i;
|
|
int best_i;
|
|
|
|
best_i = 0;
|
|
for (i = 0; i < 4; i++) {
|
|
if (rank == ranks[i])
|
|
return rank_names[i];
|
|
if (abs (rank - ranks[i]) < abs (rank - ranks[best_i])) {
|
|
best_i = i;
|
|
}
|
|
}
|
|
|
|
sprintf (s, "%s %c %d", rank_names[best_i],
|
|
(rank - ranks[best_i] > 0) ? '+' : '-', abs (ranks[best_i] - rank));
|
|
|
|
return s;
|
|
}
|
|
|
|
static void
|
|
print_factory_details_info (GstElementFactory * factory)
|
|
{
|
|
gchar **keys, **k;
|
|
GstRank rank;
|
|
char s[20];
|
|
|
|
rank = gst_plugin_feature_get_rank (GST_PLUGIN_FEATURE (factory));
|
|
n_print ("%sFactory Details:%s\n", HEADING_COLOR, RESET_COLOR);
|
|
|
|
push_indent ();
|
|
n_print ("%s%-25s%s%s (%d)%s\n", PROP_NAME_COLOR, "Rank", PROP_VALUE_COLOR,
|
|
get_rank_name (s, rank), rank, RESET_COLOR);
|
|
|
|
keys = gst_element_factory_get_metadata_keys (factory);
|
|
if (keys != NULL) {
|
|
for (k = keys; *k != NULL; ++k) {
|
|
const gchar *val;
|
|
gchar *key = *k;
|
|
|
|
val = gst_element_factory_get_metadata (factory, key);
|
|
key[0] = g_ascii_toupper (key[0]);
|
|
n_print ("%s%-25s%s%s%s\n", PROP_NAME_COLOR, key, PROP_VALUE_COLOR, val,
|
|
RESET_COLOR);
|
|
}
|
|
g_strfreev (keys);
|
|
}
|
|
pop_indent ();
|
|
n_print ("\n");
|
|
}
|
|
|
|
static void
|
|
print_hierarchy (GType type, gint level, gint * maxlevel)
|
|
{
|
|
GType parent;
|
|
gint i;
|
|
|
|
parent = g_type_parent (type);
|
|
|
|
*maxlevel = *maxlevel + 1;
|
|
level++;
|
|
|
|
if (parent)
|
|
print_hierarchy (parent, level, maxlevel);
|
|
|
|
if (_name)
|
|
g_print ("%s%s%s", DATATYPE_COLOR, _name, RESET_COLOR);
|
|
|
|
for (i = 1; i < *maxlevel - level; i++)
|
|
g_print (" ");
|
|
if (*maxlevel - level)
|
|
g_print (" %s+----%s", CHILD_LINK_COLOR, RESET_COLOR);
|
|
|
|
g_print ("%s%s%s\n", DATATYPE_COLOR, g_type_name (type), RESET_COLOR);
|
|
|
|
if (level == 1)
|
|
n_print ("\n");
|
|
}
|
|
|
|
static void
|
|
print_interfaces (GType type)
|
|
{
|
|
guint n_ifaces;
|
|
GType *iface, *ifaces = g_type_interfaces (type, &n_ifaces);
|
|
|
|
if (ifaces) {
|
|
if (n_ifaces) {
|
|
n_print (_("%sImplemented Interfaces%s:\n"), HEADING_COLOR, RESET_COLOR);
|
|
push_indent ();
|
|
iface = ifaces;
|
|
while (*iface) {
|
|
n_print ("%s%s%s\n", DATATYPE_COLOR, g_type_name (*iface), RESET_COLOR);
|
|
iface++;
|
|
}
|
|
pop_indent ();
|
|
n_print ("\n");
|
|
}
|
|
g_free (ifaces);
|
|
}
|
|
}
|
|
|
|
static gchar *
|
|
flags_to_string (GFlagsValue * vals, guint flags)
|
|
{
|
|
GString *s = NULL;
|
|
guint flags_left, i;
|
|
|
|
/* first look for an exact match and count the number of values */
|
|
for (i = 0; vals[i].value_name != NULL; ++i) {
|
|
if (vals[i].value == flags)
|
|
return g_strdup (vals[i].value_nick);
|
|
}
|
|
|
|
s = g_string_new (NULL);
|
|
|
|
/* we assume the values are sorted from lowest to highest value */
|
|
flags_left = flags;
|
|
while (i > 0) {
|
|
--i;
|
|
if (vals[i].value != 0 && (flags_left & vals[i].value) == vals[i].value) {
|
|
if (s->len > 0)
|
|
g_string_append_c (s, '+');
|
|
g_string_append (s, vals[i].value_nick);
|
|
flags_left -= vals[i].value;
|
|
if (flags_left == 0)
|
|
break;
|
|
}
|
|
}
|
|
|
|
if (s->len == 0)
|
|
g_string_assign (s, "(none)");
|
|
|
|
return g_string_free (s, FALSE);
|
|
}
|
|
|
|
#define KNOWN_PARAM_FLAGS \
|
|
(G_PARAM_CONSTRUCT | G_PARAM_CONSTRUCT_ONLY | \
|
|
G_PARAM_LAX_VALIDATION | G_PARAM_STATIC_STRINGS | \
|
|
G_PARAM_READABLE | G_PARAM_WRITABLE | G_PARAM_DEPRECATED | \
|
|
GST_PARAM_CONTROLLABLE | GST_PARAM_MUTABLE_PLAYING | \
|
|
GST_PARAM_MUTABLE_PAUSED | GST_PARAM_MUTABLE_READY)
|
|
|
|
static int
|
|
sort_gparamspecs (GParamSpec ** a, GParamSpec ** b)
|
|
{
|
|
return g_strcmp0 (g_param_spec_get_name (*a), g_param_spec_get_name (*b));
|
|
}
|
|
|
|
/* obj will be NULL if we're printing properties of pad template pads */
|
|
static void
|
|
print_object_properties_info (GObject * obj, GObjectClass * obj_class,
|
|
const gchar * desc)
|
|
{
|
|
GParamSpec **property_specs;
|
|
guint num_properties, i;
|
|
gboolean readable;
|
|
gboolean first_flag;
|
|
|
|
property_specs = g_object_class_list_properties (obj_class, &num_properties);
|
|
g_qsort_with_data (property_specs, num_properties, sizeof (gpointer),
|
|
(GCompareDataFunc) sort_gparamspecs, NULL);
|
|
|
|
n_print ("%s%s%s:\n", HEADING_COLOR, desc, RESET_COLOR);
|
|
|
|
push_indent ();
|
|
|
|
for (i = 0; i < num_properties; i++) {
|
|
GValue value = { 0, };
|
|
GParamSpec *param = property_specs[i];
|
|
GType owner_type = param->owner_type;
|
|
|
|
/* We're printing pad properties */
|
|
if (obj == NULL && (owner_type == G_TYPE_OBJECT
|
|
|| owner_type == GST_TYPE_OBJECT || owner_type == GST_TYPE_PAD))
|
|
continue;
|
|
|
|
g_value_init (&value, param->value_type);
|
|
|
|
n_print ("%s%-20s%s: %s%s%s\n", PROP_NAME_COLOR,
|
|
g_param_spec_get_name (param), RESET_COLOR, PROP_VALUE_COLOR,
|
|
g_param_spec_get_blurb (param), RESET_COLOR);
|
|
|
|
push_indent_n (11);
|
|
|
|
first_flag = TRUE;
|
|
n_print ("%sflags%s: ", PROP_ATTR_NAME_COLOR, RESET_COLOR);
|
|
readable = ! !(param->flags & G_PARAM_READABLE);
|
|
if (readable && obj != NULL) {
|
|
g_object_get_property (obj, param->name, &value);
|
|
} else {
|
|
/* if we can't read the property value, assume it's set to the default
|
|
* (which might not be entirely true for sub-classes, but that's an
|
|
* unlikely corner-case anyway) */
|
|
g_param_value_set_default (param, &value);
|
|
}
|
|
if (readable) {
|
|
g_print ("%s%s%s%s", (first_flag) ? "" : ", ", PROP_ATTR_VALUE_COLOR,
|
|
_("readable"), RESET_COLOR);
|
|
first_flag = FALSE;
|
|
}
|
|
if (param->flags & G_PARAM_WRITABLE) {
|
|
g_print ("%s%s%s%s", (first_flag) ? "" : ", ", PROP_ATTR_VALUE_COLOR,
|
|
_("writable"), RESET_COLOR);
|
|
first_flag = FALSE;
|
|
}
|
|
if (param->flags & G_PARAM_DEPRECATED) {
|
|
g_print ("%s%s%s%s", (first_flag) ? "" : ", ", PROP_ATTR_VALUE_COLOR,
|
|
_("deprecated"), RESET_COLOR);
|
|
first_flag = FALSE;
|
|
}
|
|
if (param->flags & GST_PARAM_CONTROLLABLE) {
|
|
g_print (", %s%s%s", PROP_ATTR_VALUE_COLOR, _("controllable"),
|
|
RESET_COLOR);
|
|
first_flag = FALSE;
|
|
}
|
|
if (param->flags & GST_PARAM_MUTABLE_PLAYING) {
|
|
g_print (", %s%s%s", PROP_ATTR_VALUE_COLOR,
|
|
_("changeable in NULL, READY, PAUSED or PLAYING state"), RESET_COLOR);
|
|
} else if (param->flags & GST_PARAM_MUTABLE_PAUSED) {
|
|
g_print (", %s%s%s", PROP_ATTR_VALUE_COLOR,
|
|
_("changeable only in NULL, READY or PAUSED state"), RESET_COLOR);
|
|
} else if (param->flags & GST_PARAM_MUTABLE_READY) {
|
|
g_print (", %s%s%s", PROP_ATTR_VALUE_COLOR,
|
|
_("changeable only in NULL or READY state"), RESET_COLOR);
|
|
}
|
|
if (param->flags & ~KNOWN_PARAM_FLAGS) {
|
|
g_print ("%s0x%s%0x%s", (first_flag) ? "" : ", ", PROP_ATTR_VALUE_COLOR,
|
|
param->flags & ~KNOWN_PARAM_FLAGS, RESET_COLOR);
|
|
}
|
|
g_print ("\n");
|
|
|
|
switch (G_VALUE_TYPE (&value)) {
|
|
case G_TYPE_STRING:
|
|
{
|
|
const char *string_val = g_value_get_string (&value);
|
|
|
|
n_print ("%sString%s. ", DATATYPE_COLOR, RESET_COLOR);
|
|
|
|
if (string_val == NULL)
|
|
g_print ("%sDefault%s: %snull%s", PROP_ATTR_NAME_COLOR, RESET_COLOR,
|
|
PROP_ATTR_VALUE_COLOR, RESET_COLOR);
|
|
else
|
|
g_print ("%sDefault%s: %s\"%s\"%s", PROP_ATTR_NAME_COLOR, RESET_COLOR,
|
|
PROP_ATTR_VALUE_COLOR, string_val, RESET_COLOR);
|
|
break;
|
|
}
|
|
case G_TYPE_BOOLEAN:
|
|
{
|
|
gboolean bool_val = g_value_get_boolean (&value);
|
|
|
|
n_print ("%sBoolean%s. %sDefault%s: %s%s%s", DATATYPE_COLOR,
|
|
RESET_COLOR, PROP_ATTR_NAME_COLOR, RESET_COLOR,
|
|
PROP_ATTR_VALUE_COLOR, bool_val ? "true" : "false", RESET_COLOR);
|
|
break;
|
|
}
|
|
case G_TYPE_ULONG:
|
|
{
|
|
GParamSpecULong *pulong = G_PARAM_SPEC_ULONG (param);
|
|
|
|
n_print
|
|
("%sUnsigned Long%s. %sRange%s: %s%lu - %lu%s %sDefault%s: %s%lu%s ",
|
|
DATATYPE_COLOR, RESET_COLOR, PROP_ATTR_NAME_COLOR, RESET_COLOR,
|
|
PROP_ATTR_VALUE_COLOR, pulong->minimum, pulong->maximum,
|
|
RESET_COLOR, PROP_ATTR_NAME_COLOR, RESET_COLOR,
|
|
PROP_ATTR_VALUE_COLOR, g_value_get_ulong (&value), RESET_COLOR);
|
|
|
|
GST_ERROR ("%s: property '%s' of type ulong: consider changing to "
|
|
"uint/uint64", G_OBJECT_CLASS_NAME (obj_class),
|
|
g_param_spec_get_name (param));
|
|
break;
|
|
}
|
|
case G_TYPE_LONG:
|
|
{
|
|
GParamSpecLong *plong = G_PARAM_SPEC_LONG (param);
|
|
|
|
n_print ("%sLong%s. %sRange%s: %s%ld - %ld%s %sDefault%s: %s%ld%s ",
|
|
DATATYPE_COLOR, RESET_COLOR, PROP_ATTR_NAME_COLOR, RESET_COLOR,
|
|
PROP_ATTR_VALUE_COLOR, plong->minimum, plong->maximum, RESET_COLOR,
|
|
PROP_ATTR_NAME_COLOR, RESET_COLOR, PROP_ATTR_VALUE_COLOR,
|
|
g_value_get_long (&value), RESET_COLOR);
|
|
|
|
GST_ERROR ("%s: property '%s' of type long: consider changing to "
|
|
"int/int64", G_OBJECT_CLASS_NAME (obj_class),
|
|
g_param_spec_get_name (param));
|
|
break;
|
|
}
|
|
case G_TYPE_UINT:
|
|
{
|
|
GParamSpecUInt *puint = G_PARAM_SPEC_UINT (param);
|
|
|
|
n_print
|
|
("%sUnsigned Integer%s. %sRange%s: %s%u - %u%s %sDefault%s: %s%u%s ",
|
|
DATATYPE_COLOR, RESET_COLOR, PROP_ATTR_NAME_COLOR, RESET_COLOR,
|
|
PROP_ATTR_VALUE_COLOR, puint->minimum, puint->maximum, RESET_COLOR,
|
|
PROP_ATTR_NAME_COLOR, RESET_COLOR, PROP_ATTR_VALUE_COLOR,
|
|
g_value_get_uint (&value), RESET_COLOR);
|
|
break;
|
|
}
|
|
case G_TYPE_INT:
|
|
{
|
|
GParamSpecInt *pint = G_PARAM_SPEC_INT (param);
|
|
|
|
n_print ("%sInteger%s. %sRange%s: %s%d - %d%s %sDefault%s: %s%d%s ",
|
|
DATATYPE_COLOR, RESET_COLOR, PROP_ATTR_NAME_COLOR, RESET_COLOR,
|
|
PROP_ATTR_VALUE_COLOR, pint->minimum, pint->maximum, RESET_COLOR,
|
|
PROP_ATTR_NAME_COLOR, RESET_COLOR, PROP_ATTR_VALUE_COLOR,
|
|
g_value_get_int (&value), RESET_COLOR);
|
|
break;
|
|
}
|
|
case G_TYPE_UINT64:
|
|
{
|
|
GParamSpecUInt64 *puint64 = G_PARAM_SPEC_UINT64 (param);
|
|
|
|
n_print ("%sUnsigned Integer64%s. %sRange%s: %s%" G_GUINT64_FORMAT " - "
|
|
"%" G_GUINT64_FORMAT "%s %sDefault%s: %s%" G_GUINT64_FORMAT "%s ",
|
|
DATATYPE_COLOR, RESET_COLOR, PROP_ATTR_NAME_COLOR, RESET_COLOR,
|
|
PROP_ATTR_VALUE_COLOR, puint64->minimum, puint64->maximum,
|
|
RESET_COLOR, PROP_ATTR_NAME_COLOR, RESET_COLOR,
|
|
PROP_ATTR_VALUE_COLOR, g_value_get_uint64 (&value), RESET_COLOR);
|
|
break;
|
|
}
|
|
case G_TYPE_INT64:
|
|
{
|
|
GParamSpecInt64 *pint64 = G_PARAM_SPEC_INT64 (param);
|
|
|
|
n_print ("%sInteger64%s. %sRange%s: %s%" G_GINT64_FORMAT " - %"
|
|
G_GINT64_FORMAT "%s %sDefault%s: %s%" G_GINT64_FORMAT "%s ",
|
|
DATATYPE_COLOR, RESET_COLOR, PROP_ATTR_NAME_COLOR, RESET_COLOR,
|
|
PROP_ATTR_VALUE_COLOR, pint64->minimum, pint64->maximum,
|
|
RESET_COLOR, PROP_ATTR_NAME_COLOR, RESET_COLOR,
|
|
PROP_ATTR_VALUE_COLOR, g_value_get_int64 (&value), RESET_COLOR);
|
|
break;
|
|
}
|
|
case G_TYPE_FLOAT:
|
|
{
|
|
GParamSpecFloat *pfloat = G_PARAM_SPEC_FLOAT (param);
|
|
|
|
n_print ("%sFloat%s. %sRange%s: %s%15.7g - %15.7g%s "
|
|
"%sDefault%s: %s%15.7g%s ", DATATYPE_COLOR, RESET_COLOR,
|
|
PROP_ATTR_NAME_COLOR, RESET_COLOR, PROP_ATTR_VALUE_COLOR,
|
|
pfloat->minimum, pfloat->maximum, RESET_COLOR, PROP_ATTR_NAME_COLOR,
|
|
RESET_COLOR, PROP_ATTR_VALUE_COLOR, g_value_get_float (&value),
|
|
RESET_COLOR);
|
|
break;
|
|
}
|
|
case G_TYPE_DOUBLE:
|
|
{
|
|
GParamSpecDouble *pdouble = G_PARAM_SPEC_DOUBLE (param);
|
|
|
|
n_print ("%sDouble%s. %sRange%s: %s%15.7g - %15.7g%s "
|
|
"%sDefault%s: %s%15.7g%s ", DATATYPE_COLOR, RESET_COLOR,
|
|
PROP_ATTR_NAME_COLOR, RESET_COLOR, PROP_ATTR_VALUE_COLOR,
|
|
pdouble->minimum, pdouble->maximum, RESET_COLOR,
|
|
PROP_ATTR_NAME_COLOR, RESET_COLOR, PROP_ATTR_VALUE_COLOR,
|
|
g_value_get_double (&value), RESET_COLOR);
|
|
break;
|
|
}
|
|
case G_TYPE_CHAR:
|
|
case G_TYPE_UCHAR:
|
|
GST_ERROR ("%s: property '%s' of type char: consider changing to "
|
|
"int/string", G_OBJECT_CLASS_NAME (obj_class),
|
|
g_param_spec_get_name (param));
|
|
/* fall through */
|
|
default:
|
|
if (param->value_type == GST_TYPE_CAPS) {
|
|
const GstCaps *caps = gst_value_get_caps (&value);
|
|
|
|
if (!caps)
|
|
n_print ("%sCaps%s (NULL)", DATATYPE_COLOR, RESET_COLOR);
|
|
else {
|
|
print_caps (caps, " ");
|
|
}
|
|
} else if (G_IS_PARAM_SPEC_ENUM (param)) {
|
|
GEnumValue *values;
|
|
guint j = 0;
|
|
gint enum_value;
|
|
const gchar *value_nick = "";
|
|
|
|
values = G_ENUM_CLASS (g_type_class_ref (param->value_type))->values;
|
|
enum_value = g_value_get_enum (&value);
|
|
|
|
while (values[j].value_name) {
|
|
if (values[j].value == enum_value)
|
|
value_nick = values[j].value_nick;
|
|
j++;
|
|
}
|
|
|
|
n_print ("%sEnum \"%s\"%s %sDefault%s: %s%d, \"%s\"%s",
|
|
DATATYPE_COLOR, g_type_name (G_VALUE_TYPE (&value)), RESET_COLOR,
|
|
PROP_ATTR_NAME_COLOR, RESET_COLOR, PROP_ATTR_VALUE_COLOR,
|
|
enum_value, value_nick, RESET_COLOR);
|
|
|
|
j = 0;
|
|
while (values[j].value_name) {
|
|
g_print ("\n");
|
|
n_print (" %s(%d)%s: %s%-16s%s - %s%s%s",
|
|
PROP_ATTR_NAME_COLOR, values[j].value, RESET_COLOR,
|
|
PROP_ATTR_VALUE_COLOR, values[j].value_nick, RESET_COLOR,
|
|
DESC_COLOR, values[j].value_name, RESET_COLOR);
|
|
j++;
|
|
}
|
|
/* g_type_class_unref (ec); */
|
|
} else if (G_IS_PARAM_SPEC_FLAGS (param)) {
|
|
GParamSpecFlags *pflags = G_PARAM_SPEC_FLAGS (param);
|
|
GFlagsValue *vals;
|
|
gchar *cur;
|
|
|
|
vals = pflags->flags_class->values;
|
|
|
|
cur = flags_to_string (vals, g_value_get_flags (&value));
|
|
|
|
n_print ("%sFlags \"%s\"%s %sDefault%s: %s0x%08x, \"%s\"%s",
|
|
DATATYPE_COLOR, g_type_name (G_VALUE_TYPE (&value)), RESET_COLOR,
|
|
PROP_ATTR_NAME_COLOR, RESET_COLOR, PROP_ATTR_VALUE_COLOR,
|
|
g_value_get_flags (&value), cur, RESET_COLOR);
|
|
|
|
while (vals[0].value_name) {
|
|
g_print ("\n");
|
|
n_print (" %s(0x%08x)%s: %s%-16s%s - %s%s%s",
|
|
PROP_ATTR_NAME_COLOR, vals[0].value, RESET_COLOR,
|
|
PROP_ATTR_VALUE_COLOR, vals[0].value_nick, RESET_COLOR,
|
|
DESC_COLOR, vals[0].value_name, RESET_COLOR);
|
|
++vals;
|
|
}
|
|
|
|
g_free (cur);
|
|
} else if (G_IS_PARAM_SPEC_OBJECT (param)) {
|
|
n_print ("%sObject of type%s %s\"%s\"%s", PROP_VALUE_COLOR,
|
|
RESET_COLOR, DATATYPE_COLOR,
|
|
g_type_name (param->value_type), RESET_COLOR);
|
|
} else if (G_IS_PARAM_SPEC_BOXED (param)) {
|
|
n_print ("%sBoxed pointer of type%s %s\"%s\"%s", PROP_VALUE_COLOR,
|
|
RESET_COLOR, DATATYPE_COLOR,
|
|
g_type_name (param->value_type), RESET_COLOR);
|
|
if (param->value_type == GST_TYPE_STRUCTURE) {
|
|
const GstStructure *s = gst_value_get_structure (&value);
|
|
if (s) {
|
|
g_print ("\n");
|
|
gst_structure_foreach (s, print_field,
|
|
(gpointer) " ");
|
|
}
|
|
}
|
|
} else if (G_IS_PARAM_SPEC_POINTER (param)) {
|
|
if (param->value_type != G_TYPE_POINTER) {
|
|
n_print ("%sPointer of type%s %s\"%s\"%s.", PROP_VALUE_COLOR,
|
|
RESET_COLOR, DATATYPE_COLOR, g_type_name (param->value_type),
|
|
RESET_COLOR);
|
|
} else {
|
|
n_print ("%sPointer.%s", PROP_VALUE_COLOR, RESET_COLOR);
|
|
}
|
|
} else if (param->value_type == G_TYPE_VALUE_ARRAY) {
|
|
GParamSpecValueArray *pvarray = G_PARAM_SPEC_VALUE_ARRAY (param);
|
|
|
|
if (pvarray->element_spec) {
|
|
n_print ("%sArray of GValues of type%s %s\"%s\"%s",
|
|
PROP_VALUE_COLOR, RESET_COLOR, DATATYPE_COLOR,
|
|
g_type_name (pvarray->element_spec->value_type), RESET_COLOR);
|
|
} else {
|
|
n_print ("%sArray of GValues%s", PROP_VALUE_COLOR, RESET_COLOR);
|
|
}
|
|
} else if (GST_IS_PARAM_SPEC_FRACTION (param)) {
|
|
GstParamSpecFraction *pfraction = GST_PARAM_SPEC_FRACTION (param);
|
|
|
|
n_print ("%sFraction%s. %sRange%s: %s%d/%d - %d/%d%s "
|
|
"%sDefault%s: %s%d/%d%s ", DATATYPE_COLOR, RESET_COLOR,
|
|
PROP_ATTR_NAME_COLOR, RESET_COLOR, PROP_ATTR_VALUE_COLOR,
|
|
pfraction->min_num, pfraction->min_den, pfraction->max_num,
|
|
pfraction->max_den, RESET_COLOR, PROP_ATTR_NAME_COLOR,
|
|
RESET_COLOR, PROP_ATTR_VALUE_COLOR,
|
|
gst_value_get_fraction_numerator (&value),
|
|
gst_value_get_fraction_denominator (&value), RESET_COLOR);
|
|
} else if (param->value_type == GST_TYPE_ARRAY) {
|
|
GstParamSpecArray *parray = GST_PARAM_SPEC_ARRAY_LIST (param);
|
|
|
|
if (parray->element_spec) {
|
|
n_print ("%sGstValueArray of GValues of type%s %s\"%s\"%s",
|
|
PROP_VALUE_COLOR, RESET_COLOR, DATATYPE_COLOR,
|
|
g_type_name (parray->element_spec->value_type), RESET_COLOR);
|
|
} else {
|
|
n_print ("%sGstValueArray of GValues%s", PROP_VALUE_COLOR,
|
|
RESET_COLOR);
|
|
}
|
|
} else {
|
|
n_print ("%sUnknown type %ld%s %s\"%s\"%s", PROP_VALUE_COLOR,
|
|
(glong) param->value_type, RESET_COLOR, DATATYPE_COLOR,
|
|
g_type_name (param->value_type), RESET_COLOR);
|
|
}
|
|
break;
|
|
}
|
|
if (!readable)
|
|
g_print (" %sWrite only%s\n", PROP_VALUE_COLOR, RESET_COLOR);
|
|
else
|
|
g_print ("\n");
|
|
|
|
pop_indent_n (11);
|
|
|
|
g_value_reset (&value);
|
|
}
|
|
if (num_properties == 0)
|
|
n_print ("%snone%s\n", PROP_VALUE_COLOR, RESET_COLOR);
|
|
|
|
pop_indent ();
|
|
|
|
g_free (property_specs);
|
|
}
|
|
|
|
static void
|
|
print_element_properties_info (GstElement * element)
|
|
{
|
|
g_print ("\n");
|
|
print_object_properties_info (G_OBJECT (element),
|
|
G_OBJECT_GET_CLASS (element), "Element Properties");
|
|
}
|
|
|
|
static void
|
|
print_pad_templates_info (GstElement * element, GstElementFactory * factory)
|
|
{
|
|
const GList *pads;
|
|
GstStaticPadTemplate *padtemplate;
|
|
GstPadTemplate *tmpl;
|
|
|
|
n_print ("%sPad Templates%s:\n", HEADING_COLOR, RESET_COLOR);
|
|
|
|
push_indent ();
|
|
|
|
if (gst_element_factory_get_num_pad_templates (factory) == 0) {
|
|
n_print ("%snone%s\n", PROP_VALUE_COLOR, RESET_COLOR);
|
|
goto done;
|
|
}
|
|
|
|
pads = gst_element_factory_get_static_pad_templates (factory);
|
|
while (pads) {
|
|
padtemplate = (GstStaticPadTemplate *) (pads->data);
|
|
pads = g_list_next (pads);
|
|
|
|
if (padtemplate->direction == GST_PAD_SRC)
|
|
n_print ("%sSRC template%s: %s'%s'%s\n", PROP_NAME_COLOR, RESET_COLOR,
|
|
PROP_VALUE_COLOR, padtemplate->name_template, RESET_COLOR);
|
|
else if (padtemplate->direction == GST_PAD_SINK)
|
|
n_print ("%sSINK template%s: %s'%s'%s\n", PROP_NAME_COLOR, RESET_COLOR,
|
|
PROP_VALUE_COLOR, padtemplate->name_template, RESET_COLOR);
|
|
else
|
|
n_print ("%sUNKNOWN template%s: %s'%s'%s\n", PROP_NAME_COLOR, RESET_COLOR,
|
|
PROP_VALUE_COLOR, padtemplate->name_template, RESET_COLOR);
|
|
|
|
push_indent ();
|
|
|
|
if (padtemplate->presence == GST_PAD_ALWAYS)
|
|
n_print ("%sAvailability%s: %sAlways%s\n", PROP_NAME_COLOR, RESET_COLOR,
|
|
PROP_VALUE_COLOR, RESET_COLOR);
|
|
else if (padtemplate->presence == GST_PAD_SOMETIMES)
|
|
n_print ("%sAvailability%s: %sSometimes%s\n", PROP_NAME_COLOR,
|
|
RESET_COLOR, PROP_VALUE_COLOR, RESET_COLOR);
|
|
else if (padtemplate->presence == GST_PAD_REQUEST) {
|
|
n_print ("%sAvailability%s: %sOn request%s\n", PROP_NAME_COLOR,
|
|
RESET_COLOR, PROP_VALUE_COLOR, RESET_COLOR);
|
|
} else
|
|
n_print ("%sAvailability%s: %sUNKNOWN%s\n", PROP_NAME_COLOR, RESET_COLOR,
|
|
PROP_VALUE_COLOR, RESET_COLOR);
|
|
|
|
if (padtemplate->static_caps.string) {
|
|
GstCaps *caps = gst_static_caps_get (&padtemplate->static_caps);
|
|
|
|
n_print ("%sCapabilities%s:\n", PROP_NAME_COLOR, RESET_COLOR);
|
|
|
|
push_indent ();
|
|
print_caps (caps, ""); // FIXME
|
|
pop_indent ();
|
|
|
|
gst_caps_unref (caps);
|
|
}
|
|
|
|
tmpl = gst_element_class_get_pad_template (GST_ELEMENT_GET_CLASS (element),
|
|
padtemplate->name_template);
|
|
if (tmpl != NULL) {
|
|
GType pad_type = GST_PAD_TEMPLATE_GTYPE (tmpl);
|
|
|
|
if (pad_type != G_TYPE_NONE && pad_type != GST_TYPE_PAD) {
|
|
gpointer pad_klass;
|
|
|
|
pad_klass = g_type_class_ref (pad_type);
|
|
n_print ("%sType%s: %s%s%s\n", PROP_NAME_COLOR, RESET_COLOR,
|
|
DATATYPE_COLOR, g_type_name (pad_type), RESET_COLOR);
|
|
print_object_properties_info (NULL, pad_klass, "Pad Properties");
|
|
g_type_class_unref (pad_klass);
|
|
}
|
|
}
|
|
|
|
pop_indent ();
|
|
|
|
if (pads != NULL)
|
|
n_print ("\n");
|
|
}
|
|
|
|
done:
|
|
pop_indent ();
|
|
}
|
|
|
|
static void
|
|
print_clocking_info (GstElement * element)
|
|
{
|
|
gboolean requires_clock, provides_clock;
|
|
|
|
requires_clock =
|
|
GST_OBJECT_FLAG_IS_SET (element, GST_ELEMENT_FLAG_REQUIRE_CLOCK);
|
|
provides_clock =
|
|
GST_OBJECT_FLAG_IS_SET (element, GST_ELEMENT_FLAG_PROVIDE_CLOCK);
|
|
|
|
if (!requires_clock && !provides_clock) {
|
|
n_print ("\n");
|
|
n_print ("%sElement has no clocking capabilities.%s\n", DESC_COLOR,
|
|
RESET_COLOR);
|
|
return;
|
|
}
|
|
|
|
n_print ("\n");
|
|
n_print ("%sClocking Interaction%s:\n", PROP_NAME_COLOR, RESET_COLOR);
|
|
|
|
push_indent ();
|
|
|
|
if (requires_clock) {
|
|
n_print ("%selement requires a clock%s\n", PROP_VALUE_COLOR, RESET_COLOR);
|
|
}
|
|
|
|
if (provides_clock) {
|
|
GstClock *clock;
|
|
|
|
clock = gst_element_get_clock (element);
|
|
if (clock) {
|
|
n_print ("%selement provides a clock%s: %s%s%s\n", PROP_VALUE_COLOR,
|
|
RESET_COLOR, DATATYPE_COLOR, GST_OBJECT_NAME (clock), RESET_COLOR);
|
|
gst_object_unref (clock);
|
|
} else
|
|
n_print ("%selement is supposed to provide a clock but returned NULL%s\n",
|
|
PROP_VALUE_COLOR, RESET_COLOR);
|
|
}
|
|
|
|
pop_indent ();
|
|
}
|
|
|
|
static void
|
|
print_uri_handler_info (GstElement * element)
|
|
{
|
|
if (GST_IS_URI_HANDLER (element)) {
|
|
const gchar *const *uri_protocols;
|
|
const gchar *uri_type;
|
|
|
|
if (gst_uri_handler_get_uri_type (GST_URI_HANDLER (element)) == GST_URI_SRC)
|
|
uri_type = "source";
|
|
else if (gst_uri_handler_get_uri_type (GST_URI_HANDLER (element)) ==
|
|
GST_URI_SINK)
|
|
uri_type = "sink";
|
|
else
|
|
uri_type = "unknown";
|
|
|
|
uri_protocols = gst_uri_handler_get_protocols (GST_URI_HANDLER (element));
|
|
|
|
n_print ("\n");
|
|
n_print ("%sURI handling capabilities:%s\n", HEADING_COLOR, RESET_COLOR);
|
|
|
|
push_indent ();
|
|
|
|
n_print ("%sElement can act as %s.%s\n", DESC_COLOR, uri_type, RESET_COLOR);
|
|
|
|
if (uri_protocols && *uri_protocols) {
|
|
n_print ("%sSupported URI protocols%s:\n", DESC_COLOR, RESET_COLOR);
|
|
push_indent ();
|
|
for (; *uri_protocols != NULL; uri_protocols++)
|
|
n_print ("%s%s%s\n", PROP_ATTR_VALUE_COLOR, *uri_protocols,
|
|
RESET_COLOR);
|
|
pop_indent ();
|
|
} else {
|
|
n_print ("%sNo supported URI protocols%s\n", PROP_VALUE_COLOR,
|
|
RESET_COLOR);
|
|
}
|
|
|
|
pop_indent ();
|
|
} else {
|
|
n_print ("%sElement has no URI handling capabilities.%s\n", DESC_COLOR,
|
|
RESET_COLOR);
|
|
}
|
|
}
|
|
|
|
static void
|
|
print_pad_info (GstElement * element)
|
|
{
|
|
const GList *pads;
|
|
GstPad *pad;
|
|
|
|
n_print ("\n");
|
|
n_print ("%sPads:%s\n", HEADING_COLOR, RESET_COLOR);
|
|
|
|
push_indent ();
|
|
|
|
if (!element->numpads) {
|
|
n_print ("%snone%s\n", PROP_VALUE_COLOR, RESET_COLOR);
|
|
goto done;
|
|
}
|
|
|
|
pads = element->pads;
|
|
while (pads) {
|
|
gchar *name;
|
|
GstCaps *caps;
|
|
|
|
pad = GST_PAD (pads->data);
|
|
pads = g_list_next (pads);
|
|
|
|
name = gst_pad_get_name (pad);
|
|
if (gst_pad_get_direction (pad) == GST_PAD_SRC)
|
|
n_print ("%sSRC%s: %s'%s'%s\n", PROP_NAME_COLOR, RESET_COLOR,
|
|
PROP_VALUE_COLOR, name, RESET_COLOR);
|
|
else if (gst_pad_get_direction (pad) == GST_PAD_SINK)
|
|
n_print ("%sSINK%s: %s'%s'%s\n", PROP_NAME_COLOR, RESET_COLOR,
|
|
PROP_VALUE_COLOR, name, RESET_COLOR);
|
|
else
|
|
n_print ("%sUNKNOWN%s: %s'%s'%s\n", PROP_NAME_COLOR, RESET_COLOR,
|
|
PROP_VALUE_COLOR, name, RESET_COLOR);
|
|
|
|
g_free (name);
|
|
|
|
if (pad->padtemplate) {
|
|
push_indent ();
|
|
n_print ("%sPad Template%s: %s'%s'%s\n", PROP_NAME_COLOR, RESET_COLOR,
|
|
PROP_VALUE_COLOR, pad->padtemplate->name_template, RESET_COLOR);
|
|
pop_indent ();
|
|
}
|
|
|
|
caps = gst_pad_get_current_caps (pad);
|
|
if (caps) {
|
|
n_print ("%sCapabilities:%s\n", PROP_NAME_COLOR, RESET_COLOR);
|
|
push_indent ();
|
|
print_caps (caps, ""); // FIXME
|
|
pop_indent ();
|
|
gst_caps_unref (caps);
|
|
}
|
|
}
|
|
|
|
done:
|
|
pop_indent ();
|
|
}
|
|
|
|
static gboolean
|
|
has_sometimes_template (GstElement * element)
|
|
{
|
|
GstElementClass *klass = GST_ELEMENT_GET_CLASS (element);
|
|
GList *l;
|
|
|
|
for (l = klass->padtemplates; l != NULL; l = l->next) {
|
|
if (GST_PAD_TEMPLATE (l->data)->presence == GST_PAD_SOMETIMES)
|
|
return TRUE;
|
|
}
|
|
|
|
return FALSE;
|
|
}
|
|
|
|
static gboolean
|
|
gtype_needs_ptr_marker (GType type)
|
|
{
|
|
if (type == G_TYPE_POINTER)
|
|
return FALSE;
|
|
|
|
if (G_TYPE_FUNDAMENTAL (type) == G_TYPE_POINTER || G_TYPE_IS_BOXED (type)
|
|
|| G_TYPE_IS_OBJECT (type))
|
|
return TRUE;
|
|
|
|
return FALSE;
|
|
}
|
|
|
|
static void
|
|
print_signal_info (GstElement * element)
|
|
{
|
|
/* Signals/Actions Block */
|
|
guint *signals;
|
|
guint nsignals;
|
|
gint i = 0, j, k;
|
|
GSignalQuery *query = NULL;
|
|
GType type;
|
|
GSList *found_signals, *l;
|
|
|
|
for (k = 0; k < 2; k++) {
|
|
found_signals = NULL;
|
|
|
|
/* For elements that have sometimes pads, also list a few useful GstElement
|
|
* signals. Put these first, so element-specific ones come later. */
|
|
if (k == 0 && has_sometimes_template (element)) {
|
|
query = g_new0 (GSignalQuery, 1);
|
|
g_signal_query (g_signal_lookup ("pad-added", GST_TYPE_ELEMENT), query);
|
|
found_signals = g_slist_append (found_signals, query);
|
|
query = g_new0 (GSignalQuery, 1);
|
|
g_signal_query (g_signal_lookup ("pad-removed", GST_TYPE_ELEMENT), query);
|
|
found_signals = g_slist_append (found_signals, query);
|
|
query = g_new0 (GSignalQuery, 1);
|
|
g_signal_query (g_signal_lookup ("no-more-pads", GST_TYPE_ELEMENT),
|
|
query);
|
|
found_signals = g_slist_append (found_signals, query);
|
|
}
|
|
|
|
for (type = G_OBJECT_TYPE (element); type; type = g_type_parent (type)) {
|
|
if (type == GST_TYPE_ELEMENT || type == GST_TYPE_OBJECT)
|
|
break;
|
|
|
|
if (type == GST_TYPE_BIN && G_OBJECT_TYPE (element) != GST_TYPE_BIN)
|
|
continue;
|
|
|
|
signals = g_signal_list_ids (type, &nsignals);
|
|
for (i = 0; i < nsignals; i++) {
|
|
query = g_new0 (GSignalQuery, 1);
|
|
g_signal_query (signals[i], query);
|
|
|
|
if ((k == 0 && !(query->signal_flags & G_SIGNAL_ACTION)) ||
|
|
(k == 1 && (query->signal_flags & G_SIGNAL_ACTION)))
|
|
found_signals = g_slist_append (found_signals, query);
|
|
else
|
|
g_free (query);
|
|
}
|
|
g_free (signals);
|
|
signals = NULL;
|
|
}
|
|
|
|
if (found_signals) {
|
|
n_print ("\n");
|
|
if (k == 0)
|
|
n_print ("%sElement Signals%s:\n", HEADING_COLOR, RESET_COLOR);
|
|
else
|
|
n_print ("%sElement Actions%s:\n", HEADING_COLOR, RESET_COLOR);
|
|
} else {
|
|
continue;
|
|
}
|
|
|
|
for (l = found_signals; l; l = l->next) {
|
|
gchar *indent;
|
|
const gchar *pmark;
|
|
int indent_len;
|
|
|
|
query = (GSignalQuery *) l->data;
|
|
indent_len = strlen (query->signal_name) +
|
|
strlen (g_type_name (query->return_type)) + 24;
|
|
|
|
if (gtype_needs_ptr_marker (query->return_type)) {
|
|
pmark = "* ";
|
|
indent_len += 2;
|
|
} else {
|
|
pmark = " ";
|
|
}
|
|
|
|
indent = g_new0 (gchar, indent_len + 1);
|
|
memset (indent, ' ', indent_len);
|
|
|
|
n_print (" %s\"%s\"%s : %s%s%s%suser_function%s (%s%s%s* object%s",
|
|
PROP_NAME_COLOR, query->signal_name, RESET_COLOR,
|
|
DATATYPE_COLOR, g_type_name (query->return_type), PROP_VALUE_COLOR,
|
|
pmark, RESET_COLOR, DATATYPE_COLOR, g_type_name (type),
|
|
PROP_VALUE_COLOR, RESET_COLOR);
|
|
|
|
for (j = 0; j < query->n_params; j++) {
|
|
const gchar *type_name, *asterisk;
|
|
|
|
type_name = g_type_name (query->param_types[j]);
|
|
asterisk = gtype_needs_ptr_marker (query->param_types[j]) ? "*" : "";
|
|
|
|
g_print (",\n");
|
|
n_print ("%s%s%s%s%s arg%d%s", indent, DATATYPE_COLOR, type_name,
|
|
PROP_VALUE_COLOR, asterisk, j, RESET_COLOR);
|
|
}
|
|
|
|
if (k == 0) {
|
|
g_print (",\n");
|
|
n_print ("%s%sgpointer %suser_data%s);\n", indent, DATATYPE_COLOR,
|
|
PROP_VALUE_COLOR, RESET_COLOR);
|
|
} else
|
|
g_print (");\n");
|
|
|
|
g_free (indent);
|
|
}
|
|
|
|
if (found_signals) {
|
|
g_slist_foreach (found_signals, (GFunc) g_free, NULL);
|
|
g_slist_free (found_signals);
|
|
}
|
|
}
|
|
}
|
|
|
|
static void
|
|
print_children_info (GstElement * element)
|
|
{
|
|
GList *children;
|
|
|
|
if (!GST_IS_BIN (element))
|
|
return;
|
|
|
|
children = (GList *) GST_BIN (element)->children;
|
|
if (children) {
|
|
n_print ("\n");
|
|
n_print ("%sChildren%s:\n", HEADING_COLOR, RESET_COLOR);
|
|
}
|
|
|
|
while (children) {
|
|
n_print (" %s%s%s\n", DATATYPE_COLOR,
|
|
GST_ELEMENT_NAME (GST_ELEMENT (children->data)), RESET_COLOR);
|
|
children = g_list_next (children);
|
|
}
|
|
}
|
|
|
|
static void
|
|
print_preset_list (GstElement * element)
|
|
{
|
|
gchar **presets, **preset;
|
|
|
|
if (!GST_IS_PRESET (element))
|
|
return;
|
|
|
|
presets = gst_preset_get_preset_names (GST_PRESET (element));
|
|
if (presets && *presets) {
|
|
n_print ("\n");
|
|
n_print ("%sPresets%s:\n", HEADING_COLOR, RESET_COLOR);
|
|
for (preset = presets; *preset; preset++) {
|
|
n_print (" \"%s\"\n", *preset);
|
|
}
|
|
g_strfreev (presets);
|
|
}
|
|
}
|
|
|
|
static void
|
|
print_blacklist (void)
|
|
{
|
|
GList *plugins, *cur;
|
|
gint count = 0;
|
|
|
|
g_print ("%s%s%s\n", HEADING_COLOR, _("Blacklisted files:"), RESET_COLOR);
|
|
|
|
plugins = gst_registry_get_plugin_list (gst_registry_get ());
|
|
for (cur = plugins; cur != NULL; cur = g_list_next (cur)) {
|
|
GstPlugin *plugin = (GstPlugin *) (cur->data);
|
|
if (GST_OBJECT_FLAG_IS_SET (plugin, GST_PLUGIN_FLAG_BLACKLISTED)) {
|
|
g_print (" %s\n", gst_plugin_get_name (plugin));
|
|
count++;
|
|
}
|
|
}
|
|
|
|
g_print ("\n");
|
|
g_print (_("%sTotal count%s: %s"), PROP_NAME_COLOR, RESET_COLOR,
|
|
PROP_VALUE_COLOR);
|
|
g_print (ngettext ("%d blacklisted file", "%d blacklisted files", count),
|
|
count);
|
|
g_print ("%s\n", RESET_COLOR);
|
|
gst_plugin_list_free (plugins);
|
|
}
|
|
|
|
static void
|
|
print_typefind_extensions (const gchar * const *extensions, const gchar * color)
|
|
{
|
|
guint i = 0;
|
|
|
|
while (extensions[i]) {
|
|
g_print ("%s%s%s%s", i > 0 ? ", " : "", color, extensions[i], RESET_COLOR);
|
|
i++;
|
|
}
|
|
}
|
|
|
|
static void
|
|
print_element_list (gboolean print_all, gchar * ftypes)
|
|
{
|
|
int plugincount = 0, featurecount = 0, blacklistcount = 0;
|
|
GList *plugins, *orig_plugins;
|
|
gchar **types = NULL;
|
|
|
|
if (ftypes) {
|
|
gint i;
|
|
|
|
types = g_strsplit (ftypes, "/", -1);
|
|
for (i = 0; types[i]; i++)
|
|
*types[i] = g_ascii_toupper (*types[i]);
|
|
|
|
}
|
|
|
|
orig_plugins = plugins = gst_registry_get_plugin_list (gst_registry_get ());
|
|
while (plugins) {
|
|
GList *features, *orig_features;
|
|
GstPlugin *plugin;
|
|
|
|
plugin = (GstPlugin *) (plugins->data);
|
|
plugins = g_list_next (plugins);
|
|
plugincount++;
|
|
|
|
if (GST_OBJECT_FLAG_IS_SET (plugin, GST_PLUGIN_FLAG_BLACKLISTED)) {
|
|
blacklistcount++;
|
|
continue;
|
|
}
|
|
|
|
orig_features = features =
|
|
gst_registry_get_feature_list_by_plugin (gst_registry_get (),
|
|
gst_plugin_get_name (plugin));
|
|
while (features) {
|
|
GstPluginFeature *feature;
|
|
|
|
if (G_UNLIKELY (features->data == NULL))
|
|
goto next;
|
|
feature = GST_PLUGIN_FEATURE (features->data);
|
|
featurecount++;
|
|
|
|
if (GST_IS_ELEMENT_FACTORY (feature)) {
|
|
const gchar *klass;
|
|
GstElementFactory *factory;
|
|
|
|
factory = GST_ELEMENT_FACTORY (feature);
|
|
if (types) {
|
|
gint i;
|
|
gboolean all_found = TRUE;
|
|
|
|
klass =
|
|
gst_element_factory_get_metadata (factory,
|
|
GST_ELEMENT_METADATA_KLASS);
|
|
for (i = 0; types[i]; i++) {
|
|
if (!strstr (klass, types[i])) {
|
|
all_found = FALSE;
|
|
break;
|
|
}
|
|
}
|
|
|
|
if (!all_found)
|
|
goto next;
|
|
}
|
|
if (print_all)
|
|
print_element_info (feature, TRUE);
|
|
else
|
|
g_print ("%s%s%s: %s%s%s: %s%s%s\n", PLUGIN_NAME_COLOR,
|
|
gst_plugin_get_name (plugin), RESET_COLOR, ELEMENT_NAME_COLOR,
|
|
GST_OBJECT_NAME (factory), RESET_COLOR, ELEMENT_DETAIL_COLOR,
|
|
gst_element_factory_get_metadata (factory,
|
|
GST_ELEMENT_METADATA_LONGNAME), RESET_COLOR);
|
|
} else if (GST_IS_TYPE_FIND_FACTORY (feature)) {
|
|
GstTypeFindFactory *factory;
|
|
const gchar *const *extensions;
|
|
|
|
if (types)
|
|
goto next;
|
|
factory = GST_TYPE_FIND_FACTORY (feature);
|
|
if (!print_all)
|
|
g_print ("%s%s%s: %s%s%s: ", PLUGIN_NAME_COLOR,
|
|
gst_plugin_get_name (plugin), RESET_COLOR, ELEMENT_NAME_COLOR,
|
|
gst_plugin_feature_get_name (feature), RESET_COLOR);
|
|
|
|
extensions = gst_type_find_factory_get_extensions (factory);
|
|
if (extensions != NULL) {
|
|
if (!print_all) {
|
|
print_typefind_extensions (extensions, ELEMENT_DETAIL_COLOR);
|
|
g_print ("\n");
|
|
}
|
|
} else {
|
|
if (!print_all)
|
|
g_print ("%sno extensions%s\n", ELEMENT_DETAIL_COLOR, RESET_COLOR);
|
|
}
|
|
} else {
|
|
if (types)
|
|
goto next;
|
|
if (!print_all)
|
|
n_print ("%s%s%s: %s%s%s (%s%s%s)\n", PLUGIN_NAME_COLOR,
|
|
gst_plugin_get_name (plugin), RESET_COLOR, ELEMENT_NAME_COLOR,
|
|
GST_OBJECT_NAME (feature), RESET_COLOR, ELEMENT_DETAIL_COLOR,
|
|
g_type_name (G_OBJECT_TYPE (feature)), RESET_COLOR);
|
|
}
|
|
|
|
next:
|
|
features = g_list_next (features);
|
|
}
|
|
|
|
gst_plugin_feature_list_free (orig_features);
|
|
}
|
|
|
|
gst_plugin_list_free (orig_plugins);
|
|
g_strfreev (types);
|
|
|
|
g_print ("\n");
|
|
g_print (_("%sTotal count%s: %s"), PROP_NAME_COLOR, RESET_COLOR,
|
|
PROP_VALUE_COLOR);
|
|
g_print (ngettext ("%d plugin", "%d plugins", plugincount), plugincount);
|
|
if (blacklistcount) {
|
|
g_print (" (");
|
|
g_print (ngettext ("%d blacklist entry", "%d blacklist entries",
|
|
blacklistcount), blacklistcount);
|
|
g_print (" not shown)");
|
|
}
|
|
g_print ("%s, %s", RESET_COLOR, PROP_VALUE_COLOR);
|
|
g_print (ngettext ("%d feature", "%d features", featurecount), featurecount);
|
|
g_print ("%s\n", RESET_COLOR);
|
|
}
|
|
|
|
static void
|
|
print_all_uri_handlers (void)
|
|
{
|
|
GList *plugins, *p, *features, *f;
|
|
|
|
plugins = gst_registry_get_plugin_list (gst_registry_get ());
|
|
|
|
for (p = plugins; p; p = p->next) {
|
|
GstPlugin *plugin = (GstPlugin *) (p->data);
|
|
|
|
features =
|
|
gst_registry_get_feature_list_by_plugin (gst_registry_get (),
|
|
gst_plugin_get_name (plugin));
|
|
|
|
for (f = features; f; f = f->next) {
|
|
GstPluginFeature *feature = GST_PLUGIN_FEATURE (f->data);
|
|
|
|
if (GST_IS_ELEMENT_FACTORY (feature)) {
|
|
GstElementFactory *factory;
|
|
GstElement *element;
|
|
|
|
factory = GST_ELEMENT_FACTORY (gst_plugin_feature_load (feature));
|
|
if (!factory) {
|
|
g_print ("element plugin %s couldn't be loaded\n",
|
|
gst_plugin_get_name (plugin));
|
|
continue;
|
|
}
|
|
|
|
element = gst_element_factory_create (factory, NULL);
|
|
if (!element) {
|
|
g_print ("couldn't construct element for %s for some reason\n",
|
|
GST_OBJECT_NAME (factory));
|
|
gst_object_unref (factory);
|
|
continue;
|
|
}
|
|
|
|
if (GST_IS_URI_HANDLER (element)) {
|
|
const gchar *const *uri_protocols;
|
|
const gchar *const *protocol;
|
|
const gchar *dir;
|
|
|
|
switch (gst_uri_handler_get_uri_type (GST_URI_HANDLER (element))) {
|
|
case GST_URI_SRC:
|
|
dir = "read";
|
|
break;
|
|
case GST_URI_SINK:
|
|
dir = "write";
|
|
break;
|
|
default:
|
|
dir = "unknown";
|
|
break;
|
|
}
|
|
|
|
g_print ("%s%s%s (%s%s%s, %srank %u%s): ",
|
|
FEATURE_NAME_COLOR,
|
|
gst_plugin_feature_get_name (GST_PLUGIN_FEATURE (factory)),
|
|
RESET_COLOR, FEATURE_DIR_COLOR, dir, RESET_COLOR,
|
|
FEATURE_RANK_COLOR,
|
|
gst_plugin_feature_get_rank (GST_PLUGIN_FEATURE (factory)),
|
|
RESET_COLOR);
|
|
|
|
uri_protocols =
|
|
gst_uri_handler_get_protocols (GST_URI_HANDLER (element));
|
|
for (protocol = uri_protocols; *protocol != NULL; protocol++) {
|
|
if (protocol != uri_protocols)
|
|
g_print (", ");
|
|
g_print ("%s%s%s", FEATURE_PROTO_COLOR, *protocol, RESET_COLOR);
|
|
}
|
|
g_print ("\n");
|
|
}
|
|
|
|
gst_object_unref (element);
|
|
gst_object_unref (factory);
|
|
}
|
|
}
|
|
|
|
gst_plugin_feature_list_free (features);
|
|
}
|
|
|
|
gst_plugin_list_free (plugins);
|
|
}
|
|
|
|
static void
|
|
print_plugin_info (GstPlugin * plugin)
|
|
{
|
|
const gchar *release_date = gst_plugin_get_release_date_string (plugin);
|
|
const gchar *filename = gst_plugin_get_filename (plugin);
|
|
|
|
n_print ("%sPlugin Details%s:\n", HEADING_COLOR, RESET_COLOR);
|
|
|
|
push_indent ();
|
|
|
|
n_print ("%s%-25s%s%s%s%s\n", PROP_NAME_COLOR, "Name", RESET_COLOR,
|
|
PROP_VALUE_COLOR, gst_plugin_get_name (plugin), RESET_COLOR);
|
|
n_print ("%s%-25s%s%s%s%s\n", PROP_NAME_COLOR, "Description", RESET_COLOR,
|
|
PROP_VALUE_COLOR, gst_plugin_get_description (plugin), RESET_COLOR);
|
|
n_print ("%s%-25s%s%s%s%s\n", PROP_NAME_COLOR, "Filename", RESET_COLOR,
|
|
PROP_VALUE_COLOR, (filename != NULL) ? filename : "(null)", RESET_COLOR);
|
|
n_print ("%s%-25s%s%s%s%s\n", PROP_NAME_COLOR, "Version", RESET_COLOR,
|
|
PROP_VALUE_COLOR, gst_plugin_get_version (plugin), RESET_COLOR);
|
|
n_print ("%s%-25s%s%s%s%s\n", PROP_NAME_COLOR, "License", RESET_COLOR,
|
|
PROP_VALUE_COLOR, gst_plugin_get_license (plugin), RESET_COLOR);
|
|
n_print ("%s%-25s%s%s%s%s\n", PROP_NAME_COLOR, "Source module", RESET_COLOR,
|
|
PROP_VALUE_COLOR, gst_plugin_get_source (plugin), RESET_COLOR);
|
|
|
|
if (release_date != NULL) {
|
|
const gchar *tz = "(UTC)";
|
|
gchar *str, *sep;
|
|
|
|
/* may be: YYYY-MM-DD or YYYY-MM-DDTHH:MMZ */
|
|
/* YYYY-MM-DDTHH:MMZ => YYYY-MM-DD HH:MM (UTC) */
|
|
str = g_strdup (release_date);
|
|
sep = strstr (str, "T");
|
|
if (sep != NULL) {
|
|
*sep = ' ';
|
|
sep = strstr (sep + 1, "Z");
|
|
if (sep != NULL)
|
|
*sep = ' ';
|
|
} else {
|
|
tz = "";
|
|
}
|
|
n_print ("%s%-25s%s%s%s%s%s\n", PROP_NAME_COLOR, "Source release date",
|
|
RESET_COLOR, PROP_VALUE_COLOR, str, tz, RESET_COLOR);
|
|
g_free (str);
|
|
}
|
|
n_print ("%s%-25s%s%s%s%s\n", PROP_NAME_COLOR, "Binary package", RESET_COLOR,
|
|
PROP_VALUE_COLOR, gst_plugin_get_package (plugin), RESET_COLOR);
|
|
n_print ("%s%-25s%s%s%s%s\n", PROP_NAME_COLOR, "Origin URL", RESET_COLOR,
|
|
PROP_VALUE_COLOR, gst_plugin_get_origin (plugin), RESET_COLOR);
|
|
|
|
pop_indent ();
|
|
|
|
n_print ("\n");
|
|
}
|
|
|
|
static void
|
|
print_plugin_features (GstPlugin * plugin)
|
|
{
|
|
GList *features, *origlist;
|
|
gint num_features = 0;
|
|
gint num_elements = 0;
|
|
gint num_tracers = 0;
|
|
gint num_typefinders = 0;
|
|
gint num_devproviders = 0;
|
|
gint num_other = 0;
|
|
|
|
origlist = features =
|
|
gst_registry_get_feature_list_by_plugin (gst_registry_get (),
|
|
gst_plugin_get_name (plugin));
|
|
|
|
while (features) {
|
|
GstPluginFeature *feature;
|
|
|
|
feature = GST_PLUGIN_FEATURE (features->data);
|
|
|
|
if (GST_IS_ELEMENT_FACTORY (feature)) {
|
|
GstElementFactory *factory;
|
|
|
|
factory = GST_ELEMENT_FACTORY (feature);
|
|
n_print (" %s%s%s: %s%s%s\n", ELEMENT_NAME_COLOR,
|
|
GST_OBJECT_NAME (factory), RESET_COLOR, ELEMENT_DETAIL_COLOR,
|
|
gst_element_factory_get_metadata (factory,
|
|
GST_ELEMENT_METADATA_LONGNAME), RESET_COLOR);
|
|
num_elements++;
|
|
} else if (GST_IS_TYPE_FIND_FACTORY (feature)) {
|
|
GstTypeFindFactory *factory;
|
|
const gchar *const *extensions;
|
|
|
|
factory = GST_TYPE_FIND_FACTORY (feature);
|
|
extensions = gst_type_find_factory_get_extensions (factory);
|
|
if (extensions) {
|
|
g_print (" %s%s%s: ", ELEMENT_NAME_COLOR,
|
|
gst_plugin_feature_get_name (feature), RESET_COLOR);
|
|
print_typefind_extensions (extensions, ELEMENT_DETAIL_COLOR);
|
|
g_print ("\n");
|
|
} else
|
|
g_print (" %s%s%s: %sno extensions%s\n", ELEMENT_NAME_COLOR,
|
|
gst_plugin_feature_get_name (feature), RESET_COLOR,
|
|
ELEMENT_DETAIL_COLOR, RESET_COLOR);
|
|
|
|
num_typefinders++;
|
|
} else if (GST_IS_DEVICE_PROVIDER_FACTORY (feature)) {
|
|
GstDeviceProviderFactory *factory;
|
|
|
|
factory = GST_DEVICE_PROVIDER_FACTORY (feature);
|
|
n_print (" %s%s%s: %s%s%s\n", ELEMENT_NAME_COLOR,
|
|
GST_OBJECT_NAME (factory), RESET_COLOR, ELEMENT_DETAIL_COLOR,
|
|
gst_device_provider_factory_get_metadata (factory,
|
|
GST_ELEMENT_METADATA_LONGNAME), RESET_COLOR);
|
|
num_devproviders++;
|
|
} else if (GST_IS_TRACER_FACTORY (feature)) {
|
|
n_print (" %s%s%s (%s%s%s)\n", ELEMENT_NAME_COLOR,
|
|
gst_object_get_name (GST_OBJECT (feature)), RESET_COLOR,
|
|
DATATYPE_COLOR, g_type_name (G_OBJECT_TYPE (feature)), RESET_COLOR);
|
|
num_tracers++;
|
|
} else if (feature) {
|
|
n_print (" %s%s%s (%s%s%s)\n", ELEMENT_NAME_COLOR,
|
|
gst_object_get_name (GST_OBJECT (feature)), RESET_COLOR,
|
|
DATATYPE_COLOR, g_type_name (G_OBJECT_TYPE (feature)), RESET_COLOR);
|
|
num_other++;
|
|
}
|
|
num_features++;
|
|
features = g_list_next (features);
|
|
}
|
|
|
|
gst_plugin_feature_list_free (origlist);
|
|
|
|
n_print ("\n");
|
|
n_print (" %s%d features%s:\n", HEADING_COLOR, num_features, RESET_COLOR);
|
|
if (num_elements > 0)
|
|
n_print (" %s+--%s %s%d elements%s\n", CHILD_LINK_COLOR, RESET_COLOR,
|
|
PLUGIN_FEATURE_COLOR, num_elements, RESET_COLOR);
|
|
if (num_typefinders > 0)
|
|
n_print (" %s+--%s %s%d typefinders%s\n", CHILD_LINK_COLOR, RESET_COLOR,
|
|
PLUGIN_FEATURE_COLOR, num_typefinders, RESET_COLOR);
|
|
if (num_devproviders > 0)
|
|
n_print (" %s+--%s %s%d device providers%s\n", CHILD_LINK_COLOR,
|
|
RESET_COLOR, PLUGIN_FEATURE_COLOR, num_devproviders, RESET_COLOR);
|
|
if (num_tracers > 0)
|
|
n_print (" %s+--%s %s%d tracers%s\n", CHILD_LINK_COLOR, RESET_COLOR,
|
|
PLUGIN_FEATURE_COLOR, num_tracers, RESET_COLOR);
|
|
if (num_other > 0)
|
|
n_print (" %s+--%s %s%d other objects%s\n", CHILD_LINK_COLOR, RESET_COLOR,
|
|
PLUGIN_FEATURE_COLOR, num_other, RESET_COLOR);
|
|
|
|
n_print ("\n");
|
|
}
|
|
|
|
static int
|
|
print_feature_info (const gchar * feature_name, gboolean print_all)
|
|
{
|
|
GstPluginFeature *feature;
|
|
GstRegistry *registry = gst_registry_get ();
|
|
int ret;
|
|
|
|
if ((feature = gst_registry_find_feature (registry, feature_name,
|
|
GST_TYPE_ELEMENT_FACTORY))) {
|
|
ret = print_element_info (feature, print_all);
|
|
goto handled;
|
|
}
|
|
if ((feature = gst_registry_find_feature (registry, feature_name,
|
|
GST_TYPE_TYPE_FIND_FACTORY))) {
|
|
ret = print_typefind_info (feature, print_all);
|
|
goto handled;
|
|
}
|
|
if ((feature = gst_registry_find_feature (registry, feature_name,
|
|
GST_TYPE_TRACER_FACTORY))) {
|
|
ret = print_tracer_info (feature, print_all);
|
|
goto handled;
|
|
}
|
|
|
|
/* TODO: handle DEVICE_PROVIDER_FACTORY */
|
|
|
|
return -1;
|
|
|
|
handled:
|
|
gst_object_unref (feature);
|
|
return ret;
|
|
}
|
|
|
|
static int
|
|
print_element_info (GstPluginFeature * feature, gboolean print_names)
|
|
{
|
|
GstElementFactory *factory;
|
|
GstElement *element;
|
|
GstPlugin *plugin;
|
|
gint maxlevel = 0;
|
|
|
|
factory = GST_ELEMENT_FACTORY (gst_plugin_feature_load (feature));
|
|
if (!factory) {
|
|
g_print ("%selement plugin couldn't be loaded%s\n", DESC_COLOR,
|
|
RESET_COLOR);
|
|
return -1;
|
|
}
|
|
|
|
element = gst_element_factory_create (factory, NULL);
|
|
if (!element) {
|
|
gst_object_unref (factory);
|
|
g_print ("%scouldn't construct element for some reason%s\n", DESC_COLOR,
|
|
RESET_COLOR);
|
|
return -1;
|
|
}
|
|
|
|
if (print_names)
|
|
_name =
|
|
g_strdup_printf ("%s%s%s: ", DATATYPE_COLOR, GST_OBJECT_NAME (factory),
|
|
RESET_COLOR);
|
|
else
|
|
_name = NULL;
|
|
|
|
print_factory_details_info (factory);
|
|
|
|
plugin = gst_plugin_feature_get_plugin (GST_PLUGIN_FEATURE (factory));
|
|
if (plugin) {
|
|
print_plugin_info (plugin);
|
|
gst_object_unref (plugin);
|
|
}
|
|
|
|
print_hierarchy (G_OBJECT_TYPE (element), 0, &maxlevel);
|
|
print_interfaces (G_OBJECT_TYPE (element));
|
|
|
|
print_pad_templates_info (element, factory);
|
|
print_clocking_info (element);
|
|
print_uri_handler_info (element);
|
|
print_pad_info (element);
|
|
print_element_properties_info (element);
|
|
print_signal_info (element);
|
|
print_children_info (element);
|
|
print_preset_list (element);
|
|
|
|
gst_object_unref (element);
|
|
gst_object_unref (factory);
|
|
g_free (_name);
|
|
return 0;
|
|
}
|
|
|
|
static int
|
|
print_typefind_info (GstPluginFeature * feature, gboolean print_names)
|
|
{
|
|
GstTypeFindFactory *factory;
|
|
GstPlugin *plugin;
|
|
GstCaps *caps;
|
|
GstRank rank;
|
|
char s[20];
|
|
const gchar *const *extensions;
|
|
|
|
factory = GST_TYPE_FIND_FACTORY (gst_plugin_feature_load (feature));
|
|
if (!factory) {
|
|
g_print ("%stypefind plugin couldn't be loaded%s\n", DESC_COLOR,
|
|
RESET_COLOR);
|
|
return -1;
|
|
}
|
|
|
|
if (print_names)
|
|
_name =
|
|
g_strdup_printf ("%s%s%s: ", DATATYPE_COLOR, GST_OBJECT_NAME (factory),
|
|
RESET_COLOR);
|
|
else
|
|
_name = NULL;
|
|
|
|
n_print ("%sFactory Details%s:\n", HEADING_COLOR, RESET_COLOR);
|
|
rank = gst_plugin_feature_get_rank (feature);
|
|
n_print (" %s%-25s%s%s (%d)%s\n", PROP_NAME_COLOR, "Rank", PROP_VALUE_COLOR,
|
|
get_rank_name (s, rank), rank, RESET_COLOR);
|
|
n_print (" %s%-25s%s%s%s\n", PROP_NAME_COLOR, "Name", PROP_VALUE_COLOR,
|
|
GST_OBJECT_NAME (factory), RESET_COLOR);
|
|
caps = gst_type_find_factory_get_caps (factory);
|
|
if (caps) {
|
|
gchar *caps_str = gst_caps_to_string (factory->caps);
|
|
|
|
n_print (" %s%-25s%s%s%s\n", PROP_NAME_COLOR, "Caps", PROP_VALUE_COLOR,
|
|
caps_str, RESET_COLOR);
|
|
g_free (caps_str);
|
|
}
|
|
extensions = gst_type_find_factory_get_extensions (factory);
|
|
if (extensions) {
|
|
n_print (" %s%-25s%s", PROP_NAME_COLOR, "Extensions", RESET_COLOR);
|
|
print_typefind_extensions (extensions, PROP_VALUE_COLOR);
|
|
n_print ("\n");
|
|
}
|
|
n_print ("\n");
|
|
|
|
plugin = gst_plugin_feature_get_plugin (GST_PLUGIN_FEATURE (factory));
|
|
if (plugin) {
|
|
print_plugin_info (plugin);
|
|
gst_object_unref (plugin);
|
|
}
|
|
|
|
gst_object_unref (factory);
|
|
g_free (_name);
|
|
return 0;
|
|
}
|
|
|
|
static int
|
|
print_tracer_info (GstPluginFeature * feature, gboolean print_names)
|
|
{
|
|
GstTracerFactory *factory;
|
|
GstTracer *tracer;
|
|
GstPlugin *plugin;
|
|
gint maxlevel = 0;
|
|
|
|
factory = GST_TRACER_FACTORY (gst_plugin_feature_load (feature));
|
|
if (!factory) {
|
|
g_print ("%stracer plugin couldn't be loaded%s\n", DESC_COLOR, RESET_COLOR);
|
|
return -1;
|
|
}
|
|
|
|
tracer = (GstTracer *) g_object_new (factory->type, NULL);
|
|
if (!tracer) {
|
|
gst_object_unref (factory);
|
|
g_print ("%scouldn't construct tracer for some reason%s\n", DESC_COLOR,
|
|
RESET_COLOR);
|
|
return -1;
|
|
}
|
|
|
|
if (print_names)
|
|
_name =
|
|
g_strdup_printf ("%s%s%s: ", DATATYPE_COLOR, GST_OBJECT_NAME (factory),
|
|
RESET_COLOR);
|
|
else
|
|
_name = NULL;
|
|
|
|
n_print ("%sFactory Details%s:\n", HEADING_COLOR, RESET_COLOR);
|
|
n_print (" %s%-25s%s%s%s\n", PROP_NAME_COLOR, "Name", PROP_VALUE_COLOR,
|
|
GST_OBJECT_NAME (factory), RESET_COLOR);
|
|
n_print ("\n");
|
|
|
|
plugin = gst_plugin_feature_get_plugin (GST_PLUGIN_FEATURE (factory));
|
|
if (plugin) {
|
|
print_plugin_info (plugin);
|
|
gst_object_unref (plugin);
|
|
}
|
|
|
|
print_hierarchy (G_OBJECT_TYPE (tracer), 0, &maxlevel);
|
|
print_interfaces (G_OBJECT_TYPE (tracer));
|
|
|
|
/* TODO: list what hooks it registers
|
|
* - the data is available in gsttracerutils, we need to iterate the
|
|
* _priv_tracers hashtable for each probe and then check the list of hooks
|
|
* for each probe whether hook->tracer == tracer :/
|
|
*/
|
|
|
|
/* TODO: list what records it emits
|
|
* - in class_init tracers can create GstTracerRecord instances
|
|
* - those only get logged right now and there is no association with the
|
|
* tracer that created them
|
|
* - we'd need to add them to GstTracerFactory
|
|
* gst_tracer_class_add_record (klass, record);
|
|
* - needs work in gstregistrychunks to (de)serialize specs
|
|
* - gst_tracer_register() would need to iterate the list of records and
|
|
* copy the record->spec into the factory
|
|
*/
|
|
|
|
gst_object_unref (tracer);
|
|
gst_object_unref (factory);
|
|
g_free (_name);
|
|
return 0;
|
|
}
|
|
|
|
/* NOTE: Not coloring output from automatic install functions, as their output
|
|
* is meant for machines, not humans.
|
|
*/
|
|
static void
|
|
print_plugin_automatic_install_info_codecs (GstElementFactory * factory)
|
|
{
|
|
GstPadDirection direction;
|
|
const gchar *type_name;
|
|
const gchar *klass;
|
|
const GList *static_templates, *l;
|
|
GstCaps *caps = NULL;
|
|
guint i, num;
|
|
|
|
klass =
|
|
gst_element_factory_get_metadata (factory, GST_ELEMENT_METADATA_KLASS);
|
|
g_return_if_fail (klass != NULL);
|
|
|
|
if (strstr (klass, "Demuxer") ||
|
|
strstr (klass, "Decoder") ||
|
|
strstr (klass, "Depay") || strstr (klass, "Parser")) {
|
|
type_name = "decoder";
|
|
direction = GST_PAD_SINK;
|
|
} else if (strstr (klass, "Muxer") ||
|
|
strstr (klass, "Encoder") || strstr (klass, "Pay")) {
|
|
type_name = "encoder";
|
|
direction = GST_PAD_SRC;
|
|
} else {
|
|
return;
|
|
}
|
|
|
|
/* decoder/demuxer sink pads should always be static and there should only
|
|
* be one, the same applies to encoders/muxers and source pads */
|
|
static_templates = gst_element_factory_get_static_pad_templates (factory);
|
|
for (l = static_templates; l != NULL; l = l->next) {
|
|
GstStaticPadTemplate *tmpl = NULL;
|
|
|
|
tmpl = (GstStaticPadTemplate *) l->data;
|
|
if (tmpl->direction == direction) {
|
|
caps = gst_static_pad_template_get_caps (tmpl);
|
|
break;
|
|
}
|
|
}
|
|
|
|
if (caps == NULL) {
|
|
g_printerr ("Couldn't find static pad template for %s '%s'\n",
|
|
type_name, GST_OBJECT_NAME (factory));
|
|
return;
|
|
}
|
|
|
|
caps = gst_caps_make_writable (caps);
|
|
num = gst_caps_get_size (caps);
|
|
for (i = 0; i < num; ++i) {
|
|
GstStructure *s;
|
|
gchar *s_str;
|
|
|
|
s = gst_caps_get_structure (caps, i);
|
|
/* remove fields that are almost always just MIN-MAX of some sort
|
|
* in order to make the caps look less messy */
|
|
gst_structure_remove_field (s, "pixel-aspect-ratio");
|
|
gst_structure_remove_field (s, "framerate");
|
|
gst_structure_remove_field (s, "channels");
|
|
gst_structure_remove_field (s, "width");
|
|
gst_structure_remove_field (s, "height");
|
|
gst_structure_remove_field (s, "rate");
|
|
gst_structure_remove_field (s, "depth");
|
|
gst_structure_remove_field (s, "clock-rate");
|
|
s_str = gst_structure_to_string (s);
|
|
g_print ("%s-%s\n", type_name, s_str);
|
|
g_free (s_str);
|
|
}
|
|
gst_caps_unref (caps);
|
|
}
|
|
|
|
static void
|
|
print_plugin_automatic_install_info_protocols (GstElementFactory * factory)
|
|
{
|
|
const gchar *const *protocols;
|
|
|
|
protocols = gst_element_factory_get_uri_protocols (factory);
|
|
if (protocols != NULL && *protocols != NULL) {
|
|
switch (gst_element_factory_get_uri_type (factory)) {
|
|
case GST_URI_SINK:
|
|
while (*protocols != NULL) {
|
|
g_print ("urisink-%s\n", *protocols);
|
|
++protocols;
|
|
}
|
|
break;
|
|
case GST_URI_SRC:
|
|
while (*protocols != NULL) {
|
|
g_print ("urisource-%s\n", *protocols);
|
|
++protocols;
|
|
}
|
|
break;
|
|
default:
|
|
break;
|
|
}
|
|
}
|
|
}
|
|
|
|
static void
|
|
print_plugin_automatic_install_info (GstPlugin * plugin)
|
|
{
|
|
GList *features, *l;
|
|
|
|
/* not interested in typefind factories, only element factories */
|
|
features = gst_registry_get_feature_list (gst_registry_get (),
|
|
GST_TYPE_ELEMENT_FACTORY);
|
|
|
|
for (l = features; l != NULL; l = l->next) {
|
|
GstPluginFeature *feature;
|
|
GstPlugin *feature_plugin;
|
|
|
|
feature = GST_PLUGIN_FEATURE (l->data);
|
|
|
|
/* only interested in the ones that are in the plugin we just loaded */
|
|
feature_plugin = gst_plugin_feature_get_plugin (feature);
|
|
if (feature_plugin == plugin) {
|
|
GstElementFactory *factory;
|
|
|
|
g_print ("element-%s\n", gst_plugin_feature_get_name (feature));
|
|
|
|
factory = GST_ELEMENT_FACTORY (feature);
|
|
print_plugin_automatic_install_info_protocols (factory);
|
|
print_plugin_automatic_install_info_codecs (factory);
|
|
}
|
|
if (feature_plugin)
|
|
gst_object_unref (feature_plugin);
|
|
}
|
|
|
|
g_list_foreach (features, (GFunc) gst_object_unref, NULL);
|
|
g_list_free (features);
|
|
}
|
|
|
|
static void
|
|
print_all_plugin_automatic_install_info (void)
|
|
{
|
|
GList *plugins, *orig_plugins;
|
|
|
|
orig_plugins = plugins = gst_registry_get_plugin_list (gst_registry_get ());
|
|
while (plugins) {
|
|
GstPlugin *plugin;
|
|
|
|
plugin = (GstPlugin *) (plugins->data);
|
|
plugins = g_list_next (plugins);
|
|
|
|
print_plugin_automatic_install_info (plugin);
|
|
}
|
|
gst_plugin_list_free (orig_plugins);
|
|
}
|
|
|
|
#ifdef G_OS_UNIX
|
|
static gboolean
|
|
redirect_stdout (void)
|
|
{
|
|
GError *error = NULL;
|
|
gchar **argv;
|
|
const gchar *pager;
|
|
gint stdin_fd;
|
|
gchar **envp;
|
|
|
|
pager = g_getenv ("PAGER");
|
|
if (pager == NULL)
|
|
pager = DEFAULT_PAGER;
|
|
|
|
argv = g_strsplit (pager, " ", 0);
|
|
|
|
envp = g_get_environ ();
|
|
envp = g_environ_setenv (envp, "LESS", DEFAULT_LESS_OPTS, TRUE);
|
|
|
|
if (!g_spawn_async_with_pipes (NULL, argv, envp,
|
|
G_SPAWN_DO_NOT_REAP_CHILD | G_SPAWN_SEARCH_PATH,
|
|
NULL, NULL, &child_pid, &stdin_fd,
|
|
/* pass null stdout/stderr to inherit our fds */
|
|
NULL, NULL, &error)) {
|
|
g_warning ("g_spawn_async_with_pipes() failed: %s\n",
|
|
GST_STR_NULL (error->message));
|
|
g_strfreev (argv);
|
|
g_strfreev (envp);
|
|
g_clear_error (&error);
|
|
|
|
return FALSE;
|
|
}
|
|
|
|
/* redirect our stdout to child stdin */
|
|
dup2 (stdin_fd, STDOUT_FILENO);
|
|
if (isatty (STDERR_FILENO))
|
|
dup2 (stdin_fd, STDERR_FILENO);
|
|
close (stdin_fd);
|
|
|
|
g_strfreev (argv);
|
|
g_strfreev (envp);
|
|
|
|
return TRUE;
|
|
}
|
|
|
|
static void
|
|
child_exit_cb (GPid child_pid, gint status, gpointer user_data)
|
|
{
|
|
g_spawn_close_pid (child_pid);
|
|
g_main_loop_quit (loop);
|
|
}
|
|
#endif
|
|
|
|
int
|
|
main (int argc, char *argv[])
|
|
{
|
|
gboolean print_all = FALSE;
|
|
gboolean do_print_blacklist = FALSE;
|
|
gboolean plugin_name = FALSE;
|
|
gboolean print_aii = FALSE;
|
|
gboolean uri_handlers = FALSE;
|
|
gboolean check_exists = FALSE;
|
|
gchar *min_version = NULL;
|
|
guint minver_maj = GST_VERSION_MAJOR;
|
|
guint minver_min = GST_VERSION_MINOR;
|
|
guint minver_micro = 0;
|
|
gchar *types = NULL;
|
|
const gchar *no_colors;
|
|
int exit_code = 0;
|
|
#ifndef GST_DISABLE_OPTION_PARSING
|
|
GOptionEntry options[] = {
|
|
{"print-all", 'a', 0, G_OPTION_ARG_NONE, &print_all,
|
|
N_("Print all elements"), NULL},
|
|
{"print-blacklist", 'b', 0, G_OPTION_ARG_NONE, &do_print_blacklist,
|
|
N_("Print list of blacklisted files"), NULL},
|
|
{"print-plugin-auto-install-info", '\0', 0, G_OPTION_ARG_NONE, &print_aii,
|
|
N_("Print a machine-parsable list of features the specified plugin "
|
|
"or all plugins provide.\n "
|
|
"Useful in connection with external automatic plugin "
|
|
"installation mechanisms"), NULL},
|
|
{"plugin", '\0', 0, G_OPTION_ARG_NONE, &plugin_name,
|
|
N_("List the plugin contents"), NULL},
|
|
{"types", 't', 0, G_OPTION_ARG_STRING, &types,
|
|
N_("A slashes ('/') separated list of types of elements (also known "
|
|
"as klass) to list. (unordered)"), NULL},
|
|
{"exists", '\0', 0, G_OPTION_ARG_NONE, &check_exists,
|
|
N_("Check if the specified element or plugin exists"), NULL},
|
|
{"atleast-version", '\0', 0, G_OPTION_ARG_STRING, &min_version,
|
|
N_
|
|
("When checking if an element or plugin exists, also check that its "
|
|
"version is at least the version specified"), NULL},
|
|
{"uri-handlers", 'u', 0, G_OPTION_ARG_NONE, &uri_handlers,
|
|
N_
|
|
("Print supported URI schemes, with the elements that implement them"),
|
|
NULL},
|
|
{"no-colors", '\0', G_OPTION_FLAG_REVERSE, G_OPTION_ARG_NONE,
|
|
&colored_output,
|
|
N_
|
|
("Disable colors in output. You can also achieve the same by setting"
|
|
"'GST_INSPECT_NO_COLORS' environment variable to any value."),
|
|
NULL},
|
|
GST_TOOLS_GOPTION_VERSION,
|
|
{NULL}
|
|
};
|
|
GOptionContext *ctx;
|
|
GError *err = NULL;
|
|
#endif
|
|
|
|
setlocale (LC_ALL, "");
|
|
|
|
#ifdef ENABLE_NLS
|
|
bindtextdomain (GETTEXT_PACKAGE, LOCALEDIR);
|
|
bind_textdomain_codeset (GETTEXT_PACKAGE, "UTF-8");
|
|
textdomain (GETTEXT_PACKAGE);
|
|
#endif
|
|
|
|
/* avoid glib warnings when inspecting deprecated properties */
|
|
g_setenv ("G_ENABLE_DIAGNOSTIC", "0", FALSE);
|
|
|
|
g_set_prgname ("gst-inspect-" GST_API_VERSION);
|
|
|
|
#ifndef GST_DISABLE_OPTION_PARSING
|
|
ctx = g_option_context_new ("[ELEMENT-NAME | PLUGIN-NAME]");
|
|
g_option_context_add_main_entries (ctx, options, GETTEXT_PACKAGE);
|
|
g_option_context_add_group (ctx, gst_init_get_option_group ());
|
|
if (!g_option_context_parse (ctx, &argc, &argv, &err)) {
|
|
g_printerr ("Error initializing: %s\n", err->message);
|
|
g_clear_error (&err);
|
|
g_option_context_free (ctx);
|
|
return -1;
|
|
}
|
|
g_option_context_free (ctx);
|
|
#else
|
|
gst_init (&argc, &argv);
|
|
#endif
|
|
|
|
gst_tools_print_version ();
|
|
|
|
if (print_all && argc > 1) {
|
|
g_printerr ("-a requires no extra arguments\n");
|
|
return -1;
|
|
}
|
|
|
|
if (uri_handlers && argc > 1) {
|
|
g_printerr ("-u requires no extra arguments\n");
|
|
return -1;
|
|
}
|
|
|
|
/* --atleast-version implies --exists */
|
|
if (min_version != NULL) {
|
|
if (sscanf (min_version, "%u.%u.%u", &minver_maj, &minver_min,
|
|
&minver_micro) < 2) {
|
|
g_printerr ("Can't parse version '%s' passed to --atleast-version\n",
|
|
min_version);
|
|
return -1;
|
|
}
|
|
check_exists = TRUE;
|
|
}
|
|
|
|
if (check_exists) {
|
|
if (argc == 1) {
|
|
g_printerr ("--exists requires an extra command line argument\n");
|
|
exit_code = -1;
|
|
} else {
|
|
if (!plugin_name) {
|
|
GstPluginFeature *feature;
|
|
|
|
feature = gst_registry_lookup_feature (gst_registry_get (), argv[1]);
|
|
if (feature != NULL && gst_plugin_feature_check_version (feature,
|
|
minver_maj, minver_min, minver_micro)) {
|
|
exit_code = 0;
|
|
} else {
|
|
exit_code = 1;
|
|
}
|
|
|
|
if (feature)
|
|
gst_object_unref (feature);
|
|
} else {
|
|
/* FIXME: support checking for plugins too */
|
|
g_printerr ("Checking for plugins is not supported yet\n");
|
|
exit_code = -1;
|
|
}
|
|
}
|
|
return exit_code;
|
|
}
|
|
|
|
no_colors = g_getenv ("GST_INSPECT_NO_COLORS");
|
|
/* We only support truecolor */
|
|
colored_output &= (no_colors == NULL);
|
|
|
|
#ifdef G_OS_UNIX
|
|
if (isatty (STDOUT_FILENO)) {
|
|
if (redirect_stdout ())
|
|
loop = g_main_loop_new (NULL, FALSE);
|
|
} else {
|
|
colored_output = FALSE;
|
|
}
|
|
#elif defined(G_OS_WIN32)
|
|
{
|
|
gint fd = _fileno (stdout);
|
|
/* On Windows 10, g_log_writer_supports_color will also setup the console
|
|
* so that it correctly interprets ANSI VT sequences if it's supported */
|
|
if (!_isatty (fd) || !g_log_writer_supports_color (fd))
|
|
colored_output = FALSE;
|
|
}
|
|
#endif
|
|
|
|
/* if no arguments, print out list of elements */
|
|
if (uri_handlers) {
|
|
print_all_uri_handlers ();
|
|
} else if (argc == 1 || print_all) {
|
|
if (do_print_blacklist)
|
|
print_blacklist ();
|
|
else {
|
|
if (print_aii)
|
|
print_all_plugin_automatic_install_info ();
|
|
else
|
|
print_element_list (print_all, types);
|
|
}
|
|
} else {
|
|
/* else we try to get a factory */
|
|
const char *arg = argv[argc - 1];
|
|
int retval = -1;
|
|
|
|
if (!plugin_name) {
|
|
retval = print_feature_info (arg, print_all);
|
|
}
|
|
|
|
/* otherwise check if it's a plugin */
|
|
if (retval) {
|
|
GstPlugin *plugin = gst_registry_find_plugin (gst_registry_get (), arg);
|
|
|
|
/* if there is such a plugin, print out info */
|
|
if (plugin) {
|
|
if (print_aii) {
|
|
print_plugin_automatic_install_info (plugin);
|
|
} else {
|
|
print_plugin_info (plugin);
|
|
print_plugin_features (plugin);
|
|
}
|
|
} else {
|
|
GError *error = NULL;
|
|
|
|
if (g_file_test (arg, G_FILE_TEST_EXISTS)) {
|
|
plugin = gst_plugin_load_file (arg, &error);
|
|
|
|
if (plugin) {
|
|
if (print_aii) {
|
|
print_plugin_automatic_install_info (plugin);
|
|
} else {
|
|
print_plugin_info (plugin);
|
|
print_plugin_features (plugin);
|
|
}
|
|
} else {
|
|
g_printerr (_("Could not load plugin file: %s\n"), error->message);
|
|
g_clear_error (&error);
|
|
exit_code = -1;
|
|
goto done;
|
|
}
|
|
} else {
|
|
g_printerr (_("No such element or plugin '%s'\n"), arg);
|
|
exit_code = -1;
|
|
goto done;
|
|
}
|
|
}
|
|
}
|
|
}
|
|
|
|
done:
|
|
|
|
#ifdef G_OS_UNIX
|
|
if (loop) {
|
|
fflush (stdout);
|
|
fflush (stderr);
|
|
/* So that the pipe we create in redirect_stdout() is closed */
|
|
close (STDOUT_FILENO);
|
|
close (STDERR_FILENO);
|
|
g_child_watch_add (child_pid, child_exit_cb, NULL);
|
|
g_main_loop_run (loop);
|
|
g_main_loop_unref (loop);
|
|
}
|
|
#endif
|
|
|
|
return exit_code;
|
|
}
|