Blame sysdeps/openbsd/netload.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>, October 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/netload.h>
Packit d37888
Packit d37888
#include <glibtop_suid.h>
Packit d37888
Packit d37888
#include <string.h>
Packit d37888
Packit d37888
#include <net/if.h>
Packit d37888
#include <net/if_dl.h>
Packit d37888
#include <net/if_types.h>
Packit d37888
Packit d37888
#include <sys/ioctl.h>
Packit d37888
Packit d37888
#include <net/if_var.h>
Packit d37888
Packit d37888
#include <netinet/in.h>
Packit d37888
#define _KERNEL
Packit d37888
#include <netinet/in_var.h>
Packit d37888
#undef _KERNEL
Packit d37888
#include <netinet6/in6_var.h>
Packit d37888
Packit d37888
static const unsigned long _glibtop_sysdeps_netload =
Packit d37888
(1L << GLIBTOP_NETLOAD_IF_FLAGS) +
Packit d37888
(1L << GLIBTOP_NETLOAD_PACKETS_IN) +
Packit d37888
(1L << GLIBTOP_NETLOAD_PACKETS_OUT) +
Packit d37888
(1L << GLIBTOP_NETLOAD_PACKETS_TOTAL) +
Packit d37888
(1L << GLIBTOP_NETLOAD_BYTES_IN) +
Packit d37888
(1L << GLIBTOP_NETLOAD_BYTES_OUT) +
Packit d37888
(1L << GLIBTOP_NETLOAD_BYTES_TOTAL) +
Packit d37888
(1L << GLIBTOP_NETLOAD_ERRORS_IN) +
Packit d37888
(1L << GLIBTOP_NETLOAD_ERRORS_OUT) +
Packit d37888
(1L << GLIBTOP_NETLOAD_ERRORS_TOTAL) +
Packit d37888
(1L << GLIBTOP_NETLOAD_COLLISIONS);
Packit d37888
Packit d37888
static const unsigned _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
/* nlist structure for kernel access */
Packit d37888
static struct nlist nlst [] = {
Packit d37888
    { "_ifnet" },
Packit d37888
    { 0 }
Packit d37888
};
Packit d37888
Packit d37888
/* Init function. */
Packit d37888
Packit d37888
void
Packit d37888
_glibtop_init_netload_p (glibtop *server)
Packit d37888
{
Packit d37888
    server->sysdeps.netload = _glibtop_sysdeps_netload;
Packit d37888
Packit d37888
    if (kvm_nlist (server->machine->kd, nlst) < 0)
Packit d37888
	glibtop_error_io_r (server, "kvm_nlist");
Packit d37888
}
Packit d37888
Packit d37888
/* Provides Network statistics. */
Packit d37888
Packit d37888
void
Packit d37888
glibtop_get_netload_p (glibtop *server, glibtop_netload *buf,
Packit d37888
		       const char *interface)
