Files
sysprof/src/tests/show-page-usage.c
Philip Withnall e19d70bca0 libsysprof-capture: Drop GError usage from SysprofCaptureReader
Use `errno` instead, which is icky, but given that all of the failure
modes are from POSIX I/O functions, it’s at least in keeping with them.

This is a major API break.

Signed-off-by: Philip Withnall <withnall@endlessm.com>

Helps: #40
2020-07-03 22:00:34 +01:00

198 lines
4.6 KiB
C

/* show-page-usage.c
*
* Copyright 2020 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 <cairo.h>
#include <glib.h>
#include <stddef.h>
#include <rax.h>
#include <sysprof.h>
static GMainLoop *main_loop;
static gint
u64_compare (gconstpointer a,
gconstpointer b)
{
const guint64 *aptr = a;
const guint64 *bptr = b;
if (*aptr < *bptr)
return -1;
else if (*aptr > *bptr)
return 1;
else
return 0;
}
static void
generate_cb (GObject *object,
GAsyncResult *result,
gpointer user_data)
{
SysprofProfile *profile = (SysprofProfile *)object;
g_autoptr(GError) error = NULL;
GHashTable *seen;
GHashTableIter iter;
cairo_t *cr;
cairo_surface_t *surface;
GArray *ar;
raxIterator it;
rax *r;
gpointer k,v;
g_assert (SYSPROF_IS_MEMPROF_PROFILE (profile));
g_assert (G_IS_ASYNC_RESULT (result));
if (!sysprof_profile_generate_finish (profile, result, &error))
{
g_printerr ("%s\n", error->message);
exit (EXIT_FAILURE);
}
r = sysprof_memprof_profile_get_native (SYSPROF_MEMPROF_PROFILE (profile));
seen = g_hash_table_new (NULL, NULL);
raxStart (&it, r);
raxSeek (&it, "^", NULL, 0);
while (raxNext (&it))
{
guint64 page;
guint64 addr;
memcpy (&addr, it.key, sizeof addr);
page = addr / 4096;
if (g_hash_table_contains (seen, GSIZE_TO_POINTER (page)))
continue;
g_hash_table_insert (seen, GSIZE_TO_POINTER (page), NULL);
}
raxStop (&it);
ar = g_array_sized_new (FALSE, FALSE, sizeof (guint64), g_hash_table_size (seen));
g_hash_table_iter_init (&iter, seen);
while (g_hash_table_iter_next (&iter, &k, &v))
{
guint64 key = GPOINTER_TO_SIZE (k);
g_array_append_val (ar, key);
}
g_array_sort (ar, u64_compare);
for (guint i = 0; i < ar->len; i++)
{
guint64 key = g_array_index (ar, guint64, i);
g_hash_table_insert (seen, GSIZE_TO_POINTER (key), GSIZE_TO_POINTER (i));
}
g_printerr ("We have %u pages to graph\n", ar->len);
surface = cairo_image_surface_create (CAIRO_FORMAT_RGB24, ar->len, (4096/16));
cr = cairo_create (surface);
cairo_set_line_width (cr, 1.0);
cairo_set_antialias (cr, CAIRO_ANTIALIAS_NONE);
cairo_set_source_rgb (cr, 1, 1, 1);
cairo_rectangle (cr, 0, 0, ar->len, (4096/16));
cairo_fill (cr);
cairo_set_source_rgb (cr, 0, 0, 0);
cairo_scale (cr, 1.0, 1.0/16.0);
cairo_translate (cr, .5, .5);
raxStart (&it, r);
raxSeek (&it, "^", NULL, 0);
while (raxNext (&it))
{
guint64 page;
guint64 addr;
guint64 size;
guint x;
guint y;
memcpy (&addr, it.key, sizeof addr);
page = addr / 4096;
size = GPOINTER_TO_SIZE (it.data);
x = GPOINTER_TO_UINT (g_hash_table_lookup (seen, GSIZE_TO_POINTER (page)));
y = addr % 4096;
/* TODO: Need size */
cairo_move_to (cr, x, y);
cairo_line_to (cr, x, y+size);
}
raxStop (&it);
cairo_stroke (cr);
cairo_surface_write_to_png (surface, "memory.png");
cairo_destroy (cr);
cairo_surface_destroy (surface);
g_array_unref (ar);
g_hash_table_unref (seen);
g_main_loop_quit (main_loop);
}
gint
main (gint argc,
gchar *argv[])
{
SysprofCaptureReader *reader;
const gchar *filename = argv[1];
g_autoptr(SysprofProfile) memprof = NULL;
if (argc < 2)
{
g_printerr ("usage: %s FILENAME\n", argv[0]);
return EXIT_FAILURE;
}
main_loop = g_main_loop_new (NULL, FALSE);
if (!(reader = sysprof_capture_reader_new (filename)))
{
int errsv = errno;
g_printerr ("%s\n", g_strerror (errsv));
return EXIT_FAILURE;
}
memprof = sysprof_memprof_profile_new ();
sysprof_profile_set_reader (memprof, reader);
sysprof_profile_generate (memprof, NULL, generate_cb, NULL);
g_main_loop_run (main_loop);
g_main_loop_unref (main_loop);
sysprof_capture_reader_unref (reader);
return EXIT_SUCCESS;
}