Blame sysdeps/cygwin/procmap.c

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
}