2020-04-03 20:11:46 +01:00
|
|
|
/*
|
|
|
|
* Utility function to get QEMU's own process map
|
|
|
|
*
|
|
|
|
* Copyright (c) 2020 Linaro Ltd
|
|
|
|
*
|
|
|
|
* SPDX-License-Identifier: GPL-2.0-or-later
|
|
|
|
*/
|
|
|
|
|
|
|
|
#include "qemu/osdep.h"
|
|
|
|
#include "qemu/cutils.h"
|
|
|
|
#include "qemu/selfmap.h"
|
|
|
|
|
2023-08-06 17:10:44 +00:00
|
|
|
IntervalTreeRoot *read_self_maps(void)
|
2020-04-03 20:11:46 +01:00
|
|
|
{
|
2023-08-06 17:10:44 +00:00
|
|
|
IntervalTreeRoot *root;
|
|
|
|
gchar *maps, **lines;
|
|
|
|
guint i, nlines;
|
2020-04-03 20:11:46 +01:00
|
|
|
|
2023-08-06 17:10:44 +00:00
|
|
|
if (!g_file_get_contents("/proc/self/maps", &maps, NULL, NULL)) {
|
|
|
|
return NULL;
|
|
|
|
}
|
|
|
|
|
|
|
|
root = g_new0(IntervalTreeRoot, 1);
|
|
|
|
lines = g_strsplit(maps, "\n", 0);
|
|
|
|
nlines = g_strv_length(lines);
|
|
|
|
|
|
|
|
for (i = 0; i < nlines; i++) {
|
|
|
|
gchar **fields = g_strsplit(lines[i], " ", 6);
|
|
|
|
guint nfields = g_strv_length(fields);
|
|
|
|
|
|
|
|
if (nfields > 4) {
|
|
|
|
uint64_t start, end, offset, inode;
|
|
|
|
int errors = 0;
|
|
|
|
const char *p;
|
2020-04-03 20:11:46 +01:00
|
|
|
|
2023-08-06 17:10:44 +00:00
|
|
|
errors |= qemu_strtou64(fields[0], &p, 16, &start);
|
|
|
|
errors |= qemu_strtou64(p + 1, NULL, 16, &end);
|
|
|
|
errors |= qemu_strtou64(fields[2], NULL, 16, &offset);
|
|
|
|
errors |= qemu_strtou64(fields[4], NULL, 10, &inode);
|
|
|
|
|
|
|
|
if (!errors) {
|
|
|
|
size_t dev_len, path_len;
|
|
|
|
MapInfo *e;
|
|
|
|
|
|
|
|
dev_len = strlen(fields[3]) + 1;
|
|
|
|
if (nfields == 6) {
|
|
|
|
p = fields[5];
|
|
|
|
p += strspn(p, " ");
|
|
|
|
path_len = strlen(p) + 1;
|
|
|
|
} else {
|
|
|
|
p = NULL;
|
|
|
|
path_len = 0;
|
|
|
|
}
|
2020-04-03 20:11:46 +01:00
|
|
|
|
2023-08-06 17:10:44 +00:00
|
|
|
e = g_malloc0(sizeof(*e) + dev_len + path_len);
|
|
|
|
|
|
|
|
e->itree.start = start;
|
|
|
|
e->itree.last = end - 1;
|
|
|
|
e->offset = offset;
|
|
|
|
e->inode = inode;
|
2020-04-03 20:11:46 +01:00
|
|
|
|
|
|
|
e->is_read = fields[1][0] == 'r';
|
|
|
|
e->is_write = fields[1][1] == 'w';
|
|
|
|
e->is_exec = fields[1][2] == 'x';
|
|
|
|
e->is_priv = fields[1][3] == 'p';
|
|
|
|
|
2023-08-06 17:10:44 +00:00
|
|
|
memcpy(e->dev, fields[3], dev_len);
|
|
|
|
if (path_len) {
|
|
|
|
e->path = memcpy(e->dev + dev_len, p, path_len);
|
2020-04-03 20:11:46 +01:00
|
|
|
}
|
|
|
|
|
2023-08-06 17:10:44 +00:00
|
|
|
interval_tree_insert(&e->itree, root);
|
|
|
|
}
|
2020-04-03 20:11:46 +01:00
|
|
|
}
|
2023-08-06 17:10:44 +00:00
|
|
|
g_strfreev(fields);
|
2020-04-03 20:11:46 +01:00
|
|
|
}
|
2023-08-06 17:10:44 +00:00
|
|
|
g_strfreev(lines);
|
|
|
|
g_free(maps);
|
2020-04-03 20:11:46 +01:00
|
|
|
|
2023-08-06 17:10:44 +00:00
|
|
|
return root;
|
2020-04-03 20:11:46 +01:00
|
|
|
}
|
|
|
|
|
|
|
|
/**
|
|
|
|
* free_self_maps:
|
2023-08-06 17:10:44 +00:00
|
|
|
* @root: an interval tree
|
2020-04-03 20:11:46 +01:00
|
|
|
*
|
2023-08-06 17:10:44 +00:00
|
|
|
* Free a tree of MapInfo structures.
|
|
|
|
* Since we allocated each MapInfo in one chunk, we need not consider the
|
|
|
|
* contents and can simply free each RBNode.
|
2020-04-03 20:11:46 +01:00
|
|
|
*/
|
2023-08-06 17:10:44 +00:00
|
|
|
|
|
|
|
static void free_rbnode(RBNode *n)
|
2020-04-03 20:11:46 +01:00
|
|
|
{
|
2023-08-06 17:10:44 +00:00
|
|
|
if (n) {
|
|
|
|
free_rbnode(n->rb_left);
|
|
|
|
free_rbnode(n->rb_right);
|
|
|
|
g_free(n);
|
|
|
|
}
|
2020-04-03 20:11:46 +01:00
|
|
|
}
|
|
|
|
|
2023-08-06 17:10:44 +00:00
|
|
|
void free_self_maps(IntervalTreeRoot *root)
|
2020-04-03 20:11:46 +01:00
|
|
|
{
|
2023-08-06 17:10:44 +00:00
|
|
|
if (root) {
|
|
|
|
free_rbnode(root->rb_root.rb_node);
|
|
|
|
g_free(root);
|
|
|
|
}
|
2020-04-03 20:11:46 +01:00
|
|
|
}
|