367 lines
9.0 KiB
C
367 lines
9.0 KiB
C
/*
|
|
* (C)opyright 1993-1996 by Darren Reed.
|
|
*
|
|
* Redistribution and use in source and binary forms are permitted
|
|
* provided that this notice is preserved and due credit is given
|
|
* to the original author and the contributors.
|
|
*/
|
|
|
|
#include <stdio.h>
|
|
#include <string.h>
|
|
#if !defined(__SVR4) && !defined(__svr4__)
|
|
#include <strings.h>
|
|
#endif
|
|
#include <sys/types.h>
|
|
#include <sys/param.h>
|
|
#include <sys/file.h>
|
|
#include <stdlib.h>
|
|
#include <unistd.h>
|
|
#include <fcntl.h>
|
|
#include <stddef.h>
|
|
#include <nlist.h>
|
|
#include <sys/socket.h>
|
|
#include <sys/ioctl.h>
|
|
#include <netinet/in.h>
|
|
#include <arpa/inet.h>
|
|
#include <netinet/in_systm.h>
|
|
#include <netinet/ip.h>
|
|
#include <net/if.h>
|
|
#include "ip_fil.h"
|
|
#include "ip_compat.h"
|
|
#include "ip_nat.h"
|
|
#include "ip_frag.h"
|
|
#include "ip_state.h"
|
|
#include <netdb.h>
|
|
#include <arpa/nameser.h>
|
|
#include <resolv.h>
|
|
#include "ipf.h"
|
|
#include "kmem.h"
|
|
#ifdef __NetBSD__
|
|
#include <paths.h>
|
|
#endif
|
|
|
|
#if !defined(lint) && defined(LIBC_SCCS)
|
|
static char sccsid[] = "@(#)fils.c 1.21 4/20/96 (C) 1993-1996 Darren Reed";
|
|
static char rcsid[] = "$Id: fils.c,v 2.0.1.2 1997/01/30 10:21:48 darrenr Exp $";
|
|
#endif
|
|
#ifdef _PATH_UNIX
|
|
#define VMUNIX _PATH_UNIX
|
|
#else
|
|
#define VMUNIX "/vmunix"
|
|
#endif
|
|
|
|
extern char *optarg;
|
|
|
|
#define PRINTF (void)printf
|
|
#define FPRINTF (void)fprintf
|
|
#define F_IN 0
|
|
#define F_OUT 1
|
|
#define F_AC 2
|
|
static char *filters[4] = { "ipfilter(in)", "ipfilter(out)",
|
|
"ipacct(in)", "ipacct(out)" };
|
|
|
|
int opts = 0;
|
|
|
|
static void showstats(), showfrstates();
|
|
static void showlist(), showipstates();
|
|
|
|
void Usage(name)
|
|
char *name;
|
|
{
|
|
fprintf(stderr, "Usage: %s [-afhIiosv] [-d <device>]\n", name);
|
|
exit(1);
|
|
}
|
|
|
|
|
|
int main(argc,argv)
|
|
int argc;
|
|
char *argv[];
|
|
{
|
|
friostat_t fio;
|
|
ips_stat_t ipsst;
|
|
ipfrstat_t ifrst;
|
|
char c, *name = NULL, *device = IPL_NAME;
|
|
int fd;
|
|
|
|
if (openkmem() == -1)
|
|
exit(-1);
|
|
|
|
(void)setuid(getuid());
|
|
(void)setgid(getgid());
|
|
|
|
while ((c = getopt(argc, argv, "afhIiosvd:")) != -1)
|
|
{
|
|
switch (c)
|
|
{
|
|
case 'a' :
|
|
opts |= OPT_ACCNT|OPT_SHOWLIST;
|
|
break;
|
|
case 'd' :
|
|
device = optarg;
|
|
break;
|
|
case 'f' :
|
|
opts |= OPT_FRSTATES;
|
|
break;
|
|
case 'h' :
|
|
opts |= OPT_HITS;
|
|
break;
|
|
case 'i' :
|
|
opts |= OPT_INQUE|OPT_SHOWLIST;
|
|
break;
|
|
case 'n' :
|
|
opts |= OPT_SHOWLINENO;
|
|
break;
|
|
case 'I' :
|
|
opts |= OPT_INACTIVE;
|
|
break;
|
|
case 'o' :
|
|
opts |= OPT_OUTQUE|OPT_SHOWLIST;
|
|
break;
|
|
case 's' :
|
|
opts |= OPT_IPSTATES;
|
|
break;
|
|
case 'v' :
|
|
opts |= OPT_VERBOSE;
|
|
break;
|
|
default :
|
|
Usage(argv[0]);
|
|
break;
|
|
}
|
|
}
|
|
|
|
if ((fd = open(device, O_RDONLY)) < 0) {
|
|
perror("open");
|
|
exit(-1);
|
|
}
|
|
|
|
bzero((char *)&fio, sizeof(fio));
|
|
bzero((char *)&ipsst, sizeof(ipsst));
|
|
bzero((char *)&ifrst, sizeof(ifrst));
|
|
|
|
if (ioctl(fd, SIOCGETFS, &fio) == -1) {
|
|
perror("ioctl(SIOCGETFS)");
|
|
exit(-1);
|
|
}
|
|
if ((opts & OPT_IPSTATES) && (ioctl(fd, SIOCGIPST, &ipsst) == -1)) {
|
|
perror("ioctl(SIOCGIPST)");
|
|
exit(-1);
|
|
}
|
|
if ((opts & OPT_FRSTATES) && (ioctl(fd, SIOCGFRST, &ifrst) == -1)) {
|
|
perror("ioctl(SIOCGFRST)");
|
|
exit(-1);
|
|
}
|
|
|
|
if (opts & OPT_VERBOSE)
|
|
PRINTF("opts %#x name %s\n", opts, name ? name : "<>");
|
|
if (opts & OPT_SHOWLIST) {
|
|
showlist(&fio);
|
|
if((opts & OPT_OUTQUE) && (opts & OPT_INQUE)){
|
|
opts &= ~OPT_OUTQUE;
|
|
showlist(&fio);
|
|
}
|
|
} else {
|
|
if (opts & OPT_IPSTATES)
|
|
showipstates(fd, &ipsst);
|
|
else if (opts & OPT_FRSTATES)
|
|
showfrstates(fd, &ifrst);
|
|
else
|
|
showstats(fd, &fio);
|
|
}
|
|
return 0;
|
|
}
|
|
|
|
|
|
/*
|
|
* read the kernel stats for packets blocked and passed
|
|
*/
|
|
static void showstats(fd, fp)
|
|
int fd;
|
|
struct friostat *fp;
|
|
{
|
|
int frf = 0;
|
|
|
|
if (ioctl(fd, SIOCGETFF, &frf) == -1)
|
|
perror("ioctl(SIOCGETFF)");
|
|
|
|
#if SOLARIS
|
|
PRINTF("dropped packets:\tin %lu\tout %lu\n",
|
|
fp->f_st[0].fr_drop, fp->f_st[1].fr_drop);
|
|
PRINTF("non-ip packets:\t\tin %lu\tout %lu\n",
|
|
fp->f_st[0].fr_notip, fp->f_st[1].fr_notip);
|
|
PRINTF(" bad packets:\t\tin %lu\tout %lu\n",
|
|
fp->f_st[0].fr_bad, fp->f_st[1].fr_bad);
|
|
#endif
|
|
PRINTF(" input packets:\t\tblocked %lu passed %lu nomatch %lu",
|
|
fp->f_st[0].fr_block, fp->f_st[0].fr_pass,
|
|
fp->f_st[0].fr_nom);
|
|
PRINTF(" counted %lu\n", fp->f_st[0].fr_acct);
|
|
PRINTF("output packets:\t\tblocked %lu passed %lu nomatch %lu",
|
|
fp->f_st[1].fr_block, fp->f_st[1].fr_pass,
|
|
fp->f_st[1].fr_nom);
|
|
PRINTF(" counted %lu\n", fp->f_st[0].fr_acct);
|
|
PRINTF(" input packets logged:\tblocked %lu passed %lu\n",
|
|
fp->f_st[0].fr_bpkl, fp->f_st[0].fr_ppkl);
|
|
PRINTF("output packets logged:\tblocked %lu passed %lu\n",
|
|
fp->f_st[1].fr_bpkl, fp->f_st[1].fr_ppkl);
|
|
PRINTF(" packets logged:\tinput %lu output %lu\n",
|
|
fp->f_st[0].fr_pkl, fp->f_st[1].fr_pkl);
|
|
PRINTF(" log failures:\t\tinput %lu output %lu\n",
|
|
fp->f_st[0].fr_skip, fp->f_st[1].fr_skip);
|
|
PRINTF("fragment state(in):\tkept %lu\tlost %lu\n",
|
|
fp->f_st[0].fr_nfr, fp->f_st[0].fr_bnfr);
|
|
PRINTF("fragment state(out):\tkept %lu\tlost %lu\n",
|
|
fp->f_st[1].fr_nfr, fp->f_st[1].fr_bnfr);
|
|
PRINTF("packet state(in):\tkept %lu\tlost %lu\n",
|
|
fp->f_st[0].fr_ads, fp->f_st[0].fr_bads);
|
|
PRINTF("packet state(out):\tkept %lu\tlost %lu\n",
|
|
fp->f_st[1].fr_ads, fp->f_st[1].fr_bads);
|
|
PRINTF("ICMP replies:\t%lu\tTCP RSTs sent:\t%lu\n",
|
|
fp->f_st[0].fr_ret, fp->f_st[1].fr_ret);
|
|
PRINTF("Result cache hits(in):\t%lu\t(out):\t%lu\n",
|
|
fp->f_st[0].fr_chit, fp->f_st[1].fr_chit);
|
|
PRINTF("IN Pullups succeeded:\t%lu\tfailed:\t%lu\n",
|
|
fp->f_st[0].fr_pull[0], fp->f_st[0].fr_pull[1]);
|
|
PRINTF("OUT Pullups succeeded:\t%lu\tfailed:\t%lu\n",
|
|
fp->f_st[1].fr_pull[0], fp->f_st[1].fr_pull[1]);
|
|
|
|
PRINTF("Packet log flags set: (%#x)\n", frf);
|
|
if (frf & FF_LOGPASS)
|
|
PRINTF("\tpackets passed through filter\n");
|
|
if (frf & FF_LOGBLOCK)
|
|
PRINTF("\tpackets blocked by filter\n");
|
|
if (!frf)
|
|
PRINTF("\tnone\n");
|
|
}
|
|
|
|
/*
|
|
* print out filter rule list
|
|
*/
|
|
static void showlist(fiop)
|
|
struct friostat *fiop;
|
|
{
|
|
struct frentry fb;
|
|
struct frentry *fp = NULL;
|
|
int i, set, n;
|
|
|
|
set = fiop->f_active;
|
|
if (opts & OPT_INACTIVE)
|
|
set = 1 - set;
|
|
if (opts & OPT_ACCNT) {
|
|
i = F_AC;
|
|
if (opts & OPT_OUTQUE) {
|
|
fp = (struct frentry *)fiop->f_acctout[set];
|
|
i++;
|
|
} else if (opts & OPT_INQUE)
|
|
fp = (struct frentry *)fiop->f_acctin[set];
|
|
} else if (opts & OPT_OUTQUE) {
|
|
i = F_OUT;
|
|
fp = (struct frentry *)fiop->f_fout[set];
|
|
} else if (opts & OPT_INQUE) {
|
|
i = F_IN;
|
|
fp = (struct frentry *)fiop->f_fin[set];
|
|
} else
|
|
return;
|
|
if (opts & OPT_VERBOSE)
|
|
FPRINTF(stderr, "showlist:opts %#x i %d\n", opts, i);
|
|
|
|
if (opts & OPT_VERBOSE)
|
|
PRINTF("fp %#x set %d\n", (u_int)fp, set);
|
|
if (!fp) {
|
|
FPRINTF(stderr, "empty list for %s%s\n",
|
|
(opts & OPT_INACTIVE) ? "inactive " : "", filters[i]);
|
|
return;
|
|
}
|
|
|
|
for (n = 1; fp; n++) {
|
|
if (kmemcpy((char *)&fb, (u_long)fp, sizeof(fb)) == -1) {
|
|
perror("kmemcpy");
|
|
return;
|
|
}
|
|
fp = &fb;
|
|
if (opts & OPT_OUTQUE)
|
|
fp->fr_flags |= FR_OUTQUE;
|
|
if (opts & (OPT_HITS|OPT_VERBOSE))
|
|
PRINTF("%ld ", fp->fr_hits);
|
|
if (opts & (OPT_ACCNT|OPT_VERBOSE))
|
|
PRINTF("%ld ", fp->fr_bytes);
|
|
if (opts & OPT_SHOWLINENO)
|
|
PRINTF("@%d ", n);
|
|
printfr(fp);
|
|
if (opts & OPT_VERBOSE)
|
|
binprint(fp);
|
|
fp = fp->fr_next;
|
|
}
|
|
}
|
|
|
|
|
|
static void showipstates(fd, ipsp)
|
|
int fd;
|
|
ips_stat_t *ipsp;
|
|
{
|
|
ipstate_t *istab[IPSTATE_SIZE], ips;
|
|
int i;
|
|
|
|
PRINTF("IP states added:\n\t%lu TCP\n\t%lu UDP\n\t%lu ICMP\n",
|
|
ipsp->iss_tcp, ipsp->iss_udp, ipsp->iss_icmp);
|
|
PRINTF("\t%lu hits\n\t%lu misses\n", ipsp->iss_hits, ipsp->iss_miss);
|
|
PRINTF("\t%lu maximum\n\t%lu no memory\n",
|
|
ipsp->iss_max, ipsp->iss_nomem);
|
|
PRINTF("\t%lu active\n\t%lu expired\n\t%lu closed\n",
|
|
ipsp->iss_active, ipsp->iss_expire, ipsp->iss_fin);
|
|
if (kmemcpy((char *)istab, (u_long)ipsp->iss_table, sizeof(istab)))
|
|
return;
|
|
for (i = 0; i < IPSTATE_SIZE; i++)
|
|
while (istab[i]) {
|
|
if (kmemcpy(&ips, istab[i], sizeof(ips)) == -1)
|
|
break;
|
|
PRINTF("%s -> ", inet_ntoa(ips.is_src));
|
|
PRINTF("%s age %d pass %d pr %d state %d/%d\n",
|
|
inet_ntoa(ips.is_dst), ips.is_age,
|
|
ips.is_pass, ips.is_p, ips.is_state[0],
|
|
ips.is_state[1]);
|
|
if (ips.is_p == IPPROTO_TCP)
|
|
PRINTF("\t%hu -> %hu %lu:%lu %hu:%hu\n",
|
|
ntohs(ips.is_sport),
|
|
ntohs(ips.is_dport),
|
|
ips.is_seq, ips.is_ack,
|
|
ips.is_swin, ips.is_dwin);
|
|
else if (ips.is_p == IPPROTO_UDP)
|
|
PRINTF("\t%hu -> %hu\n", ntohs(ips.is_sport),
|
|
ntohs(ips.is_dport));
|
|
else if (ips.is_p == IPPROTO_ICMP)
|
|
PRINTF("\t%hu %hu %d\n", ips.is_icmp.ics_id,
|
|
ips.is_icmp.ics_seq,
|
|
ips.is_icmp.ics_type);
|
|
istab[i] = ips.is_next;
|
|
}
|
|
}
|
|
|
|
|
|
static void showfrstates(fd, ifsp)
|
|
int fd;
|
|
ipfrstat_t *ifsp;
|
|
{
|
|
struct ipfr *ipfrtab[IPFT_SIZE], ifr;
|
|
int i;
|
|
|
|
PRINTF("IP fragment states:\n\t%lu new\n\t%lu expired\n\t%lu hits\n",
|
|
ifsp->ifs_new, ifsp->ifs_expire, ifsp->ifs_hits);
|
|
PRINTF("\t%lu no memory\n\t%lu already exist\n",
|
|
ifsp->ifs_nomem, ifsp->ifs_exists);
|
|
PRINTF("\t%lu inuse\n", ifsp->ifs_inuse);
|
|
if (kmemcpy((char *)ipfrtab, (u_long)ifsp->ifs_table, sizeof(ipfrtab)))
|
|
return;
|
|
for (i = 0; i < IPFT_SIZE; i++)
|
|
while (ipfrtab[i]) {
|
|
if (kmemcpy(&ifr, (u_long)ipfrtab[i],
|
|
sizeof(ifr)) == -1)
|
|
break;
|
|
PRINTF("%s -> ", inet_ntoa(ifr.ipfr_src));
|
|
PRINTF("%s %d %d %d %#02x = %#x\n",
|
|
inet_ntoa(ifr.ipfr_dst), ifr.ipfr_id,
|
|
ifr.ipfr_ttl, ifr.ipfr_p, ifr.ipfr_tos,
|
|
ifr.ipfr_pass);
|
|
ipfrtab[i] = ifr.ipfr_next;
|
|
}
|
|
}
|