292 lines
7.7 KiB
C
292 lines
7.7 KiB
C
/* gstdotstracer.c */
|
|
/**
|
|
* SECTION:tracer-dots
|
|
* @short_description: Tracer for dot file generation setup and pipeline
|
|
* snapshot integration
|
|
* @title: GstDotsTracer
|
|
*
|
|
* The Dots tracer handles dot file generation setup and integrates with the
|
|
* pipeline-snapshot tracer when available. It ensures proper directory setup
|
|
* to collaborate with the `gst-dots-viewer` tool, and it handles file cleanup.
|
|
*
|
|
* The tracer determines the output directory in the following order:
|
|
* 1. Uses GST_DEBUG_DUMP_DOT_DIR if set
|
|
* 2. Falls back to $XDG_CACHE_HOME/gstreamer-dots otherwise
|
|
*
|
|
* The determined directory is created if it doesn't exist and set as
|
|
* `GST_DEBUG_DUMP_DOT_DIR` for the entire process.
|
|
*
|
|
* When available, it instantiates the pipeline-snapshot tracer with the
|
|
* following configuration:
|
|
* - dots-viewer-ws-url=ws://127.0.0.1:3000/snapshot/
|
|
* - xdg-cache=true
|
|
* - folder-mode=numbered
|
|
*
|
|
* ## Examples:
|
|
*
|
|
* ```
|
|
* # Basic usage - will delete existing .dot files
|
|
* GST_TRACERS=dots gst-launch-1.0 videotestsrc ! autovideosink
|
|
*
|
|
* # Keep existing .dot files
|
|
* GST_TRACERS="dots(no-delete=true)" gst-launch-1.0 videotestsrc ! autovideosink
|
|
* ```
|
|
*
|
|
* Since: 1.26
|
|
*/
|
|
|
|
#include "gst/gsttracerfactory.h"
|
|
#include <gio/gio.h>
|
|
#include <glib/gstdio.h>
|
|
#include <gst/gst.h>
|
|
#include <gst/gsttracer.h>
|
|
|
|
#define GST_TYPE_DOTS_TRACER (gst_dots_tracer_get_type())
|
|
G_DECLARE_FINAL_TYPE (GstDotsTracer, gst_dots_tracer, GST, DOTS_TRACER,
|
|
GstTracer)
|
|
/**
|
|
* GstDotsTracer:
|
|
*
|
|
* The #GstDotsTracer structure.
|
|
*
|
|
* Since: 1.26
|
|
*/
|
|
/* *INDENT-OFF* */
|
|
struct _GstDotsTracer
|
|
{
|
|
GstTracer parent;
|
|
|
|
gboolean no_delete;
|
|
gchar *output_dir;
|
|
GstTracer *pipeline_snapshot_tracer;
|
|
};
|
|
|
|
G_DEFINE_TYPE (GstDotsTracer, gst_dots_tracer, GST_TYPE_TRACER);
|
|
|
|
enum {
|
|
PROP_0,
|
|
PROP_NO_DELETE,
|
|
N_PROPERTIES
|
|
};
|
|
|
|
static GParamSpec *properties[N_PROPERTIES] = {
|
|
NULL,
|
|
};
|
|
|
|
GST_DEBUG_CATEGORY_STATIC (dots_debug);
|
|
#define GST_CAT_DEFAULT dots_debug
|
|
|
|
static void
|
|
gst_dots_tracer_set_property (GObject * object, guint prop_id,
|
|
const GValue * value, GParamSpec * pspec)
|
|
/* *INDENT-ON* */
|
|
{
|
|
GstDotsTracer *self = GST_DOTS_TRACER (object);
|
|
|
|
switch (prop_id) {
|
|
case PROP_NO_DELETE:
|
|
self->no_delete = g_value_get_boolean (value);
|
|
break;
|
|
default:
|
|
G_OBJECT_WARN_INVALID_PROPERTY_ID (object, prop_id, pspec);
|
|
break;
|
|
}
|
|
}
|
|
|
|
static void
|
|
gst_dots_tracer_get_property (GObject * object, guint prop_id,
|
|
GValue * value, GParamSpec * pspec)
|
|
{
|
|
GstDotsTracer *self = GST_DOTS_TRACER (object);
|
|
|
|
switch (prop_id) {
|
|
case PROP_NO_DELETE:
|
|
g_value_set_boolean (value, self->no_delete);
|
|
break;
|
|
default:
|
|
G_OBJECT_WARN_INVALID_PROPERTY_ID (object, prop_id, pspec);
|
|
break;
|
|
}
|
|
}
|
|
|
|
static void
|
|
gst_dots_tracer_finalize (GObject * obj)
|
|
{
|
|
GstDotsTracer *self = GST_DOTS_TRACER (obj);
|
|
|
|
g_free (self->output_dir);
|
|
|
|
if (self->pipeline_snapshot_tracer) {
|
|
gst_object_unref (self->pipeline_snapshot_tracer);
|
|
}
|
|
|
|
G_OBJECT_CLASS (gst_dots_tracer_parent_class)->finalize (obj);
|
|
}
|
|
|
|
static void
|
|
clean_dot_files (const gchar * dir_path)
|
|
{
|
|
GDir *dir;
|
|
const gchar *filename;
|
|
GError *error = NULL;
|
|
GSList *paths = NULL, *l;
|
|
GSList *dirs = NULL;
|
|
|
|
/* Build directory list starting with root dir */
|
|
dirs = g_slist_prepend (dirs, g_strdup (dir_path));
|
|
|
|
/* Find all matching files */
|
|
while (dirs) {
|
|
gchar *current_dir = dirs->data;
|
|
dirs = g_slist_delete_link (dirs, dirs);
|
|
|
|
dir = g_dir_open (current_dir, 0, &error);
|
|
if (!dir) {
|
|
GST_WARNING ("Could not open directory %s: %s", current_dir,
|
|
error ? error->message : "unknown error");
|
|
g_clear_error (&error);
|
|
g_free (current_dir);
|
|
continue;
|
|
}
|
|
|
|
while ((filename = g_dir_read_name (dir))) {
|
|
gchar *path = g_build_filename (current_dir, filename, NULL);
|
|
|
|
if (g_file_test (path, G_FILE_TEST_IS_DIR)) {
|
|
dirs = g_slist_prepend (dirs, path);
|
|
} else if (g_str_has_suffix (path, ".dot")) {
|
|
paths = g_slist_prepend (paths, path);
|
|
} else {
|
|
g_free (path);
|
|
}
|
|
}
|
|
g_dir_close (dir);
|
|
g_free (current_dir);
|
|
}
|
|
|
|
/* Delete all matched files */
|
|
for (l = paths; l; l = l->next) {
|
|
if (g_unlink (l->data) != 0) {
|
|
GST_WARNING ("Could not delete file %s", (gchar *) l->data);
|
|
}
|
|
}
|
|
|
|
g_slist_free_full (paths, g_free);
|
|
}
|
|
|
|
static gboolean
|
|
try_create_pipeline_snapshot_tracer (GstDotsTracer * self)
|
|
{
|
|
GstRegistry *registry;
|
|
GstPluginFeature *feature;
|
|
GstTracerFactory *factory;
|
|
|
|
registry = gst_registry_get ();
|
|
feature = gst_registry_lookup_feature (registry, "pipeline-snapshot");
|
|
|
|
if (!feature) {
|
|
GST_WARNING ("pipeline-snapshot tracer not found. \
|
|
Please ensure that the `rstracers` plugin is installed.");
|
|
return FALSE;
|
|
}
|
|
|
|
factory = GST_TRACER_FACTORY (gst_plugin_feature_load (feature));
|
|
gst_object_unref (feature);
|
|
|
|
if (!factory) {
|
|
GST_WARNING ("Could not load pipeline-snapshot factory. \
|
|
Please ensure GStreamer is properly installed.");
|
|
return FALSE;
|
|
}
|
|
|
|
GType tracer_type = gst_tracer_factory_get_tracer_type (factory);
|
|
GObjectClass *tracer_class = g_type_class_ref (tracer_type);
|
|
|
|
if (g_object_class_find_property (tracer_class, "dots-viewer-ws-url"))
|
|
self->pipeline_snapshot_tracer = g_object_new (gst_tracer_factory_get_tracer_type (factory), "dot-dir", self->output_dir, "dots-viewer-ws-url", "ws://127.0.0.1:3000/snapshot/", "folder-mode", 1, /*numbered */
|
|
NULL);
|
|
else
|
|
self->pipeline_snapshot_tracer =
|
|
g_object_new (gst_tracer_factory_get_tracer_type (factory), NULL);
|
|
gst_object_unref (factory);
|
|
g_type_class_unref (tracer_class);
|
|
|
|
if (!self->pipeline_snapshot_tracer) {
|
|
GST_WARNING ("Could not create pipeline-snapshot tracer instance");
|
|
return FALSE;
|
|
}
|
|
|
|
GST_INFO ("Successfully created and configured pipeline-snapshot tracer");
|
|
return TRUE;
|
|
}
|
|
|
|
static void
|
|
setup_output_directory (GstDotsTracer * self)
|
|
{
|
|
const gchar *env_dir;
|
|
|
|
// Check GST_DEBUG_DUMP_DOT_DIR first
|
|
env_dir = g_getenv ("GST_DEBUG_DUMP_DOT_DIR");
|
|
if (env_dir) {
|
|
self->output_dir = g_strdup (env_dir);
|
|
} else {
|
|
// Use XDG cache directory if GST_DEBUG_DUMP_DOT_DIR is not set
|
|
self->output_dir =
|
|
g_build_filename (g_get_user_cache_dir (), "gstreamer-dots", NULL);
|
|
|
|
GST_DEBUG ("Setting GST_DEBUG_DUMP_DOT_DIR to %s", self->output_dir);
|
|
|
|
g_setenv ("GST_DEBUG_DUMP_DOT_DIR", self->output_dir, TRUE);
|
|
}
|
|
|
|
// Create output directory if it doesn't exist
|
|
g_mkdir_with_parents (self->output_dir, 0755);
|
|
|
|
// Clean existing .dot files unless no-delete is set
|
|
if (!self->no_delete) {
|
|
clean_dot_files (self->output_dir);
|
|
}
|
|
}
|
|
|
|
static void
|
|
gst_dots_tracer_init (GstDotsTracer * self)
|
|
{
|
|
self->no_delete = FALSE;
|
|
self->pipeline_snapshot_tracer = NULL;
|
|
|
|
setup_output_directory (self);
|
|
|
|
// Try to create pipeline-snapshot tracer with exact same configuration as
|
|
// gstdump.rs
|
|
try_create_pipeline_snapshot_tracer (self);
|
|
}
|
|
|
|
static void
|
|
gst_dots_tracer_class_init (GstDotsTracerClass * klass)
|
|
{
|
|
GObjectClass *gobject_class = G_OBJECT_CLASS (klass);
|
|
|
|
gobject_class->set_property = gst_dots_tracer_set_property;
|
|
gobject_class->get_property = gst_dots_tracer_get_property;
|
|
gobject_class->finalize = gst_dots_tracer_finalize;
|
|
|
|
gst_tracer_class_set_use_structure_params (GST_TRACER_CLASS (gobject_class),
|
|
TRUE);
|
|
|
|
/**
|
|
* GstDotsTracer:no-delete:
|
|
*
|
|
* Don't delete existing .dot files on startup.
|
|
*
|
|
* Since: 1.26
|
|
*/
|
|
properties[PROP_NO_DELETE] =
|
|
g_param_spec_boolean ("no-delete", "No Delete",
|
|
"Don't delete existing .dot files on startup", FALSE,
|
|
G_PARAM_READWRITE | G_PARAM_STATIC_STRINGS);
|
|
|
|
g_object_class_install_properties (gobject_class, N_PROPERTIES, properties);
|
|
|
|
GST_DEBUG_CATEGORY_INIT (dots_debug, "dots", 0, "dots tracer");
|
|
}
|