mirror of
https://github.com/varun-r-mallya/sysprof.git
synced 2025-12-31 20:36:25 +00:00
libsysprof-analyze: add SysprofTimeSeries
This includes a helper to generate timeseries items such as marks which have a point in time and optionally a duration. The TimeSpan has also been beefed up to gain a few operations that are useful for implementing that.
This commit is contained in:
@ -32,6 +32,7 @@ libsysprof_analyze_public_sources = [
|
||||
'sysprof-no-symbolizer.c',
|
||||
'sysprof-symbol.c',
|
||||
'sysprof-symbolizer.c',
|
||||
'sysprof-time-series.c',
|
||||
'sysprof-time-span.c',
|
||||
]
|
||||
|
||||
@ -48,6 +49,7 @@ libsysprof_analyze_private_sources = [
|
||||
'sysprof-process-info.c',
|
||||
'sysprof-strings.c',
|
||||
'sysprof-symbol-cache.c',
|
||||
'sysprof-time-series.h',
|
||||
'sysprof-time-span.h',
|
||||
]
|
||||
|
||||
|
||||
@ -59,6 +59,8 @@ G_BEGIN_DECLS
|
||||
# include "sysprof-no-symbolizer.h"
|
||||
# include "sysprof-symbol.h"
|
||||
# include "sysprof-symbolizer.h"
|
||||
# include "sysprof-time-series.h"
|
||||
# include "sysprof-time-span.h"
|
||||
#undef SYSPROF_ANALYZE_INSIDE
|
||||
|
||||
G_END_DECLS
|
||||
|
||||
183
src/libsysprof-analyze/sysprof-time-series.c
Normal file
183
src/libsysprof-analyze/sysprof-time-series.c
Normal file
@ -0,0 +1,183 @@
|
||||
/* sysprof-time-series.c
|
||||
*
|
||||
* Copyright 2023 Christian Hergert <chergert@redhat.com>
|
||||
*
|
||||
* This program is free software: you can redistribute it and/or modify
|
||||
* it under the terms of the GNU General Public License as published by
|
||||
* the Free Software Foundation, either version 3 of the License, or
|
||||
* (at your option) any later version.
|
||||
*
|
||||
* This program is distributed in the hope that it will be useful,
|
||||
* but WITHOUT ANY WARRANTY; without even the implied warranty of
|
||||
* MERCHANTABILITY or FITNESS FOR A PARTICULAR PURPOSE. See the
|
||||
* GNU General Public License for more details.
|
||||
*
|
||||
* You should have received a copy of the GNU General Public License
|
||||
* along with this program. If not, see <http://www.gnu.org/licenses/>.
|
||||
*
|
||||
* SPDX-License-Identifier: GPL-3.0-or-later
|
||||
*/
|
||||
|
||||
#include "config.h"
|
||||
|
||||
#include "sysprof-time-series.h"
|
||||
|
||||
struct _SysprofTimeSeries
|
||||
{
|
||||
/* Model of SysprofDocumentFrame */
|
||||
GListModel *model;
|
||||
|
||||
/* Array of SysprofTimeSeriesValue */
|
||||
GArray *values;
|
||||
|
||||
/* The timespan contained in this series */
|
||||
SysprofTimeSpan time_span;
|
||||
|
||||
/* The duration of the series */
|
||||
gint64 duration;
|
||||
|
||||
/* Used for warning on items-changed */
|
||||
gulong items_changed_handler;
|
||||
};
|
||||
|
||||
G_DEFINE_BOXED_TYPE (SysprofTimeSeries,
|
||||
sysprof_time_series,
|
||||
sysprof_time_series_ref,
|
||||
sysprof_time_series_unref)
|
||||
|
||||
static void
|
||||
warn_on_items_changed_cb (GListModel *model,
|
||||
guint position,
|
||||
guint removed,
|
||||
guint added,
|
||||
gpointer user_data)
|
||||
{
|
||||
g_critical ("%s @ %p emitted items changed while a timeseries is active! Expect errors.",
|
||||
G_OBJECT_TYPE_NAME (model), model);
|
||||
}
|
||||
|
||||
/**
|
||||
* sysprof_time_series_new:
|
||||
* @model: a #GListModel
|
||||
* @time_span: the span of the time series
|
||||
*
|
||||
* Used for creating normalized time spans (between 0..1) for times
|
||||
* within a timespan. Useful for creating time-based charts at any
|
||||
* size or scale.
|
||||
*
|
||||
* It is required that @model does not change during the lifetime
|
||||
* of the time series.
|
||||
*/
|
||||
SysprofTimeSeries *
|
||||
sysprof_time_series_new (GListModel *model,
|
||||
SysprofTimeSpan time_span)
|
||||
{
|
||||
SysprofTimeSeries *self;
|
||||
|
||||
g_return_val_if_fail (G_IS_LIST_MODEL (model), NULL);
|
||||
|
||||
self = g_atomic_rc_box_new0 (SysprofTimeSeries);
|
||||
self->model = g_object_ref (model);
|
||||
self->values = g_array_new (FALSE, FALSE, sizeof (SysprofTimeSeriesValue));
|
||||
self->time_span = sysprof_time_span_order (time_span);
|
||||
self->duration = MAX (1, self->time_span.end_nsec - self->time_span.begin_nsec);
|
||||
self->items_changed_handler =
|
||||
g_signal_connect (self->model,
|
||||
"items-changed",
|
||||
G_CALLBACK (warn_on_items_changed_cb),
|
||||
NULL);
|
||||
|
||||
return self;
|
||||
}
|
||||
|
||||
SysprofTimeSeries *
|
||||
sysprof_time_series_ref (SysprofTimeSeries *self)
|
||||
{
|
||||
return g_atomic_rc_box_acquire (self);
|
||||
}
|
||||
|
||||
static void
|
||||
_sysprof_time_series_finalize (SysprofTimeSeries *self)
|
||||
{
|
||||
g_clear_signal_handler (&self->items_changed_handler, self->model);
|
||||
g_clear_object (&self->model);
|
||||
g_clear_pointer (&self->values, g_array_unref);
|
||||
}
|
||||
|
||||
void
|
||||
sysprof_time_series_unref (SysprofTimeSeries *self)
|
||||
{
|
||||
g_atomic_rc_box_release_full (self,
|
||||
(GDestroyNotify)_sysprof_time_series_finalize);
|
||||
}
|
||||
|
||||
void
|
||||
sysprof_time_series_add (SysprofTimeSeries *self,
|
||||
SysprofTimeSpan time_span,
|
||||
guint index)
|
||||
{
|
||||
SysprofTimeSeriesValue value;
|
||||
|
||||
time_span = sysprof_time_span_order (time_span);
|
||||
|
||||
if (!sysprof_time_span_clamp (&time_span, self->time_span))
|
||||
return;
|
||||
|
||||
value.index = index;
|
||||
sysprof_time_span_normalize (time_span, self->time_span, value.time);
|
||||
|
||||
g_array_append_val (self->values, value);
|
||||
}
|
||||
|
||||
/**
|
||||
* sysprof_time_series_get_model:
|
||||
* @self: a #SysprofTimeSeries
|
||||
*
|
||||
* Gets the underlying model for the time series.
|
||||
*
|
||||
* Returns: (transfer none): a #GListModel
|
||||
*/
|
||||
GListModel *
|
||||
sysprof_time_series_get_model (SysprofTimeSeries *self)
|
||||
{
|
||||
g_return_val_if_fail (self != NULL, NULL);
|
||||
|
||||
return self->model;
|
||||
}
|
||||
|
||||
const SysprofTimeSeriesValue *
|
||||
sysprof_time_series_get_values (const SysprofTimeSeries *self,
|
||||
guint *n_values)
|
||||
{
|
||||
*n_values = self->values->len;
|
||||
return &g_array_index (self->values, SysprofTimeSeriesValue, 0);
|
||||
}
|
||||
|
||||
static int
|
||||
compare_by_time (gconstpointer a,
|
||||
gconstpointer b)
|
||||
{
|
||||
const SysprofTimeSeriesValue *aval = a;
|
||||
const SysprofTimeSeriesValue *bval = b;
|
||||
|
||||
if (aval->begin < bval->begin)
|
||||
return -1;
|
||||
else if (aval->begin > bval->begin)
|
||||
return 1;
|
||||
|
||||
if (aval->end < bval->end)
|
||||
return -1;
|
||||
else if (aval->end > bval->end)
|
||||
return 1;
|
||||
|
||||
return 0;
|
||||
}
|
||||
|
||||
void
|
||||
sysprof_time_series_sort (SysprofTimeSeries *self)
|
||||
{
|
||||
qsort (self->values->data,
|
||||
self->values->len,
|
||||
sizeof (SysprofTimeSeriesValue),
|
||||
compare_by_time);
|
||||
}
|
||||
74
src/libsysprof-analyze/sysprof-time-series.h
Normal file
74
src/libsysprof-analyze/sysprof-time-series.h
Normal file
@ -0,0 +1,74 @@
|
||||
/* sysprof-time-series.h
|
||||
*
|
||||
* Copyright 2023 Christian Hergert <chergert@redhat.com>
|
||||
*
|
||||
* This program is free software: you can redistribute it and/or modify
|
||||
* it under the terms of the GNU General Public License as published by
|
||||
* the Free Software Foundation, either version 3 of the License, or
|
||||
* (at your option) any later version.
|
||||
*
|
||||
* This program is distributed in the hope that it will be useful,
|
||||
* but WITHOUT ANY WARRANTY; without even the implied warranty of
|
||||
* MERCHANTABILITY or FITNESS FOR A PARTICULAR PURPOSE. See the
|
||||
* GNU General Public License for more details.
|
||||
*
|
||||
* You should have received a copy of the GNU General Public License
|
||||
* along with this program. If not, see <http://www.gnu.org/licenses/>.
|
||||
*
|
||||
* SPDX-License-Identifier: GPL-3.0-or-later
|
||||
*/
|
||||
|
||||
#pragma once
|
||||
|
||||
#include <sysprof-capture.h>
|
||||
|
||||
#include <gio/gio.h>
|
||||
|
||||
#include "sysprof-time-span.h"
|
||||
|
||||
G_BEGIN_DECLS
|
||||
|
||||
#define SYSPROF_TYPE_TIME_SERIES (sysprof_time_series_get_type())
|
||||
|
||||
typedef struct _SysprofTimeSeries SysprofTimeSeries;
|
||||
typedef struct _SysprofTimeSeriesValue SysprofTimeSeriesValue;
|
||||
|
||||
struct _SysprofTimeSeriesValue
|
||||
{
|
||||
/* Normalized begin/end value between 0..1 */
|
||||
union {
|
||||
float time[2];
|
||||
struct {
|
||||
float begin;
|
||||
float end;
|
||||
};
|
||||
};
|
||||
|
||||
/* Index of SysprofDocumentFrame */
|
||||
guint index;
|
||||
};
|
||||
|
||||
SYSPROF_AVAILABLE_IN_ALL
|
||||
GType sysprof_time_series_get_type (void) G_GNUC_CONST;
|
||||
SYSPROF_AVAILABLE_IN_ALL
|
||||
SysprofTimeSeries *sysprof_time_series_new (GListModel *model,
|
||||
SysprofTimeSpan time_span);
|
||||
SYSPROF_AVAILABLE_IN_ALL
|
||||
SysprofTimeSeries *sysprof_time_series_ref (SysprofTimeSeries *self);
|
||||
SYSPROF_AVAILABLE_IN_ALL
|
||||
void sysprof_time_series_unref (SysprofTimeSeries *self);
|
||||
SYSPROF_AVAILABLE_IN_ALL
|
||||
void sysprof_time_series_add (SysprofTimeSeries *self,
|
||||
SysprofTimeSpan time_span,
|
||||
guint index);
|
||||
SYSPROF_AVAILABLE_IN_ALL
|
||||
void sysprof_time_series_sort (SysprofTimeSeries *self);
|
||||
SYSPROF_AVAILABLE_IN_ALL
|
||||
GListModel *sysprof_time_series_get_model (SysprofTimeSeries *self);
|
||||
SYSPROF_AVAILABLE_IN_ALL
|
||||
const SysprofTimeSeriesValue *sysprof_time_series_get_values (const SysprofTimeSeries *self,
|
||||
guint *n_values);
|
||||
|
||||
G_DEFINE_AUTOPTR_CLEANUP_FUNC (SysprofTimeSeries, sysprof_time_series_unref)
|
||||
|
||||
G_END_DECLS
|
||||
@ -41,4 +41,56 @@ SysprofTimeSpan *sysprof_time_span_copy (const SysprofTimeSpan *self);
|
||||
SYSPROF_AVAILABLE_IN_ALL
|
||||
void sysprof_time_span_free (SysprofTimeSpan *self);
|
||||
|
||||
static inline SysprofTimeSpan
|
||||
sysprof_time_span_relative_to (SysprofTimeSpan time_span,
|
||||
gint64 point)
|
||||
{
|
||||
return (SysprofTimeSpan) {
|
||||
time_span.begin_nsec - point,
|
||||
time_span.end_nsec - point
|
||||
};
|
||||
}
|
||||
|
||||
static inline void
|
||||
sysprof_time_span_normalize (SysprofTimeSpan time_span,
|
||||
SysprofTimeSpan allowed,
|
||||
float values[restrict 2])
|
||||
{
|
||||
double duration = allowed.end_nsec - allowed.begin_nsec;
|
||||
|
||||
time_span = sysprof_time_span_relative_to (time_span, allowed.begin_nsec);
|
||||
|
||||
values[0] = time_span.begin_nsec / duration;
|
||||
values[1] = time_span.end_nsec / duration;
|
||||
}
|
||||
|
||||
static inline SysprofTimeSpan
|
||||
sysprof_time_span_order (SysprofTimeSpan time_span)
|
||||
{
|
||||
if (time_span.begin_nsec > time_span.end_nsec)
|
||||
return (SysprofTimeSpan) { time_span.end_nsec, time_span.begin_nsec };
|
||||
|
||||
return time_span;
|
||||
}
|
||||
|
||||
static inline gboolean
|
||||
sysprof_time_span_clamp (SysprofTimeSpan *time_span,
|
||||
SysprofTimeSpan allowed)
|
||||
{
|
||||
if (time_span->end_nsec <= allowed.begin_nsec ||
|
||||
time_span->begin_nsec >= allowed.end_nsec)
|
||||
{
|
||||
time_span->begin_nsec = time_span->end_nsec = 0;
|
||||
return FALSE;
|
||||
}
|
||||
|
||||
if (time_span->begin_nsec < allowed.begin_nsec)
|
||||
time_span->begin_nsec = allowed.begin_nsec;
|
||||
|
||||
if (time_span->end_nsec > allowed.end_nsec)
|
||||
time_span->end_nsec = allowed.end_nsec;
|
||||
|
||||
return TRUE;
|
||||
}
|
||||
|
||||
G_END_DECLS
|
||||
|
||||
Reference in New Issue
Block a user