tracers: Add a dots
tracer which is meant to be used with gst-dots-viewer
See documentation Part-of: <https://gitlab.freedesktop.org/gstreamer/gstreamer/-/merge_requests/7999>
This commit is contained in:
parent
f103692205
commit
642ad6c927
@ -2948,6 +2948,29 @@
|
||||
"package": "GStreamer",
|
||||
"source": "gstreamer",
|
||||
"tracers": {
|
||||
"dots": {
|
||||
"hierarchy": [
|
||||
"GstDotsTracer",
|
||||
"GstTracer",
|
||||
"GstObject",
|
||||
"GInitiallyUnowned",
|
||||
"GObject"
|
||||
],
|
||||
"properties": {
|
||||
"no-delete": {
|
||||
"blurb": "Don't delete existing .dot files on startup",
|
||||
"conditionally-available": false,
|
||||
"construct": false,
|
||||
"construct-only": false,
|
||||
"controllable": false,
|
||||
"default": "false",
|
||||
"mutable": "null",
|
||||
"readable": true,
|
||||
"type": "gboolean",
|
||||
"writable": true
|
||||
}
|
||||
}
|
||||
},
|
||||
"factories": {
|
||||
"hierarchy": [
|
||||
"GstFactoriesTracer",
|
||||
|
291
subprojects/gstreamer/plugins/tracers/gstdots.c
Normal file
291
subprojects/gstreamer/plugins/tracers/gstdots.c
Normal file
@ -0,0 +1,291 @@
|
||||
/* 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)
|
||||
{
|
||||
GPatternSpec *pspec;
|
||||
GDir *dir;
|
||||
const gchar *filename;
|
||||
GError *error = NULL;
|
||||
GSList *paths = NULL, *l;
|
||||
GSList *dirs = NULL;
|
||||
gchar *pattern;
|
||||
|
||||
/* Create glob pattern for .dot files */
|
||||
pattern = g_build_filename (dir_path, "**", "*.dot", NULL);
|
||||
pspec = g_pattern_spec_new (pattern);
|
||||
g_free (pattern);
|
||||
|
||||
/* 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_pattern_spec_match_string (pspec, path)) {
|
||||
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_pattern_spec_free (pspec);
|
||||
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;
|
||||
}
|
||||
|
||||
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);
|
||||
gst_object_unref (factory);
|
||||
|
||||
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_ERROR ("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");
|
||||
}
|
@ -31,9 +31,14 @@
|
||||
#include "gstleaks.h"
|
||||
#include "gstfactories.h"
|
||||
|
||||
GType gst_dots_tracer_get_type (void);
|
||||
|
||||
static gboolean
|
||||
plugin_init (GstPlugin * plugin)
|
||||
{
|
||||
if (!gst_tracer_register (plugin, "dots", gst_dots_tracer_get_type ()))
|
||||
return FALSE;
|
||||
|
||||
if (!gst_tracer_register (plugin, "latency", gst_latency_tracer_get_type ()))
|
||||
return FALSE;
|
||||
#ifndef GST_DISABLE_GST_DEBUG
|
||||
|
@ -1,4 +1,5 @@
|
||||
gst_tracers_sources = [
|
||||
'gstdots.c',
|
||||
'gstlatency.c',
|
||||
'gstleaks.c',
|
||||
'gststats.c',
|
||||
|
Loading…
x
Reference in New Issue
Block a user