|
Packit Service |
407539 |
/* Copyright (C) 1998-99 Martin Baulig
|
|
Packit Service |
407539 |
This file is part of LibGTop 1.0.
|
|
Packit Service |
407539 |
|
|
Packit Service |
407539 |
Contributed by Martin Baulig <martin@home-of-linux.org>, April 1998.
|
|
Packit Service |
407539 |
|
|
Packit Service |
407539 |
LibGTop is free software; you can redistribute it and/or modify it
|
|
Packit Service |
407539 |
under the terms of the GNU General Public License as published by
|
|
Packit Service |
407539 |
the Free Software Foundation; either version 2 of the License,
|
|
Packit Service |
407539 |
or (at your option) any later version.
|
|
Packit Service |
407539 |
|
|
Packit Service |
407539 |
LibGTop is distributed in the hope that it will be useful, but WITHOUT
|
|
Packit Service |
407539 |
ANY WARRANTY; without even the implied warranty of MERCHANTABILITY or
|
|
Packit Service |
407539 |
FITNESS FOR A PARTICULAR PURPOSE. See the GNU General Public License
|
|
Packit Service |
407539 |
for more details.
|
|
Packit Service |
407539 |
|
|
Packit Service |
407539 |
You should have received a copy of the GNU General Public License
|
|
Packit Service |
407539 |
along with LibGTop; see the file COPYING. If not, write to the
|
|
Packit Service |
407539 |
Free Software Foundation, Inc., 51 Franklin Street, Fifth Floor,
|
|
Packit Service |
407539 |
Boston, MA 02110-1301, USA.
|
|
Packit Service |
407539 |
*/
|
|
Packit Service |
407539 |
|
|
Packit Service |
407539 |
#include <config.h>
|
|
Packit Service |
407539 |
#include <glib.h>
|
|
Packit Service |
407539 |
|
|
Packit Service |
407539 |
#include <glibtop.h>
|
|
Packit Service |
407539 |
#include <glibtop/error.h>
|
|
Packit Service |
407539 |
#include <glibtop/procmap.h>
|
|
Packit Service |
407539 |
|
|
Packit Service |
407539 |
#include <stddef.h>
|
|
Packit Service |
407539 |
|
|
Packit Service |
407539 |
#include "glibtop_private.h"
|
|
Packit Service |
407539 |
|
|
Packit Service |
407539 |
#define MKDEV(ma,mi) (((ma) << 20) | (mi))
|
|
Packit Service |
407539 |
|
|
Packit Service |
407539 |
#define MAPS_FILE "/proc/%u/maps"
|
|
Packit Service |
407539 |
|
|
Packit Service |
407539 |
|
|
Packit Service |
407539 |
#define PROC_MAPS_FORMAT "%16" G_GINT64_MODIFIER "x-%16" G_GINT64_MODIFIER "x %4c %16" G_GINT64_MODIFIER "x %02hx:%02hx %" G_GINT64_MODIFIER "u%*[ ]%n"
|
|
Packit Service |
407539 |
|
|
Packit Service |
407539 |
|
|
Packit Service |
407539 |
static const unsigned long _glibtop_sysdeps_proc_map =
|
|
Packit Service |
407539 |
(1L << GLIBTOP_PROC_MAP_NUMBER) + (1L << GLIBTOP_PROC_MAP_TOTAL) +
|
|
Packit Service |
407539 |
(1L << GLIBTOP_PROC_MAP_SIZE);
|
|
Packit Service |
407539 |
|
|
Packit Service |
407539 |
static const unsigned long _glibtop_sysdeps_map_entry =
|
|
Packit Service |
407539 |
(1L << GLIBTOP_MAP_ENTRY_START) + (1L << GLIBTOP_MAP_ENTRY_END) +
|
|
Packit Service |
407539 |
(1L << GLIBTOP_MAP_ENTRY_OFFSET) + (1L << GLIBTOP_MAP_ENTRY_PERM) +
|
|
Packit Service |
407539 |
(1L << GLIBTOP_MAP_ENTRY_INODE) + (1L << GLIBTOP_MAP_ENTRY_DEVICE) +
|
|
Packit Service |
407539 |
(1L << GLIBTOP_MAP_ENTRY_FILENAME);
|
|
Packit Service |
407539 |
|
|
Packit Service |
407539 |
|
|
Packit Service |
407539 |
/* Init function. */
|
|
Packit Service |
407539 |
|
|
Packit Service |
407539 |
void
|
|
Packit Service |
407539 |
_glibtop_init_proc_map_s (glibtop *server)
|
|
Packit Service |
407539 |
{
|
|
Packit Service |
407539 |
server->sysdeps.proc_map = _glibtop_sysdeps_proc_map;
|
|
Packit Service |
407539 |
}
|
|
Packit Service |
407539 |
|
|
Packit Service |
407539 |
|
|
Packit Service |
407539 |
glibtop_map_entry *
|
|
Packit Service |
407539 |
glibtop_get_proc_map_s (glibtop *server, glibtop_proc_map *buf, pid_t pid)
|
|
Packit Service |
407539 |
{
|
|
Packit Service |
407539 |
char procfilename[GLIBTOP_MAP_FILENAME_LEN+1];
|
|
Packit Service |
407539 |
|
|
Packit Service |
407539 |
/*
|
|
Packit Service |
407539 |
default size of 100 maybe inaccurate.
|
|
Packit Service |
407539 |
It's the average number of entry per process on my laptop
|
|
Packit Service |
407539 |
*/
|
|
Packit Service |
407539 |
|
|
Packit Service |
407539 |
GArray *entry_list = g_array_sized_new(FALSE, FALSE,
|
|
Packit Service |
407539 |
sizeof(glibtop_map_entry),
|
|
Packit Service |
407539 |
100);
|
|
Packit Service |
407539 |
FILE *maps;
|
|
Packit Service |
407539 |
const char *filename;
|
|
Packit Service |
407539 |
char *line = NULL;
|
|
Packit Service |
407539 |
size_t line_size = 0;
|
|
Packit Service |
407539 |
|
|
Packit Service |
407539 |
memset (buf, 0, sizeof (glibtop_proc_map));
|
|
Packit Service |
407539 |
|
|
Packit Service |
407539 |
filename = MAPS_FILE;
|
|
Packit Service |
407539 |
|
|
Packit Service |
407539 |
snprintf (procfilename, sizeof procfilename, filename, (unsigned)pid);
|
|
Packit Service |
407539 |
|
|
Packit Service |
407539 |
if((maps = fopen (procfilename, "r")) == NULL) {
|
|
Packit Service |
407539 |
return (glibtop_map_entry*) g_array_free(entry_list, TRUE);
|
|
Packit Service |
407539 |
}
|
|
Packit Service |
407539 |
|
|
Packit Service |
407539 |
while(TRUE)
|
|
Packit Service |
407539 |
{
|
|
Packit Service |
407539 |
unsigned long perm;
|
|
Packit Service |
407539 |
guint len;
|
|
Packit Service |
407539 |
int line_end;
|
|
Packit Service |
407539 |
|
|
Packit Service |
407539 |
unsigned short dev_major, dev_minor;
|
|
Packit Service |
407539 |
guint64 start, end, offset, inode;
|
|
Packit Service |
407539 |
char flags[4];
|
|
Packit Service |
407539 |
char *filename;
|
|
Packit Service |
407539 |
|
|
Packit Service |
407539 |
glibtop_map_entry *entry;
|
|
Packit Service |
407539 |
|
|
Packit Service |
407539 |
if (getline(&line, &line_size, maps) == -1)
|
|
Packit Service |
407539 |
break;
|
|
Packit Service |
407539 |
|
|
Packit Service |
407539 |
new_entry_line:
|
|
Packit Service |
407539 |
|
|
Packit Service |
407539 |
if (sscanf(line, PROC_MAPS_FORMAT,
|
|
Packit Service |
407539 |
&start, &end, flags, &offset,
|
|
Packit Service |
407539 |
&dev_major, &dev_minor, &inode, &line_end) != 7)
|
|
Packit Service |
407539 |
continue;
|
|
Packit Service |
407539 |
|
|
Packit Service |
407539 |
filename = line + line_end;
|
|
Packit Service |
407539 |
g_strstrip(filename);
|
|
Packit Service |
407539 |
|
|
Packit Service |
407539 |
/* Compute access permissions. */
|
|
Packit Service |
407539 |
perm = 0;
|
|
Packit Service |
407539 |
|
|
Packit Service |
407539 |
if (flags [0] == 'r')
|
|
Packit Service |
407539 |
perm |= GLIBTOP_MAP_PERM_READ;
|
|
Packit Service |
407539 |
|
|
Packit Service |
407539 |
if (flags [1] == 'w')
|
|
Packit Service |
407539 |
perm |= GLIBTOP_MAP_PERM_WRITE;
|
|
Packit Service |
407539 |
|
|
Packit Service |
407539 |
if (flags [2] == 'x')
|
|
Packit Service |
407539 |
perm |= GLIBTOP_MAP_PERM_EXECUTE;
|
|
Packit Service |
407539 |
|
|
Packit Service |
407539 |
if (flags [3] == 's')
|
|
Packit Service |
407539 |
perm |= GLIBTOP_MAP_PERM_SHARED;
|
|
Packit Service |
407539 |
else if (flags [3] == 'p')
|
|
Packit Service |
407539 |
perm |= GLIBTOP_MAP_PERM_PRIVATE;
|
|
Packit Service |
407539 |
|
|
Packit Service |
407539 |
/*
|
|
Packit Service |
407539 |
avoid copying the entry, grow by 1 and point to the last
|
|
Packit Service |
407539 |
element.
|
|
Packit Service |
407539 |
*/
|
|
Packit Service |
407539 |
len = entry_list->len;
|
|
Packit Service |
407539 |
g_array_set_size(entry_list, len + 1);
|
|
Packit Service |
407539 |
entry = &g_array_index(entry_list, glibtop_map_entry, len);
|
|
Packit Service |
407539 |
|
|
Packit Service |
407539 |
entry->flags = _glibtop_sysdeps_map_entry;
|
|
Packit Service |
407539 |
entry->start = start;
|
|
Packit Service |
407539 |
entry->end = end;
|
|
Packit Service |
407539 |
entry->offset = offset;
|
|
Packit Service |
407539 |
entry->perm = perm;
|
|
Packit Service |
407539 |
entry->device = MKDEV(dev_major, dev_minor);
|
|
Packit Service |
407539 |
entry->inode = inode;
|
|
Packit Service |
407539 |
g_strlcpy(entry->filename, filename, sizeof entry->filename);
|
|
Packit Service |
407539 |
}
|
|
Packit Service |
407539 |
|
|
Packit Service |
407539 |
eof:
|
|
Packit Service |
407539 |
|
|
Packit Service |
407539 |
free(line);
|
|
Packit Service |
407539 |
fclose (maps);
|
|
Packit Service |
407539 |
|
|
Packit Service |
407539 |
buf->flags = _glibtop_sysdeps_proc_map;
|
|
Packit Service |
407539 |
|
|
Packit Service |
407539 |
buf->number = entry_list->len;
|
|
Packit Service |
407539 |
buf->size = sizeof (glibtop_map_entry);
|
|
Packit Service |
407539 |
buf->total = buf->number * buf->size;
|
|
Packit Service |
407539 |
|
|
Packit Service |
407539 |
return (glibtop_map_entry*) g_array_free(entry_list, FALSE);
|
|
Packit Service |
407539 |
}
|