Blame sysdeps/linux/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 <linux/kdev_t.h>
Packit Service 407539
#include <stddef.h>
Packit Service 407539
Packit Service 407539
#include "glibtop_private.h"
Packit Service 407539
Packit Service 407539
#include "procmap_smaps.c"
Packit Service 407539
Packit Service 407539
#define  MAPS_FILE "/proc/%u/maps"
Packit Service 407539
#define SMAPS_FILE "/proc/%u/smaps"
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
static const unsigned long _glibtop_sysdeps_map_entry_smaps =
Packit Service 407539
(1UL << GLIBTOP_MAP_ENTRY_SIZE) + (1UL << GLIBTOP_MAP_ENTRY_RSS) +
Packit Service 407539
(1UL << GLIBTOP_MAP_ENTRY_PSS) + (1UL << GLIBTOP_MAP_ENTRY_SWAP) +
Packit Service 407539
(1UL << GLIBTOP_MAP_ENTRY_SHARED_DIRTY) + (1UL << GLIBTOP_MAP_ENTRY_SHARED_CLEAN) +
Packit Service 407539
(1UL << GLIBTOP_MAP_ENTRY_PRIVATE_DIRTY) + (1UL << GLIBTOP_MAP_ENTRY_PRIVATE_CLEAN);
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
/* Provides detailed information about a process. */
Packit Service 407539
Packit Service 407539
Packit Service 407539
static const char*
Packit Service 407539
is_smap_value(const char* s)
Packit Service 407539
{
Packit Service 407539
	for ( ; *s; ++s) {
Packit Service 407539
Packit Service 407539
		if (isspace(*s))
Packit Service 407539
			return NULL;
Packit Service 407539
Packit Service 407539
		if (*s == ':')
Packit Service 407539
			return s;
Packit Service 407539
	}
Packit Service 407539
Packit Service 407539
	return NULL;
Packit Service 407539
}
Packit Service 407539
Packit Service 407539
Packit Service 407539
/*
Packit Service 407539
  Returns whether line is a 'value' line
Packit Service 407539
  and add if we know its meaning
Packit Service 407539
*/
Packit Service 407539
static gboolean
Packit Service 407539
parse_smaps(glibtop_map_entry *entry, const char* line)
Packit Service 407539
{
Packit Service 407539
Packit Service 407539
	const struct smap_value* smap;
Packit Service 407539
	size_t len;
Packit Service 407539
	const char* colon;
Packit Service 407539
Packit Service 407539
	if ((colon = is_smap_value(line)) == NULL)
Packit Service 407539
		return FALSE;
Packit Service 407539
Packit Service 407539
	len = colon - line;
Packit Service 407539
	smap = _glibtop_find_smap(line, len);
Packit Service 407539
Packit Service 407539
//		g_debug("smap %s -> %p", line, smap);
Packit Service 407539
Packit Service 407539
	if (smap) {
Packit Service 407539
		char *offset;
Packit Service 407539
		guint64 *value;
Packit Service 407539
Packit Service 407539
		offset = (void*) entry;
Packit Service 407539
		offset += smap->offset;
Packit Service 407539
		value = (void*) offset;
Packit Service 407539
Packit Service 407539
		*value = get_scaled(line + len, NULL);
Packit Service 407539
	}
Packit Service 407539
Packit Service 407539
	return TRUE;
Packit Service 407539
}
Packit Service 407539
Packit Service 407539
Packit Service 407539
/*
Packit Service 407539
  sscanf is too slow
Packit Service 407539
  and system-monitor calls procmap for each pid every second
Packit Service 407539
Packit Service 407539
  manual parsing is faster
Packit Service 407539
Packit Service 407539
  error checking is weaker
Packit Service 407539
*/
Packit Service 407539
Packit Service 407539
static gboolean
Packit Service 407539
parse_line(char* line,
Packit Service 407539
	   guint64* start, guint64* end, char flags[4], guint64* offset,
Packit Service 407539
	   gushort* dev_major, gushort* dev_minor, guint64* inode,
Packit Service 407539
	   char** filename)
