dmake: do not set MAKEFLAGS=k
[unleashed/tickless.git] / usr / src / cmd / ipf / tools / ipfstat.c
blobaab11c01a0cefb25c96335e2e91dedff64f205f5
1 /*
2 * Copyright (C) 1993-2001, 2003 by Darren Reed.
4 * See the IPFILTER.LICENCE file for details on licencing.
6 * Copyright 2008 Sun Microsystems, Inc. All rights reserved.
7 * Use is subject to license terms.
9 * Copyright (c) 2014, Joyent, Inc. All rights reserved.
12 #ifdef __FreeBSD__
13 # ifndef __FreeBSD_cc_version
14 # include <osreldate.h>
15 # else
16 # if __FreeBSD_cc_version < 430000
17 # include <osreldate.h>
18 # endif
19 # endif
20 #endif
21 #include <sys/ioctl.h>
22 #include <fcntl.h>
23 #ifdef linux
24 # include <linux/a.out.h>
25 #else
26 # include <nlist.h>
27 #endif
28 #include <ctype.h>
29 #if defined(sun) && (defined(__svr4__) || defined(__SVR4))
30 # include <stddef.h>
31 #endif
32 #include "ipf.h"
33 #include "netinet/ipl.h"
34 #if defined(STATETOP)
35 # if defined(_BSDI_VERSION)
36 # undef STATETOP
37 # endif
38 # if defined(__FreeBSD__) && \
39 (!defined(__FreeBSD_version) || (__FreeBSD_version < 430000))
40 # undef STATETOP
41 # endif
42 # if defined(__NetBSD_Version__) && (__NetBSD_Version__ < 105000000)
43 # undef STATETOP
44 # endif
45 # if defined(sun)
46 # if defined(__svr4__) || defined(__SVR4)
47 # include <sys/select.h>
48 # else
49 # undef STATETOP /* NOT supported on SunOS4 */
50 # endif
51 # endif
52 #endif
53 #if defined(STATETOP) && !defined(linux)
54 # include <netinet/ip_var.h>
55 # include <netinet/tcp_fsm.h>
56 #endif
57 #ifdef STATETOP
58 # include <ctype.h>
59 # include <signal.h>
60 # if defined(SOLARIS) || defined(__NetBSD__) || defined(_BSDI_VERSION) || \
61 defined(__sgi)
62 # ifdef ERR
63 # undef ERR
64 # endif
65 # undef ISASCII
66 # undef ISPRINT
67 # include <curses.h>
68 # else /* SOLARIS */
69 # include <curses.h>
70 # endif /* SOLARIS */
71 #endif /* STATETOP */
72 #include "kmem.h"
73 #if defined(__NetBSD__) || (__OpenBSD__)
74 # include <paths.h>
75 #endif
76 #include "ipfzone.h"
78 static const char sccsid[] = "@(#)fils.c 1.21 4/20/96 (C) 1993-2000 Darren Reed";
79 static const char rcsid[] = "@(#)$Id: ipfstat.c,v 1.44.2.12 2005/06/12 07:18:46 darrenr Exp $";
81 #ifdef __hpux
82 # define nlist nlist64
83 #endif
85 extern char *optarg;
86 extern int optind;
87 extern int opterr;
89 #define PRINTF (void)printf
90 #define FPRINTF (void)fprintf
91 static char *filters[4] = { "ipfilter(in)", "ipfilter(out)",
92 "ipacct(in)", "ipacct(out)" };
93 static int state_logging = -1;
95 int opts = 0;
96 int use_inet6 = 0;
97 int live_kernel = 1;
98 int state_fd = -1;
99 int ipf_fd = -1;
101 #ifdef STATETOP
102 #define STSTRSIZE 80
103 #define STGROWSIZE 16
104 #define HOSTNMLEN 40
106 #define STSORT_PR 0
107 #define STSORT_PKTS 1
108 #define STSORT_BYTES 2
109 #define STSORT_TTL 3
110 #define STSORT_SRCIP 4
111 #define STSORT_SRCPT 5
112 #define STSORT_DSTIP 6
113 #define STSORT_DSTPT 7
114 #define STSORT_MAX STSORT_DSTPT
115 #define STSORT_DEFAULT STSORT_BYTES
118 typedef struct statetop {
119 i6addr_t st_src;
120 i6addr_t st_dst;
121 u_short st_sport;
122 u_short st_dport;
123 u_char st_p;
124 u_char st_v;
125 u_char st_state[2];
126 U_QUAD_T st_pkts;
127 U_QUAD_T st_bytes;
128 u_long st_age;
129 } statetop_t;
130 #endif
132 int main __P((int, char *[]));
134 static void showstats __P((friostat_t *, u_32_t));
135 static void showfrstates __P((ipfrstat_t *, u_long));
136 static void showlist __P((friostat_t *));
137 static void showipstates __P((ips_stat_t *));
138 static void showauthstates __P((fr_authstat_t *));
139 static void showgroups __P((friostat_t *));
140 static void usage __P((char *));
141 static void printlivelist __P((int, int, frentry_t *, char *, char *));
142 static void printdeadlist __P((int, int, frentry_t *, char *, char *));
143 static void printlist __P((frentry_t *, char *));
144 static void parse_ipportstr __P((const char *, i6addr_t *, int *));
145 static void ipfstate_live __P((char *, friostat_t **, ips_stat_t **,
146 ipfrstat_t **, fr_authstat_t **, u_32_t *));
147 static void ipfstate_dead __P((char *, friostat_t **, ips_stat_t **,
148 ipfrstat_t **, fr_authstat_t **, u_32_t *));
149 #ifdef STATETOP
150 static void topipstates __P((i6addr_t, i6addr_t, int, int, int,
151 int, int, int));
152 static void sig_break __P((int));
153 static void sig_resize __P((int));
154 static char *getip __P((int, i6addr_t *));
155 static char *ttl_to_string __P((long));
156 static int sort_p __P((const void *, const void *));
157 static int sort_pkts __P((const void *, const void *));
158 static int sort_bytes __P((const void *, const void *));
159 static int sort_ttl __P((const void *, const void *));
160 static int sort_srcip __P((const void *, const void *));
161 static int sort_srcpt __P((const void *, const void *));
162 static int sort_dstip __P((const void *, const void *));
163 static int sort_dstpt __P((const void *, const void *));
164 #endif
167 static void usage(name)
168 char *name;
170 #ifdef USE_INET6
171 fprintf(stderr, "Usage: %s [-6aAdfghIilnoRsv]\n", name);
172 #else
173 fprintf(stderr, "Usage: %s [-aAdfghIilnoRsv]\n", name);
174 #endif
175 fprintf(stderr, " %s [-M corefile] [-N symbol-list]\n", name);
176 #ifdef USE_INET6
177 fprintf(stderr, " %s -t [-6C] ", name);
178 #else
179 fprintf(stderr, " %s -t [-C] ", name);
180 #endif
181 fprintf(stderr, "[-G|-z zonename] ");
182 fprintf(stderr, "[-D destination address] [-P protocol] [-S source address] [-T refresh time]\n");
183 exit(1);
187 int main(argc,argv)
188 int argc;
189 char *argv[];
191 fr_authstat_t frauthst;
192 fr_authstat_t *frauthstp = &frauthst;
193 friostat_t fio;
194 friostat_t *fiop = &fio;
195 ips_stat_t ipsst;
196 ips_stat_t *ipsstp = &ipsst;
197 ipfrstat_t ifrst;
198 ipfrstat_t *ifrstp = &ifrst;
199 char *device = IPL_NAME, *memf = NULL;
200 char *options, *kern = NULL;
201 int c, myoptind;
203 int protocol = -1; /* -1 = wild card for any protocol */
204 int refreshtime = 1; /* default update time */
205 int sport = -1; /* -1 = wild card for any source port */
206 int dport = -1; /* -1 = wild card for any dest port */
207 int topclosed = 0; /* do not show closed tcp sessions */
208 i6addr_t saddr, daddr;
209 u_32_t frf;
211 #ifdef USE_INET6
212 options = "6aACdfgG:hIilnostvD:M:N:P:RS:T:z:";
213 #else
214 options = "aACdfgG:hIilnostvD:M:N:P:RS:T:z:";
215 #endif
217 saddr.in4.s_addr = INADDR_ANY; /* default any v4 source addr */
218 daddr.in4.s_addr = INADDR_ANY; /* default any v4 dest addr */
219 #ifdef USE_INET6
220 saddr.in6 = in6addr_any; /* default any v6 source addr */
221 daddr.in6 = in6addr_any; /* default any v6 dest addr */
222 #endif
224 /* Don't warn about invalid flags when we run getopt for the 1st time */
225 opterr = 0;
228 * Parse these four arguments now lest there be any buffer overflows
229 * in the parsing of the rest.
231 myoptind = optind;
232 while ((c = getopt(argc, argv, options)) != -1) {
233 switch (c)
235 case 'G' :
236 setzonename_global(optarg);
237 break;
238 case 'M' :
239 memf = optarg;
240 live_kernel = 0;
241 break;
242 case 'N' :
243 kern = optarg;
244 live_kernel = 0;
245 break;
246 case 'z' :
247 setzonename(optarg);
248 break;
251 optind = myoptind;
253 if (live_kernel == 1) {
254 if ((state_fd = open(IPSTATE_NAME, O_RDONLY)) == -1) {
255 perror("open(IPSTATE_NAME)");
256 exit(-1);
259 if (setzone(state_fd) != 0) {
260 close(state_fd);
261 exit(-1);
264 if ((ipf_fd = open(device, O_RDONLY)) == -1) {
265 fprintf(stderr, "open(%s)", device);
266 perror("");
267 exit(-1);
270 if (setzone(ipf_fd) != 0) {
271 close(ipf_fd);
272 exit(-1);
276 if (kern != NULL || memf != NULL) {
277 (void)setgid(getgid());
278 (void)setreuid(getuid(), getuid());
279 if (openkmem(kern, memf) == -1)
280 exit(-1);
283 if (live_kernel == 1)
284 (void) checkrev(device);
285 (void)setgid(getgid());
286 (void)setreuid(getuid(), getuid());
288 opterr = 1;
290 while ((c = getopt(argc, argv, options)) != -1)
292 switch (c)
294 #ifdef USE_INET6
295 case '6' :
296 use_inet6 = 1;
297 break;
298 #endif
299 case 'a' :
300 opts |= OPT_ACCNT|OPT_SHOWLIST;
301 break;
302 case 'A' :
303 opts |= OPT_AUTHSTATS;
304 break;
305 case 'C' :
306 topclosed = 1;
307 break;
308 case 'd' :
309 opts |= OPT_DEBUG;
310 break;
311 case 'D' :
312 parse_ipportstr(optarg, &daddr, &dport);
313 break;
314 case 'f' :
315 opts |= OPT_FRSTATES;
316 break;
317 case 'g' :
318 opts |= OPT_GROUPS;
319 break;
320 case 'G' :
321 /* Already handled by getzoneopt() above */
322 break;
323 case 'h' :
324 opts |= OPT_HITS;
325 break;
326 case 'i' :
327 opts |= OPT_INQUE|OPT_SHOWLIST;
328 break;
329 case 'I' :
330 opts |= OPT_INACTIVE;
331 break;
332 case 'l' :
333 opts |= OPT_SHOWLIST;
334 break;
335 case 'M' :
336 break;
337 case 'N' :
338 break;
339 case 'n' :
340 opts |= OPT_SHOWLINENO;
341 break;
342 case 'o' :
343 opts |= OPT_OUTQUE|OPT_SHOWLIST;
344 break;
345 case 'P' :
346 protocol = getproto(optarg);
347 if (protocol == -1) {
348 fprintf(stderr, "%s: Invalid protocol: %s\n",
349 argv[0], optarg);
350 exit(-2);
352 break;
353 case 'R' :
354 opts |= OPT_NORESOLVE;
355 break;
356 case 's' :
357 opts |= OPT_IPSTATES;
358 break;
359 case 'S' :
360 parse_ipportstr(optarg, &saddr, &sport);
361 break;
362 case 't' :
363 #ifdef STATETOP
364 opts |= OPT_STATETOP;
365 break;
366 #else
367 fprintf(stderr,
368 "%s: state top facility not compiled in\n",
369 argv[0]);
370 exit(-2);
371 #endif
372 case 'T' :
373 if (!sscanf(optarg, "%d", &refreshtime) ||
374 (refreshtime <= 0)) {
375 fprintf(stderr,
376 "%s: Invalid refreshtime < 1 : %s\n",
377 argv[0], optarg);
378 exit(-2);
380 break;
381 case 'v' :
382 opts |= OPT_VERBOSE;
383 opts |= OPT_UNDEF;
384 break;
385 case 'z' :
386 /* Already handled by getzoneopt() above */
387 break;
388 default :
389 usage(argv[0]);
390 break;
394 if (live_kernel == 1) {
395 bzero((char *)&fio, sizeof(fio));
396 bzero((char *)&ipsst, sizeof(ipsst));
397 bzero((char *)&ifrst, sizeof(ifrst));
399 ipfstate_live(device, &fiop, &ipsstp, &ifrstp,
400 &frauthstp, &frf);
401 } else
402 ipfstate_dead(kern, &fiop, &ipsstp, &ifrstp, &frauthstp, &frf);
404 if (opts & OPT_IPSTATES) {
405 showipstates(ipsstp);
406 } else if (opts & OPT_SHOWLIST) {
407 showlist(fiop);
408 if ((opts & OPT_OUTQUE) && (opts & OPT_INQUE)){
409 opts &= ~OPT_OUTQUE;
410 showlist(fiop);
412 } else if (opts & OPT_FRSTATES)
413 showfrstates(ifrstp, fiop->f_ticks);
414 #ifdef STATETOP
415 else if (opts & OPT_STATETOP)
416 topipstates(saddr, daddr, sport, dport, protocol,
417 use_inet6 ? 6 : 4, refreshtime, topclosed);
418 #endif
419 else if (opts & OPT_AUTHSTATS)
420 showauthstates(frauthstp);
421 else if (opts & OPT_GROUPS)
422 showgroups(fiop);
423 else
424 showstats(fiop, frf);
426 return 0;
431 * Fill in the stats structures from the live kernel, using a combination
432 * of ioctl's and copying directly from kernel memory.
434 static void ipfstate_live(device, fiopp, ipsstpp, ifrstpp, frauthstpp, frfp)
435 char *device;
436 friostat_t **fiopp;
437 ips_stat_t **ipsstpp;
438 ipfrstat_t **ifrstpp;
439 fr_authstat_t **frauthstpp;
440 u_32_t *frfp;
442 ipfobj_t ipfo;
444 if (checkrev(device) == -1) {
445 fprintf(stderr, "User/kernel version check failed\n");
446 exit(1);
449 if ((opts & OPT_AUTHSTATS) == 0) {
450 bzero((caddr_t)&ipfo, sizeof(ipfo));
451 ipfo.ipfo_rev = IPFILTER_VERSION;
452 ipfo.ipfo_size = sizeof(friostat_t);
453 ipfo.ipfo_ptr = (void *)*fiopp;
454 ipfo.ipfo_type = IPFOBJ_IPFSTAT;
456 if (ioctl(ipf_fd, SIOCGETFS, &ipfo) == -1) {
457 perror("ioctl(ipf:SIOCGETFS)");
458 exit(-1);
461 if (ioctl(ipf_fd, SIOCGETFF, frfp) == -1)
462 perror("ioctl(SIOCGETFF)");
465 if ((opts & OPT_IPSTATES) != 0) {
467 bzero((caddr_t)&ipfo, sizeof(ipfo));
468 ipfo.ipfo_rev = IPFILTER_VERSION;
469 ipfo.ipfo_size = sizeof(ips_stat_t);
470 ipfo.ipfo_ptr = (void *)*ipsstpp;
471 ipfo.ipfo_type = IPFOBJ_STATESTAT;
473 if ((ioctl(state_fd, SIOCGETFS, &ipfo) == -1)) {
474 perror("ioctl(state:SIOCGETFS)");
475 exit(-1);
477 if (ioctl(state_fd, SIOCGETLG, &state_logging) == -1) {
478 perror("ioctl(state:SIOCGETLG)");
479 exit(-1);
483 if ((opts & OPT_FRSTATES) != 0) {
484 bzero((caddr_t)&ipfo, sizeof(ipfo));
485 ipfo.ipfo_rev = IPFILTER_VERSION;
486 ipfo.ipfo_size = sizeof(ipfrstat_t);
487 ipfo.ipfo_ptr = (void *)*ifrstpp;
488 ipfo.ipfo_type = IPFOBJ_FRAGSTAT;
490 if (ioctl(ipf_fd, SIOCGFRST, &ipfo) == -1) {
491 perror("ioctl(SIOCGFRST)");
492 exit(-1);
496 if (opts & OPT_VERBOSE)
497 PRINTF("opts %#x name %s\n", opts, device);
499 if ((opts & OPT_AUTHSTATS) != 0) {
500 if (ipf_fd >= 0) {
501 close(ipf_fd);
502 ipf_fd = -1;
504 device = IPAUTH_NAME;
505 if ((ipf_fd = open(device, O_RDONLY)) == -1) {
506 perror("open");
507 exit(-1);
510 if (setzone(ipf_fd) != 0) {
511 close(ipf_fd);
512 exit(-1);
515 bzero((caddr_t)&ipfo, sizeof(ipfo));
516 ipfo.ipfo_rev = IPFILTER_VERSION;
517 ipfo.ipfo_size = sizeof(fr_authstat_t);
518 ipfo.ipfo_ptr = (void *)*frauthstpp;
519 ipfo.ipfo_type = IPFOBJ_AUTHSTAT;
521 if (ioctl(ipf_fd, SIOCATHST, &ipfo) == -1) {
522 perror("ioctl(SIOCATHST)");
523 exit(-1);
530 * Build up the stats structures from data held in the "core" memory.
531 * This is mainly useful when looking at data in crash dumps and ioctl's
532 * just won't work any more.
534 static void ipfstate_dead(kernel, fiopp, ipsstpp, ifrstpp, frauthstpp, frfp)
535 char *kernel;
536 friostat_t **fiopp;
537 ips_stat_t **ipsstpp;
538 ipfrstat_t **ifrstpp;
539 fr_authstat_t **frauthstpp;
540 u_32_t *frfp;
542 static fr_authstat_t frauthst, *frauthstp;
543 static ips_stat_t ipsst, *ipsstp;
544 static ipfrstat_t ifrst, *ifrstp;
545 static friostat_t fio, *fiop;
546 int temp;
548 void *rules[2][2];
549 struct nlist deadlist[43] = {
550 { "fr_authstats" }, /* 0 */
551 { "fae_list" },
552 { "ipauth" },
553 { "fr_authlist" },
554 { "fr_authstart" },
555 { "fr_authend" }, /* 5 */
556 { "fr_authnext" },
557 { "fr_auth" },
558 { "fr_authused" },
559 { "fr_authsize" },
560 { "fr_defaultauthage" }, /* 10 */
561 { "fr_authpkts" },
562 { "fr_auth_lock" },
563 { "frstats" },
564 { "ips_stats" },
565 { "ips_num" }, /* 15 */
566 { "ips_wild" },
567 { "ips_list" },
568 { "ips_table" },
569 { "fr_statemax" },
570 { "fr_statesize" }, /* 20 */
571 { "fr_state_doflush" },
572 { "fr_state_lock" },
573 { "ipfr_heads" },
574 { "ipfr_nattab" },
575 { "ipfr_stats" }, /* 25 */
576 { "ipfr_inuse" },
577 { "fr_ipfrttl" },
578 { "fr_frag_lock" },
579 { "ipfr_timer_id" },
580 { "fr_nat_lock" }, /* 30 */
581 { "ipfilter" },
582 { "ipfilter6" },
583 { "ipacct" },
584 { "ipacct6" },
585 { "ipl_frouteok" }, /* 35 */
586 { "fr_running" },
587 { "ipfgroups" },
588 { "fr_active" },
589 { "fr_pass" },
590 { "fr_flags" }, /* 40 */
591 { "ipstate_logging" },
592 { NULL }
596 frauthstp = &frauthst;
597 ipsstp = &ipsst;
598 ifrstp = &ifrst;
599 fiop = &fio;
601 *frfp = 0;
602 *fiopp = fiop;
603 *ipsstpp = ipsstp;
604 *ifrstpp = ifrstp;
605 *frauthstpp = frauthstp;
607 bzero((char *)fiop, sizeof(*fiop));
608 bzero((char *)ipsstp, sizeof(*ipsstp));
609 bzero((char *)ifrstp, sizeof(*ifrstp));
610 bzero((char *)frauthstp, sizeof(*frauthstp));
612 if (nlist(kernel, deadlist) == -1) {
613 fprintf(stderr, "nlist error\n");
614 return;
618 * This is for SIOCGETFF.
620 kmemcpy((char *)frfp, (u_long)deadlist[40].n_value, sizeof(*frfp));
623 * f_locks is a combination of the lock variable from each part of
624 * ipfilter (state, auth, nat, fragments).
626 kmemcpy((char *)fiop, (u_long)deadlist[13].n_value, sizeof(*fiop));
627 kmemcpy((char *)&fiop->f_locks[0], (u_long)deadlist[22].n_value,
628 sizeof(fiop->f_locks[0]));
629 kmemcpy((char *)&fiop->f_locks[0], (u_long)deadlist[30].n_value,
630 sizeof(fiop->f_locks[1]));
631 kmemcpy((char *)&fiop->f_locks[2], (u_long)deadlist[28].n_value,
632 sizeof(fiop->f_locks[2]));
633 kmemcpy((char *)&fiop->f_locks[3], (u_long)deadlist[12].n_value,
634 sizeof(fiop->f_locks[3]));
637 * Get pointers to each list of rules (active, inactive, in, out)
639 kmemcpy((char *)&rules, (u_long)deadlist[31].n_value, sizeof(rules));
640 fiop->f_fin[0] = rules[0][0];
641 fiop->f_fin[1] = rules[0][1];
642 fiop->f_fout[0] = rules[1][0];
643 fiop->f_fout[1] = rules[1][1];
646 * Same for IPv6, except make them null if support for it is not
647 * being compiled in.
649 #ifdef USE_INET6
650 kmemcpy((char *)&rules, (u_long)deadlist[32].n_value, sizeof(rules));
651 fiop->f_fin6[0] = rules[0][0];
652 fiop->f_fin6[1] = rules[0][1];
653 fiop->f_fout6[0] = rules[1][0];
654 fiop->f_fout6[1] = rules[1][1];
655 #else
656 fiop->f_fin6[0] = NULL;
657 fiop->f_fin6[1] = NULL;
658 fiop->f_fout6[0] = NULL;
659 fiop->f_fout6[1] = NULL;
660 #endif
663 * Now get accounting rules pointers.
665 kmemcpy((char *)&rules, (u_long)deadlist[33].n_value, sizeof(rules));
666 fiop->f_acctin[0] = rules[0][0];
667 fiop->f_acctin[1] = rules[0][1];
668 fiop->f_acctout[0] = rules[1][0];
669 fiop->f_acctout[1] = rules[1][1];
671 #ifdef USE_INET6
672 kmemcpy((char *)&rules, (u_long)deadlist[34].n_value, sizeof(rules));
673 fiop->f_acctin6[0] = rules[0][0];
674 fiop->f_acctin6[1] = rules[0][1];
675 fiop->f_acctout6[0] = rules[1][0];
676 fiop->f_acctout6[1] = rules[1][1];
677 #else
678 fiop->f_acctin6[0] = NULL;
679 fiop->f_acctin6[1] = NULL;
680 fiop->f_acctout6[0] = NULL;
681 fiop->f_acctout6[1] = NULL;
682 #endif
685 * A collection of "global" variables used inside the kernel which
686 * are all collected in friostat_t via ioctl.
688 kmemcpy((char *)&fiop->f_froute, (u_long)deadlist[35].n_value,
689 sizeof(fiop->f_froute));
690 kmemcpy((char *)&fiop->f_running, (u_long)deadlist[36].n_value,
691 sizeof(fiop->f_running));
692 kmemcpy((char *)&fiop->f_groups, (u_long)deadlist[37].n_value,
693 sizeof(fiop->f_groups));
694 kmemcpy((char *)&fiop->f_active, (u_long)deadlist[38].n_value,
695 sizeof(fiop->f_active));
696 kmemcpy((char *)&fiop->f_defpass, (u_long)deadlist[39].n_value,
697 sizeof(fiop->f_defpass));
700 * Build up the state information stats structure.
702 kmemcpy((char *)ipsstp, (u_long)deadlist[14].n_value, sizeof(*ipsstp));
703 kmemcpy((char *)&temp, (u_long)deadlist[15].n_value, sizeof(temp));
704 ipsstp->iss_active = temp;
705 ipsstp->iss_table = (void *)deadlist[18].n_value;
706 ipsstp->iss_list = (void *)deadlist[17].n_value;
709 * Build up the authentiation information stats structure.
711 kmemcpy((char *)frauthstp, (u_long)deadlist[0].n_value,
712 sizeof(*frauthstp));
713 frauthstp->fas_faelist = (void *)deadlist[1].n_value;
716 * Build up the fragment information stats structure.
718 kmemcpy((char *)ifrstp, (u_long)deadlist[25].n_value,
719 sizeof(*ifrstp));
720 ifrstp->ifs_table = (void *)deadlist[23].n_value;
721 ifrstp->ifs_nattab = (void *)deadlist[24].n_value;
722 kmemcpy((char *)&ifrstp->ifs_inuse, (u_long)deadlist[26].n_value,
723 sizeof(ifrstp->ifs_inuse));
726 * Get logging on/off switches
728 kmemcpy((char *)&state_logging, (u_long)deadlist[41].n_value,
729 sizeof(state_logging));
734 * Display the kernel stats for packets blocked and passed and other
735 * associated running totals which are kept.
737 static void showstats(fp, frf)
738 struct friostat *fp;
739 u_32_t frf;
742 PRINTF("bad packets:\t\tin %lu\tout %lu\n",
743 fp->f_st[0].fr_bad, fp->f_st[1].fr_bad);
744 #ifdef USE_INET6
745 PRINTF(" IPv6 packets:\t\tin %lu out %lu\n",
746 fp->f_st[0].fr_ipv6, fp->f_st[1].fr_ipv6);
747 #endif
748 PRINTF(" input packets:\t\tblocked %lu passed %lu nomatch %lu",
749 fp->f_st[0].fr_block, fp->f_st[0].fr_pass,
750 fp->f_st[0].fr_nom);
751 PRINTF(" counted %lu short %lu\n",
752 fp->f_st[0].fr_acct, fp->f_st[0].fr_short);
753 PRINTF("output packets:\t\tblocked %lu passed %lu nomatch %lu",
754 fp->f_st[1].fr_block, fp->f_st[1].fr_pass,
755 fp->f_st[1].fr_nom);
756 PRINTF(" counted %lu short %lu\n",
757 fp->f_st[1].fr_acct, fp->f_st[1].fr_short);
758 PRINTF(" input packets logged:\tblocked %lu passed %lu\n",
759 fp->f_st[0].fr_bpkl, fp->f_st[0].fr_ppkl);
760 PRINTF("output packets logged:\tblocked %lu passed %lu\n",
761 fp->f_st[1].fr_bpkl, fp->f_st[1].fr_ppkl);
762 PRINTF(" packets logged:\tinput %lu output %lu\n",
763 fp->f_st[0].fr_pkl, fp->f_st[1].fr_pkl);
764 PRINTF(" log failures:\t\tinput %lu output %lu\n",
765 fp->f_st[0].fr_skip, fp->f_st[1].fr_skip);
766 PRINTF("fragment state(in):\tkept %lu\tlost %lu\tnot fragmented %lu\n",
767 fp->f_st[0].fr_nfr, fp->f_st[0].fr_bnfr,
768 fp->f_st[0].fr_cfr);
769 PRINTF("fragment state(out):\tkept %lu\tlost %lu\tnot fragmented %lu\n",
770 fp->f_st[1].fr_nfr, fp->f_st[1].fr_bnfr,
771 fp->f_st[0].fr_cfr);
772 PRINTF("packet state(in):\tkept %lu\tlost %lu\n",
773 fp->f_st[0].fr_ads, fp->f_st[0].fr_bads);
774 PRINTF("packet state(out):\tkept %lu\tlost %lu\n",
775 fp->f_st[1].fr_ads, fp->f_st[1].fr_bads);
776 PRINTF("ICMP replies:\t%lu\tTCP RSTs sent:\t%lu\n",
777 fp->f_st[0].fr_ret, fp->f_st[1].fr_ret);
778 PRINTF("Invalid source(in):\t%lu\n", fp->f_st[0].fr_badsrc);
779 PRINTF("Result cache hits(in):\t%lu\t(out):\t%lu\n",
780 fp->f_st[0].fr_chit, fp->f_st[1].fr_chit);
781 PRINTF("IN Pullups succeeded:\t%lu\tfailed:\t%lu\n",
782 fp->f_st[0].fr_pull[0], fp->f_st[0].fr_pull[1]);
783 PRINTF("OUT Pullups succeeded:\t%lu\tfailed:\t%lu\n",
784 fp->f_st[1].fr_pull[0], fp->f_st[1].fr_pull[1]);
785 PRINTF("Fastroute successes:\t%lu\tfailures:\t%lu\n",
786 fp->f_froute[0], fp->f_froute[1]);
787 PRINTF("TCP cksum fails(in):\t%lu\t(out):\t%lu\n",
788 fp->f_st[0].fr_tcpbad, fp->f_st[1].fr_tcpbad);
789 PRINTF("IPF Ticks:\t%lu\n", fp->f_ticks);
791 PRINTF("Packet log flags set: (%#x)\n", frf);
792 if (frf & FF_LOGPASS)
793 PRINTF("\tpackets passed through filter\n");
794 if (frf & FF_LOGBLOCK)
795 PRINTF("\tpackets blocked by filter\n");
796 if (frf & FF_LOGNOMATCH)
797 PRINTF("\tpackets not matched by filter\n");
798 if (!frf)
799 PRINTF("\tnone\n");
804 * Print out a list of rules from the kernel, starting at the one passed.
806 static void printlivelist(out, set, fp, group, comment)
807 int out, set;
808 frentry_t *fp;
809 char *group, *comment;
811 frgroup_t *grtop, *grtail, *g;
812 struct frentry fb, *fg;
813 int n;
814 ipfruleiter_t rule;
815 ipfobj_t obj;
817 fb.fr_next = fp;
818 n = 0;
820 grtop = NULL;
821 grtail = NULL;
822 rule.iri_ver = use_inet6? AF_INET6 : AF_INET;
823 rule.iri_inout = out;
824 rule.iri_active = set;
825 rule.iri_nrules = 1;
826 rule.iri_rule = &fb;
827 if (group != NULL)
828 strncpy(rule.iri_group, group, FR_GROUPLEN);
829 else
830 rule.iri_group[0] = '\0';
832 bzero((char *)&obj, sizeof(obj));
833 obj.ipfo_rev = IPFILTER_VERSION;
834 obj.ipfo_type = IPFOBJ_IPFITER;
835 obj.ipfo_size = sizeof(rule);
836 obj.ipfo_ptr = &rule;
838 do {
839 u_long array[1000];
841 memset(array, 0xff, sizeof(array));
842 fp = (frentry_t *)array;
843 rule.iri_rule = fp;
844 if (ioctl(ipf_fd, SIOCIPFITER, &obj) == -1) {
845 perror("ioctl(SIOCIPFITER)");
846 return;
848 if (fp->fr_data != NULL)
849 fp->fr_data = (char *)fp + sizeof(*fp);
851 n++;
853 if (opts & (OPT_HITS|OPT_VERBOSE))
854 #ifdef USE_QUAD_T
855 PRINTF("%qu ", (unsigned long long) fp->fr_hits);
856 #else
857 PRINTF("%lu ", fp->fr_hits);
858 #endif
859 if (opts & (OPT_ACCNT|OPT_VERBOSE))
860 #ifdef USE_QUAD_T
861 PRINTF("%qu ", (unsigned long long) fp->fr_bytes);
862 #else
863 PRINTF("%lu ", fp->fr_bytes);
864 #endif
865 if (opts & OPT_SHOWLINENO)
866 PRINTF("@%d ", n);
868 printfr(fp, ioctl);
869 if (opts & OPT_DEBUG) {
870 binprint(fp, sizeof(*fp));
871 if (fp->fr_data != NULL && fp->fr_dsize > 0)
872 binprint(fp->fr_data, fp->fr_dsize);
875 if (fp->fr_grhead[0] != '\0') {
876 g = calloc(1, sizeof(*g));
878 if (g != NULL) {
879 strncpy(g->fg_name, fp->fr_grhead,
880 FR_GROUPLEN);
881 if (grtop == NULL) {
882 grtop = g;
883 grtail = g;
884 } else {
885 grtail->fg_next = g;
886 grtail = g;
890 } while (fp->fr_next != NULL);
892 while ((g = grtop) != NULL) {
893 printlivelist(out, set, NULL, g->fg_name, comment);
894 grtop = g->fg_next;
895 free(g);
900 static void printdeadlist(out, set, fp, group, comment)
901 int out, set;
902 frentry_t *fp;
903 char *group, *comment;
905 frgroup_t *grtop, *grtail, *g;
906 struct frentry fb, *fg;
907 char *data;
908 u_32_t type;
909 int n;
911 fb.fr_next = fp;
912 n = 0;
913 grtop = NULL;
914 grtail = NULL;
916 do {
917 fp = fb.fr_next;
918 if (kmemcpy((char *)&fb, (u_long)fb.fr_next,
919 sizeof(fb)) == -1) {
920 perror("kmemcpy");
921 return;
924 data = NULL;
925 type = fb.fr_type & ~FR_T_BUILTIN;
926 if (type == FR_T_IPF || type == FR_T_BPFOPC) {
927 if (fb.fr_dsize) {
928 data = malloc(fb.fr_dsize);
930 if (kmemcpy(data, (u_long)fb.fr_data,
931 fb.fr_dsize) == -1) {
932 perror("kmemcpy");
933 return;
935 fb.fr_data = data;
939 n++;
941 if (opts & (OPT_HITS|OPT_VERBOSE))
942 #ifdef USE_QUAD_T
943 PRINTF("%qu ", (unsigned long long) fb.fr_hits);
944 #else
945 PRINTF("%lu ", fb.fr_hits);
946 #endif
947 if (opts & (OPT_ACCNT|OPT_VERBOSE))
948 #ifdef USE_QUAD_T
949 PRINTF("%qu ", (unsigned long long) fb.fr_bytes);
950 #else
951 PRINTF("%lu ", fb.fr_bytes);
952 #endif
953 if (opts & OPT_SHOWLINENO)
954 PRINTF("@%d ", n);
956 printfr(fp, ioctl);
957 if (opts & OPT_DEBUG) {
958 binprint(fp, sizeof(*fp));
959 if (fb.fr_data != NULL && fb.fr_dsize > 0)
960 binprint(fb.fr_data, fb.fr_dsize);
962 free(data);
963 if (fb.fr_grhead[0] != '\0') {
964 g = calloc(1, sizeof(*g));
966 if (g != NULL) {
967 strncpy(g->fg_name, fb.fr_grhead,
968 FR_GROUPLEN);
969 if (grtop == NULL) {
970 grtop = g;
971 grtail = g;
972 } else {
973 grtail->fg_next = g;
974 grtail = g;
978 if (type == FR_T_CALLFUNC) {
979 printdeadlist(out, set, fb.fr_data, group,
980 "# callfunc: ");
982 } while (fb.fr_next != NULL);
984 while ((g = grtop) != NULL) {
985 printdeadlist(out, set, NULL, g->fg_name, comment);
986 grtop = g->fg_next;
987 free(g);
993 * print out all of the asked for rule sets, using the stats struct as
994 * the base from which to get the pointers.
996 static void showlist(fiop)
997 struct friostat *fiop;
999 struct frentry *fp = NULL;
1000 int i, set;
1002 set = fiop->f_active;
1003 if (opts & OPT_INACTIVE)
1004 set = 1 - set;
1005 if (opts & OPT_ACCNT) {
1006 #ifdef USE_INET6
1007 if ((use_inet6) && (opts & OPT_OUTQUE)) {
1008 i = F_ACOUT;
1009 fp = (struct frentry *)fiop->f_acctout6[set];
1010 } else if ((use_inet6) && (opts & OPT_INQUE)) {
1011 i = F_ACIN;
1012 fp = (struct frentry *)fiop->f_acctin6[set];
1013 } else
1014 #endif
1015 if (opts & OPT_OUTQUE) {
1016 i = F_ACOUT;
1017 fp = (struct frentry *)fiop->f_acctout[set];
1018 } else if (opts & OPT_INQUE) {
1019 i = F_ACIN;
1020 fp = (struct frentry *)fiop->f_acctin[set];
1021 } else {
1022 FPRINTF(stderr, "No -i or -o given with -a\n");
1023 return;
1025 } else {
1026 #ifdef USE_INET6
1027 if ((use_inet6) && (opts & OPT_OUTQUE)) {
1028 i = F_OUT;
1029 fp = (struct frentry *)fiop->f_fout6[set];
1030 } else if ((use_inet6) && (opts & OPT_INQUE)) {
1031 i = F_IN;
1032 fp = (struct frentry *)fiop->f_fin6[set];
1033 } else
1034 #endif
1035 if (opts & OPT_OUTQUE) {
1036 i = F_OUT;
1037 fp = (struct frentry *)fiop->f_fout[set];
1038 } else if (opts & OPT_INQUE) {
1039 i = F_IN;
1040 fp = (struct frentry *)fiop->f_fin[set];
1041 } else
1042 return;
1044 if (opts & OPT_VERBOSE)
1045 FPRINTF(stderr, "showlist:opts %#x i %d\n", opts, i);
1047 if (opts & OPT_VERBOSE)
1048 PRINTF("fp %p set %d\n", fp, set);
1049 if (!fp) {
1050 FPRINTF(stderr, "empty list for %s%s\n",
1051 (opts & OPT_INACTIVE) ? "inactive " : "", filters[i]);
1052 return;
1054 if (live_kernel == 1)
1055 printlivelist(i, set, fp, NULL, NULL);
1056 else
1057 printdeadlist(i, set, fp, NULL, NULL);
1062 * Display ipfilter stateful filtering information
1064 static void showipstates(ipsp)
1065 ips_stat_t *ipsp;
1067 u_long minlen, maxlen, totallen, *buckets;
1068 int i, sz;
1070 sz = sizeof(*buckets) * ipsp->iss_statesize;
1071 buckets = (u_long *)malloc(sz);
1072 if (buckets == NULL) {
1073 perror("malloc");
1074 exit(1);
1076 if (kmemcpy((char *)buckets, (u_long)ipsp->iss_bucketlen, sz)) {
1077 free(buckets);
1078 return;
1082 * If a list of states hasn't been asked for, only print out stats
1084 if (!(opts & OPT_SHOWLIST)) {
1085 PRINTF("IP states added:\n\t%lu TCP\n\t%lu UDP\n\t%lu ICMP\n",
1086 ipsp->iss_tcp, ipsp->iss_udp, ipsp->iss_icmp);
1087 PRINTF("\t%lu hits\n\t%lu misses\n",
1088 ipsp->iss_hits, ipsp->iss_miss);
1089 PRINTF("\t%lu maximum\n\t%lu no memory\n", ipsp->iss_max,
1090 ipsp->iss_nomem);
1091 PRINTF("\t%lu active\n\t%lu expired\n",
1092 ipsp->iss_active, ipsp->iss_expire);
1093 PRINTF("\t%lu closed\n\t%u orphans\n",
1094 ipsp->iss_fin, ipsp->iss_orphans);
1096 PRINTF("State logging %sabled\n",
1097 state_logging ? "en" : "dis");
1099 PRINTF("\nState table bucket statistics:\n");
1100 PRINTF("\t%lu in use\n\t%lu max bucket\n", ipsp->iss_inuse,
1101 ipsp->iss_bucketfull);
1103 minlen = ipsp->iss_max;
1104 totallen = 0;
1105 maxlen = 0;
1107 for (i = 0; i < ipsp->iss_statesize; i++) {
1108 if (buckets[i] > maxlen)
1109 maxlen = buckets[i];
1110 if (buckets[i] < minlen)
1111 minlen = buckets[i];
1112 totallen += buckets[i];
1115 PRINTF("\t%2.2f%% bucket usage\n\t%lu minimal length\n",
1116 ((float)ipsp->iss_inuse / ipsp->iss_statesize) * 100.0,
1117 minlen);
1118 PRINTF("\t%lu maximal length\n\t%.3f average length\n",
1119 maxlen,
1120 ipsp->iss_inuse ? (float) totallen/ ipsp->iss_inuse :
1121 0.0);
1123 #define ENTRIES_PER_LINE 5
1125 if (opts & OPT_VERBOSE) {
1126 PRINTF("\nCurrent bucket sizes :\n");
1127 for (i = 0; i < ipsp->iss_statesize; i++) {
1128 if ((i % ENTRIES_PER_LINE) == 0)
1129 PRINTF("\t");
1130 PRINTF("%4d -> %4lu", i, buckets[i]);
1131 if ((i % ENTRIES_PER_LINE) ==
1132 (ENTRIES_PER_LINE - 1))
1133 PRINTF("\n");
1134 else
1135 PRINTF(" ");
1137 PRINTF("\n");
1139 PRINTF("\n");
1141 free(buckets);
1142 return;
1146 * Print out all the state information currently held in the kernel.
1148 while (ipsp->iss_list != NULL) {
1149 ipsp->iss_list = printstate(ipsp->iss_list, opts,
1150 ipsp->iss_ticks);
1153 free(buckets);
1157 #ifdef STATETOP
1158 static int handle_resize = 0, handle_break = 0;
1160 static void topipstates(saddr, daddr, sport, dport, protocol, ver,
1161 refreshtime, topclosed)
1162 i6addr_t saddr;
1163 i6addr_t daddr;
1164 int sport;
1165 int dport;
1166 int protocol;
1167 int ver;
1168 int refreshtime;
1169 int topclosed;
1171 char str1[STSTRSIZE], str2[STSTRSIZE], str3[STSTRSIZE], str4[STSTRSIZE];
1172 int maxtsentries = 0, reverse = 0, sorting = STSORT_DEFAULT;
1173 int i, j, winy, tsentry, maxx, maxy, redraw = 0, ret = 0;
1174 int len, srclen, dstlen, forward = 1, c = 0;
1175 ips_stat_t ipsst, *ipsstp = &ipsst;
1176 statetop_t *tstable = NULL, *tp;
1177 const char *errstr = "";
1178 ipstate_t ips;
1179 ipfobj_t ipfo;
1180 struct timeval selecttimeout;
1181 char hostnm[HOSTNMLEN];
1182 struct protoent *proto;
1183 fd_set readfd;
1184 time_t t;
1186 /* install signal handlers */
1187 signal(SIGINT, sig_break);
1188 signal(SIGQUIT, sig_break);
1189 signal(SIGTERM, sig_break);
1190 signal(SIGWINCH, sig_resize);
1192 /* init ncurses stuff */
1193 initscr();
1194 cbreak();
1195 noecho();
1196 curs_set(0);
1197 timeout(0);
1198 getmaxyx(stdscr, maxy, maxx);
1200 /* init hostname */
1201 gethostname(hostnm, sizeof(hostnm) - 1);
1202 hostnm[sizeof(hostnm) - 1] = '\0';
1204 /* init ipfobj_t stuff */
1205 bzero((caddr_t)&ipfo, sizeof(ipfo));
1206 ipfo.ipfo_rev = IPFILTER_VERSION;
1207 ipfo.ipfo_size = sizeof(*ipsstp);
1208 ipfo.ipfo_ptr = (void *)ipsstp;
1209 ipfo.ipfo_type = IPFOBJ_STATESTAT;
1211 /* repeat until user aborts */
1212 while ( 1 ) {
1214 /* get state table */
1215 bzero((char *)&ipsst, sizeof(ipsst));
1216 if ((ioctl(state_fd, SIOCGETFS, &ipfo) == -1)) {
1217 errstr = "ioctl(SIOCGETFS)";
1218 ret = -1;
1219 goto out;
1222 /* clear the history */
1223 tsentry = -1;
1225 /* reset max str len */
1226 srclen = dstlen = 0;
1228 /* read the state table and store in tstable */
1229 for (; ipsstp->iss_list; ipsstp->iss_list = ips.is_next) {
1231 if (kmemcpy((char *)&ips, (u_long)ipsstp->iss_list,
1232 sizeof(ips)))
1233 break;
1235 if (ips.is_v != ver)
1236 continue;
1238 /* check v4 src/dest addresses */
1239 if (ips.is_v == 4) {
1240 if ((saddr.in4.s_addr != INADDR_ANY &&
1241 saddr.in4.s_addr != ips.is_saddr) ||
1242 (daddr.in4.s_addr != INADDR_ANY &&
1243 daddr.in4.s_addr != ips.is_daddr))
1244 continue;
1246 #ifdef USE_INET6
1247 /* check v6 src/dest addresses */
1248 if (ips.is_v == 6) {
1249 if ((IP6_NEQ(&saddr, &in6addr_any) &&
1250 IP6_NEQ(&saddr, &ips.is_src)) ||
1251 (IP6_NEQ(&daddr, &in6addr_any) &&
1252 IP6_NEQ(&daddr, &ips.is_dst)))
1253 continue;
1255 #endif
1256 /* check protocol */
1257 if (protocol > 0 && protocol != ips.is_p)
1258 continue;
1260 /* check ports if protocol is TCP or UDP */
1261 if (((ips.is_p == IPPROTO_TCP) ||
1262 (ips.is_p == IPPROTO_UDP)) &&
1263 (((sport > 0) && (htons(sport) != ips.is_sport)) ||
1264 ((dport > 0) && (htons(dport) != ips.is_dport))))
1265 continue;
1267 /* show closed TCP sessions ? */
1268 if ((topclosed == 0) && (ips.is_p == IPPROTO_TCP) &&
1269 (ips.is_state[0] >= IPF_TCPS_LAST_ACK) &&
1270 (ips.is_state[1] >= IPF_TCPS_LAST_ACK))
1271 continue;
1274 * if necessary make room for this state
1275 * entry
1277 tsentry++;
1278 if (!maxtsentries || tsentry == maxtsentries) {
1279 maxtsentries += STGROWSIZE;
1280 tstable = reallocarray(tstable, maxtsentries,
1281 sizeof (statetop_t));
1282 if (tstable == NULL) {
1283 perror("realloc");
1284 exit(-1);
1288 /* get max src/dest address string length */
1289 len = strlen(getip(ips.is_v, &ips.is_src));
1290 if (srclen < len)
1291 srclen = len;
1292 len = strlen(getip(ips.is_v, &ips.is_dst));
1293 if (dstlen < len)
1294 dstlen = len;
1296 /* fill structure */
1297 tp = tstable + tsentry;
1298 tp->st_src = ips.is_src;
1299 tp->st_dst = ips.is_dst;
1300 tp->st_p = ips.is_p;
1301 tp->st_v = ips.is_v;
1302 tp->st_state[0] = ips.is_state[0];
1303 tp->st_state[1] = ips.is_state[1];
1304 if (forward) {
1305 tp->st_pkts = ips.is_pkts[0]+ips.is_pkts[1];
1306 tp->st_bytes = ips.is_bytes[0]+ips.is_bytes[1];
1307 } else {
1308 tp->st_pkts = ips.is_pkts[2]+ips.is_pkts[3];
1309 tp->st_bytes = ips.is_bytes[2]+ips.is_bytes[3];
1311 tp->st_age = ips.is_die - ipsstp->iss_ticks;
1312 if ((ips.is_p == IPPROTO_TCP) ||
1313 (ips.is_p == IPPROTO_UDP)) {
1314 tp->st_sport = ips.is_sport;
1315 tp->st_dport = ips.is_dport;
1320 /* sort the array */
1321 if (tsentry != -1) {
1322 switch (sorting)
1324 case STSORT_PR:
1325 qsort(tstable, tsentry + 1,
1326 sizeof(statetop_t), sort_p);
1327 break;
1328 case STSORT_PKTS:
1329 qsort(tstable, tsentry + 1,
1330 sizeof(statetop_t), sort_pkts);
1331 break;
1332 case STSORT_BYTES:
1333 qsort(tstable, tsentry + 1,
1334 sizeof(statetop_t), sort_bytes);
1335 break;
1336 case STSORT_TTL:
1337 qsort(tstable, tsentry + 1,
1338 sizeof(statetop_t), sort_ttl);
1339 break;
1340 case STSORT_SRCIP:
1341 qsort(tstable, tsentry + 1,
1342 sizeof(statetop_t), sort_srcip);
1343 break;
1344 case STSORT_SRCPT:
1345 qsort(tstable, tsentry +1,
1346 sizeof(statetop_t), sort_srcpt);
1347 break;
1348 case STSORT_DSTIP:
1349 qsort(tstable, tsentry + 1,
1350 sizeof(statetop_t), sort_dstip);
1351 break;
1352 case STSORT_DSTPT:
1353 qsort(tstable, tsentry + 1,
1354 sizeof(statetop_t), sort_dstpt);
1355 break;
1356 default:
1357 break;
1361 /* handle window resizes */
1362 if (handle_resize) {
1363 endwin();
1364 initscr();
1365 cbreak();
1366 noecho();
1367 curs_set(0);
1368 timeout(0);
1369 getmaxyx(stdscr, maxy, maxx);
1370 redraw = 1;
1371 handle_resize = 0;
1374 /* stop program? */
1375 if (handle_break)
1376 break;
1378 /* print title */
1379 erase();
1380 attron(A_BOLD);
1381 winy = 0;
1382 move(winy,0);
1383 sprintf(str1, "%s - %s - state top", hostnm, IPL_VERSION);
1384 for (j = 0 ; j < (maxx - 8 - strlen(str1)) / 2; j++)
1385 printw(" ");
1386 printw("%s", str1);
1387 attroff(A_BOLD);
1389 /* just for fun add a clock */
1390 move(winy, maxx - 8);
1391 t = time(NULL);
1392 strftime(str1, 80, "%T", localtime(&t));
1393 printw("%s\n", str1);
1396 * print the display filters, this is placed in the loop,
1397 * because someday I might add code for changing these
1398 * while the programming is running :-)
1400 if (sport >= 0)
1401 sprintf(str1, "%s,%d", getip(ver, &saddr), sport);
1402 else
1403 sprintf(str1, "%s", getip(ver, &saddr));
1405 if (dport >= 0)
1406 sprintf(str2, "%s,%d", getip(ver, &daddr), dport);
1407 else
1408 sprintf(str2, "%s", getip(ver, &daddr));
1410 if (protocol < 0)
1411 strcpy(str3, "any");
1412 else if ((proto = getprotobynumber(protocol)) != NULL)
1413 sprintf(str3, "%s", proto->p_name);
1414 else
1415 sprintf(str3, "%d", protocol);
1417 switch (sorting)
1419 case STSORT_PR:
1420 sprintf(str4, "proto");
1421 break;
1422 case STSORT_PKTS:
1423 sprintf(str4, "# pkts");
1424 break;
1425 case STSORT_BYTES:
1426 sprintf(str4, "# bytes");
1427 break;
1428 case STSORT_TTL:
1429 sprintf(str4, "ttl");
1430 break;
1431 case STSORT_SRCIP:
1432 sprintf(str4, "src ip");
1433 break;
1434 case STSORT_SRCPT:
1435 sprintf(str4, "src port");
1436 break;
1437 case STSORT_DSTIP:
1438 sprintf(str4, "dest ip");
1439 break;
1440 case STSORT_DSTPT:
1441 sprintf(str4, "dest port");
1442 break;
1443 default:
1444 sprintf(str4, "unknown");
1445 break;
1448 if (reverse)
1449 strcat(str4, " (reverse)");
1451 winy += 2;
1452 move(winy,0);
1453 printw("Src: %s, Dest: %s, Proto: %s, Sorted by: %s\n\n",
1454 str1, str2, str3, str4);
1457 * For an IPv4 IP address we need at most 15 characters,
1458 * 4 tuples of 3 digits, separated by 3 dots. Enforce this
1459 * length, so the colums do not change positions based
1460 * on the size of the IP address. This length makes the
1461 * output fit in a 80 column terminal.
1462 * We are lacking a good solution for IPv6 addresses (that
1463 * can be longer that 15 characters), so we do not enforce
1464 * a maximum on the IP field size.
1466 if (srclen < 15)
1467 srclen = 15;
1468 if (dstlen < 15)
1469 dstlen = 15;
1471 /* print column description */
1472 winy += 2;
1473 move(winy,0);
1474 attron(A_BOLD);
1475 printw("%-*s %-*s %3s %4s %7s %9s %9s\n",
1476 srclen + 6, "Source IP", dstlen + 6, "Destination IP",
1477 "ST", "PR", "#pkts", "#bytes", "ttl");
1478 attroff(A_BOLD);
1480 /* print all the entries */
1481 tp = tstable;
1482 if (reverse)
1483 tp += tsentry;
1485 if (tsentry > maxy - 6)
1486 tsentry = maxy - 6;
1487 for (i = 0; i <= tsentry; i++) {
1488 /* print src/dest and port */
1489 if ((tp->st_p == IPPROTO_TCP) ||
1490 (tp->st_p == IPPROTO_UDP)) {
1491 sprintf(str1, "%s,%hu",
1492 getip(tp->st_v, &tp->st_src),
1493 ntohs(tp->st_sport));
1494 sprintf(str2, "%s,%hu",
1495 getip(tp->st_v, &tp->st_dst),
1496 ntohs(tp->st_dport));
1497 } else {
1498 sprintf(str1, "%s", getip(tp->st_v,
1499 &tp->st_src));
1500 sprintf(str2, "%s", getip(tp->st_v,
1501 &tp->st_dst));
1503 winy++;
1504 move(winy, 0);
1505 printw("%-*s %-*s", srclen + 6, str1, dstlen + 6, str2);
1507 /* print state */
1508 sprintf(str1, "%X/%X", tp->st_state[0],
1509 tp->st_state[1]);
1510 printw(" %3s", str1);
1512 /* print protocol */
1513 proto = getprotobynumber(tp->st_p);
1514 if (proto) {
1515 strncpy(str1, proto->p_name, 4);
1516 str1[4] = '\0';
1517 } else {
1518 sprintf(str1, "%d", tp->st_p);
1520 /* just print icmp for IPv6-ICMP */
1521 if (tp->st_p == IPPROTO_ICMPV6)
1522 strcpy(str1, "icmp");
1523 printw(" %4s", str1);
1525 /* print #pkt/#bytes */
1526 #ifdef USE_QUAD_T
1527 printw(" %7qu %9qu", (unsigned long long) tp->st_pkts,
1528 (unsigned long long) tp->st_bytes);
1529 #else
1530 printw(" %7lu %9lu", tp->st_pkts, tp->st_bytes);
1531 #endif
1532 printw(" %9s", ttl_to_string(tp->st_age));
1534 if (reverse)
1535 tp--;
1536 else
1537 tp++;
1540 /* screen data structure is filled, now update the screen */
1541 if (redraw)
1542 clearok(stdscr,1);
1544 if (refresh() == ERR)
1545 break;
1546 if (redraw) {
1547 clearok(stdscr,0);
1548 redraw = 0;
1551 /* wait for key press or a 1 second time out period */
1552 selecttimeout.tv_sec = refreshtime;
1553 selecttimeout.tv_usec = 0;
1554 FD_ZERO(&readfd);
1555 FD_SET(0, &readfd);
1556 select(1, &readfd, NULL, NULL, &selecttimeout);
1558 /* if key pressed, read all waiting keys */
1559 if (FD_ISSET(0, &readfd)) {
1560 c = wgetch(stdscr);
1561 if (c == ERR)
1562 continue;
1564 if (ISALPHA(c) && ISUPPER(c))
1565 c = TOLOWER(c);
1566 if (c == 'l') {
1567 redraw = 1;
1568 } else if (c == 'q') {
1569 break;
1570 } else if (c == 'r') {
1571 reverse = !reverse;
1572 } else if (c == 'b') {
1573 forward = 0;
1574 } else if (c == 'f') {
1575 forward = 1;
1576 } else if (c == 's') {
1577 if (++sorting > STSORT_MAX)
1578 sorting = 0;
1581 } /* while */
1583 out:
1584 printw("\n");
1585 refresh();
1586 endwin();
1587 free(tstable);
1588 if (ret != 0)
1589 perror(errstr);
1591 #endif
1595 * Show fragment cache information that's held in the kernel.
1597 static void showfrstates(ifsp, ticks)
1598 ipfrstat_t *ifsp;
1599 u_long ticks;
1601 struct ipfr *ipfrtab[IPFT_SIZE], ifr;
1602 int i;
1605 * print out the numeric statistics
1607 PRINTF("IP fragment states:\n\t%lu new\n\t%lu expired\n\t%lu hits\n",
1608 ifsp->ifs_new, ifsp->ifs_expire, ifsp->ifs_hits);
1609 PRINTF("\t%lu retrans\n\t%lu too short\n",
1610 ifsp->ifs_retrans0, ifsp->ifs_short);
1611 PRINTF("\t%lu no memory\n\t%lu already exist\n",
1612 ifsp->ifs_nomem, ifsp->ifs_exists);
1613 PRINTF("\t%lu inuse\n", ifsp->ifs_inuse);
1614 if (kmemcpy((char *)ipfrtab, (u_long)ifsp->ifs_table, sizeof(ipfrtab)))
1615 return;
1618 * Print out the contents (if any) of the fragment cache table.
1620 PRINTF("\n");
1621 for (i = 0; i < IPFT_SIZE; i++)
1622 while (ipfrtab[i] != NULL) {
1623 if (kmemcpy((char *)&ifr, (u_long)ipfrtab[i],
1624 sizeof(ifr)) == -1)
1625 break;
1626 ifr.ipfr_ttl -= ticks;
1627 printfraginfo("", &ifr);
1628 ipfrtab[i] = ifr.ipfr_next;
1631 * Print out the contents (if any) of the NAT fragment cache table.
1633 if (kmemcpy((char *)ipfrtab, (u_long)ifsp->ifs_nattab,sizeof(ipfrtab)))
1634 return;
1635 for (i = 0; i < IPFT_SIZE; i++)
1636 while (ipfrtab[i] != NULL) {
1637 if (kmemcpy((char *)&ifr, (u_long)ipfrtab[i],
1638 sizeof(ifr)) == -1)
1639 break;
1640 ifr.ipfr_ttl -= ticks;
1641 printfraginfo("NAT: ", &ifr);
1642 ipfrtab[i] = ifr.ipfr_next;
1648 * Show stats on how auth within IPFilter has been used
1650 static void showauthstates(asp)
1651 fr_authstat_t *asp;
1653 frauthent_t *frap, fra;
1655 #ifdef USE_QUAD_T
1656 printf("Authorisation hits: %qu\tmisses %qu\n",
1657 (unsigned long long) asp->fas_hits,
1658 (unsigned long long) asp->fas_miss);
1659 #else
1660 printf("Authorisation hits: %ld\tmisses %ld\n", asp->fas_hits,
1661 asp->fas_miss);
1662 #endif
1663 printf("nospace %ld\nadded %ld\nsendfail %ld\nsendok %ld\n",
1664 asp->fas_nospace, asp->fas_added, asp->fas_sendfail,
1665 asp->fas_sendok);
1666 printf("queok %ld\nquefail %ld\nexpire %ld\n",
1667 asp->fas_queok, asp->fas_quefail, asp->fas_expire);
1669 frap = asp->fas_faelist;
1670 while (frap) {
1671 if (kmemcpy((char *)&fra, (u_long)frap, sizeof(fra)) == -1)
1672 break;
1674 printf("age %ld\t", fra.fae_age);
1675 printfr(&fra.fae_fr, ioctl);
1676 frap = fra.fae_next;
1682 * Display groups used for each of filter rules, accounting rules and
1683 * authentication, separately.
1685 static void showgroups(fiop)
1686 struct friostat *fiop;
1688 static char *gnames[3] = { "Filter", "Accounting", "Authentication" };
1689 static int gnums[3] = { IPL_LOGIPF, IPL_LOGCOUNT, IPL_LOGAUTH };
1690 frgroup_t *fp, grp;
1691 int on, off, i;
1693 on = fiop->f_active;
1694 off = 1 - on;
1696 for (i = 0; i < 3; i++) {
1697 printf("%s groups (active):\n", gnames[i]);
1698 for (fp = fiop->f_groups[gnums[i]][on]; fp != NULL;
1699 fp = grp.fg_next)
1700 if (kmemcpy((char *)&grp, (u_long)fp, sizeof(grp)))
1701 break;
1702 else
1703 printf("%s\n", grp.fg_name);
1704 printf("%s groups (inactive):\n", gnames[i]);
1705 for (fp = fiop->f_groups[gnums[i]][off]; fp != NULL;
1706 fp = grp.fg_next)
1707 if (kmemcpy((char *)&grp, (u_long)fp, sizeof(grp)))
1708 break;
1709 else
1710 printf("%s\n", grp.fg_name);
1714 static void parse_ipportstr(argument, ip, port)
1715 const char *argument;
1716 i6addr_t *ip;
1717 int *port;
1719 char *s, *comma;
1720 int ok = 0;
1722 /* make working copy of argument, Theoretically you must be able
1723 * to write to optarg, but that seems very ugly to me....
1725 s = strdup(argument);
1726 if (s == NULL)
1727 return;
1729 /* get port */
1730 if ((comma = strchr(s, ',')) != NULL) {
1731 if (!strcasecmp(comma + 1, "any")) {
1732 *port = -1;
1733 } else if (!sscanf(comma + 1, "%d", port) ||
1734 (*port < 0) || (*port > 65535)) {
1735 fprintf(stderr, "Invalid port specfication in %s\n",
1736 argument);
1737 free(s);
1738 exit(-2);
1740 *comma = '\0';
1744 /* get ip address */
1745 if (!strcasecmp(s, "any")) {
1746 ip->in4.s_addr = INADDR_ANY;
1747 #ifdef USE_INET6
1748 ip->in6 = in6addr_any;
1749 } else if (use_inet6 && inet_pton(AF_INET6, s, &ip->in6)) {
1750 ok = 1;
1751 #endif
1752 } else if (inet_aton(s, &ip->in4))
1753 ok = 1;
1755 if (ok == 0) {
1756 fprintf(stderr, "Invalid IP address: %s\n", s);
1757 free(s);
1758 exit(-2);
1761 /* free allocated memory */
1762 free(s);
1766 #ifdef STATETOP
1767 static void sig_resize(s)
1768 int s;
1770 handle_resize = 1;
1773 static void sig_break(s)
1774 int s;
1776 handle_break = 1;
1779 static char *getip(v, addr)
1780 int v;
1781 i6addr_t *addr;
1783 #ifdef USE_INET6
1784 static char hostbuf[MAXHOSTNAMELEN+1];
1785 #endif
1787 if (v == 4)
1788 return inet_ntoa(addr->in4);
1790 #ifdef USE_INET6
1791 (void) inet_ntop(AF_INET6, &addr->in6, hostbuf, sizeof(hostbuf) - 1);
1792 hostbuf[MAXHOSTNAMELEN] = '\0';
1793 return hostbuf;
1794 #else
1795 return "IPv6";
1796 #endif
1800 static char *ttl_to_string(ttl)
1801 long int ttl;
1803 static char ttlbuf[STSTRSIZE];
1804 int hours, minutes, seconds;
1806 /* ttl is in half seconds */
1807 ttl /= 2;
1809 hours = ttl / 3600;
1810 ttl = ttl % 3600;
1811 minutes = ttl / 60;
1812 seconds = ttl % 60;
1814 if (hours > 0)
1815 sprintf(ttlbuf, "%2d:%02d:%02d", hours, minutes, seconds);
1816 else
1817 sprintf(ttlbuf, "%2d:%02d", minutes, seconds);
1818 return ttlbuf;
1822 static int sort_pkts(a, b)
1823 const void *a;
1824 const void *b;
1827 register const statetop_t *ap = a;
1828 register const statetop_t *bp = b;
1830 if (ap->st_pkts == bp->st_pkts)
1831 return 0;
1832 else if (ap->st_pkts < bp->st_pkts)
1833 return 1;
1834 return -1;
1838 static int sort_bytes(a, b)
1839 const void *a;
1840 const void *b;
1842 register const statetop_t *ap = a;
1843 register const statetop_t *bp = b;
1845 if (ap->st_bytes == bp->st_bytes)
1846 return 0;
1847 else if (ap->st_bytes < bp->st_bytes)
1848 return 1;
1849 return -1;
1853 static int sort_p(a, b)
1854 const void *a;
1855 const void *b;
1857 register const statetop_t *ap = a;
1858 register const statetop_t *bp = b;
1860 if (ap->st_p == bp->st_p)
1861 return 0;
1862 else if (ap->st_p < bp->st_p)
1863 return 1;
1864 return -1;
1868 static int sort_ttl(a, b)
1869 const void *a;
1870 const void *b;
1872 register const statetop_t *ap = a;
1873 register const statetop_t *bp = b;
1875 if (ap->st_age == bp->st_age)
1876 return 0;
1877 else if (ap->st_age < bp->st_age)
1878 return 1;
1879 return -1;
1882 static int sort_srcip(a, b)
1883 const void *a;
1884 const void *b;
1886 register const statetop_t *ap = a;
1887 register const statetop_t *bp = b;
1889 #ifdef USE_INET6
1890 if (use_inet6) {
1891 if (IP6_EQ(&ap->st_src, &bp->st_src))
1892 return 0;
1893 else if (IP6_GT(&ap->st_src, &bp->st_src))
1894 return 1;
1895 } else
1896 #endif
1898 if (ntohl(ap->st_src.in4.s_addr) ==
1899 ntohl(bp->st_src.in4.s_addr))
1900 return 0;
1901 else if (ntohl(ap->st_src.in4.s_addr) >
1902 ntohl(bp->st_src.in4.s_addr))
1903 return 1;
1905 return -1;
1908 static int sort_srcpt(a, b)
1909 const void *a;
1910 const void *b;
1912 register const statetop_t *ap = a;
1913 register const statetop_t *bp = b;
1915 if (htons(ap->st_sport) == htons(bp->st_sport))
1916 return 0;
1917 else if (htons(ap->st_sport) > htons(bp->st_sport))
1918 return 1;
1919 return -1;
1922 static int sort_dstip(a, b)
1923 const void *a;
1924 const void *b;
1926 register const statetop_t *ap = a;
1927 register const statetop_t *bp = b;
1929 #ifdef USE_INET6
1930 if (use_inet6) {
1931 if (IP6_EQ(&ap->st_dst, &bp->st_dst))
1932 return 0;
1933 else if (IP6_GT(&ap->st_dst, &bp->st_dst))
1934 return 1;
1935 } else
1936 #endif
1938 if (ntohl(ap->st_dst.in4.s_addr) ==
1939 ntohl(bp->st_dst.in4.s_addr))
1940 return 0;
1941 else if (ntohl(ap->st_dst.in4.s_addr) >
1942 ntohl(bp->st_dst.in4.s_addr))
1943 return 1;
1945 return -1;
1948 static int sort_dstpt(a, b)
1949 const void *a;
1950 const void *b;
1952 register const statetop_t *ap = a;
1953 register const statetop_t *bp = b;
1955 if (htons(ap->st_dport) == htons(bp->st_dport))
1956 return 0;
1957 else if (htons(ap->st_dport) > htons(bp->st_dport))
1958 return 1;
1959 return -1;
1962 #endif