Blame sysdeps/solaris/procmap.c

Packit d37888
/* Copyright (C) 1998-99 Martin Baulig
Packit d37888
   This file is part of LibGTop 1.0.
Packit d37888
Packit d37888
   Contributed by Martin Baulig <martin@home-of-linux.org>, April 1998.
Packit d37888
Packit d37888
   LibGTop is free software; you can redistribute it and/or modify it
Packit d37888
   under the terms of the GNU General Public License as published by
Packit d37888
   the Free Software Foundation; either version 2 of the License,
Packit d37888
   or (at your option) any later version.
Packit d37888
Packit d37888
   LibGTop is distributed in the hope that it will be useful, but WITHOUT
Packit d37888
   ANY WARRANTY; without even the implied warranty of MERCHANTABILITY or
Packit d37888
   FITNESS FOR A PARTICULAR PURPOSE.  See the GNU General Public License
Packit d37888
   for more details.
Packit d37888
Packit d37888
   You should have received a copy of the GNU General Public License
Packit d37888
   along with LibGTop; see the file COPYING. If not, write to the
Packit d37888
   Free Software Foundation, Inc., 51 Franklin Street, Fifth Floor,
Packit d37888
   Boston, MA 02110-1301, USA.
Packit d37888
*/
Packit d37888
Packit d37888
Packit d37888
#include <config.h>
Packit d37888
#include <glibtop.h>
Packit d37888
#include <glibtop/error.h>
Packit d37888
#include <glibtop/procmap.h>
Packit d37888
Packit d37888
#include <errno.h>
Packit d37888
Packit d37888
#include "safeio.h"
Packit d37888
Packit d37888
Packit d37888
static const unsigned long _glibtop_sysdeps_proc_map =
Packit d37888
(1L << GLIBTOP_PROC_MAP_NUMBER) + (1L << GLIBTOP_PROC_MAP_TOTAL) +
Packit d37888
(1L << GLIBTOP_PROC_MAP_SIZE);
Packit d37888
static const unsigned long _glibtop_sysdeps_map_entry =
Packit d37888
(1L << GLIBTOP_MAP_ENTRY_START) + (1L << GLIBTOP_MAP_ENTRY_END) +
Packit d37888
(1L << GLIBTOP_MAP_ENTRY_OFFSET) + (1L << GLIBTOP_MAP_ENTRY_PERM);
Packit d37888
static const unsigned long _glibtop_sysdeps_map_device =
Packit d37888
(1L << GLIBTOP_MAP_ENTRY_DEVICE) + (1L << GLIBTOP_MAP_ENTRY_INODE);
Packit d37888
Packit d37888
Packit d37888
/* Init function. */
Packit d37888
Packit d37888
void
Packit d37888
_glibtop_init_proc_map_s (glibtop *server)
Packit d37888
{
Packit d37888
	server->sysdeps.proc_map = _glibtop_sysdeps_proc_map;
Packit d37888
}
Packit d37888
Packit d37888
/* Provides detailed information about a process. */
Packit d37888
Packit d37888
glibtop_map_entry *
Packit d37888
glibtop_get_proc_map_s (glibtop *server, glibtop_proc_map *buf,	pid_t pid)
Packit d37888
{
Packit d37888
   	int fd, i, nmaps, pr_err, heap;
Packit d37888
#if GLIBTOP_SOLARIS_RELEASE >= 50600
Packit d37888
	prxmap_t *maps;
Packit d37888
	struct ps_prochandle *Pr = NULL;
Packit d37888
#else
Packit d37888
	prmap_t *maps;
Packit d37888
#endif
Packit d37888
Packit d37888
	/* A few defines, to make it shorter down there */
Packit d37888
Packit d37888
#ifdef HAVE_PROCFS_H
Packit d37888
# define OFFSET  pr_offset
Packit d37888
#else
Packit d37888
# define OFFSET  pr_off
Packit d37888
#endif
Packit d37888
Packit d37888
	glibtop_map_entry *entry;
Packit d37888
	struct stat inode;
Packit d37888
	char buffer[BUFSIZ];
Packit d37888
Packit d37888
	memset (buf, 0, sizeof (glibtop_proc_map));
Packit d37888
Packit d37888
#ifdef HAVE_PROCFS_H
Packit d37888
	sprintf(buffer, "/proc/%d/xmap", (int)pid);
Packit d37888
#else
Packit d37888
	sprintf(buffer, "/proc/%d", (int)pid);
Packit d37888
#endif
Packit d37888
	if((fd = s_open(buffer, O_RDONLY)) < 0)
Packit d37888
	{
Packit d37888
	   	if (errno != EPERM && errno != EACCES)
Packit d37888
		   	glibtop_warn_io_r(server, "open (%s)", buffer);
Packit d37888
		return NULL;
Packit d37888
	}
Packit d37888
#ifdef HAVE_PROCFS_H
Packit d37888
	if(fstat(fd, &inode) < 0)
Packit d37888
	{
Packit d37888
	   	if(errno != EOVERFLOW)
Packit d37888
		   	glibtop_warn_io_r(server, "fstat (%s)", buffer);
Packit d37888
		/* else call daemon for 64-bit support */
Packit d37888
		s_close(fd);
Packit d37888
		return NULL;
Packit d37888
	}
Packit d37888
	maps = g_alloca(inode.st_size);
Packit d37888
	nmaps = inode.st_size / sizeof(prxmap_t);
Packit d37888
	if(s_pread(fd, maps, inode.st_size, 0) != inode.st_size)
Packit d37888
	{
Packit d37888
	   	glibtop_warn_io_r(server, "pread (%s)", buffer);
Packit d37888
		s_close(fd);
Packit d37888
		return NULL;
Packit d37888
	}
Packit d37888
#else
Packit d37888
	if(ioctl(fd, PIOCNMAP, &nmaps) < 0)
Packit d37888
	{
Packit d37888
	   	glibtop_warn_io_r(server, "ioctl(%s, PIOCNMAP)", buffer);
Packit d37888
		s_close(fd);
Packit d37888
		return NULL;
Packit d37888
	}
Packit d37888
	maps = g_alloca((nmaps + 1) * sizeof(prmap_t));
Packit d37888
	if(ioctl(fd, PIOCMAP, maps) < 0)
Packit d37888
	{
Packit d37888
	   	glibtop_warn_io_r(server, "ioctl(%s, PIOCMAP)", buffer);
Packit d37888
		s_close(fd);
Packit d37888
		return NULL;
Packit d37888
	}
Packit d37888
#endif
Packit d37888
	buf->number = nmaps;
Packit d37888
	buf->size = sizeof(glibtop_map_entry);
Packit d37888
	buf->total = nmaps * sizeof(glibtop_map_entry);
Packit d37888
	entry = g_malloc0(buf->total);
Packit d37888
Packit d37888
#if GLIBTOP_SOLARIS_RELEASE >= 50600
Packit d37888
Packit d37888
	if(server->machine->objname && server->machine->pgrab &&
Packit d37888
	   server->machine->pfree)
Packit d37888
	   Pr = (server->machine->pgrab)(pid, 1, &pr_err);
Packit d37888
#endif
Packit d37888
	for(heap = 0,i = 0; i < nmaps; ++i)
Packit d37888
	{
Packit d37888
	   	int len;
Packit d37888
Packit d37888
	   	entry[i].start = maps[i].pr_vaddr;
Packit d37888
		entry[i].end = maps[i].pr_vaddr + maps[i].pr_size;
Packit d37888
Packit d37888
#if GLIBTOP_SOLARIS_RELEASE >= 50600
Packit d37888
Packit d37888
		if(maps[i].pr_dev != PRNODEV)
Packit d37888
		{
Packit d37888
		   entry[i].device = maps[i].pr_dev;
Packit d37888
		   entry[i].inode = maps[i].pr_ino;
Packit d37888
		   entry[i].flags |= _glibtop_sysdeps_map_device;
Packit d37888
		}
Packit d37888
#endif
Packit d37888
		entry[i].offset = maps[i].OFFSET;
Packit d37888
		if(maps[i].pr_mflags & MA_READ)
Packit d37888
		   	entry[i].perm |= GLIBTOP_MAP_PERM_READ;
Packit d37888
		if(maps[i].pr_mflags & MA_WRITE){
Packit d37888
		   	entry[i].perm |= GLIBTOP_MAP_PERM_WRITE;
Packit d37888
		   	entry[i].size = maps[i].pr_size;
Packit d37888
		}
Packit d37888
		if(maps[i].pr_mflags & MA_EXEC)
Packit d37888
		   	entry[i].perm |= GLIBTOP_MAP_PERM_EXECUTE;
Packit d37888
		if(maps[i].pr_mflags & MA_SHARED)
Packit d37888
		   	entry[i].perm |= GLIBTOP_MAP_PERM_SHARED;
Packit d37888
		else
Packit d37888
		   	entry[i].perm |= GLIBTOP_MAP_PERM_PRIVATE;
Packit d37888
		entry[i].flags = _glibtop_sysdeps_map_entry;
Packit d37888
Packit d37888
#if GLIBTOP_SOLARIS_RELEASE >= 50600
Packit d37888
Packit d37888
		if(maps[i].pr_mflags & MA_ANON)
Packit d37888
		{
Packit d37888
		   if(!heap)
Packit d37888
		   {
Packit d37888
		      ++heap;
Packit d37888
		      strcpy(entry[i].filename, "[ heap ]");
Packit d37888
		   }
Packit d37888
		   else
Packit d37888
		      if(i == nmaps - 1)
Packit d37888
			 strcpy(entry[i].filename, "[ stack ]");
Packit d37888
		      else
Packit d37888
			 strcpy(entry[i].filename, "[ anon ]");
Packit d37888
		   entry[i].flags |= (1L << GLIBTOP_MAP_ENTRY_FILENAME);
Packit d37888
		}
Packit d37888
		else
Packit d37888
		   if(Pr)
Packit d37888
		   {
Packit d37888
		      server->machine->objname(Pr, maps[i].pr_vaddr, buffer,
Packit d37888
					      BUFSIZ);
Packit d37888
		      if((len = resolvepath(buffer, entry[i].filename,
Packit d37888
					    GLIBTOP_MAP_FILENAME_LEN)) > 0)
Packit d37888
		      {
Packit d37888
			 entry[i].filename[len] = 0;
Packit d37888
			 entry[i].flags |= (1L << GLIBTOP_MAP_ENTRY_FILENAME);
Packit d37888
		      }
Packit d37888
		   }
Packit d37888
#endif
Packit d37888
	}
Packit d37888
Packit d37888
#if GLIBTOP_SOLARIS_RELEASE >= 50600
Packit d37888
Packit d37888
	if(Pr)
Packit d37888
	   	server->machine->pfree(Pr);
Packit d37888
#endif
Packit d37888
	buf->flags = _glibtop_sysdeps_proc_map;
Packit d37888
	s_close(fd);
Packit d37888
	return entry;
Packit d37888
}