Packit d37888
{
Packit d37888
    struct ifnet ifnet;
Packit d37888
    u_long ifnetaddr, ifnetfound;
Packit d37888
    struct sockaddr *sa = NULL;
Packit d37888
    char name [32];
Packit d37888
Packit d37888
    union {
Packit d37888
	struct ifaddr ifa;
Packit d37888
	struct in_ifaddr in;
Packit d37888
    } ifaddr;
Packit d37888
Packit d37888
    glibtop_init_p (server, (1L << GLIBTOP_SYSDEPS_NETLOAD), 0);
Packit d37888
Packit d37888
    memset (buf, 0, sizeof (glibtop_netload));
Packit d37888
Packit d37888
    if (kvm_read (server->machine->kd, nlst [0].n_value,
Packit d37888
		  &ifnetaddr, sizeof (ifnetaddr)) != sizeof (ifnetaddr))
Packit d37888
	glibtop_error_io_r (server, "kvm_read (ifnet)");
Packit d37888
Packit d37888
    while (ifnetaddr) {
Packit d37888
	struct sockaddr_in *sin;
Packit d37888
	register char *cp;
Packit d37888
	u_long ifaddraddr;
Packit d37888
Packit d37888
	{
Packit d37888
	    ifnetfound = ifnetaddr;
Packit d37888
Packit d37888
	    if (kvm_read (server->machine->kd, ifnetaddr, &ifnet,
Packit d37888
			  sizeof (ifnet)) != sizeof (ifnet))
Packit d37888
		    glibtop_error_io_r (server, "kvm_read (ifnetaddr)");
Packit d37888
Packit d37888
	    g_strlcpy (name, ifnet.if_xname, sizeof(name));
Packit d37888
	    ifnetaddr = (u_long) ifnet.if_list.tqe_next;
Packit d37888
Packit d37888
	    if (strcmp (name, interface) != 0)
Packit d37888
		    continue;
Packit d37888
Packit d37888
	    ifaddraddr = (u_long) ifnet.if_addrlist.tqh_first;
Packit d37888
	}
Packit d37888
	if (ifnet.if_flags & IFF_UP)
Packit d37888
		buf->if_flags |= (1L << GLIBTOP_IF_FLAGS_UP);
Packit d37888
	if (ifnet.if_flags & IFF_BROADCAST)
Packit d37888
		buf->if_flags |= (1L << GLIBTOP_IF_FLAGS_BROADCAST);
Packit d37888
	if (ifnet.if_flags & IFF_DEBUG)
Packit d37888
		buf->if_flags |= (1L << GLIBTOP_IF_FLAGS_DEBUG);
Packit d37888
	if (ifnet.if_flags & IFF_LOOPBACK)
Packit d37888
		buf->if_flags |= (1L << GLIBTOP_IF_FLAGS_LOOPBACK);
Packit d37888
	if (ifnet.if_flags & IFF_POINTOPOINT)
Packit d37888
		buf->if_flags |= (1L << GLIBTOP_IF_FLAGS_POINTOPOINT);
Packit d37888
#ifdef IFF_DRV_RUNNING
Packit d37888
	if (ifnet.if_drv_flags & IFF_DRV_RUNNING)
Packit d37888
#else
Packit d37888
	if (ifnet.if_flags & IFF_RUNNING)
Packit d37888
#endif
Packit d37888
		buf->if_flags |= (1L << GLIBTOP_IF_FLAGS_RUNNING);
Packit d37888
	if (ifnet.if_flags & IFF_NOARP)
Packit d37888
		buf->if_flags |= (1L << GLIBTOP_IF_FLAGS_NOARP);
Packit d37888
	if (ifnet.if_flags & IFF_PROMISC)
Packit d37888
		buf->if_flags |= (1L << GLIBTOP_IF_FLAGS_PROMISC);
Packit d37888
	if (ifnet.if_flags & IFF_ALLMULTI)
Packit d37888
		buf->if_flags |= (1L << GLIBTOP_IF_FLAGS_ALLMULTI);
Packit d37888
#ifdef IFF_DRV_OACTIVE
Packit d37888
	if (ifnet.if_drv_flags & IFF_DRV_OACTIVE)
Packit d37888
#else
Packit d37888
	if (ifnet.if_flags & IFF_OACTIVE)
Packit d37888
#endif
Packit d37888
		buf->if_flags |= (1L << GLIBTOP_IF_FLAGS_OACTIVE);
Packit d37888
	if (ifnet.if_flags & IFF_SIMPLEX)
Packit d37888
		buf->if_flags |= (1L << GLIBTOP_IF_FLAGS_SIMPLEX);
Packit d37888
	if (ifnet.if_flags & IFF_LINK0)
Packit d37888
		buf->if_flags |= (1L << GLIBTOP_IF_FLAGS_LINK0);
Packit d37888
	if (ifnet.if_flags & IFF_LINK1)
Packit d37888
		buf->if_flags |= (1L << GLIBTOP_IF_FLAGS_LINK1);
Packit d37888
	if (ifnet.if_flags & IFF_LINK2)
Packit d37888
		buf->if_flags |= (1L << GLIBTOP_IF_FLAGS_LINK2);
Packit d37888
	if (ifnet.if_flags & IFF_MULTICAST)
Packit d37888
		buf->if_flags |= (1L << GLIBTOP_IF_FLAGS_MULTICAST);
Packit d37888
Packit d37888
	buf->packets_in = ifnet.if_ipackets;
Packit d37888
	buf->packets_out = ifnet.if_opackets;
Packit d37888
	buf->packets_total = buf->packets_in + buf->packets_out;
Packit d37888
Packit d37888
	buf->bytes_in = ifnet.if_ibytes;
Packit d37888
	buf->bytes_out = ifnet.if_obytes;
Packit d37888
	buf->bytes_total = buf->bytes_in + buf->bytes_out;
Packit d37888
Packit d37888
	buf->errors_in = ifnet.if_ierrors;
Packit d37888
	buf->errors_out = ifnet.if_oerrors;
Packit d37888
	buf->errors_total = buf->errors_in + buf->errors_out;
Packit d37888
Packit d37888
	buf->collisions = ifnet.if_collisions;
Packit d37888
	buf->flags = _glibtop_sysdeps_netload;
Packit d37888
Packit d37888
	while (ifaddraddr) {
Packit d37888
	    if ((kvm_read (server->machine->kd, ifaddraddr, &ifaddr,
Packit d37888
			   sizeof (ifaddr)) != sizeof (ifaddr)))
Packit d37888
		glibtop_error_io_r (server, "kvm_read (ifaddraddr)");
Packit d37888
Packit d37888
#define CP(x) ((char *)(x))
Packit d37888
	    cp = (CP(ifaddr.ifa.ifa_addr) - CP(ifaddraddr)) +
Packit d37888
		CP(&ifaddr);
Packit d37888
	    sa = (struct sockaddr *)cp;
Packit d37888
Packit d37888
	    if (sa->sa_family == AF_LINK) {
Packit d37888
		struct sockaddr_dl *dl = (struct sockaddr_dl *) sa;
Packit d37888
Packit d37888
		memcpy (buf->hwaddress, LLADDR (dl), sizeof (buf->hwaddress));
Packit d37888
		buf->flags |= (1L << GLIBTOP_NETLOAD_HWADDRESS);
Packit d37888
	    } else if (sa->sa_family == AF_INET) {
Packit d37888
		sin = (struct sockaddr_in *)sa;
Packit d37888
		buf->subnet = ifaddr.in.ia_netmask;
Packit d37888
		buf->address = sin->sin_addr.s_addr;
Packit d37888
		buf->mtu = ifnet.if_mtu;
Packit d37888
Packit d37888
		buf->flags |= _glibtop_sysdeps_netload_data;
Packit d37888
	    } else if (sa->sa_family == AF_INET6) {
Packit d37888
		struct sockaddr_in6 *sin6 = (struct sockaddr_in6 *) sa;
Packit d37888
		int in6fd;
Packit d37888
Packit d37888
		memcpy (buf->address6, &sin6->sin6_addr,
Packit d37888
		    sizeof (buf->address6));
Packit d37888
		buf->flags |= (1L << GLIBTOP_NETLOAD_ADDRESS6);
Packit d37888
Packit d37888
		if (IN6_IS_ADDR_LINKLOCAL(&sin6->sin6_addr)) {
Packit d37888
			sin6->sin6_scope_id =
Packit d37888
				ntohs(*(u_int16_t *)&sin6->sin6_addr.s6_addr[2]);
Packit d37888
			sin6->sin6_addr.s6_addr[2] = sin6->sin6_addr.s6_addr[3] = 0;
Packit d37888
		}
Packit d37888
Packit d37888
		buf->scope6 = (guint8) sin6->sin6_scope_id;
Packit d37888
		buf->flags |= (1L << GLIBTOP_NETLOAD_SCOPE6);
Packit d37888
Packit d37888
		in6fd = socket (AF_INET6, SOCK_DGRAM, 0);
Packit d37888
		if (in6fd >= 0) {
Packit d37888
			struct in6_ifreq ifr;
Packit d37888
Packit d37888
			memset (&ifr, 0, sizeof (ifr));
Packit d37888
			ifr.ifr_addr = *sin6;
Packit d37888
			g_strlcpy (ifr.ifr_name, interface,
Packit d37888
			    sizeof (ifr.ifr_name));
Packit d37888
			if (ioctl (in6fd, SIOCGIFNETMASK_IN6, (char *) &ifr) >= 0) {
Packit d37888
				memcpy (buf->prefix6, &ifr.ifr_addr.sin6_addr,
Packit d37888
				    sizeof (buf->prefix6));
Packit d37888
				buf->flags |= (1L << GLIBTOP_NETLOAD_PREFIX6);
Packit d37888
			}
Packit d37888
			close (in6fd);
Packit d37888
		}
Packit d37888
	    }
Packit d37888
	    ifaddraddr = (u_long) ifaddr.ifa.ifa_list.tqe_next;
Packit d37888
	}
Packit d37888
	return;
Packit d37888
    }
Packit d37888
}