summaryrefslogtreecommitdiff
path: root/interface.c
diff options
context:
space:
mode:
Diffstat (limited to 'interface.c')
-rw-r--r--interface.c282
1 files changed, 172 insertions, 110 deletions
diff --git a/interface.c b/interface.c
index 648888b..b647874 100644
--- a/interface.c
+++ b/interface.c
@@ -3,7 +3,7 @@
* that either displays or sets the characteristics of
* one or more of the system's networking interfaces.
*
- * Version: $Id: interface.c,v 1.3 2001/03/10 02:00:54 mjn3 Exp $
+ * Version: $Id: interface.c,v 1.4 2001/03/12 09:57:59 mjn3 Exp $
*
* Author: Fred N. van Kempen, <waltje@uwalt.nl.mugnet.org>
* and others. Copyright 1993 MicroWalt Corporation
@@ -26,6 +26,17 @@
* Erik Andersen <andersee@debian.org>
*/
+/*
+ * Heavily modified by Manuel Novoa III Mar 12, 2001
+ *
+ * Pruned unused code using KEEP_UNUSED define.
+ * Added print_bytes_scaled function to reduce code size.
+ * Added some (potentially) missing defines.
+ * Improved display support for -a and for a named interface.
+ */
+
+/* #define KEEP_UNUSED */
+
#include "busybox.h"
/*
@@ -75,7 +86,7 @@
#define new(p) ((p) = xcalloc(1,sizeof(*(p))))
#define KRELEASE(maj,min,patch) ((maj) * 10000 + (min)*1000 + (patch))
-int procnetdev_vsn = 1;
+static int procnetdev_vsn = 1;
/* Ugh. But libc5 doesn't provide POSIX types. */
@@ -120,10 +131,29 @@ struct in6_ifreq {
#include "util.h"
#endif
+/* Defines for glibc2.0 users. */
+#ifndef SIOCSIFTXQLEN
+#define SIOCSIFTXQLEN 0x8943
+#define SIOCGIFTXQLEN 0x8942
+#endif
+
+/* ifr_qlen is ifru_ivalue, but it isn't present in 2.0 kernel headers */
+#ifndef ifr_qlen
+#define ifr_qlen ifr_ifru.ifru_mtu
+#endif
+
+#ifndef HAVE_TXQUEUELEN
+#define HAVE_TXQUEUELEN 1
+#endif
+
+#ifndef IFF_DYNAMIC
+#define IFF_DYNAMIC 0x8000 /* dialup device with changing addresses */
+#endif
+
/* This structure defines protocol families and their handlers. */
struct aftype {
- char *name;
- char *title;
+ const char *name;
+ const char *title;
int af;
int alen;
char *(*print) (unsigned char *);
@@ -140,20 +170,23 @@ struct aftype {
char *flag_file;
};
-extern struct aftype *aftypes[];
-int flag_unx;
-int flag_ipx;
-int flag_ax25;
-int flag_ddp;
-int flag_netrom;
-int flag_inet;
-int flag_inet6;
-int flag_econet;
-int flag_x25 = 0;
-int flag_ash;
+static struct aftype *aftypes[];
+
+#ifdef KEEP_UNUSED
+static int flag_unx;
+static int flag_ipx;
+static int flag_ax25;
+static int flag_ddp;
+static int flag_netrom;
+static int flag_inet;
+static int flag_inet6;
+static int flag_econet;
+static int flag_x25 = 0;
+static int flag_ash;
-struct aftrans_t {
+
+static struct aftrans_t {
char *alias;
char *name;
int *flag;
@@ -206,7 +239,8 @@ struct aftrans_t {
}
};
-char afname[256] = "";
+static char afname[256] = "";
+#endif /* KEEP_UNUSED */
#if HAVE_AFUNIX
@@ -228,9 +262,9 @@ static char *UNIX_sprint(struct sockaddr *sap, int numeric)
}
-struct aftype unix_aftype =
+static struct aftype unix_aftype =
{
- "unix", NULL, /*"UNIX Domain", */ AF_UNIX, 0,
+ "unix", "UNIX Domain", AF_UNIX, 0,
UNIX_print, UNIX_sprint, NULL, NULL,
NULL, NULL, NULL,
-1,
@@ -240,7 +274,9 @@ struct aftype unix_aftype =
#if HAVE_AFINET
+#if 0
extern int h_errno; /* some netdb.h versions don't export this */
+#endif
/* cache */
struct addr {
@@ -252,6 +288,7 @@ struct addr {
static struct addr *INET_nn = NULL; /* addr-to-name cache */
+#ifdef KEEP_UNUSED
static int INET_resolve(char *name, struct sockaddr_in *sin, int hostfirst)
{
struct hostent *hp;
@@ -310,7 +347,7 @@ static int INET_resolve(char *name, struct sockaddr_in *sin, int hostfirst)
return 0;
}
-
+#endif /* KEEP_UNUSED */
/* numeric: & 0x8000: default instead of *,
* & 0x4000: host instead of net,
@@ -399,19 +436,18 @@ static int INET_rresolve(char *name, size_t len, struct sockaddr_in *sin,
return (0);
}
-
+#ifdef KEEP_UNUSED
static void INET_reserror(char *text)
{
herror(text);
}
-
/* Display an Internet socket address. */
static char *INET_print(unsigned char *ptr)
{
return (inet_ntoa((*(struct in_addr *) ptr)));
}
-
+#endif /* KEEP_UNUSED */
/* Display an Internet socket address. */
static char *INET_sprint(struct sockaddr *sap, int numeric)
@@ -428,7 +464,8 @@ static char *INET_sprint(struct sockaddr *sap, int numeric)
return (buff);
}
-char *INET_sprintmask(struct sockaddr *sap, int numeric,
+#ifdef KEEP_UNUSED
+static char *INET_sprintmask(struct sockaddr *sap, int numeric,
unsigned int netmask)
{
static char buff[128];
@@ -441,7 +478,6 @@ char *INET_sprintmask(struct sockaddr *sap, int numeric,
return (buff);
}
-
static int INET_getsock(char *bufp, struct sockaddr *sap)
{
char *sp = bufp, *bp;
@@ -516,14 +552,15 @@ static int INET_getnetmask(char *adr, struct sockaddr *m, char *name)
mask->sin_addr.s_addr = htonl(~(0xffffffffU >> prefix));
return 1;
}
+#endif /* KEEP_UNUSED */
-
-struct aftype inet_aftype =
+static struct aftype inet_aftype =
{
- "inet", NULL, /*"DARPA Internet", */ AF_INET, sizeof(unsigned long),
- INET_print, INET_sprint, INET_input, INET_reserror,
+ "inet", "DARPA Internet", AF_INET, sizeof(unsigned long),
+ NULL /* UNUSED INET_print */, INET_sprint,
+ NULL /* UNUSED INET_input */, NULL /* UNUSED INET_reserror */,
NULL /*INET_rprint */ , NULL /*INET_rinput */ ,
- INET_getnetmask,
+ NULL /* UNUSED INET_getnetmask */,
-1,
NULL
};
@@ -555,16 +592,14 @@ static char *UNSPEC_sprint(struct sockaddr *sap, int numeric)
return (UNSPEC_print(sap->sa_data));
}
-struct aftype unspec_aftype =
+static struct aftype unspec_aftype =
{
- "unspec", NULL, /*"UNSPEC", */ AF_UNSPEC, 0,
+ "unspec", "UNSPEC", AF_UNSPEC, 0,
UNSPEC_print, UNSPEC_sprint, NULL, NULL,
NULL,
};
-static short sVafinit = 0;
-
-struct aftype *aftypes[] =
+static struct aftype *aftypes[] =
{
#if HAVE_AFUNIX
&unix_aftype,
@@ -603,7 +638,10 @@ struct aftype *aftypes[] =
NULL
};
-void afinit()
+#ifdef KEEP_UNUSED
+static short sVafinit = 0;
+
+static void afinit()
{
unspec_aftype.title = _("UNSPEC");
#if HAVE_AFUNIX
@@ -642,7 +680,7 @@ void afinit()
sVafinit = 1;
}
-int aftrans_opt(const char *arg)
+static int aftrans_opt(const char *arg)
{
struct aftrans_t *paft;
char *tmp1, *tmp2;
@@ -685,7 +723,7 @@ int aftrans_opt(const char *arg)
}
/* set the default AF list from the program name or a constant value */
-void aftrans_def(char *tool, char *argv0, char *dflt)
+static void aftrans_def(char *tool, char *argv0, char *dflt)
{
char *tmp;
char *buf;
@@ -721,14 +759,15 @@ void aftrans_def(char *tool, char *argv0, char *dflt)
free(buf);
}
-
/* Check our protocol family table for this family. */
-struct aftype *get_aftype(const char *name)
+static struct aftype *get_aftype(const char *name)
{
struct aftype **afp;
+#ifdef KEEP_UNUSED
if (!sVafinit)
afinit();
+#endif /* KEEP_UNUSED */
afp = aftypes;
while (*afp != NULL) {
@@ -740,15 +779,17 @@ struct aftype *get_aftype(const char *name)
fprintf(stderr, _("Please don't supply more than one address family.\n"));
return (NULL);
}
-
+#endif /* KEEP_UNUSED */
/* Check our protocol family table for this family. */
-struct aftype *get_afntype(int af)
+static struct aftype *get_afntype(int af)
{
struct aftype **afp;
+#ifdef KEEP_UNUSED
if (!sVafinit)
afinit();
+#endif /* KEEP_UNUSED */
afp = aftypes;
while (*afp != NULL) {
@@ -760,12 +801,14 @@ struct aftype *get_afntype(int af)
}
/* Check our protocol family table for this family and return its socket */
-int get_socket_for_af(int af)
+static int get_socket_for_af(int af)
{
struct aftype **afp;
+#ifdef KEEP_UNUSED
if (!sVafinit)
afinit();
+#endif /* KEEP_UNUSED */
afp = aftypes;
while (*afp != NULL) {
@@ -776,14 +819,17 @@ int get_socket_for_af(int af)
return -1;
}
+#ifdef KEEP_UNUSED
/* type: 0=all, 1=getroute */
-void print_aflist(int type) {
+static void print_aflist(int type) {
int count = 0;
char * txt;
struct aftype **afp;
+#ifdef KEEP_UNUSED
if (!sVafinit)
afinit();
+#endif /* KEEP_UNUSED */
afp = aftypes;
while (*afp != NULL) {
@@ -792,12 +838,13 @@ void print_aflist(int type) {
}
if ((count % 3) == 0) fprintf(stderr,count?"\n ":" ");
txt = (*afp)->name; if (!txt) txt = "..";
- fprintf(stderr,"%s (%s) ",txt,(*afp)->title);
+ fprintf(stderr,"%s (%s) ",txt,_((*afp)->title));
count++;
afp++;
}
fprintf(stderr,"\n");
}
+#endif /* KEEP_UNUSED */
struct user_net_device_stats {
unsigned long long rx_packets; /* total packets received */
@@ -863,16 +910,20 @@ struct interface {
};
-int opt_a = 0; /* show all interfaces */
-int opt_i = 0; /* show the statistics */
-int opt_v = 0; /* debugging output flag */
+int interface_opt_a = 0; /* show all interfaces */
+
+#ifdef KEEP_UNUSED
+static int opt_i = 0; /* show the statistics */
+static int opt_v = 0; /* debugging output flag */
+
+static int addr_family = 0; /* currently selected AF */
+#endif /* KEEP_UNUSED */
-int addr_family = 0; /* currently selected AF */
static struct interface *int_list, *int_last;
-int skfd = -1; /* generic raw socket desc. */
+static int skfd = -1; /* generic raw socket desc. */
-int sockets_open(int family)
+static int sockets_open(int family)
{
struct aftype **aft;
int sfd = -1;
@@ -919,7 +970,7 @@ int sockets_open(int family)
}
/* like strcmp(), but knows about numbers */
-int nstrcmp(const char *astr, const char *b)
+static int nstrcmp(const char *astr, const char *b)
{
const char *a = astr;
@@ -1053,7 +1104,7 @@ static int get_dev_fields(char *bp, struct interface *ife)
switch (procnetdev_vsn) {
case 3:
sscanf(bp,
- "%llu %llu %lu %lu %lu %lu %lu %lu %llu %llu %lu %lu %lu %lu %lu %lu",
+ "%Lu %Lu %lu %lu %lu %lu %lu %lu %Lu %Lu %lu %lu %lu %lu %lu %lu",
&ife->stats.rx_bytes,
&ife->stats.rx_packets,
&ife->stats.rx_errors,
@@ -1073,7 +1124,7 @@ static int get_dev_fields(char *bp, struct interface *ife)
&ife->stats.tx_compressed);
break;
case 2:
- sscanf(bp, "%llu %llu %lu %lu %lu %lu %llu %llu %lu %lu %lu %lu %lu",
+ sscanf(bp, "%Lu %Lu %lu %lu %lu %lu %Lu %Lu %lu %lu %lu %lu %lu",
&ife->stats.rx_bytes,
&ife->stats.rx_packets,
&ife->stats.rx_errors,
@@ -1091,7 +1142,7 @@ static int get_dev_fields(char *bp, struct interface *ife)
ife->stats.rx_multicast = 0;
break;
case 1:
- sscanf(bp, "%llu %lu %lu %lu %lu %llu %lu %lu %lu %lu %lu",
+ sscanf(bp, "%Lu %lu %lu %lu %lu %Lu %lu %lu %lu %lu %lu",
&ife->stats.rx_packets,
&ife->stats.rx_errors,
&ife->stats.rx_dropped,
@@ -1112,7 +1163,7 @@ static int get_dev_fields(char *bp, struct interface *ife)
return 0;
}
-static int procnetdev_version(char *buf)
+static inline int procnetdev_version(char *buf)
{
if (strstr(buf, "compressed"))
return 3;
@@ -1192,7 +1243,7 @@ static int if_readlist_proc(char *target)
return err;
}
-int if_readlist(void)
+static int if_readlist(void)
{
int err = if_readlist_proc(NULL);
if (!err)
@@ -1200,7 +1251,7 @@ int if_readlist(void)
return err;
}
-int for_all_interfaces(int (*doit) (struct interface *, void *), void *cookie)
+static int for_all_interfaces(int (*doit) (struct interface *, void *), void *cookie)
{
struct interface *ife;
@@ -1226,7 +1277,7 @@ static int ipx_getaddr(int sock, int ft, struct ifreq *ifr)
/* Fetch the interface configuration from the kernel. */
-int if_fetch(struct interface *ife)
+static int if_fetch(struct interface *ife)
{
struct ifreq ifr;
int fd;
@@ -1386,7 +1437,7 @@ int if_fetch(struct interface *ife)
}
-int do_if_fetch(struct interface *ife)
+static int do_if_fetch(struct interface *ife)
{
if (if_fetch(ife) < 0) {
char *errmsg;
@@ -1405,8 +1456,8 @@ int do_if_fetch(struct interface *ife)
/* This structure defines hardware protocols and their handlers. */
struct hwtype {
- char *name;
- char *title;
+ const char *name;
+ const char *title;
int type;
int alen;
char *(*print) (unsigned char *);
@@ -1430,15 +1481,15 @@ static char *pr_unspec(unsigned char *ptr)
return (buff);
}
-struct hwtype unspec_hwtype =
+static struct hwtype unspec_hwtype =
{
- "unspec", NULL, /*"UNSPEC", */ -1, 0,
+ "unspec", "UNSPEC", -1, 0,
pr_unspec, NULL, NULL
};
-struct hwtype loop_hwtype =
+static struct hwtype loop_hwtype =
{
- "loop", NULL, /*"Local Loopback", */ ARPHRD_LOOPBACK, 0,
+ "loop", "Local Loopback", ARPHRD_LOOPBACK, 0,
NULL, NULL, NULL
};
@@ -1446,7 +1497,7 @@ struct hwtype loop_hwtype =
#include <net/if_arp.h>
#include <linux/if_ether.h>
-extern struct hwtype ether_hwtype;
+static struct hwtype ether_hwtype;
/* Display an Ethernet address in readable format. */
static char *pr_ether(unsigned char *ptr)
@@ -1460,7 +1511,7 @@ static char *pr_ether(unsigned char *ptr)
return (buff);
}
-
+#ifdef KEEP_UNUSED
/* Input an Ethernet address and convert to binary. */
static int in_ether(char *bufp, struct sockaddr *sap)
{
@@ -1539,12 +1590,13 @@ static int in_ether(char *bufp, struct sockaddr *sap)
return (0);
}
+#endif /* KEEP_UNUSED */
-struct hwtype ether_hwtype =
+static struct hwtype ether_hwtype =
{
- "ether", NULL, /*"10Mbps Ethernet", */ ARPHRD_ETHER, ETH_ALEN,
- pr_ether, in_ether, NULL
+ "ether", "Ethernet", ARPHRD_ETHER, ETH_ALEN,
+ pr_ether, NULL /* UNUSED in_ether */, NULL
};
@@ -1555,18 +1607,19 @@ struct hwtype ether_hwtype =
#include <net/if_arp.h>
+#ifdef KEEP_UNUSED
/* Start the PPP encapsulation on the file descriptor. */
static int do_ppp(int fd)
{
fprintf(stderr, _("You cannot start PPP with this program.\n"));
return -1;
}
+#endif /* KEEP_UNUSED */
-
-struct hwtype ppp_hwtype =
+static struct hwtype ppp_hwtype =
{
- "ppp", NULL, /*"Point-Point Protocol", */ ARPHRD_PPP, 0,
- NULL, NULL, do_ppp, 0
+ "ppp", "Point-Point Protocol", ARPHRD_PPP, 0,
+ NULL, NULL, NULL /* UNUSED do_ppp */, 0
};
@@ -1647,9 +1700,10 @@ static struct hwtype *hwtypes[] =
NULL
};
+#ifdef KEEP_UNUSED
static short sVhwinit = 0;
-void hwinit()
+static void hwinit()
{
loop_hwtype.title = _("Local Loopback");
unspec_hwtype.title = _("UNSPEC");
@@ -1718,9 +1772,10 @@ void hwinit()
#endif
sVhwinit = 1;
}
+#endif /* KEEP_UNUSED */
#ifdef IFF_PORTSEL
-const char *if_port_text[][4] =
+static const char *if_port_text[][4] =
{
/* Keep in step with <linux/netdevice.h> */
{"unknown", NULL, NULL, NULL},
@@ -1735,12 +1790,14 @@ const char *if_port_text[][4] =
#endif
/* Check our hardware type table for this type. */
-struct hwtype *get_hwntype(int type)
+static struct hwtype *get_hwntype(int type)
{
struct hwtype **hwp;
+#ifdef KEEP_UNUSED
if (!sVhwinit)
hwinit();
+#endif /* KEEP_UNUSED */
hwp = hwtypes;
while (*hwp != NULL) {
@@ -1752,7 +1809,7 @@ struct hwtype *get_hwntype(int type)
}
/* return 1 if address is all zeros */
-int hw_null_address(struct hwtype *hw, void *ap)
+static int hw_null_address(struct hwtype *hw, void *ap)
{
unsigned int i;
unsigned char *address = (unsigned char *)ap;
@@ -1762,15 +1819,35 @@ int hw_null_address(struct hwtype *hw, void *ap)
return 1;
}
-void ife_print(struct interface *ptr)
+static const char TRext[] = "\0\0k\0M";
+
+static void print_bytes_scaled(unsigned long long ull, const char *end)
+{
+ unsigned long long int_part;
+ unsigned long frac_part;
+ const char *ext;
+ int i;
+
+ frac_part = 0;
+ ext = TRext;
+ int_part = ull;
+ for (i=0 ; i<2 ; i++) {
+ if (int_part >= 1024) {
+ frac_part = ((int_part % 1024) * 10) / 1024;
+ int_part /= 1024;
+ ext += 2; /* Kb, Mb */
+ }
+ }
+
+ printf("X bytes:%Lu (%Lu.%lu %sb)%s", ull, int_part, frac_part, ext, end);
+}
+
+static void ife_print(struct interface *ptr)
{
struct aftype *ap;
struct hwtype *hw;
int hf;
int can_compress = 0;
- unsigned long long rx, tx, short_rx, short_tx;
- char Rext[5]="b";
- char Text[5]="b";
#if HAVE_AFIPX
static struct aftype *ipxtype = NULL;
@@ -1803,7 +1880,7 @@ void ife_print(struct interface *ptr)
if (hw == NULL)
hw = get_hwntype(-1);
- printf(_("%-9.9s Link encap:%s "), ptr->name, hw->title);
+ printf(_("%-9.9s Link encap:%s "), ptr->name, _(hw->title));
/* For some hardware types (eg Ash, ATM) we don't print the
hardware address if it's null. */
if (hw->print != NULL && (! (hw_null_address(hw, ptr->hwaddr) &&
@@ -1965,24 +2042,14 @@ void ife_print(struct interface *ptr)
*/
printf(" ");
- printf(_("RX packets:%llu errors:%lu dropped:%lu overruns:%lu frame:%lu\n"),
+ printf(_("RX packets:%Lu errors:%lu dropped:%lu overruns:%lu frame:%lu\n"),
ptr->stats.rx_packets, ptr->stats.rx_errors,
ptr->stats.rx_dropped, ptr->stats.rx_fifo_errors,
ptr->stats.rx_frame_errors);
if (can_compress)
printf(_(" compressed:%lu\n"), ptr->stats.rx_compressed);
-
- rx = ptr->stats.rx_bytes;
- tx = ptr->stats.tx_bytes;
- short_rx = rx * 10;
- short_tx = tx * 10;
- if (rx > 1048576) { short_rx /= 1048576; strcpy(Rext, "Mb"); }
- else if (rx > 1024) { short_rx /= 1024; strcpy(Rext, "Kb"); }
- if (tx > 1048576) { short_tx /= 1048576; strcpy(Text, "Mb"); }
- else if (tx > 1024) { short_tx /= 1024; strcpy(Text, "Kb"); }
-
printf(" ");
- printf(_("TX packets:%llu errors:%lu dropped:%lu overruns:%lu carrier:%lu\n"),
+ printf(_("TX packets:%Lu errors:%lu dropped:%lu overruns:%lu carrier:%lu\n"),
ptr->stats.tx_packets, ptr->stats.tx_errors,
ptr->stats.tx_dropped, ptr->stats.tx_fifo_errors,
ptr->stats.tx_carrier_errors);
@@ -1991,12 +2058,10 @@ void ife_print(struct interface *ptr)
printf(_("compressed:%lu "), ptr->stats.tx_compressed);
if (ptr->tx_queue_len != -1)
printf(_("txqueuelen:%d "), ptr->tx_queue_len);
- printf("\n ");
- printf(_("RX bytes:%llu (%lu.%lu %s) TX bytes:%llu (%lu.%lu %s)\n"),
- rx, (unsigned long)(short_rx / 10),
- (unsigned long)(short_rx % 10), Rext,
- tx, (unsigned long)(short_tx / 10),
- (unsigned long)(short_tx % 10), Text);
+ printf("\n R");
+ print_bytes_scaled(ptr->stats.rx_bytes, " T");
+ print_bytes_scaled(ptr->stats.rx_bytes, "\n");
+
}
if ((ptr->map.irq || ptr->map.mem_start || ptr->map.dma ||
@@ -2018,7 +2083,7 @@ void ife_print(struct interface *ptr)
}
-int do_if_print(struct interface *ife, void *cookie)
+static int do_if_print(struct interface *ife, void *cookie)
{
int *opt_a = (int *) cookie;
int res;
@@ -2031,7 +2096,7 @@ int do_if_print(struct interface *ife, void *cookie)
return res;
}
-struct interface *lookup_interface(char *name)
+static struct interface *lookup_interface(char *name)
{
struct interface *ife = NULL;
@@ -2047,7 +2112,7 @@ static int if_print(char *ifname)
int res;
if (!ifname) {
- res = for_all_interfaces(do_if_print, &opt_a);
+ res = for_all_interfaces(do_if_print, &interface_opt_a);
} else {
struct interface *ife;
@@ -2059,20 +2124,17 @@ static int if_print(char *ifname)
return res;
}
-int display_interfaces(int opt_all)
+int display_interfaces(char *ifname)
{
int status;
- opt_a = opt_all;
-
/* Create a channel to the NET kernel. */
if ((skfd = sockets_open(0)) < 0) {
perror_msg_and_die("socket");
}
/* Do we have to show the current setup? */
- status = if_print((char *) NULL);
+ status = if_print(ifname);
close(skfd);
exit(status < 0);
}
-