Packit Service 407539
{
Packit Service 407539
	/* %16llx-%16llx %4c %16llx %02hx:%02hx %llu%*[ ]%n */
Packit Service 407539
Packit Service 407539
	char *p, *next;
Packit Service 407539
Packit Service 407539
	p = line;
Packit Service 407539
Packit Service 407539
	*start = strtoull(p, &p, 16);
Packit Service 407539
Packit Service 407539
	if (G_UNLIKELY(*p != '-'))
Packit Service 407539
		return FALSE;
Packit Service 407539
	p++;
Packit Service 407539
Packit Service 407539
	*end = strtoull(p, &p, 16);
Packit Service 407539
Packit Service 407539
	p = next_token(p);
Packit Service 407539
Packit Service 407539
	memcpy(flags, p, 4);
Packit Service 407539
	p += 4;
Packit Service 407539
Packit Service 407539
	*offset = strtoull(p, &p, 16);
Packit Service 407539
Packit Service 407539
	*dev_major = strtoul(p, &p, 16);
Packit Service 407539
Packit Service 407539
	if (G_UNLIKELY(*p != ':'))
Packit Service 407539
		return FALSE;
Packit Service 407539
	p++;
Packit Service 407539
Packit Service 407539
	*dev_minor = strtoul(p, &p, 16);
Packit Service 407539
Packit Service 407539
	*inode = strtoull(p, &p, 10);
Packit Service 407539
Packit Service 407539
	p = next_token(p);
Packit Service 407539
Packit Service 407539
	*filename = p;
Packit Service 407539
Packit Service 407539
	for ( ; *p; p++) {
Packit Service 407539
		if (*p == '\n') {
Packit Service 407539
			*p = '\0';
Packit Service 407539
			break;
Packit Service 407539
		}
Packit Service 407539
	}
Packit Service 407539
Packit Service 407539
	return TRUE;
Packit Service 407539
}
Packit Service 407539
Packit Service 407539
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
	size_t added = 0, entry_list_capacity = 100;
Packit Service 407539
	GArray *entry_list = g_array_sized_new(FALSE, FALSE,
Packit Service 407539
					       sizeof(glibtop_map_entry),
Packit Service 407539
					       entry_list_capacity);
Packit Service 407539
	FILE *maps;
Packit Service 407539
	const char *filename;
Packit Service 407539
	gboolean has_smaps;
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
	has_smaps = server->os_version_code >= LINUX_VERSION_CODE(2, 6, 14);
Packit Service 407539
Packit Service 407539
	if (has_smaps)
Packit Service 407539
		filename = SMAPS_FILE;
Packit Service 407539
	else
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
		/* 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 (!parse_line(line,
Packit Service 407539
				&start, &end, flags, &offset,
Packit Service 407539
				&dev_major, &dev_minor, &inode, &filename))
Packit Service 407539
			continue;
Packit Service 407539
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
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
Packit Service 407539
		if (G_UNLIKELY(added >= entry_list_capacity)) {
Packit Service 407539
			entry_list_capacity *= 2;
Packit Service 407539
			g_array_set_size(entry_list, entry_list_capacity);
Packit Service 407539
		}
Packit Service 407539
Packit Service 407539
		entry = &g_array_index(entry_list, glibtop_map_entry, added++);
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
		if (has_smaps) {
Packit Service 407539
			ssize_t ret;
Packit Service 407539
			entry->flags |= _glibtop_sysdeps_map_entry_smaps;
Packit Service 407539
Packit Service 407539
			while ((ret = getline(&line, &line_size, maps)) != -1) {
Packit Service 407539
				if (!parse_smaps(entry, line))
Packit Service 407539
					goto new_entry_line;
Packit Service 407539
			}
Packit Service 407539
Packit Service 407539
			if (ret == -1)
Packit Service 407539
				goto eof;
Packit Service 407539
		}
Packit Service 407539
	}
Packit Service 407539
Packit Service 407539
eof:
Packit Service 407539
Packit Service 407539
	g_array_set_size(entry_list, added);
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 = added;
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
}