nicely separate conn2inode stuff into its own file
This commit is contained in:
2
Makefile
2
Makefile
@@ -13,7 +13,7 @@ all: nethogs decpcap_test
|
|||||||
|
|
||||||
CFLAGS=-g -Wall
|
CFLAGS=-g -Wall
|
||||||
#CFLAGS=-O2
|
#CFLAGS=-O2
|
||||||
OBJS=packet.o connection.o process.o refresh.o decpcap.o cui.o inode2prog.o
|
OBJS=packet.o connection.o process.o refresh.o decpcap.o cui.o inode2prog.o conninode.o
|
||||||
.PHONY: tgz
|
.PHONY: tgz
|
||||||
|
|
||||||
tgz: clean
|
tgz: clean
|
||||||
|
|||||||
169
conninode.cpp
Normal file
169
conninode.cpp
Normal file
@@ -0,0 +1,169 @@
|
|||||||
|
#include <netinet/in.h>
|
||||||
|
#include <map>
|
||||||
|
|
||||||
|
#
|
||||||
|
#include "nethogs.h"
|
||||||
|
#include "conninode.h"
|
||||||
|
|
||||||
|
extern local_addr * local_addrs;
|
||||||
|
|
||||||
|
/*
|
||||||
|
* connection-inode table. takes information from /proc/net/tcp.
|
||||||
|
* key contains source ip, source port, destination ip, destination
|
||||||
|
* port in format: '1.2.3.4:5-1.2.3.4:5'
|
||||||
|
*/
|
||||||
|
std::map <std::string, unsigned long> conninode;
|
||||||
|
|
||||||
|
|
||||||
|
/*
|
||||||
|
* parses a /proc/net/tcp-line of the form:
|
||||||
|
* sl local_address rem_address st tx_queue rx_queue tr tm->when retrnsmt uid timeout inode
|
||||||
|
* 10: 020310AC:1770 9DD8A9C3:A525 01 00000000:00000000 00:00000000 00000000 0 0 2119 1 c0f4f0c0 206 40 10 3 -1
|
||||||
|
* 11: 020310AC:0404 936B2ECF:0747 01 00000000:00000000 00:00000000 00000000 1000 0 2109 1 c0f4fc00 368 40 20 2 -1
|
||||||
|
*
|
||||||
|
* and of the form:
|
||||||
|
* 2: 0000000000000000FFFF0000020310AC:0016 0000000000000000FFFF00009DD8A9C3:A526 01 00000000:00000000 02:000A7214 00000000 0 0 2525 2 c732eca0 201 40 1 2 -1
|
||||||
|
*
|
||||||
|
*/
|
||||||
|
void addtoconninode (char * buffer)
|
||||||
|
{
|
||||||
|
short int sa_family;
|
||||||
|
struct in6_addr result_addr_local;
|
||||||
|
struct in6_addr result_addr_remote;
|
||||||
|
|
||||||
|
char rem_addr[128], local_addr[128];
|
||||||
|
int local_port, rem_port;
|
||||||
|
struct in6_addr in6_local;
|
||||||
|
struct in6_addr in6_remote;
|
||||||
|
|
||||||
|
// this leaked memory
|
||||||
|
//unsigned long * inode = (unsigned long *) malloc (sizeof(unsigned long));
|
||||||
|
unsigned long inode;
|
||||||
|
|
||||||
|
int matches = sscanf(buffer, "%*d: %64[0-9A-Fa-f]:%X %64[0-9A-Fa-f]:%X %*X %*X:%*X %*X:%*X %*X %*d %*d %ld %*512s\n",
|
||||||
|
local_addr, &local_port, rem_addr, &rem_port, &inode);
|
||||||
|
|
||||||
|
if (matches != 5) {
|
||||||
|
fprintf(stderr,"Unexpected buffer: '%s'\n",buffer);
|
||||||
|
exit(0);
|
||||||
|
}
|
||||||
|
|
||||||
|
if (inode == 0) {
|
||||||
|
/* connection is in TIME_WAIT state. We rely on
|
||||||
|
* the old data still in the table. */
|
||||||
|
return;
|
||||||
|
}
|
||||||
|
|
||||||
|
if (strlen(local_addr) > 8)
|
||||||
|
{
|
||||||
|
/* this is an IPv6-style row */
|
||||||
|
|
||||||
|
/* Demangle what the kernel gives us */
|
||||||
|
sscanf(local_addr, "%08X%08X%08X%08X",
|
||||||
|
&in6_local.s6_addr32[0], &in6_local.s6_addr32[1],
|
||||||
|
&in6_local.s6_addr32[2], &in6_local.s6_addr32[3]);
|
||||||
|
sscanf(rem_addr, "%08X%08X%08X%08X",
|
||||||
|
&in6_remote.s6_addr32[0], &in6_remote.s6_addr32[1],
|
||||||
|
&in6_remote.s6_addr32[2], &in6_remote.s6_addr32[3]);
|
||||||
|
|
||||||
|
if ((in6_local.s6_addr32[0] == 0x0) && (in6_local.s6_addr32[1] == 0x0)
|
||||||
|
&& (in6_local.s6_addr32[2] == 0xFFFF0000))
|
||||||
|
{
|
||||||
|
/* IPv4-compatible address */
|
||||||
|
result_addr_local = *((struct in6_addr*) &(in6_local.s6_addr32[3]));
|
||||||
|
result_addr_remote = *((struct in6_addr*) &(in6_remote.s6_addr32[3]));
|
||||||
|
sa_family = AF_INET;
|
||||||
|
} else {
|
||||||
|
/* real IPv6 address */
|
||||||
|
//inet_ntop(AF_INET6, &in6_local, addr6, sizeof(addr6));
|
||||||
|
//INET6_getsock(addr6, (struct sockaddr *) &localaddr);
|
||||||
|
//inet_ntop(AF_INET6, &in6_remote, addr6, sizeof(addr6));
|
||||||
|
//INET6_getsock(addr6, (struct sockaddr *) &remaddr);
|
||||||
|
//localaddr.sin6_family = AF_INET6;
|
||||||
|
//remaddr.sin6_family = AF_INET6;
|
||||||
|
result_addr_local = in6_local;
|
||||||
|
result_addr_remote = in6_remote;
|
||||||
|
sa_family = AF_INET6;
|
||||||
|
}
|
||||||
|
}
|
||||||
|
else
|
||||||
|
{
|
||||||
|
/* this is an IPv4-style row */
|
||||||
|
sscanf(local_addr, "%X", (unsigned int *) &result_addr_local);
|
||||||
|
sscanf(rem_addr, "%X", (unsigned int *) &result_addr_remote);
|
||||||
|
sa_family = AF_INET;
|
||||||
|
}
|
||||||
|
|
||||||
|
char * hashkey = (char *) malloc (HASHKEYSIZE * sizeof(char));
|
||||||
|
char * local_string = (char*) malloc (50);
|
||||||
|
char * remote_string = (char*) malloc (50);
|
||||||
|
inet_ntop(sa_family, &result_addr_local, local_string, 49);
|
||||||
|
inet_ntop(sa_family, &result_addr_remote, remote_string, 49);
|
||||||
|
|
||||||
|
snprintf(hashkey, HASHKEYSIZE * sizeof(char), "%s:%d-%s:%d", local_string, local_port, remote_string, rem_port);
|
||||||
|
free (local_string);
|
||||||
|
|
||||||
|
//if (DEBUG)
|
||||||
|
// fprintf (stderr, "Hashkey: %s\n", hashkey);
|
||||||
|
|
||||||
|
//std::cout << "Adding to conninode\n" << std::endl;
|
||||||
|
|
||||||
|
conninode[hashkey] = inode;
|
||||||
|
|
||||||
|
/* workaround: sometimes, when a connection is actually from 172.16.3.1 to
|
||||||
|
* 172.16.3.3, packages arrive from 195.169.216.157 to 172.16.3.3, where
|
||||||
|
* 172.16.3.1 and 195.169.216.157 are the local addresses of different
|
||||||
|
* interfaces */
|
||||||
|
struct local_addr * current_local_addr = local_addrs;
|
||||||
|
while (current_local_addr != NULL) {
|
||||||
|
/* TODO maybe only add the ones with the same sa_family */
|
||||||
|
snprintf(hashkey, HASHKEYSIZE * sizeof(char), "%s:%d-%s:%d", current_local_addr->string, local_port, remote_string, rem_port);
|
||||||
|
conninode[hashkey] = inode;
|
||||||
|
current_local_addr = current_local_addr->next;
|
||||||
|
}
|
||||||
|
free (hashkey);
|
||||||
|
free (remote_string);
|
||||||
|
}
|
||||||
|
|
||||||
|
/* opens /proc/net/tcp[6] and adds its contents line by line */
|
||||||
|
int addprocinfo (const char * filename) {
|
||||||
|
FILE * procinfo = fopen (filename, "r");
|
||||||
|
|
||||||
|
char buffer[8192];
|
||||||
|
|
||||||
|
if (procinfo == NULL)
|
||||||
|
return 0;
|
||||||
|
|
||||||
|
fgets(buffer, sizeof(buffer), procinfo);
|
||||||
|
|
||||||
|
do
|
||||||
|
{
|
||||||
|
if (fgets(buffer, sizeof(buffer), procinfo))
|
||||||
|
addtoconninode(buffer);
|
||||||
|
} while (!feof(procinfo));
|
||||||
|
|
||||||
|
fclose(procinfo);
|
||||||
|
|
||||||
|
return 1;
|
||||||
|
}
|
||||||
|
|
||||||
|
|
||||||
|
|
||||||
|
|
||||||
|
void refreshconninode ()
|
||||||
|
{
|
||||||
|
/* we don't forget old mappings, just overwrite */
|
||||||
|
//delete conninode;
|
||||||
|
//conninode = new HashTable (256);
|
||||||
|
|
||||||
|
if (! addprocinfo ("/proc/net/tcp"))
|
||||||
|
{
|
||||||
|
std::cout << "Error: couldn't open /proc/net/tcp\n";
|
||||||
|
exit(0);
|
||||||
|
}
|
||||||
|
addprocinfo ("/proc/net/tcp6");
|
||||||
|
|
||||||
|
//if (DEBUG)
|
||||||
|
// reviewUnknown();
|
||||||
|
|
||||||
|
}
|
||||||
2
conninode.h
Normal file
2
conninode.h
Normal file
@@ -0,0 +1,2 @@
|
|||||||
|
// handling the connection->inode mapping
|
||||||
|
void refreshconninode ();
|
||||||
163
process.cpp
163
process.cpp
@@ -15,20 +15,21 @@
|
|||||||
#include "nethogs.h"
|
#include "nethogs.h"
|
||||||
/* #include "inodeproc.cpp" */
|
/* #include "inodeproc.cpp" */
|
||||||
#include "inode2prog.h"
|
#include "inode2prog.h"
|
||||||
|
#include "conninode.h"
|
||||||
|
|
||||||
extern local_addr * local_addrs;
|
extern local_addr * local_addrs;
|
||||||
|
|
||||||
/* this file includes:
|
|
||||||
* - code to convert from connection to inode
|
|
||||||
* - calls to inodeproc to get the pid that belongs to that inode
|
|
||||||
*/
|
|
||||||
|
|
||||||
/*
|
/*
|
||||||
* connection-inode table. takes information from /proc/net/tcp.
|
* connection-inode table. takes information from /proc/net/tcp.
|
||||||
* key contains source ip, source port, destination ip, destination
|
* key contains source ip, source port, destination ip, destination
|
||||||
* port in format: '1.2.3.4:5-1.2.3.4:5'
|
* port in format: '1.2.3.4:5-1.2.3.4:5'
|
||||||
*/
|
*/
|
||||||
std::map <std::string, unsigned long> conninode;
|
extern std::map <std::string, unsigned long> conninode;
|
||||||
|
|
||||||
|
|
||||||
|
/* this file includes:
|
||||||
|
* - calls to inodeproc to get the pid that belongs to that inode
|
||||||
|
*/
|
||||||
|
|
||||||
/*
|
/*
|
||||||
* Initialise the global process-list with some special processes:
|
* Initialise the global process-list with some special processes:
|
||||||
@@ -76,138 +77,6 @@ int Process::getLastPacket()
|
|||||||
return lastpacket;
|
return lastpacket;
|
||||||
}
|
}
|
||||||
|
|
||||||
/*
|
|
||||||
* parses a /proc/net/tcp-line of the form:
|
|
||||||
* sl local_address rem_address st tx_queue rx_queue tr tm->when retrnsmt uid timeout inode
|
|
||||||
* 10: 020310AC:1770 9DD8A9C3:A525 01 00000000:00000000 00:00000000 00000000 0 0 2119 1 c0f4f0c0 206 40 10 3 -1
|
|
||||||
* 11: 020310AC:0404 936B2ECF:0747 01 00000000:00000000 00:00000000 00000000 1000 0 2109 1 c0f4fc00 368 40 20 2 -1
|
|
||||||
*
|
|
||||||
* and of the form:
|
|
||||||
* 2: 0000000000000000FFFF0000020310AC:0016 0000000000000000FFFF00009DD8A9C3:A526 01 00000000:00000000 02:000A7214 00000000 0 0 2525 2 c732eca0 201 40 1 2 -1
|
|
||||||
*
|
|
||||||
*/
|
|
||||||
void addtoconninode (char * buffer)
|
|
||||||
{
|
|
||||||
short int sa_family;
|
|
||||||
struct in6_addr result_addr_local;
|
|
||||||
struct in6_addr result_addr_remote;
|
|
||||||
|
|
||||||
char rem_addr[128], local_addr[128];
|
|
||||||
int local_port, rem_port;
|
|
||||||
struct in6_addr in6_local;
|
|
||||||
struct in6_addr in6_remote;
|
|
||||||
|
|
||||||
// this leaked memory
|
|
||||||
//unsigned long * inode = (unsigned long *) malloc (sizeof(unsigned long));
|
|
||||||
unsigned long inode;
|
|
||||||
|
|
||||||
int matches = sscanf(buffer, "%*d: %64[0-9A-Fa-f]:%X %64[0-9A-Fa-f]:%X %*X %*X:%*X %*X:%*X %*X %*d %*d %ld %*512s\n",
|
|
||||||
local_addr, &local_port, rem_addr, &rem_port, &inode);
|
|
||||||
|
|
||||||
if (matches != 5) {
|
|
||||||
fprintf(stderr,"Unexpected buffer: '%s'\n",buffer);
|
|
||||||
exit(0);
|
|
||||||
}
|
|
||||||
|
|
||||||
if (inode == 0) {
|
|
||||||
/* connection is in TIME_WAIT state. We rely on
|
|
||||||
* the old data still in the table. */
|
|
||||||
return;
|
|
||||||
}
|
|
||||||
|
|
||||||
if (strlen(local_addr) > 8)
|
|
||||||
{
|
|
||||||
/* this is an IPv6-style row */
|
|
||||||
|
|
||||||
/* Demangle what the kernel gives us */
|
|
||||||
sscanf(local_addr, "%08X%08X%08X%08X",
|
|
||||||
&in6_local.s6_addr32[0], &in6_local.s6_addr32[1],
|
|
||||||
&in6_local.s6_addr32[2], &in6_local.s6_addr32[3]);
|
|
||||||
sscanf(rem_addr, "%08X%08X%08X%08X",
|
|
||||||
&in6_remote.s6_addr32[0], &in6_remote.s6_addr32[1],
|
|
||||||
&in6_remote.s6_addr32[2], &in6_remote.s6_addr32[3]);
|
|
||||||
|
|
||||||
if ((in6_local.s6_addr32[0] == 0x0) && (in6_local.s6_addr32[1] == 0x0)
|
|
||||||
&& (in6_local.s6_addr32[2] == 0xFFFF0000))
|
|
||||||
{
|
|
||||||
/* IPv4-compatible address */
|
|
||||||
result_addr_local = *((struct in6_addr*) &(in6_local.s6_addr32[3]));
|
|
||||||
result_addr_remote = *((struct in6_addr*) &(in6_remote.s6_addr32[3]));
|
|
||||||
sa_family = AF_INET;
|
|
||||||
} else {
|
|
||||||
/* real IPv6 address */
|
|
||||||
//inet_ntop(AF_INET6, &in6_local, addr6, sizeof(addr6));
|
|
||||||
//INET6_getsock(addr6, (struct sockaddr *) &localaddr);
|
|
||||||
//inet_ntop(AF_INET6, &in6_remote, addr6, sizeof(addr6));
|
|
||||||
//INET6_getsock(addr6, (struct sockaddr *) &remaddr);
|
|
||||||
//localaddr.sin6_family = AF_INET6;
|
|
||||||
//remaddr.sin6_family = AF_INET6;
|
|
||||||
result_addr_local = in6_local;
|
|
||||||
result_addr_remote = in6_remote;
|
|
||||||
sa_family = AF_INET6;
|
|
||||||
}
|
|
||||||
}
|
|
||||||
else
|
|
||||||
{
|
|
||||||
/* this is an IPv4-style row */
|
|
||||||
sscanf(local_addr, "%X", (unsigned int *) &result_addr_local);
|
|
||||||
sscanf(rem_addr, "%X", (unsigned int *) &result_addr_remote);
|
|
||||||
sa_family = AF_INET;
|
|
||||||
}
|
|
||||||
|
|
||||||
char * hashkey = (char *) malloc (HASHKEYSIZE * sizeof(char));
|
|
||||||
char * local_string = (char*) malloc (50);
|
|
||||||
char * remote_string = (char*) malloc (50);
|
|
||||||
inet_ntop(sa_family, &result_addr_local, local_string, 49);
|
|
||||||
inet_ntop(sa_family, &result_addr_remote, remote_string, 49);
|
|
||||||
|
|
||||||
snprintf(hashkey, HASHKEYSIZE * sizeof(char), "%s:%d-%s:%d", local_string, local_port, remote_string, rem_port);
|
|
||||||
free (local_string);
|
|
||||||
|
|
||||||
//if (DEBUG)
|
|
||||||
// fprintf (stderr, "Hashkey: %s\n", hashkey);
|
|
||||||
|
|
||||||
//std::cout << "Adding to conninode\n" << std::endl;
|
|
||||||
|
|
||||||
conninode[hashkey] = inode;
|
|
||||||
|
|
||||||
/* workaround: sometimes, when a connection is actually from 172.16.3.1 to
|
|
||||||
* 172.16.3.3, packages arrive from 195.169.216.157 to 172.16.3.3, where
|
|
||||||
* 172.16.3.1 and 195.169.216.157 are the local addresses of different
|
|
||||||
* interfaces */
|
|
||||||
struct local_addr * current_local_addr = local_addrs;
|
|
||||||
while (current_local_addr != NULL) {
|
|
||||||
/* TODO maybe only add the ones with the same sa_family */
|
|
||||||
snprintf(hashkey, HASHKEYSIZE * sizeof(char), "%s:%d-%s:%d", current_local_addr->string, local_port, remote_string, rem_port);
|
|
||||||
conninode[hashkey] = inode;
|
|
||||||
current_local_addr = current_local_addr->next;
|
|
||||||
}
|
|
||||||
free (hashkey);
|
|
||||||
free (remote_string);
|
|
||||||
}
|
|
||||||
|
|
||||||
/* opens /proc/net/tcp[6] and adds its contents line by line */
|
|
||||||
int addprocinfo (const char * filename) {
|
|
||||||
FILE * procinfo = fopen (filename, "r");
|
|
||||||
|
|
||||||
char buffer[8192];
|
|
||||||
|
|
||||||
if (procinfo == NULL)
|
|
||||||
return 0;
|
|
||||||
|
|
||||||
fgets(buffer, sizeof(buffer), procinfo);
|
|
||||||
|
|
||||||
do
|
|
||||||
{
|
|
||||||
if (fgets(buffer, sizeof(buffer), procinfo))
|
|
||||||
addtoconninode(buffer);
|
|
||||||
} while (!feof(procinfo));
|
|
||||||
|
|
||||||
fclose(procinfo);
|
|
||||||
|
|
||||||
return 1;
|
|
||||||
}
|
|
||||||
|
|
||||||
Process * findProcess (struct prg_node * node)
|
Process * findProcess (struct prg_node * node)
|
||||||
{
|
{
|
||||||
ProcList * current = processes;
|
ProcList * current = processes;
|
||||||
@@ -282,24 +151,6 @@ void reviewUnknown ()
|
|||||||
}
|
}
|
||||||
}
|
}
|
||||||
|
|
||||||
void refreshconninode ()
|
|
||||||
{
|
|
||||||
/* we don't forget old mappings, just overwrite */
|
|
||||||
//delete conninode;
|
|
||||||
//conninode = new HashTable (256);
|
|
||||||
|
|
||||||
if (! addprocinfo ("/proc/net/tcp"))
|
|
||||||
{
|
|
||||||
std::cout << "Error: couldn't open /proc/net/tcp\n";
|
|
||||||
exit(0);
|
|
||||||
}
|
|
||||||
addprocinfo ("/proc/net/tcp6");
|
|
||||||
|
|
||||||
if (DEBUG)
|
|
||||||
reviewUnknown();
|
|
||||||
|
|
||||||
}
|
|
||||||
|
|
||||||
int ProcList::size ()
|
int ProcList::size ()
|
||||||
{
|
{
|
||||||
int i=1;
|
int i=1;
|
||||||
|
|||||||
Reference in New Issue
Block a user