Blame sysdeps/linux/procopenfiles.c

Packit d37888
/* Copyright (C) 1998-99 Martin Baulig
Packit d37888
   Copyright (C) 2004 Nicolás Lichtmaier
Packit d37888
   This file is part of LibGTop 1.0.
Packit d37888
Packit d37888
   Modified by Nicolás Lichtmaier to give a process open files.
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
#include <config.h>
Packit d37888
#include <glibtop.h>
Packit d37888
#include <glibtop/error.h>
Packit d37888
#include <glibtop/procopenfiles.h>
Packit d37888
#include <sys/types.h>
Packit d37888
#include <dirent.h>
Packit d37888
#include <string.h>
Packit d37888
#include <stdio.h>
Packit d37888
#include <arpa/inet.h>
Packit d37888
Packit d37888
#include "glibtop_private.h"
Packit d37888
Packit d37888
static const unsigned long _glibtop_sysdeps_proc_open_files =
Packit d37888
(1L << GLIBTOP_PROC_OPEN_FILES_NUMBER)|
Packit d37888
(1L << GLIBTOP_PROC_OPEN_FILES_TOTAL)|
Packit d37888
(1L << GLIBTOP_PROC_OPEN_FILES_SIZE);
Packit d37888
Packit d37888
/* Init function. */
Packit d37888
Packit d37888
void
Packit d37888
_glibtop_init_proc_open_files_s (glibtop *server)
Packit d37888
{
Packit d37888
	server->sysdeps.proc_open_files = _glibtop_sysdeps_proc_open_files;
Packit d37888
}
Packit d37888
Packit d37888
Packit d37888
Packit d37888
typedef void (*LineParser)(GHashTable *dict, const char *line);
Packit d37888
Packit d37888
Packit d37888
static void
Packit d37888
parse_file(const char *filename, LineParser parser, GHashTable *dict)
Packit d37888
{
Packit d37888
	FILE *f;
Packit d37888
	char *line = NULL;
Packit d37888
	size_t size = 0;
Packit d37888
Packit d37888
	f = fopen(filename, "r");
Packit d37888
Packit d37888
	if(!f) {
Packit d37888
		g_warning("Cannot open '%s'", filename);
Packit d37888
		return;
Packit d37888
	}
Packit d37888
Packit d37888
Packit d37888
	/* skip the first line */
Packit d37888
	if (getline(&line, &size, f) == -1)
Packit d37888
		goto eof;
Packit d37888
Packit d37888
	while (getline(&line, &size, f) != -1)
Packit d37888
		parser(dict, line);
Packit d37888
Packit d37888
 eof:
Packit d37888
	free(line);
Packit d37888
	fclose(f);
Packit d37888
}
Packit d37888
Packit d37888
Packit d37888
static GHashTable*
Packit d37888
get_all(const char *filename, LineParser parser)
Packit d37888
{
Packit d37888
	GHashTable *dict;
Packit d37888
Packit d37888
	dict = g_hash_table_new_full(g_direct_hash, g_direct_equal,
Packit d37888
				     NULL, g_free);
Packit d37888
Packit d37888
	parse_file(filename, parser, dict);
Packit d37888
Packit d37888
	return dict;
Packit d37888
}
Packit d37888
Packit d37888
Packit d37888
Packit d37888
Packit d37888
Packit d37888
struct Inet6SocketEntry
Packit d37888
{
Packit d37888
	char host[GLIBTOP_OPEN_DEST_HOST_LEN + 1];
Packit d37888
	int port;
Packit d37888
};
Packit d37888
Packit d37888
Packit d37888
static void
Packit d37888
inet6_socket_parser(GHashTable *dict, const char* line)
Packit d37888
{
Packit d37888
	struct Inet6SocketEntry *se;
Packit d37888
	int sock;
Packit d37888
	struct in6_addr addr;
Packit d37888
Packit d37888
	se = g_malloc0(sizeof *se);
Packit d37888
Packit d37888
	if(sscanf(line, "%*d: %*s %8x%8x%8x%8x:%4x %*x %*x:%*x %*x:%*x %*d %*d %*d %d",
Packit d37888
		  &addr.s6_addr32[0], &addr.s6_addr32[1], &addr.s6_addr32[2],
Packit d37888
		  &addr.s6_addr32[3], &se->port, &sock) != 6)
Packit d37888
		goto error;
Packit d37888
Packit d37888
	if(!inet_ntop(AF_INET6, &addr, se->host, sizeof se->host))
Packit d37888
		goto error;
Packit d37888
Packit d37888
	g_hash_table_insert(dict, GINT_TO_POINTER(sock), se);
Packit d37888
	return;
Packit d37888
Packit d37888
 error:
Packit d37888
	g_free(se);
Packit d37888
}
Packit d37888
Packit d37888
Packit d37888
static inline GHashTable *
Packit d37888
get_all_inet6_sockets()
Packit d37888
{
Packit d37888
	return get_all("/proc/net/tcp6", inet6_socket_parser);
Packit d37888
}
Packit d37888
Packit d37888
Packit d37888
Packit d37888
Packit d37888
Packit d37888
struct InetSocketEntry
Packit d37888
{
Packit d37888
	char host[GLIBTOP_OPEN_DEST_HOST_LEN + 1];
Packit d37888
	int port;
Packit d37888
};
Packit d37888
Packit d37888
Packit d37888
static void
Packit d37888
inet_socket_parser(GHashTable *dict, const char* line)
Packit d37888
{
Packit d37888
	struct InetSocketEntry *se;
Packit d37888
	int sock;
Packit d37888
	unsigned addr;
Packit d37888
Packit d37888
	se = g_malloc0(sizeof *se);
Packit d37888
Packit d37888
	if(sscanf(line, "%*d: %*x:%*x %8x:%4x %*x %*x:%*x %*x:%*x %*d %*d %*d %d",
Packit d37888
		  &addr, &se->port, &sock) != 3)
Packit d37888
		goto error;
Packit d37888
Packit d37888
	if(!inet_ntop(AF_INET, &addr, se->host, sizeof se->host))
Packit d37888
		goto error;
Packit d37888
Packit d37888
	g_hash_table_insert(dict, GINT_TO_POINTER(sock), se);
Packit d37888
	return;
Packit d37888
Packit d37888
 error:
Packit d37888
	g_free(se);
Packit d37888
}
Packit d37888
Packit d37888
Packit d37888
static inline GHashTable *
Packit d37888
get_all_inet_sockets()
Packit d37888
{
Packit d37888
	return get_all("/proc/net/tcp", inet_socket_parser);
Packit d37888
}
Packit d37888
Packit d37888
Packit d37888
Packit d37888
Packit d37888
Packit d37888
struct LocalSocketEntry
Packit d37888
{
Packit d37888
	char name[GLIBTOP_OPEN_DEST_HOST_LEN + 1];
Packit d37888
};
Packit d37888
Packit d37888
Packit d37888
static void
Packit d37888
local_socket_parser(GHashTable *dict, const char *line)
Packit d37888
{
Packit d37888
	int sock;
Packit d37888
	struct LocalSocketEntry *use;
Packit d37888
	char *p;
Packit d37888
Packit d37888
	use = g_malloc0(sizeof *use);
Packit d37888
Packit d37888
	/* dfaf1640: 00000003 00000000 00000000 0001 03  6457 /dev/log */
Packit d37888
	p = skip_multiple_token(line, 6);
Packit d37888
Packit d37888
	sock = strtoul(p, &p, 10);
Packit d37888
	g_strlcpy(use->name, p, sizeof use->name);
Packit d37888
	g_strstrip(use->name);
Packit d37888
	g_hash_table_insert(dict, GINT_TO_POINTER(sock), use);
Packit d37888
}
Packit d37888
Packit d37888
Packit d37888
static inline GHashTable *
Packit d37888
get_all_local_sockets()
Packit d37888
{
Packit d37888
	return get_all("/proc/net/unix", local_socket_parser);
Packit d37888
}
Packit d37888
Packit d37888
Packit d37888
Packit d37888
/* Provides detailed information about a process' open files */
Packit d37888
Packit d37888
glibtop_open_files_entry *
Packit d37888
glibtop_get_proc_open_files_s (glibtop *server, glibtop_proc_open_files *buf,	pid_t pid)
Packit d37888
{
Packit d37888
	char fn [BUFSIZ];
Packit d37888
	GArray *entries;
Packit d37888
	GHashTable *inet6_sockets = NULL, *inet_sockets = NULL, *local_sockets = NULL;
Packit d37888
	struct dirent *direntry;
Packit d37888
	DIR *dir;
Packit d37888
Packit d37888
	memset (buf, 0, sizeof (glibtop_proc_open_files));
Packit d37888
Packit d37888
	sprintf (fn, "/proc/%d/fd", pid);
Packit d37888
Packit d37888
	dir = opendir (fn);
Packit d37888
	if (!dir) return NULL;
Packit d37888
Packit d37888
	entries = g_array_new(FALSE, FALSE, sizeof(glibtop_open_files_entry));
Packit d37888
Packit d37888
	while((direntry = readdir(dir))) {
Packit d37888
		char tgt [BUFSIZ];
Packit d37888
		glibtop_open_files_entry entry = {0};
Packit d37888
Packit d37888
		if(direntry->d_name[0] == '.')
Packit d37888
			continue;
Packit d37888
Packit d37888
		g_snprintf(fn, sizeof fn, "/proc/%d/fd/%s",
Packit d37888
			   pid, direntry->d_name);
Packit d37888
Packit d37888
		if (!safe_readlink(fn, tgt, sizeof tgt))
Packit d37888
			continue;
Packit d37888
Packit d37888
		entry.fd = atoi(direntry->d_name);
Packit d37888
Packit d37888
		if(g_str_has_prefix(tgt, "socket:["))
Packit d37888
		{
Packit d37888
			int sockfd;
Packit d37888
			struct Inet6SocketEntry *i6se;
Packit d37888
			struct InetSocketEntry *ise;
Packit d37888
			struct LocalSocketEntry *lse;
Packit d37888
Packit d37888
			if(!inet6_sockets) inet6_sockets = get_all_inet6_sockets();
Packit d37888
			if(!inet_sockets) inet_sockets = get_all_inet_sockets();
Packit d37888
			if(!local_sockets) local_sockets = get_all_local_sockets();
Packit d37888
Packit d37888
			sockfd = atoi(tgt + 8);
Packit d37888
Packit d37888
			i6se = g_hash_table_lookup(inet6_sockets,
Packit d37888
						 GINT_TO_POINTER(sockfd));
Packit d37888
Packit d37888
			if(i6se) {
Packit d37888
				entry.type = GLIBTOP_FILE_TYPE_INET6SOCKET;
Packit d37888
				entry.info.sock.dest_port = i6se->port;
Packit d37888
				g_strlcpy(entry.info.sock.dest_host, i6se->host,
Packit d37888
					  sizeof entry.info.sock.dest_host);
Packit d37888
				goto found;
Packit d37888
			}
Packit d37888
Packit d37888
			ise = g_hash_table_lookup(inet_sockets,
Packit d37888
						 GINT_TO_POINTER(sockfd));
Packit d37888
Packit d37888
			if(ise) {
Packit d37888
				entry.type = GLIBTOP_FILE_TYPE_INETSOCKET;
Packit d37888
				entry.info.sock.dest_port = ise->port;
Packit d37888
				g_strlcpy(entry.info.sock.dest_host, ise->host,
Packit d37888
					  sizeof entry.info.sock.dest_host);
Packit d37888
				goto found;
Packit d37888
			}
Packit d37888
Packit d37888
			lse = g_hash_table_lookup(local_sockets,
Packit d37888
						  GINT_TO_POINTER(sockfd));
Packit d37888
Packit d37888
			if(lse) {
Packit d37888
				entry.type = GLIBTOP_FILE_TYPE_LOCALSOCKET;
Packit d37888
				g_strlcpy(entry.info.localsock.name, lse->name,
Packit d37888
					  sizeof entry.info.localsock.name);
Packit d37888
				goto found;
Packit d37888
			}
Packit d37888
Packit d37888
		found:
Packit d37888
			(void)0; /* kills warning */
Packit d37888
		}
Packit d37888
		else if(g_str_has_prefix(tgt, "pipe:["))
Packit d37888
		{
Packit d37888
			entry.type = GLIBTOP_FILE_TYPE_PIPE;
Packit d37888
		}
Packit d37888
		else
Packit d37888
		{
Packit d37888
			entry.type = GLIBTOP_FILE_TYPE_FILE;
Packit d37888
			g_strlcpy(entry.info.file.name, tgt, sizeof entry.info.file.name);
Packit d37888
		}
Packit d37888
Packit d37888
		g_array_append_val(entries, entry);
Packit d37888
	}
Packit d37888
Packit d37888
	closedir (dir);
Packit d37888
Packit d37888
	if(inet_sockets) g_hash_table_destroy(inet_sockets);
Packit d37888
	if(inet6_sockets) g_hash_table_destroy(inet6_sockets);
Packit d37888
	if(local_sockets) g_hash_table_destroy(local_sockets);
Packit d37888
Packit d37888
	buf->flags = _glibtop_sysdeps_proc_open_files;
Packit d37888
	buf->number = entries->len;
Packit d37888
	buf->size = sizeof(glibtop_open_files_entry);
Packit d37888
	buf->total = buf->number * buf->size;
Packit d37888
Packit d37888
	return (glibtop_open_files_entry*)g_array_free(entries, FALSE);
Packit d37888
}