libsysprof-analyze: improve progress notifications

This commit is contained in:
Christian Hergert
2023-06-21 22:06:42 -07:00
parent 40778febae
commit 13323c5e79
6 changed files with 85 additions and 13 deletions

View File

@ -44,6 +44,7 @@ struct _SysprofDocumentLoader
double fraction; double fraction;
int fd; int fd;
guint notify_source; guint notify_source;
guint symbolizing : 1;
}; };
enum { enum {
@ -85,8 +86,14 @@ set_progress (double fraction,
g_assert (SYSPROF_IS_DOCUMENT_LOADER (self)); g_assert (SYSPROF_IS_DOCUMENT_LOADER (self));
g_mutex_lock (&self->mutex); g_mutex_lock (&self->mutex);
self->fraction = fraction;
self->fraction = fraction * .5;
if (self->symbolizing)
self->fraction += .5;
g_set_str (&self->message, message); g_set_str (&self->message, message);
if (!self->notify_source) if (!self->notify_source)
self->notify_source = g_idle_add_full (G_PRIORITY_LOW, self->notify_source = g_idle_add_full (G_PRIORITY_LOW,
progress_notify_in_idle, progress_notify_in_idle,
@ -421,7 +428,7 @@ sysprof_document_loader_load_symbols_cb (GObject *object,
self = g_task_get_source_object (task); self = g_task_get_source_object (task);
set_progress (1., _("Document loaded"), self); set_progress (0., _("Document loaded"), self);
if (!_sysprof_document_symbolize_finish (document, result, &error)) if (!_sysprof_document_symbolize_finish (document, result, &error))
g_task_return_error (task, g_steal_pointer (&error)); g_task_return_error (task, g_steal_pointer (&error));
@ -449,16 +456,24 @@ sysprof_document_loader_load_document_cb (GObject *object,
g_assert (symbolizer != NULL); g_assert (symbolizer != NULL);
g_assert (SYSPROF_IS_SYMBOLIZER (symbolizer)); g_assert (SYSPROF_IS_SYMBOLIZER (symbolizer));
set_progress (.9, _("Symbolizing stack traces"), self);
if (!(document = _sysprof_document_new_finish (result, &error))) if (!(document = _sysprof_document_new_finish (result, &error)))
g_task_return_error (task, g_steal_pointer (&error)); {
else g_task_return_error (task, g_steal_pointer (&error));
_sysprof_document_symbolize_async (document, set_progress (1., _("Loading failed"), self);
symbolizer, }
g_task_get_cancellable (task),
sysprof_document_loader_load_symbols_cb, self->symbolizing = TRUE;
g_object_ref (task));
set_progress (.0, _("Symbolizing stack traces"), self);
_sysprof_document_symbolize_async (document,
symbolizer,
set_progress,
g_object_ref (self),
g_object_unref,
g_task_get_cancellable (task),
sysprof_document_loader_load_symbols_cb,
g_object_ref (task));
} }
static void static void

View File

@ -53,6 +53,9 @@ SysprofDocument *_sysprof_document_new_finish (GAsyncResult *resul
GError **error); GError **error);
void _sysprof_document_symbolize_async (SysprofDocument *self, void _sysprof_document_symbolize_async (SysprofDocument *self,
SysprofSymbolizer *symbolizer, SysprofSymbolizer *symbolizer,
ProgressFunc progress_func,
gpointer progress_data,
GDestroyNotify progress_data_destroy,
GCancellable *cancellable, GCancellable *cancellable,
GAsyncReadyCallback callback, GAsyncReadyCallback callback,
gpointer user_data); gpointer user_data);

View File

@ -43,6 +43,9 @@ void _sysprof_document_symbols_new (SysprofDocument
SysprofStrings *strings, SysprofStrings *strings,
SysprofSymbolizer *symbolizer, SysprofSymbolizer *symbolizer,
GHashTable *pid_to_process_info, GHashTable *pid_to_process_info,
ProgressFunc progress_func,
gpointer progress_data,
GDestroyNotify progress_data_destroy,
GCancellable *cancellable, GCancellable *cancellable,
GAsyncReadyCallback callback, GAsyncReadyCallback callback,
gpointer user_data); gpointer user_data);

View File

@ -20,6 +20,8 @@
#include "config.h" #include "config.h"
#include <glib/gi18n.h>
#include "sysprof-address-layout-private.h" #include "sysprof-address-layout-private.h"
#include "sysprof-document-private.h" #include "sysprof-document-private.h"
#include "sysprof-document-symbols-private.h" #include "sysprof-document-symbols-private.h"
@ -65,11 +67,16 @@ typedef struct _Symbolize
SysprofDocumentSymbols *symbols; SysprofDocumentSymbols *symbols;
SysprofStrings *strings; SysprofStrings *strings;
GHashTable *pid_to_process_info; GHashTable *pid_to_process_info;
ProgressFunc progress_func;
gpointer progress_data;
GDestroyNotify progress_data_destroy;
} Symbolize; } Symbolize;
static void static void
symbolize_free (Symbolize *state) symbolize_free (Symbolize *state)
{ {
if (state->progress_data_destroy)
state->progress_data_destroy (state->progress_data);
g_clear_object (&state->document); g_clear_object (&state->document);
g_clear_object (&state->symbolizer); g_clear_object (&state->symbolizer);
g_clear_object (&state->symbols); g_clear_object (&state->symbols);
@ -155,6 +162,7 @@ sysprof_document_symbols_worker (GTask *task,
EggBitsetIter iter; EggBitsetIter iter;
EggBitset *bitset; EggBitset *bitset;
GListModel *model; GListModel *model;
guint count = 0;
guint i; guint i;
g_assert (source_object == NULL); g_assert (source_object == NULL);
@ -189,6 +197,8 @@ sysprof_document_symbols_worker (GTask *task,
if (!SYSPROF_IS_NO_SYMBOLIZER (state->symbolizer) && if (!SYSPROF_IS_NO_SYMBOLIZER (state->symbolizer) &&
egg_bitset_iter_init_first (&iter, bitset, &i)) egg_bitset_iter_init_first (&iter, bitset, &i))
{ {
guint n_items = egg_bitset_get_size (bitset);
do do
{ {
g_autoptr(SysprofDocumentTraceable) traceable = g_list_model_get_item (model, i); g_autoptr(SysprofDocumentTraceable) traceable = g_list_model_get_item (model, i);
@ -200,6 +210,11 @@ sysprof_document_symbols_worker (GTask *task,
process_info, process_info,
traceable, traceable,
state->symbolizer); state->symbolizer);
count++;
if (state->progress_func != NULL && count % 100 == 0)
state->progress_func (count / (double)n_items, _("Symbolizing stack traces"), state->progress_data);
} }
while (egg_bitset_iter_next (&iter, &i)); while (egg_bitset_iter_next (&iter, &i));
} }
@ -214,6 +229,9 @@ _sysprof_document_symbols_new (SysprofDocument *document,
SysprofStrings *strings, SysprofStrings *strings,
SysprofSymbolizer *symbolizer, SysprofSymbolizer *symbolizer,
GHashTable *pid_to_process_info, GHashTable *pid_to_process_info,
ProgressFunc progress_func,
gpointer progress_data,
GDestroyNotify progress_data_destroy,
GCancellable *cancellable, GCancellable *cancellable,
GAsyncReadyCallback callback, GAsyncReadyCallback callback,
gpointer user_data) gpointer user_data)
@ -230,6 +248,9 @@ _sysprof_document_symbols_new (SysprofDocument *document,
state->symbols = g_object_new (SYSPROF_TYPE_DOCUMENT_SYMBOLS, NULL); state->symbols = g_object_new (SYSPROF_TYPE_DOCUMENT_SYMBOLS, NULL);
state->strings = sysprof_strings_ref (strings); state->strings = sysprof_strings_ref (strings);
state->pid_to_process_info = g_hash_table_ref (pid_to_process_info); state->pid_to_process_info = g_hash_table_ref (pid_to_process_info);
state->progress_func = progress_func;
state->progress_data = progress_data;
state->progress_data_destroy = progress_data_destroy;
task = g_task_new (NULL, cancellable, callback, user_data); task = g_task_new (NULL, cancellable, callback, user_data);
g_task_set_source_tag (task, _sysprof_document_symbols_new); g_task_set_source_tag (task, _sysprof_document_symbols_new);

View File

@ -1003,6 +1003,21 @@ sysprof_document_symbolize_symbols_cb (GObject *object,
g_task_return_boolean (task, TRUE); g_task_return_boolean (task, TRUE);
} }
typedef struct _Symbolize
{
ProgressFunc progress_func;
gpointer progress_data;
GDestroyNotify progress_data_destroy;
} Symbolize;
static void
symbolize_free (Symbolize *state)
{
if (state->progress_data_destroy)
state->progress_data_destroy (state->progress_data);
g_free (state);
}
static void static void
sysprof_document_symbolize_prepare_cb (GObject *object, sysprof_document_symbolize_prepare_cb (GObject *object,
GAsyncResult *result, GAsyncResult *result,
@ -1012,6 +1027,7 @@ sysprof_document_symbolize_prepare_cb (GObject *object,
g_autoptr(GTask) task = user_data; g_autoptr(GTask) task = user_data;
g_autoptr(GError) error = NULL; g_autoptr(GError) error = NULL;
SysprofDocument *self; SysprofDocument *self;
Symbolize *state;
g_assert (SYSPROF_IS_SYMBOLIZER (symbolizer)); g_assert (SYSPROF_IS_SYMBOLIZER (symbolizer));
g_assert (G_IS_ASYNC_RESULT (result)); g_assert (G_IS_ASYNC_RESULT (result));
@ -1023,6 +1039,8 @@ sysprof_document_symbolize_prepare_cb (GObject *object,
g_assert (SYSPROF_IS_DOCUMENT (self)); g_assert (SYSPROF_IS_DOCUMENT (self));
g_assert (self->pid_to_process_info != NULL); g_assert (self->pid_to_process_info != NULL);
state = g_task_get_task_data (task);
if (!_sysprof_symbolizer_prepare_finish (symbolizer, result, &error)) if (!_sysprof_symbolizer_prepare_finish (symbolizer, result, &error))
g_task_return_error (task, g_steal_pointer (&error)); g_task_return_error (task, g_steal_pointer (&error));
else else
@ -1030,6 +1048,9 @@ sysprof_document_symbolize_prepare_cb (GObject *object,
self->strings, self->strings,
symbolizer, symbolizer,
self->pid_to_process_info, self->pid_to_process_info,
state->progress_func,
state->progress_data,
NULL,
g_task_get_cancellable (task), g_task_get_cancellable (task),
sysprof_document_symbolize_symbols_cb, sysprof_document_symbolize_symbols_cb,
g_object_ref (task)); g_object_ref (task));
@ -1038,12 +1059,15 @@ sysprof_document_symbolize_prepare_cb (GObject *object,
void void
_sysprof_document_symbolize_async (SysprofDocument *self, _sysprof_document_symbolize_async (SysprofDocument *self,
SysprofSymbolizer *symbolizer, SysprofSymbolizer *symbolizer,
ProgressFunc progress_func,
gpointer progress_data,
GDestroyNotify progress_data_destroy,
GCancellable *cancellable, GCancellable *cancellable,
GAsyncReadyCallback callback, GAsyncReadyCallback callback,
gpointer user_data) gpointer user_data)
{ {
g_autoptr(SysprofDocumentSymbols) symbols = NULL;
g_autoptr(GTask) task = NULL; g_autoptr(GTask) task = NULL;
Symbolize *state;
g_return_if_fail (SYSPROF_IS_DOCUMENT (self)); g_return_if_fail (SYSPROF_IS_DOCUMENT (self));
g_return_if_fail (SYSPROF_IS_SYMBOLIZER (symbolizer)); g_return_if_fail (SYSPROF_IS_SYMBOLIZER (symbolizer));
@ -1052,6 +1076,12 @@ _sysprof_document_symbolize_async (SysprofDocument *self,
task = g_task_new (self, cancellable, callback, user_data); task = g_task_new (self, cancellable, callback, user_data);
g_task_set_source_tag (task, _sysprof_document_symbolize_async); g_task_set_source_tag (task, _sysprof_document_symbolize_async);
state = g_new0 (Symbolize, 1);
state->progress_func = progress_func;
state->progress_data = progress_data;
state->progress_data_destroy = progress_data_destroy;
g_task_set_task_data(task, state, (GDestroyNotify)symbolize_free);
_sysprof_symbolizer_prepare_async (symbolizer, _sysprof_symbolizer_prepare_async (symbolizer,
self, self,
cancellable, cancellable,

View File

@ -147,7 +147,7 @@ main (int argc,
"spacing", 6, "spacing", 6,
NULL); NULL);
message = g_object_new (GTK_TYPE_LABEL, message = g_object_new (GTK_TYPE_LABEL,
"xalign", .0f, "xalign", 1.f,
"hexpand", TRUE, "hexpand", TRUE,
NULL); NULL);
progress = g_object_new (GTK_TYPE_PROGRESS_BAR, progress = g_object_new (GTK_TYPE_PROGRESS_BAR,