libsysprof-analyze: add SysprofDocumentJitmap

This adds a specific frame type for the Jitmap frames in the capture files.
You can iterate them without having to bswap as well, which is why this
does not use the SysprofCaptureJitmapIter (which does require bswap'd
frames).
This commit is contained in:
Christian Hergert
2023-05-22 12:18:57 -07:00
parent c7f025d4d3
commit d28080b9ad
9 changed files with 304 additions and 0 deletions

View File

@ -7,6 +7,7 @@ libsysprof_analyze_public_sources = [
'sysprof-document-file-chunk.c',
'sysprof-document-fork.c',
'sysprof-document-frame.c',
'sysprof-document-jitmap.c',
'sysprof-document-loader.c',
'sysprof-document-log.c',
'sysprof-document-mark.c',
@ -51,6 +52,7 @@ libsysprof_analyze_public_headers = [
'sysprof-document-file-chunk.h',
'sysprof-document-fork.h',
'sysprof-document-frame.h',
'sysprof-document-jitmap.h',
'sysprof-document-loader.h',
'sysprof-document-log.h',
'sysprof-document-mark.h',

View File

@ -34,6 +34,7 @@ G_BEGIN_DECLS
# include "sysprof-document-fork.h"
# include "sysprof-document-frame.h"
# include "sysprof-document-loader.h"
# include "sysprof-document-jitmap.h"
# include "sysprof-document-log.h"
# include "sysprof-document-mark.h"
# include "sysprof-document-metadata.h"

View File

@ -46,6 +46,9 @@ SysprofDocumentFrame *_sysprof_document_frame_new (GMappedFile *ma
guint16 frame_len,
gboolean needs_swap);
#define SYSPROF_DOCUMENT_FRAME_ENDPTR(obj) \
(&((const guint8 *)SYSPROF_DOCUMENT_FRAME(obj)->frame)[SYSPROF_DOCUMENT_FRAME(obj)->frame_len])
#define SYSPROF_DOCUMENT_FRAME_GET(obj, type) \
((const type *)(SYSPROF_DOCUMENT_FRAME(obj)->frame))

View File

@ -27,6 +27,7 @@
#include "sysprof-document-file-chunk.h"
#include "sysprof-document-fork.h"
#include "sysprof-document-log.h"
#include "sysprof-document-jitmap.h"
#include "sysprof-document-mark.h"
#include "sysprof-document-mmap.h"
#include "sysprof-document-overlay.h"
@ -173,6 +174,10 @@ _sysprof_document_frame_new (GMappedFile *mapped_file,
gtype = SYSPROF_TYPE_DOCUMENT_OVERLAY;
break;
case SYSPROF_CAPTURE_FRAME_JITMAP:
gtype = SYSPROF_TYPE_DOCUMENT_JITMAP;
break;
default:
gtype = SYSPROF_TYPE_DOCUMENT_FRAME;
break;

View File

@ -0,0 +1,166 @@
/* sysprof-document-jitmap.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-document-frame-private.h"
#include "sysprof-document-jitmap.h"
typedef struct _Jitmap
{
SysprofAddress address;
const char *name;
} Jitmap;
struct _SysprofDocumentJitmap
{
SysprofDocumentFrame parent_instance;
GArray *jitmaps;
guint initialized : 1;
};
struct _SysprofDocumentJitmapClass
{
SysprofDocumentFrameClass parent_class;
};
enum {
PROP_0,
PROP_SIZE,
N_PROPS
};
G_DEFINE_FINAL_TYPE (SysprofDocumentJitmap, sysprof_document_jitmap, SYSPROF_TYPE_DOCUMENT_FRAME)
static GParamSpec *properties [N_PROPS];
static void
sysprof_document_jitmap_get_property (GObject *object,
guint prop_id,
GValue *value,
GParamSpec *pspec)
{
SysprofDocumentJitmap *self = SYSPROF_DOCUMENT_JITMAP (object);
switch (prop_id)
{
case PROP_SIZE:
g_value_set_uint (value, sysprof_document_jitmap_get_size (self));
break;
default:
G_OBJECT_WARN_INVALID_PROPERTY_ID (object, prop_id, pspec);
}
}
static void
sysprof_document_jitmap_class_init (SysprofDocumentJitmapClass *klass)
{
GObjectClass *object_class = G_OBJECT_CLASS (klass);
object_class->get_property = sysprof_document_jitmap_get_property;
properties [PROP_SIZE] =
g_param_spec_string ("size", NULL, NULL,
NULL,
(G_PARAM_READABLE | G_PARAM_STATIC_STRINGS));
g_object_class_install_properties (object_class, N_PROPS, properties);
}
static void
sysprof_document_jitmap_init (SysprofDocumentJitmap *self)
{
self->jitmaps = g_array_new (FALSE, FALSE, sizeof (Jitmap));
}
guint
sysprof_document_jitmap_get_size (SysprofDocumentJitmap *self)
{
const SysprofCaptureJitmap *jitmap;
g_return_val_if_fail (SYSPROF_IS_DOCUMENT_JITMAP (self), 0);
jitmap = SYSPROF_DOCUMENT_FRAME_GET (self, SysprofCaptureJitmap);
return SYSPROF_DOCUMENT_FRAME_UINT32 (self, jitmap->n_jitmaps);
}
/**
* sysprof_document_jitmap_get_mapping:
* @self: a #SysprofDocumentJitmap
* @nth: the index of the mapping
* @address: (out): a location for the address
*
* Gets the @nth mapping and returns it as @address and the return value
* of this function.
*
* Returns: (nullable): the name of the symbol, or %NULL if @nth is
* out of bounds.
*/
const char *
sysprof_document_jitmap_get_mapping (SysprofDocumentJitmap *self,
guint nth,
SysprofAddress *address)
{
const SysprofCaptureJitmap *frame;
const Jitmap *j;
g_return_val_if_fail (SYSPROF_IS_DOCUMENT_JITMAP (self), NULL);
g_return_val_if_fail (address != NULL, NULL);
if G_UNLIKELY (!self->initialized)
{
const guint8 *pos;
const guint8 *endptr;
self->initialized = TRUE;
frame = SYSPROF_DOCUMENT_FRAME_GET (self, SysprofCaptureJitmap);
endptr = SYSPROF_DOCUMENT_FRAME_ENDPTR (self);
pos = frame->data;
while (pos < endptr)
{
Jitmap map;
if (pos + sizeof map.address >= endptr)
break;
memcpy (&map.address, pos, sizeof map.address);
pos += sizeof map.address;
if (!(map.name = SYSPROF_DOCUMENT_FRAME_CSTRING (self, (const char *)pos)))
break;
pos += strlen (map.name) + 1;
g_array_append_val (self->jitmaps, map);
}
}
if (nth >= self->jitmaps->len)
return NULL;
j = &g_array_index (self->jitmaps, Jitmap, nth);
*address = j->address;
return j->name;
}

View File

@ -0,0 +1,49 @@
/* sysprof-document-jitmap.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 <gio/gio.h>
#include "sysprof-document-frame.h"
G_BEGIN_DECLS
#define SYSPROF_TYPE_DOCUMENT_JITMAP (sysprof_document_jitmap_get_type())
#define SYSPROF_IS_DOCUMENT_JITMAP(obj) G_TYPE_CHECK_INSTANCE_TYPE(obj, SYSPROF_TYPE_DOCUMENT_JITMAP)
#define SYSPROF_DOCUMENT_JITMAP(obj) G_TYPE_CHECK_INSTANCE_CAST(obj, SYSPROF_TYPE_DOCUMENT_JITMAP, SysprofDocumentJitmap)
#define SYSPROF_DOCUMENT_JITMAP_CLASS(klass) G_TYPE_CHECK_CLASS_CAST(klass, SYSPROF_TYPE_DOCUMENT_JITMAP, SysprofDocumentJitmapClass)
typedef struct _SysprofDocumentJitmap SysprofDocumentJitmap;
typedef struct _SysprofDocumentJitmapClass SysprofDocumentJitmapClass;
SYSPROF_AVAILABLE_IN_ALL
GType sysprof_document_jitmap_get_type (void) G_GNUC_CONST;
SYSPROF_AVAILABLE_IN_ALL
guint sysprof_document_jitmap_get_size (SysprofDocumentJitmap *self);
SYSPROF_AVAILABLE_IN_ALL
const char *sysprof_document_jitmap_get_mapping (SysprofDocumentJitmap *self,
guint position,
SysprofAddress *address);
G_DEFINE_AUTOPTR_CLEANUP_FUNC (SysprofDocumentJitmap, g_object_unref)
G_END_DECLS

View File

@ -15,6 +15,7 @@ libsysprof_analyze_testsuite = {
'test-capture-model' : {'skip': true},
'test-elf-loader' : {'skip': true},
'test-list-files' : {'skip': true},
'test-list-jitmap' : {'skip': true},
'test-print-file' : {'skip': true},
'test-list-processes' : {'skip': true},
'test-list-address-layout' : {'skip': true},

View File

@ -81,6 +81,9 @@ main (int argc,
sysprof_document_mmap_get_end_address (SYSPROF_DOCUMENT_MMAP (frame)),
sysprof_document_mmap_get_file_offset (SYSPROF_DOCUMENT_MMAP (frame)),
sysprof_document_mmap_get_file (SYSPROF_DOCUMENT_MMAP (frame)));
else if (SYSPROF_IS_DOCUMENT_JITMAP (frame))
g_print (" n_jitmaps=%u",
sysprof_document_jitmap_get_size (SYSPROF_DOCUMENT_JITMAP (frame)));
else if (SYSPROF_IS_DOCUMENT_ALLOCATION (frame))
{
if (sysprof_document_allocation_is_free (SYSPROF_DOCUMENT_ALLOCATION (frame)))

View File

@ -0,0 +1,74 @@
/* test-list-jitmap.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 <sysprof-analyze.h>
#include "sysprof-document-private.h"
int
main (int argc,
char *argv[])
{
g_autoptr(SysprofDocumentLoader) loader = NULL;
g_autoptr(SysprofDocument) document = NULL;
g_autoptr(GError) error = NULL;
guint n_items;
if (argc < 2)
{
g_printerr ("usage: %s CAPTURE_FILE\n", argv[0]);
return 1;
}
loader = sysprof_document_loader_new (argv[1]);
sysprof_document_loader_set_symbolizer (loader, sysprof_no_symbolizer_get ());
if (!(document = sysprof_document_loader_load (loader, NULL, &error)))
{
g_printerr ("Failed to open capture: %s\n", error->message);
return 1;
}
n_items = g_list_model_get_n_items (G_LIST_MODEL (document));
for (guint i = 0; i < n_items; i++)
{
g_autoptr(SysprofDocumentFrame) frame = g_list_model_get_item ((GListModel *)document, i);
if (SYSPROF_IS_DOCUMENT_JITMAP (frame))
{
SysprofDocumentJitmap *jitmap = SYSPROF_DOCUMENT_JITMAP (frame);
guint size = sysprof_document_jitmap_get_size (jitmap);
for (guint j = 0; j < size; j++)
{
SysprofAddress address;
const char *name;
if (!(name = sysprof_document_jitmap_get_mapping (jitmap, j, &address)))
break;
g_print ("0x%"G_GINT64_MODIFIER"x: %s\n", address, name);
}
}
}
return 0;
}