|
Packit |
d37888 |
/* Copyright (C) 2007 Henry Zhang
|
|
Packit |
d37888 |
This file is part of LibGTop 2.20.
|
|
Packit |
d37888 |
|
|
Packit |
d37888 |
Contributed by Henry Zhang <hua.zhang@sun.com>, October 2007.
|
|
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/netload.h>
|
|
Packit |
d37888 |
|
|
Packit |
d37888 |
#include "glibtop_private.h"
|
|
Packit |
d37888 |
|
|
Packit |
d37888 |
#include <errno.h>
|
|
Packit |
d37888 |
#include <string.h>
|
|
Packit |
d37888 |
#include <kstat.h>
|
|
Packit |
d37888 |
|
|
Packit |
d37888 |
#include <sys/types.h>
|
|
Packit |
d37888 |
#include <sys/socket.h>
|
|
Packit |
d37888 |
#include <sys/ioctl.h>
|
|
Packit |
d37888 |
#include <sys/sockio.h>
|
|
Packit |
d37888 |
|
|
Packit |
d37888 |
#include <net/if.h>
|
|
Packit |
d37888 |
|
|
Packit |
d37888 |
|
|
Packit |
d37888 |
static const unsigned long _glibtop_sysdeps_netload =
|
|
Packit |
d37888 |
(1L << GLIBTOP_NETLOAD_ERRORS_IN) +
|
|
Packit |
d37888 |
(1L << GLIBTOP_NETLOAD_ERRORS_OUT) +
|
|
Packit |
d37888 |
(1L << GLIBTOP_NETLOAD_COLLISIONS);
|
|
Packit |
d37888 |
|
|
Packit |
d37888 |
static const unsigned long _glibtop_sysdeps_netload_data =
|
|
Packit |
d37888 |
(1L << GLIBTOP_NETLOAD_ADDRESS) +
|
|
Packit |
d37888 |
(1L << GLIBTOP_NETLOAD_SUBNET) +
|
|
Packit |
d37888 |
(1L << GLIBTOP_NETLOAD_MTU);
|
|
Packit |
d37888 |
|
|
Packit |
d37888 |
static const unsigned long _glibtop_sysdeps_netload_bytes =
|
|
Packit |
d37888 |
(1L << GLIBTOP_NETLOAD_BYTES_IN) +
|
|
Packit |
d37888 |
(1L << GLIBTOP_NETLOAD_BYTES_OUT) +
|
|
Packit |
d37888 |
(1L << GLIBTOP_NETLOAD_BYTES_TOTAL);
|
|
Packit |
d37888 |
|
|
Packit |
d37888 |
static const unsigned long _glibtop_sysdeps_netload_packets =
|
|
Packit |
d37888 |
(1L << GLIBTOP_NETLOAD_PACKETS_IN) +
|
|
Packit |
d37888 |
(1L << GLIBTOP_NETLOAD_PACKETS_OUT) +
|
|
Packit |
d37888 |
(1L << GLIBTOP_NETLOAD_PACKETS_TOTAL);
|
|
Packit |
d37888 |
|
|
Packit |
d37888 |
static const unsigned long _glibtop_sysdeps_netload_total =
|
|
Packit |
d37888 |
(1L << GLIBTOP_NETLOAD_PACKETS_TOTAL) +
|
|
Packit |
d37888 |
(1L << GLIBTOP_NETLOAD_BYTES_TOTAL);
|
|
Packit |
d37888 |
|
|
Packit |
d37888 |
static const unsigned long _glibtop_sysdeps_netload_in =
|
|
Packit |
d37888 |
(1L << GLIBTOP_NETLOAD_PACKETS_TOTAL) +
|
|
Packit |
d37888 |
(1L << GLIBTOP_NETLOAD_BYTES_TOTAL) +
|
|
Packit |
d37888 |
(1L << GLIBTOP_NETLOAD_PACKETS_IN) +
|
|
Packit |
d37888 |
(1L << GLIBTOP_NETLOAD_BYTES_IN);
|
|
Packit |
d37888 |
|
|
Packit |
d37888 |
static const unsigned long _glibtop_sysdeps_netload_out =
|
|
Packit |
d37888 |
(1L << GLIBTOP_NETLOAD_PACKETS_TOTAL) +
|
|
Packit |
d37888 |
(1L << GLIBTOP_NETLOAD_BYTES_TOTAL) +
|
|
Packit |
d37888 |
(1L << GLIBTOP_NETLOAD_PACKETS_OUT) +
|
|
Packit |
d37888 |
(1L << GLIBTOP_NETLOAD_BYTES_OUT);
|
|
Packit |
d37888 |
|
|
Packit |
d37888 |
static const unsigned long _glibtop_sysdeps_netload_6 =
|
|
Packit |
d37888 |
(1L << GLIBTOP_NETLOAD_ADDRESS6) +
|
|
Packit |
d37888 |
(1L << GLIBTOP_NETLOAD_PREFIX6) +
|
|
Packit |
d37888 |
(1L << GLIBTOP_NETLOAD_SCOPE6);
|
|
Packit |
d37888 |
|
|
Packit |
d37888 |
/* Init function. */
|
|
Packit |
d37888 |
|
|
Packit |
d37888 |
void
|
|
Packit |
d37888 |
_glibtop_init_netload_s (glibtop *server)
|
|
Packit |
d37888 |
{
|
|
Packit |
d37888 |
server->sysdeps.netload = _glibtop_sysdeps_netload |
|
|
Packit |
d37888 |
_glibtop_sysdeps_netload_bytes |
|
|
Packit |
d37888 |
_glibtop_sysdeps_netload_packets;
|
|
Packit |
d37888 |
}
|
|
Packit |
d37888 |
|
|
Packit |
d37888 |
static int
|
|
Packit |
d37888 |
solaris_stats(glibtop *server,
|
|
Packit |
d37888 |
glibtop_netload *buf,
|
|
Packit |
d37888 |
const char *interface)
|
|
Packit |
d37888 |
{
|
|
Packit |
d37888 |
char *name = interface;
|
|
Packit |
d37888 |
char *module;
|
|
Packit |
d37888 |
char *ptr;
|
|
Packit |
d37888 |
kstat_ctl_t * const kctl = server->machine->kc;
|
|
Packit |
d37888 |
kstat_t *ksp;
|
|
Packit |
d37888 |
kstat_named_t *kdata;
|
|
Packit |
d37888 |
int have_bytes = 1;
|
|
Packit |
d37888 |
|
|
Packit |
d37888 |
/*
|
|
Packit |
d37888 |
* chop off the trailing interface
|
|
Packit |
d37888 |
*/
|
|
Packit |
d37888 |
module = strdup( name );
|
|
Packit |
d37888 |
ptr = module + strlen( module ) - 1;
|
|
Packit |
d37888 |
while( (ptr > module) && isdigit( (int) *ptr ) ) {
|
|
Packit |
d37888 |
*ptr = '\0';
|
|
Packit |
d37888 |
ptr--;
|
|
Packit |
d37888 |
}
|
|
Packit |
d37888 |
|
|
Packit |
d37888 |
/*
|
|
Packit |
d37888 |
* get a kstat handle and update the user's kstat chain
|
|
Packit |
d37888 |
*/
|
|
Packit |
d37888 |
if( kctl == NULL ){
|
|
Packit |
d37888 |
glibtop_warn_io_r (server, "kstat_open ()");
|
|
Packit |
d37888 |
free( module );
|
|
Packit |
d37888 |
return( 0 );
|
|
Packit |
d37888 |
}
|
|
Packit |
d37888 |
while( kstat_chain_update( kctl ) != 0 )
|
|
Packit |
d37888 |
;
|
|
Packit |
d37888 |
|
|
Packit |
d37888 |
/*
|
|
Packit |
d37888 |
* traverse the kstat chain
|
|
Packit |
d37888 |
* to find the appropriate statistics
|
|
Packit |
d37888 |
*/
|
|
Packit |
d37888 |
if( (ksp = kstat_lookup( kctl,
|
|
Packit |
d37888 |
module, 0, name )) == NULL ) {
|
|
Packit |
d37888 |
free( module );
|
|
Packit |
d37888 |
return( 0 );
|
|
Packit |
d37888 |
}
|
|
Packit |
d37888 |
if( kstat_read( kctl, ksp, NULL ) == -1 ) {
|
|
Packit |
d37888 |
free( module );
|
|
Packit |
d37888 |
return( 0 );
|
|
Packit |
d37888 |
}
|
|
Packit |
d37888 |
free( module );
|
|
Packit |
d37888 |
|
|
Packit |
d37888 |
/*
|
|
Packit |
d37888 |
* lookup & store the data
|
|
Packit |
d37888 |
*/
|
|
Packit |
d37888 |
kdata = (kstat_named_t *) kstat_data_lookup( ksp, "ipackets" );
|
|
Packit |
d37888 |
if( kdata != NULL ) {
|
|
Packit |
d37888 |
buf->packets_in= kdata->value.ul;
|
|
Packit |
d37888 |
}
|
|
Packit |
d37888 |
kdata = (kstat_named_t *) kstat_data_lookup( ksp, "opackets" );
|
|
Packit |
d37888 |
if( kdata != NULL ) {
|
|
Packit |
d37888 |
buf->packets_out = kdata->value.ul;
|
|
Packit |
d37888 |
}
|
|
Packit |
d37888 |
kdata = (kstat_named_t *) kstat_data_lookup( ksp, "rbytes" );
|
|
Packit |
d37888 |
if( kdata != NULL ) {
|
|
Packit |
d37888 |
buf->bytes_in =kdata->value.ul;
|
|
Packit |
d37888 |
}
|
|
Packit |
d37888 |
kdata = (kstat_named_t *) kstat_data_lookup( ksp, "obytes" );
|
|
Packit |
d37888 |
if( kdata != NULL ) {
|
|
Packit |
d37888 |
buf->bytes_out =kdata->value.ul;
|
|
Packit |
d37888 |
}
|
|
Packit |
d37888 |
kdata = (kstat_named_t *) kstat_data_lookup( ksp, "ierrors" );
|
|
Packit |
d37888 |
if( kdata != NULL ) {
|
|
Packit |
d37888 |
buf->errors_in = kdata->value.ul;
|
|
Packit |
d37888 |
}
|
|
Packit |
d37888 |
kdata = (kstat_named_t *) kstat_data_lookup( ksp, "oerrors" );
|
|
Packit |
d37888 |
if( kdata != NULL ) {
|
|
Packit |
d37888 |
buf->errors_out = kdata->value.ul;
|
|
Packit |
d37888 |
}
|
|
Packit |
d37888 |
kdata = (kstat_named_t *) kstat_data_lookup( ksp, "collisions" );
|
|
Packit |
d37888 |
if( kdata != NULL ) {
|
|
Packit |
d37888 |
buf->collisions = kdata->value.ul;
|
|
Packit |
d37888 |
}
|
|
Packit |
d37888 |
|
|
Packit |
d37888 |
/* Compute total valules. */
|
|
Packit |
d37888 |
|
|
Packit |
d37888 |
buf->bytes_total = buf->bytes_in + buf->bytes_out;
|
|
Packit |
d37888 |
buf->packets_total = buf->packets_in + buf->packets_out;
|
|
Packit |
d37888 |
buf->errors_total = buf->errors_in + buf->errors_out;
|
|
Packit |
d37888 |
/* And now the flags. */
|
|
Packit |
d37888 |
buf->flags |= _glibtop_sysdeps_netload;
|
|
Packit |
d37888 |
buf->flags |= _glibtop_sysdeps_netload_bytes;
|
|
Packit |
d37888 |
buf->flags |= _glibtop_sysdeps_netload_packets;
|
|
Packit |
d37888 |
|
|
Packit |
d37888 |
/* finished */
|
|
Packit |
d37888 |
}
|
|
Packit |
d37888 |
|
|
Packit |
d37888 |
/* Provides network statistics. */
|
|
Packit |
d37888 |
|
|
Packit |
d37888 |
void
|
|
Packit |
d37888 |
glibtop_get_netload_s (glibtop *server, glibtop_netload *buf,
|
|
Packit |
d37888 |
const char *interface)
|
|
Packit |
d37888 |
{
|
|
Packit |
d37888 |
int skfd;
|
|
Packit |
d37888 |
memset (buf, 0, sizeof (glibtop_netload));
|
|
Packit |
d37888 |
|
|
Packit |
d37888 |
/* set flag */
|
|
Packit |
d37888 |
skfd = socket (PF_INET, SOCK_DGRAM, 0);
|
|
Packit |
d37888 |
if (skfd) {
|
|
Packit |
d37888 |
struct ifreq ifr;
|
|
Packit |
d37888 |
|
|
Packit |
d37888 |
g_strlcpy (ifr.ifr_name, interface, sizeof ifr.ifr_name);
|
|
Packit |
d37888 |
if (ioctl (skfd, SIOCGIFFLAGS, &ifr) >= 0) {
|
|
Packit |
d37888 |
const unsigned long long flags = ifr.ifr_flags;
|
|
Packit |
d37888 |
|
|
Packit |
d37888 |
buf->flags |= (1L << GLIBTOP_NETLOAD_IF_FLAGS);
|
|
Packit |
d37888 |
|
|
Packit |
d37888 |
if (flags & IFF_UP)
|
|
Packit |
d37888 |
buf->if_flags |= (1L << GLIBTOP_IF_FLAGS_UP);
|
|
Packit |
d37888 |
|
|
Packit |
d37888 |
if (flags & IFF_BROADCAST)
|
|
Packit |
d37888 |
buf->if_flags |= (1L << GLIBTOP_IF_FLAGS_BROADCAST);
|
|
Packit |
d37888 |
|
|
Packit |
d37888 |
if (flags & IFF_DEBUG)
|
|
Packit |
d37888 |
buf->if_flags |= (1L << GLIBTOP_IF_FLAGS_DEBUG);
|
|
Packit |
d37888 |
|
|
Packit |
d37888 |
if (flags & IFF_LOOPBACK)
|
|
Packit |
d37888 |
buf->if_flags |= (1L << GLIBTOP_IF_FLAGS_LOOPBACK);
|
|
Packit |
d37888 |
|
|
Packit |
d37888 |
if (flags & IFF_POINTOPOINT)
|
|
Packit |
d37888 |
buf->if_flags |= (1L << GLIBTOP_IF_FLAGS_POINTOPOINT);
|
|
Packit |
d37888 |
|
|
Packit |
d37888 |
if (flags & IFF_RUNNING)
|
|
Packit |
d37888 |
buf->if_flags |= (1L << GLIBTOP_IF_FLAGS_RUNNING);
|
|
Packit |
d37888 |
|
|
Packit |
d37888 |
if (flags & IFF_NOARP)
|
|
Packit |
d37888 |
buf->if_flags |= (1L << GLIBTOP_IF_FLAGS_NOARP);
|
|
Packit |
d37888 |
|
|
Packit |
d37888 |
if (flags & IFF_PROMISC)
|
|
Packit |
d37888 |
buf->if_flags |= (1L << GLIBTOP_IF_FLAGS_PROMISC);
|
|
Packit |
d37888 |
|
|
Packit |
d37888 |
if (flags & IFF_ALLMULTI)
|
|
Packit |
d37888 |
buf->if_flags |= (1L << GLIBTOP_IF_FLAGS_ALLMULTI);
|
|
Packit |
d37888 |
|
|
Packit |
d37888 |
if (flags & IFF_MULTICAST)
|
|
Packit |
d37888 |
buf->if_flags |= (1L << GLIBTOP_IF_FLAGS_MULTICAST);
|
|
Packit |
d37888 |
}
|
|
Packit |
d37888 |
|
|
Packit |
d37888 |
g_strlcpy (ifr.ifr_name, interface, sizeof ifr.ifr_name);
|
|
Packit |
d37888 |
if (!ioctl (skfd, SIOCGIFADDR, &ifr)) {
|
|
Packit |
d37888 |
buf->address = ((struct sockaddr_in *) &ifr.ifr_addr)->sin_addr.s_addr;
|
|
Packit |
d37888 |
buf->flags |= (1L << GLIBTOP_NETLOAD_ADDRESS);
|
|
Packit |
d37888 |
}
|
|
Packit |
d37888 |
|
|
Packit |
d37888 |
g_strlcpy (ifr.ifr_name, interface, sizeof ifr.ifr_name);
|
|
Packit |
d37888 |
if (!ioctl (skfd, SIOCGIFNETMASK, &ifr)) {
|
|
Packit |
d37888 |
buf->subnet = ((struct sockaddr_in *) &ifr.ifr_addr)->sin_addr.s_addr;
|
|
Packit |
d37888 |
buf->flags |= (1L << GLIBTOP_NETLOAD_SUBNET);
|
|
Packit |
d37888 |
}
|
|
Packit |
d37888 |
close (skfd);
|
|
Packit |
d37888 |
}
|
|
Packit |
d37888 |
|
|
Packit |
d37888 |
/*
|
|
Packit |
d37888 |
* Statistics
|
|
Packit |
d37888 |
*/
|
|
Packit |
d37888 |
|
|
Packit |
d37888 |
solaris_stats(server, buf, interface);
|
|
Packit |
d37888 |
|
|
Packit |
d37888 |
}
|