mirror of
https://github.com/varun-r-mallya/sysprof.git
synced 2025-12-31 20:36:25 +00:00
995 lines
31 KiB
C
995 lines
31 KiB
C
/* sysprof-capture-view.c
|
|
*
|
|
* Copyright 2019 Christian Hergert <unknown@domain.org>
|
|
*
|
|
* 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
|
|
*/
|
|
|
|
#define G_LOG_DOMAIN "sysprof-capture-view"
|
|
|
|
#include "config.h"
|
|
|
|
#include <glib/gi18n.h>
|
|
|
|
#include "sysprof-callgraph-view.h"
|
|
#include "sysprof-capture-view.h"
|
|
#include "sysprof-details-view.h"
|
|
#include "sysprof-marks-view.h"
|
|
#include "sysprof-ui-private.h"
|
|
#include "sysprof-visualizer-view.h"
|
|
|
|
#define NSEC_PER_SEC (G_USEC_PER_SEC * 1000L)
|
|
|
|
typedef struct
|
|
{
|
|
gint64 begin_time;
|
|
gint64 end_time;
|
|
guint has_samples : 1;
|
|
guint has_counters : 1;
|
|
guint has_marks : 1;
|
|
} SysprofCaptureFeatures;
|
|
|
|
typedef struct
|
|
{
|
|
SysprofCaptureReader *reader;
|
|
GCancellable *cancellable;
|
|
GHashTable *mark_stats;
|
|
|
|
SysprofCaptureFeatures features;
|
|
|
|
/* Template Objects */
|
|
GtkAdjustment *time_adj;
|
|
GtkStack *stack;
|
|
SysprofCallgraphView *callgraph_view;
|
|
SysprofDetailsView *details_view;
|
|
SysprofMarksView *marks_view;
|
|
SysprofVisualizerView *visualizer_view;
|
|
SysprofZoomManager *zoom_manager;
|
|
GtkStackSwitcher *stack_switcher;
|
|
|
|
guint busy;
|
|
|
|
guint needs_fit : 1;
|
|
} SysprofCaptureViewPrivate;
|
|
|
|
typedef struct
|
|
{
|
|
SysprofCaptureReader *reader;
|
|
SysprofSelection *selection;
|
|
gint n_active;
|
|
guint has_error : 1;
|
|
} LoadAsync;
|
|
|
|
G_DEFINE_TYPE_WITH_PRIVATE (SysprofCaptureView, sysprof_capture_view, GTK_TYPE_BIN)
|
|
|
|
enum {
|
|
PROP_0,
|
|
PROP_BUSY,
|
|
N_PROPS
|
|
};
|
|
|
|
static GParamSpec *properties [N_PROPS];
|
|
|
|
static void
|
|
load_async_free (gpointer data)
|
|
{
|
|
LoadAsync *state = data;
|
|
|
|
if (state != NULL)
|
|
{
|
|
g_clear_pointer (&state->reader, sysprof_capture_reader_unref);
|
|
g_clear_object (&state->selection);
|
|
g_slice_free (LoadAsync, state);
|
|
}
|
|
}
|
|
|
|
static gboolean
|
|
do_best_fit_in_idle (gpointer user_data)
|
|
{
|
|
SysprofCaptureView *self = user_data;
|
|
|
|
if (gtk_widget_get_visible (GTK_WIDGET (self)))
|
|
sysprof_capture_view_fit_to_width (self);
|
|
|
|
return G_SOURCE_REMOVE;
|
|
}
|
|
|
|
SysprofMarkStat *
|
|
_sysprof_mark_stat_new (const gchar *name)
|
|
{
|
|
SysprofMarkStat *ret;
|
|
|
|
ret = g_slice_new0 (SysprofMarkStat);
|
|
ret->name = g_strdup (name);
|
|
|
|
return ret;
|
|
}
|
|
|
|
void
|
|
_sysprof_mark_stat_free (SysprofMarkStat *self)
|
|
{
|
|
g_clear_pointer (&self->name, g_free);
|
|
g_slice_free (SysprofMarkStat, self);
|
|
}
|
|
|
|
/**
|
|
* sysprof_capture_view_new:
|
|
*
|
|
* Create a new #SysprofCaptureView.
|
|
*
|
|
* Returns: (transfer full): a newly created #SysprofCaptureView
|
|
*
|
|
* Since: 3.34
|
|
*/
|
|
GtkWidget *
|
|
sysprof_capture_view_new (void)
|
|
{
|
|
return g_object_new (SYSPROF_TYPE_CAPTURE_VIEW, NULL);
|
|
}
|
|
|
|
static void
|
|
add_marks_to_details (SysprofCaptureView *self)
|
|
{
|
|
SysprofCaptureViewPrivate *priv = sysprof_capture_view_get_instance_private (self);
|
|
GHashTableIter iter;
|
|
gpointer k, v;
|
|
guint count = 0;
|
|
|
|
g_assert (SYSPROF_IS_CAPTURE_VIEW (self));
|
|
|
|
if (priv->mark_stats == NULL)
|
|
return;
|
|
|
|
if (g_hash_table_size (priv->mark_stats) == 0)
|
|
return;
|
|
|
|
g_hash_table_iter_init (&iter, priv->mark_stats);
|
|
while (count < 100 && g_hash_table_iter_next (&iter, &k, &v))
|
|
{
|
|
g_autofree gchar *str = NULL;
|
|
SysprofMarkStat *st = v;
|
|
GtkLabel *left_label;
|
|
GtkLabel *center_label;
|
|
|
|
if (st->avg == 0)
|
|
continue;
|
|
|
|
left_label = g_object_new (GTK_TYPE_LABEL,
|
|
"selectable", TRUE,
|
|
"visible", TRUE,
|
|
"xalign", 1.0f,
|
|
"label", st->name,
|
|
"ellipsize", PANGO_ELLIPSIZE_START,
|
|
NULL);
|
|
gtk_style_context_add_class (gtk_widget_get_style_context (GTK_WIDGET (left_label)),
|
|
GTK_STYLE_CLASS_DIM_LABEL);
|
|
str = g_strdup_printf ("<span fgalpha='32767'>Min:</span> %.4lf "
|
|
"<span fgalpha='32767'>Max:</span> %.4lf "
|
|
"<span fgalpha='32767'>Ø:</span> %.4lf",
|
|
st->min / (gdouble)NSEC_PER_SEC,
|
|
st->max / (gdouble)NSEC_PER_SEC,
|
|
st->avg / (gdouble)NSEC_PER_SEC);
|
|
center_label = g_object_new (GTK_TYPE_LABEL,
|
|
"label", str,
|
|
"selectable", TRUE,
|
|
"use-markup", TRUE,
|
|
"visible", TRUE,
|
|
"xalign", 0.0f,
|
|
NULL);
|
|
|
|
sysprof_details_view_add_item (priv->details_view,
|
|
GTK_WIDGET (left_label),
|
|
GTK_WIDGET (center_label));
|
|
|
|
count++;
|
|
}
|
|
}
|
|
|
|
static void
|
|
sysprof_capture_view_task_completed (SysprofCaptureView *self,
|
|
GParamSpec *pspec,
|
|
GTask *task)
|
|
{
|
|
SysprofCaptureViewPrivate *priv = sysprof_capture_view_get_instance_private (self);
|
|
|
|
g_assert (SYSPROF_IS_CAPTURE_VIEW (self));
|
|
g_assert (G_IS_TASK (task));
|
|
|
|
g_signal_handlers_disconnect_by_func (task,
|
|
G_CALLBACK (sysprof_capture_view_task_completed),
|
|
self);
|
|
|
|
priv->busy--;
|
|
|
|
if (priv->busy == 0)
|
|
g_object_notify_by_pspec (G_OBJECT (self), properties [PROP_BUSY]);
|
|
}
|
|
|
|
static void
|
|
sysprof_capture_view_monitor_task (SysprofCaptureView *self,
|
|
GTask *task)
|
|
{
|
|
SysprofCaptureViewPrivate *priv = sysprof_capture_view_get_instance_private (self);
|
|
|
|
g_assert (SYSPROF_IS_CAPTURE_VIEW (self));
|
|
g_assert (G_IS_TASK (task));
|
|
|
|
priv->busy++;
|
|
|
|
if (priv->busy == 1)
|
|
g_object_notify_by_pspec (G_OBJECT (self), properties [PROP_BUSY]);
|
|
|
|
g_signal_connect_object (task,
|
|
"notify::completed",
|
|
G_CALLBACK (sysprof_capture_view_task_completed),
|
|
self,
|
|
G_CONNECT_SWAPPED);
|
|
}
|
|
|
|
static void
|
|
sysprof_capture_view_generate_callgraph_cb (GObject *object,
|
|
GAsyncResult *result,
|
|
gpointer user_data)
|
|
{
|
|
SysprofCallgraphProfile *callgraph = (SysprofCallgraphProfile *)object;
|
|
g_autoptr(GError) error = NULL;
|
|
g_autoptr(GTask) task = user_data;
|
|
SysprofCaptureView *self;
|
|
SysprofCaptureViewPrivate *priv;
|
|
|
|
g_assert (SYSPROF_IS_CALLGRAPH_PROFILE (callgraph));
|
|
g_assert (G_IS_ASYNC_RESULT (result));
|
|
g_assert (G_IS_TASK (task));
|
|
|
|
if (!sysprof_profile_generate_finish (SYSPROF_PROFILE (callgraph), result, &error))
|
|
{
|
|
g_task_return_error (task, g_steal_pointer (&error));
|
|
return;
|
|
}
|
|
|
|
self = g_task_get_source_object (task);
|
|
priv = sysprof_capture_view_get_instance_private (self);
|
|
|
|
sysprof_callgraph_view_set_profile (priv->callgraph_view, callgraph);
|
|
|
|
g_task_return_boolean (task, TRUE);
|
|
}
|
|
|
|
static void
|
|
sysprof_capture_view_generate_callgraph_async (SysprofCaptureView *self,
|
|
SysprofCaptureReader *reader,
|
|
SysprofSelection *selection,
|
|
GCancellable *cancellable,
|
|
GAsyncReadyCallback callback,
|
|
gpointer user_data)
|
|
{
|
|
g_autoptr(SysprofCaptureReader) copy = NULL;
|
|
g_autoptr(SysprofProfile) callgraph = NULL;
|
|
g_autoptr(GTask) task = NULL;
|
|
|
|
g_return_if_fail (SYSPROF_IS_CAPTURE_VIEW (self));
|
|
g_return_if_fail (reader != NULL);
|
|
g_return_if_fail (!selection || SYSPROF_IS_SELECTION (selection));
|
|
g_return_if_fail (!cancellable || G_IS_CANCELLABLE (cancellable));
|
|
|
|
task = g_task_new (self, cancellable, callback, user_data);
|
|
g_task_set_source_tag (task, sysprof_capture_view_generate_callgraph_async);
|
|
sysprof_capture_view_monitor_task (self, task);
|
|
|
|
copy = sysprof_capture_reader_copy (reader);
|
|
callgraph = sysprof_callgraph_profile_new_with_selection (selection);
|
|
sysprof_profile_set_reader (callgraph, copy);
|
|
sysprof_profile_generate (callgraph,
|
|
cancellable,
|
|
sysprof_capture_view_generate_callgraph_cb,
|
|
g_steal_pointer (&task));
|
|
}
|
|
|
|
static gboolean
|
|
sysprof_capture_view_generate_callgraph_finish (SysprofCaptureView *self,
|
|
GAsyncResult *result,
|
|
GError **error)
|
|
{
|
|
g_assert (SYSPROF_IS_CAPTURE_VIEW (self));
|
|
g_assert (G_IS_TASK (result));
|
|
|
|
return g_task_propagate_boolean (G_TASK (result), error);
|
|
}
|
|
|
|
static void
|
|
sysprof_capture_view_scan_worker (GTask *task,
|
|
gpointer source_object,
|
|
gpointer task_data,
|
|
GCancellable *cancellable)
|
|
{
|
|
SysprofCaptureView *self = source_object;
|
|
SysprofCaptureViewPrivate *priv = sysprof_capture_view_get_instance_private (self);
|
|
SysprofCaptureReader *reader = task_data;
|
|
g_autoptr(GHashTable) mark_stats = NULL;
|
|
SysprofCaptureFeatures features = {0};
|
|
SysprofCaptureFrame frame;
|
|
SysprofCaptureStat st_buf = {{0}};
|
|
|
|
g_assert (SYSPROF_IS_CAPTURE_VIEW (self));
|
|
g_assert (G_IS_TASK (task));
|
|
g_assert (reader != NULL);
|
|
g_assert (!cancellable || G_IS_CANCELLABLE (cancellable));
|
|
|
|
mark_stats = g_hash_table_new_full (g_str_hash, g_str_equal, NULL,
|
|
(GDestroyNotify)_sysprof_mark_stat_free);
|
|
|
|
features.begin_time = sysprof_capture_reader_get_start_time (reader);
|
|
features.end_time = sysprof_capture_reader_get_end_time (reader);
|
|
|
|
while (sysprof_capture_reader_peek_frame (reader, &frame))
|
|
{
|
|
gint64 begin_time = frame.time;
|
|
gint64 end_time = G_MININT64;
|
|
|
|
g_assert (frame.type < G_N_ELEMENTS (st_buf.frame_count));
|
|
|
|
st_buf.frame_count[frame.type]++;
|
|
|
|
if (frame.type == SYSPROF_CAPTURE_FRAME_MARK)
|
|
{
|
|
const SysprofCaptureMark *mark;
|
|
|
|
if ((mark = sysprof_capture_reader_read_mark (reader)))
|
|
{
|
|
SysprofMarkStat *mstat;
|
|
gchar name[128];
|
|
|
|
end_time = frame.time + mark->duration;
|
|
|
|
g_snprintf (name, sizeof name, "%s:%s", mark->group, mark->name);
|
|
|
|
if (!(mstat = g_hash_table_lookup (mark_stats, name)))
|
|
{
|
|
mstat = _sysprof_mark_stat_new (name);
|
|
g_hash_table_insert (mark_stats, mstat->name, mstat);
|
|
}
|
|
|
|
if (mark->duration > 0)
|
|
{
|
|
if (mstat->min == 0 || mark->duration < mstat->min)
|
|
mstat->min = mark->duration;
|
|
}
|
|
|
|
if (mark->duration > mstat->max)
|
|
mstat->max = mark->duration;
|
|
|
|
if (mark->duration > 0)
|
|
{
|
|
mstat->avg += mark->duration;
|
|
mstat->avg_count++;
|
|
}
|
|
|
|
mstat->count++;
|
|
}
|
|
|
|
features.has_marks = TRUE;
|
|
}
|
|
else if (frame.type == SYSPROF_CAPTURE_FRAME_CTRDEF)
|
|
{
|
|
features.has_counters = TRUE;
|
|
sysprof_capture_reader_skip (reader);
|
|
}
|
|
else if (frame.type == SYSPROF_CAPTURE_FRAME_SAMPLE)
|
|
{
|
|
features.has_samples = TRUE;
|
|
sysprof_capture_reader_skip (reader);
|
|
}
|
|
else
|
|
{
|
|
sysprof_capture_reader_skip (reader);
|
|
}
|
|
|
|
if (begin_time < features.begin_time)
|
|
features.begin_time = begin_time;
|
|
|
|
if (end_time > features.end_time)
|
|
features.end_time = end_time;
|
|
}
|
|
|
|
{
|
|
GHashTableIter iter;
|
|
gpointer k,v;
|
|
|
|
g_hash_table_iter_init (&iter, mark_stats);
|
|
while (g_hash_table_iter_next (&iter, &k, &v))
|
|
{
|
|
SysprofMarkStat *mstat = v;
|
|
|
|
if (mstat->avg_count > 0 && mstat->avg > 0)
|
|
mstat->avg /= mstat->avg_count;
|
|
|
|
#if 0
|
|
g_print ("%s: count=%ld avg=%ld min=%ld max=%ld\n",
|
|
(gchar*)k,
|
|
((SysprofMarkStat *)v)->count,
|
|
((SysprofMarkStat *)v)->avg,
|
|
((SysprofMarkStat *)v)->min,
|
|
((SysprofMarkStat *)v)->max);
|
|
#endif
|
|
}
|
|
}
|
|
|
|
sysprof_capture_reader_set_stat (reader, &st_buf);
|
|
|
|
g_object_set_data_full (G_OBJECT (task),
|
|
"MARK_STAT",
|
|
g_steal_pointer (&mark_stats),
|
|
(GDestroyNotify) g_hash_table_unref);
|
|
|
|
if (!g_task_return_error_if_cancelled (task))
|
|
{
|
|
priv->features = features;
|
|
sysprof_capture_reader_reset (reader);
|
|
g_task_return_boolean (task, TRUE);
|
|
}
|
|
}
|
|
|
|
static void
|
|
sysprof_capture_view_scan_async (SysprofCaptureView *self,
|
|
SysprofCaptureReader *reader,
|
|
GCancellable *cancellable,
|
|
GAsyncReadyCallback callback,
|
|
gpointer user_data)
|
|
{
|
|
g_autoptr(GTask) task = NULL;
|
|
|
|
g_assert (SYSPROF_IS_CAPTURE_VIEW (self));
|
|
g_assert (reader != NULL);
|
|
g_assert (!cancellable || G_IS_CANCELLABLE (cancellable));
|
|
|
|
task = g_task_new (self, cancellable, callback, user_data);
|
|
g_task_set_source_tag (task, sysprof_capture_view_scan_async);
|
|
g_task_set_task_data (task,
|
|
sysprof_capture_reader_ref (reader),
|
|
(GDestroyNotify) sysprof_capture_reader_unref);
|
|
sysprof_capture_view_monitor_task (self, task);
|
|
g_task_run_in_thread (task, sysprof_capture_view_scan_worker);
|
|
}
|
|
|
|
static gboolean
|
|
sysprof_capture_view_scan_finish (SysprofCaptureView *self,
|
|
GAsyncResult *result,
|
|
GError **error)
|
|
{
|
|
SysprofCaptureViewPrivate *priv = sysprof_capture_view_get_instance_private (self);
|
|
GHashTable *stats;
|
|
|
|
g_assert (SYSPROF_IS_CAPTURE_VIEW (self));
|
|
g_assert (G_IS_TASK (result));
|
|
|
|
if (!priv->features.has_samples && priv->features.has_marks)
|
|
gtk_stack_set_visible_child_name (priv->stack, "timings");
|
|
|
|
g_clear_pointer (&priv->mark_stats, g_hash_table_unref);
|
|
if ((stats = g_object_get_data (G_OBJECT (result), "MARK_STAT")))
|
|
priv->mark_stats = g_hash_table_ref (stats);
|
|
|
|
add_marks_to_details (self);
|
|
|
|
priv->needs_fit = TRUE;
|
|
|
|
return g_task_propagate_boolean (G_TASK (result), error);
|
|
}
|
|
|
|
static void
|
|
sysprof_capture_view_load_marks_cb (GObject *object,
|
|
GAsyncResult *result,
|
|
gpointer user_data)
|
|
{
|
|
SysprofMarksView *view = (SysprofMarksView *)object;
|
|
g_autoptr(GError) error = NULL;
|
|
g_autoptr(GTask) task = user_data;
|
|
LoadAsync *state;
|
|
|
|
g_assert (SYSPROF_IS_MARKS_VIEW (view));
|
|
g_assert (G_IS_ASYNC_RESULT (result));
|
|
g_assert (G_IS_TASK (task));
|
|
|
|
state = g_task_get_task_data (task);
|
|
g_assert (state != NULL);
|
|
g_assert (state->reader != NULL);
|
|
g_assert (state->n_active > 0);
|
|
|
|
state->n_active--;
|
|
|
|
if (!sysprof_marks_view_load_finish (view, result, &error))
|
|
{
|
|
if (!state->has_error)
|
|
{
|
|
state->has_error = TRUE;
|
|
g_task_return_error (task, g_steal_pointer (&error));
|
|
}
|
|
|
|
return;
|
|
}
|
|
|
|
if (state->n_active == 0)
|
|
g_task_return_boolean (task, TRUE);
|
|
}
|
|
|
|
static void
|
|
sysprof_capture_view_load_callgraph_cb (GObject *object,
|
|
GAsyncResult *result,
|
|
gpointer user_data)
|
|
{
|
|
SysprofCaptureView *self = (SysprofCaptureView *)object;
|
|
g_autoptr(GError) error = NULL;
|
|
g_autoptr(GTask) task = user_data;
|
|
LoadAsync *state;
|
|
|
|
g_assert (SYSPROF_IS_CAPTURE_VIEW (self));
|
|
g_assert (G_IS_ASYNC_RESULT (result));
|
|
g_assert (G_IS_TASK (task));
|
|
|
|
state = g_task_get_task_data (task);
|
|
g_assert (state != NULL);
|
|
g_assert (state->reader != NULL);
|
|
g_assert (state->n_active > 0);
|
|
|
|
state->n_active--;
|
|
|
|
if (!sysprof_capture_view_generate_callgraph_finish (self, result, &error))
|
|
{
|
|
if (!state->has_error)
|
|
{
|
|
state->has_error = TRUE;
|
|
g_task_return_error (task, g_steal_pointer (&error));
|
|
}
|
|
|
|
return;
|
|
}
|
|
|
|
if (state->n_active == 0)
|
|
g_task_return_boolean (task, TRUE);
|
|
}
|
|
|
|
static void
|
|
sysprof_capture_view_load_scan_cb (GObject *object,
|
|
GAsyncResult *result,
|
|
gpointer user_data)
|
|
{
|
|
SysprofCaptureView *self = (SysprofCaptureView *)object;
|
|
SysprofCaptureViewPrivate *priv = sysprof_capture_view_get_instance_private (self);
|
|
g_autoptr(GTask) task = user_data;
|
|
g_autoptr(GError) error = NULL;
|
|
LoadAsync *state;
|
|
|
|
g_assert (SYSPROF_IS_CAPTURE_VIEW (self));
|
|
g_assert (G_IS_ASYNC_RESULT (result));
|
|
g_assert (G_IS_TASK (task));
|
|
|
|
if (!sysprof_capture_view_scan_finish (self, result, &error))
|
|
{
|
|
g_task_return_error (task, g_steal_pointer (&error));
|
|
return;
|
|
}
|
|
|
|
state = g_task_get_task_data (task);
|
|
g_assert (state != NULL);
|
|
g_assert (state->reader != NULL);
|
|
|
|
if (priv->features.has_samples)
|
|
{
|
|
state->n_active++;
|
|
sysprof_capture_view_generate_callgraph_async (self,
|
|
state->reader,
|
|
state->selection,
|
|
g_task_get_cancellable (task),
|
|
sysprof_capture_view_load_callgraph_cb,
|
|
g_object_ref (task));
|
|
}
|
|
|
|
if (priv->features.has_counters || priv->features.has_marks)
|
|
sysprof_visualizer_view_set_reader (priv->visualizer_view, state->reader);
|
|
|
|
state->n_active++;
|
|
sysprof_marks_view_load_async (priv->marks_view,
|
|
state->reader,
|
|
state->selection,
|
|
g_task_get_cancellable (task),
|
|
sysprof_capture_view_load_marks_cb,
|
|
g_object_ref (task));
|
|
|
|
sysprof_details_view_set_reader (priv->details_view, priv->reader);
|
|
|
|
if (state->n_active == 0)
|
|
g_task_return_boolean (task, TRUE);
|
|
}
|
|
|
|
static void
|
|
sysprof_capture_view_real_load_async (SysprofCaptureView *self,
|
|
SysprofCaptureReader *reader,
|
|
GCancellable *cancellable,
|
|
GAsyncReadyCallback callback,
|
|
gpointer user_data)
|
|
{
|
|
SysprofCaptureViewPrivate *priv = sysprof_capture_view_get_instance_private (self);
|
|
g_autoptr(GTask) task = NULL;
|
|
SysprofSelection *selection;
|
|
LoadAsync *state;
|
|
|
|
g_assert (SYSPROF_IS_CAPTURE_VIEW (self));
|
|
g_assert (reader != NULL);
|
|
g_assert (!cancellable || G_IS_CANCELLABLE (cancellable));
|
|
|
|
g_clear_pointer (&priv->reader, sysprof_capture_reader_unref);
|
|
priv->reader = sysprof_capture_reader_ref (reader);
|
|
|
|
selection = sysprof_visualizer_view_get_selection (priv->visualizer_view);
|
|
|
|
state = g_slice_new0 (LoadAsync);
|
|
state->reader = sysprof_capture_reader_copy (reader);
|
|
state->selection = g_object_ref (selection);
|
|
state->n_active = 0;
|
|
|
|
task = g_task_new (self, cancellable, callback, user_data);
|
|
g_task_set_source_tag (task, sysprof_capture_view_real_load_async);
|
|
g_task_set_task_data (task,
|
|
g_steal_pointer (&state),
|
|
load_async_free);
|
|
sysprof_capture_view_monitor_task (self, task);
|
|
|
|
/* Zoom after loading state */
|
|
g_signal_connect_object (task,
|
|
"notify::completed",
|
|
G_CALLBACK (sysprof_capture_view_fit_to_width),
|
|
self,
|
|
G_CONNECT_SWAPPED);
|
|
|
|
/* Cancel any previously in-flight operations and save a cancellable so
|
|
* that any supplimental calls will result in the previous being cancelled.
|
|
*/
|
|
if (priv->cancellable != cancellable)
|
|
{
|
|
g_cancellable_cancel (priv->cancellable);
|
|
g_clear_object (&priv->cancellable);
|
|
g_set_object (&priv->cancellable, cancellable);
|
|
}
|
|
|
|
/* First, discover the the time range for the display */
|
|
sysprof_capture_view_scan_async (self,
|
|
reader,
|
|
cancellable,
|
|
sysprof_capture_view_load_scan_cb,
|
|
g_steal_pointer (&task));
|
|
}
|
|
|
|
static gboolean
|
|
sysprof_capture_view_real_load_finish (SysprofCaptureView *self,
|
|
GAsyncResult *result,
|
|
GError **error)
|
|
{
|
|
g_assert (SYSPROF_IS_CAPTURE_VIEW (self));
|
|
g_assert (G_IS_TASK (result));
|
|
|
|
return g_task_propagate_boolean (G_TASK (result), error);
|
|
}
|
|
|
|
static void
|
|
sysprof_capture_view_selection_changed_cb (SysprofCaptureView *self,
|
|
SysprofSelection *selection)
|
|
{
|
|
SysprofCaptureViewPrivate *priv = sysprof_capture_view_get_instance_private (self);
|
|
|
|
g_assert (SYSPROF_IS_CAPTURE_VIEW (self));
|
|
g_assert (SYSPROF_IS_SELECTION (selection));
|
|
|
|
if (priv->reader == NULL)
|
|
return;
|
|
|
|
sysprof_capture_view_generate_callgraph_async (self,
|
|
priv->reader,
|
|
selection,
|
|
NULL, NULL, NULL);
|
|
sysprof_marks_view_load_async (priv->marks_view,
|
|
priv->reader,
|
|
selection,
|
|
NULL, NULL, NULL);
|
|
}
|
|
|
|
static void
|
|
fit_zoom_cb (GSimpleAction *action,
|
|
GVariant *param,
|
|
gpointer user_data)
|
|
{
|
|
SysprofCaptureView *self = user_data;
|
|
|
|
g_assert (G_IS_SIMPLE_ACTION (action));
|
|
g_assert (SYSPROF_IS_CAPTURE_VIEW (self));
|
|
|
|
sysprof_capture_view_fit_to_width (self);
|
|
}
|
|
|
|
static void
|
|
set_use_underline_cb (GtkWidget *widget,
|
|
gpointer user_data)
|
|
{
|
|
if (GTK_IS_RADIO_BUTTON (widget))
|
|
{
|
|
GtkWidget *child = gtk_bin_get_child (GTK_BIN (widget));
|
|
|
|
if (GTK_IS_LABEL (child))
|
|
gtk_label_set_use_underline (GTK_LABEL (child), TRUE);
|
|
}
|
|
}
|
|
|
|
static void
|
|
sysprof_capture_view_map (GtkWidget *widget)
|
|
{
|
|
SysprofCaptureView *self = (SysprofCaptureView *)widget;
|
|
SysprofCaptureViewPrivate *priv = sysprof_capture_view_get_instance_private (self);
|
|
|
|
g_assert (SYSPROF_IS_CAPTURE_VIEW (self));
|
|
|
|
GTK_WIDGET_CLASS (sysprof_capture_view_parent_class)->map (widget);
|
|
|
|
if (priv->needs_fit)
|
|
g_timeout_add_full (G_PRIORITY_LOW,
|
|
25,
|
|
do_best_fit_in_idle,
|
|
g_object_ref (self),
|
|
g_object_unref);
|
|
}
|
|
|
|
static void
|
|
sysprof_capture_view_finalize (GObject *object)
|
|
{
|
|
SysprofCaptureView *self = (SysprofCaptureView *)object;
|
|
SysprofCaptureViewPrivate *priv = sysprof_capture_view_get_instance_private (self);
|
|
|
|
g_clear_pointer (&priv->mark_stats, g_hash_table_unref);
|
|
g_clear_pointer (&priv->reader, sysprof_capture_reader_unref);
|
|
g_clear_object (&priv->cancellable);
|
|
|
|
G_OBJECT_CLASS (sysprof_capture_view_parent_class)->finalize (object);
|
|
}
|
|
|
|
static void
|
|
sysprof_capture_view_get_property (GObject *object,
|
|
guint prop_id,
|
|
GValue *value,
|
|
GParamSpec *pspec)
|
|
{
|
|
SysprofCaptureView *self = (SysprofCaptureView *)object;
|
|
|
|
switch (prop_id)
|
|
{
|
|
case PROP_BUSY:
|
|
g_value_set_boolean (value, sysprof_capture_view_get_busy (self));
|
|
break;
|
|
|
|
default:
|
|
G_OBJECT_WARN_INVALID_PROPERTY_ID (object, prop_id, pspec);
|
|
}
|
|
}
|
|
|
|
static void
|
|
sysprof_capture_view_class_init (SysprofCaptureViewClass *klass)
|
|
{
|
|
GObjectClass *object_class = G_OBJECT_CLASS (klass);
|
|
GtkWidgetClass *widget_class = GTK_WIDGET_CLASS (klass);
|
|
|
|
object_class->finalize = sysprof_capture_view_finalize;
|
|
object_class->get_property = sysprof_capture_view_get_property;
|
|
|
|
widget_class->map = sysprof_capture_view_map;
|
|
|
|
klass->load_async = sysprof_capture_view_real_load_async;
|
|
klass->load_finish = sysprof_capture_view_real_load_finish;
|
|
|
|
gtk_widget_class_set_template_from_resource (widget_class, "/org/gnome/sysprof/ui/sysprof-capture-view.ui");
|
|
gtk_widget_class_bind_template_child_private (widget_class, SysprofCaptureView, callgraph_view);
|
|
gtk_widget_class_bind_template_child_private (widget_class, SysprofCaptureView, details_view);
|
|
gtk_widget_class_bind_template_child_private (widget_class, SysprofCaptureView, marks_view);
|
|
gtk_widget_class_bind_template_child_private (widget_class, SysprofCaptureView, stack);
|
|
gtk_widget_class_bind_template_child_private (widget_class, SysprofCaptureView, stack_switcher);
|
|
gtk_widget_class_bind_template_child_private (widget_class, SysprofCaptureView, time_adj);
|
|
gtk_widget_class_bind_template_child_private (widget_class, SysprofCaptureView, visualizer_view);
|
|
gtk_widget_class_bind_template_child_private (widget_class, SysprofCaptureView, zoom_manager);
|
|
|
|
properties [PROP_BUSY] =
|
|
g_param_spec_boolean ("busy",
|
|
"Busy",
|
|
"If the widget is busy",
|
|
FALSE,
|
|
(G_PARAM_READABLE | G_PARAM_STATIC_STRINGS));
|
|
|
|
g_object_class_install_properties (object_class, N_PROPS, properties);
|
|
|
|
g_type_ensure (SYSPROF_TYPE_DETAILS_VIEW);
|
|
}
|
|
|
|
static void
|
|
sysprof_capture_view_init (SysprofCaptureView *self)
|
|
{
|
|
SysprofCaptureViewPrivate *priv = sysprof_capture_view_get_instance_private (self);
|
|
g_autoptr(GSimpleActionGroup) group = NULL;
|
|
SysprofSelection *selection;
|
|
static GActionEntry actions[] = {
|
|
{ "fit-zoom", fit_zoom_cb },
|
|
};
|
|
|
|
gtk_widget_init_template (GTK_WIDGET (self));
|
|
|
|
gtk_container_foreach (GTK_CONTAINER (priv->stack_switcher),
|
|
set_use_underline_cb,
|
|
NULL);
|
|
|
|
selection = sysprof_visualizer_view_get_selection (priv->visualizer_view);
|
|
g_signal_connect_object (selection,
|
|
"changed",
|
|
G_CALLBACK (sysprof_capture_view_selection_changed_cb),
|
|
self,
|
|
G_CONNECT_SWAPPED);
|
|
|
|
gtk_widget_insert_action_group (GTK_WIDGET (self),
|
|
"zoom",
|
|
G_ACTION_GROUP (priv->zoom_manager));
|
|
|
|
group = g_simple_action_group_new ();
|
|
g_action_map_add_action_entries (G_ACTION_MAP (group),
|
|
actions,
|
|
G_N_ELEMENTS (actions),
|
|
self);
|
|
gtk_widget_insert_action_group (GTK_WIDGET (self),
|
|
"capture-view",
|
|
G_ACTION_GROUP (group));
|
|
|
|
_sysprof_marks_view_set_hadjustment (priv->marks_view, priv->time_adj);
|
|
_sysprof_visualizer_view_set_hadjustment (priv->visualizer_view, priv->time_adj);
|
|
}
|
|
|
|
/**
|
|
* sysprof_capture_view_load_async:
|
|
* @self: a #SysprofCaptureView
|
|
*
|
|
* Asynchronously loads a capture.
|
|
*
|
|
* Since: 3.34
|
|
*/
|
|
void
|
|
sysprof_capture_view_load_async (SysprofCaptureView *self,
|
|
SysprofCaptureReader *reader,
|
|
GCancellable *cancellable,
|
|
GAsyncReadyCallback callback,
|
|
gpointer user_data)
|
|
{
|
|
g_return_if_fail (SYSPROF_IS_CAPTURE_VIEW (self));
|
|
g_return_if_fail (reader != NULL);
|
|
g_return_if_fail (!cancellable || G_IS_CANCELLABLE (cancellable));
|
|
|
|
SYSPROF_CAPTURE_VIEW_GET_CLASS (self)->load_async (self, reader, cancellable, callback, user_data);
|
|
}
|
|
|
|
/**
|
|
* sysprof_capture_view_load_finish:
|
|
* @self: a #SysprofCaptureView
|
|
*
|
|
* Completes a request to load a capture.
|
|
*
|
|
* Since: 3.34
|
|
*/
|
|
gboolean
|
|
sysprof_capture_view_load_finish (SysprofCaptureView *self,
|
|
GAsyncResult *result,
|
|
GError **error)
|
|
{
|
|
g_return_val_if_fail (SYSPROF_IS_CAPTURE_VIEW (self), FALSE);
|
|
g_return_val_if_fail (G_IS_ASYNC_RESULT (result), FALSE);
|
|
|
|
return SYSPROF_CAPTURE_VIEW_GET_CLASS (self)->load_finish (self, result, error);
|
|
}
|
|
|
|
/**
|
|
* sysprof_capture_view_get_busy:
|
|
* @self: a #SysprofCaptureView
|
|
*
|
|
* Returns: %TRUE if the view is busy loading capture contents
|
|
*
|
|
* Since: 3.34
|
|
*/
|
|
gboolean
|
|
sysprof_capture_view_get_busy (SysprofCaptureView *self)
|
|
{
|
|
SysprofCaptureViewPrivate *priv = sysprof_capture_view_get_instance_private (self);
|
|
|
|
g_return_val_if_fail (SYSPROF_IS_CAPTURE_VIEW (self), FALSE);
|
|
|
|
return priv->busy > 0;
|
|
}
|
|
|
|
void
|
|
sysprof_capture_view_reset (SysprofCaptureView *self)
|
|
{
|
|
g_return_if_fail (SYSPROF_IS_CAPTURE_VIEW (self));
|
|
|
|
/* TODO: reset */
|
|
g_warning ("Clear all loaded state");
|
|
}
|
|
|
|
/**
|
|
* sysprof_capture_view_get_zoom_manager:
|
|
* @self: a #SysprofCaptureView
|
|
*
|
|
* Returns: (transfer none): a #SysprofZoomManager
|
|
*
|
|
* Since: 3.34
|
|
*/
|
|
SysprofZoomManager *
|
|
sysprof_capture_view_get_zoom_manager (SysprofCaptureView *self)
|
|
{
|
|
SysprofCaptureViewPrivate *priv = sysprof_capture_view_get_instance_private (self);
|
|
|
|
g_return_val_if_fail (SYSPROF_IS_CAPTURE_VIEW (self), NULL);
|
|
|
|
return priv->zoom_manager;
|
|
}
|
|
|
|
void
|
|
sysprof_capture_view_fit_to_width (SysprofCaptureView *self)
|
|
{
|
|
SysprofCaptureViewPrivate *priv = sysprof_capture_view_get_instance_private (self);
|
|
GtkAllocation alloc;
|
|
gdouble zoom;
|
|
gint64 duration;
|
|
gint width;
|
|
|
|
g_return_if_fail (SYSPROF_IS_CAPTURE_VIEW (self));
|
|
|
|
if (priv->reader == NULL)
|
|
return;
|
|
|
|
duration = priv->features.end_time - priv->features.begin_time;
|
|
|
|
g_print ("DURATION: %ld\n", duration);
|
|
|
|
if (duration <= 0)
|
|
return;
|
|
|
|
priv->needs_fit = FALSE;
|
|
|
|
/* Trim a bit off the width to avoid drawing past edges */
|
|
gtk_widget_get_allocation (GTK_WIDGET (self), &alloc);
|
|
width = MAX (100, alloc.width - 25);
|
|
|
|
zoom = sysprof_zoom_manager_fit_zoom_for_duration (priv->zoom_manager, duration, width);
|
|
sysprof_zoom_manager_set_zoom (priv->zoom_manager, zoom);
|
|
}
|
|
|
|
/**
|
|
* sysprof_capture_view_get_reader:
|
|
*
|
|
* Gets the reader for the view, if any.
|
|
*
|
|
* Returns: (transfer none): a #SysprofCaptureReader or %NULL
|
|
*
|
|
* Since: 3.34
|
|
*/
|
|
SysprofCaptureReader *
|
|
sysprof_capture_view_get_reader (SysprofCaptureView *self)
|
|
{
|
|
SysprofCaptureViewPrivate *priv = sysprof_capture_view_get_instance_private (self);
|
|
|
|
g_return_val_if_fail (SYSPROF_IS_CAPTURE_VIEW (self), NULL);
|
|
|
|
return priv->reader;
|
|
}
|