141edb306SCy Schubert
241edb306SCy Schubert /*
341edb306SCy Schubert * Copyright (C) 2012 by Darren Reed.
441edb306SCy Schubert *
541edb306SCy Schubert * See the IPFILTER.LICENCE file for details on licencing.
641edb306SCy Schubert */
741edb306SCy Schubert #include "ipf.h"
841edb306SCy Schubert #include "ipmon.h"
941edb306SCy Schubert #include <sys/ioctl.h>
1041edb306SCy Schubert #include <sys/stat.h>
1141edb306SCy Schubert #include <syslog.h>
1241edb306SCy Schubert #include <ctype.h>
1341edb306SCy Schubert #include <fcntl.h>
1441edb306SCy Schubert #include <signal.h>
1541edb306SCy Schubert
1641edb306SCy Schubert
1741edb306SCy Schubert
1841edb306SCy Schubert #define STRERROR(x) strerror(x)
1941edb306SCy Schubert
2041edb306SCy Schubert extern int optind;
2141edb306SCy Schubert extern char *optarg;
2241edb306SCy Schubert
2341edb306SCy Schubert extern ipmon_saver_t executesaver;
2441edb306SCy Schubert extern ipmon_saver_t filesaver;
2541edb306SCy Schubert extern ipmon_saver_t nothingsaver;
2641edb306SCy Schubert extern ipmon_saver_t snmpv1saver;
2741edb306SCy Schubert extern ipmon_saver_t snmpv2saver;
2841edb306SCy Schubert extern ipmon_saver_t syslogsaver;
2941edb306SCy Schubert
3041edb306SCy Schubert
3141edb306SCy Schubert struct flags {
3241edb306SCy Schubert int value;
3341edb306SCy Schubert char flag;
3441edb306SCy Schubert };
3541edb306SCy Schubert
3641edb306SCy Schubert typedef struct logsource {
3741edb306SCy Schubert int fd;
3841edb306SCy Schubert int logtype;
3941edb306SCy Schubert char *file;
4041edb306SCy Schubert int regular;
4141edb306SCy Schubert size_t size;
4241edb306SCy Schubert } logsource_t;
4341edb306SCy Schubert
4441edb306SCy Schubert typedef struct config {
4541edb306SCy Schubert int opts;
4641edb306SCy Schubert int maxfd;
4741edb306SCy Schubert logsource_t logsrc[3];
4841edb306SCy Schubert fd_set fdmr;
4941edb306SCy Schubert FILE *blog;
5041edb306SCy Schubert char *bfile;
5141edb306SCy Schubert FILE *log;
5241edb306SCy Schubert char *file;
5341edb306SCy Schubert char *cfile;
5441edb306SCy Schubert } config_t;
5541edb306SCy Schubert
5641edb306SCy Schubert typedef struct icmp_subtype {
5741edb306SCy Schubert int ist_val;
5841edb306SCy Schubert char *ist_name;
5941edb306SCy Schubert } icmp_subtype_t;
6041edb306SCy Schubert
6141edb306SCy Schubert typedef struct icmp_type {
6241edb306SCy Schubert int it_val;
6341edb306SCy Schubert struct icmp_subtype *it_subtable;
6441edb306SCy Schubert size_t it_stsize;
6541edb306SCy Schubert char *it_name;
6641edb306SCy Schubert } icmp_type_t;
6741edb306SCy Schubert
6841edb306SCy Schubert
6941edb306SCy Schubert #define IST_SZ(x) (sizeof(x)/sizeof(icmp_subtype_t))
7041edb306SCy Schubert
7141edb306SCy Schubert
7241edb306SCy Schubert struct flags tcpfl[] = {
7341edb306SCy Schubert { TH_ACK, 'A' },
7441edb306SCy Schubert { TH_RST, 'R' },
7541edb306SCy Schubert { TH_SYN, 'S' },
7641edb306SCy Schubert { TH_FIN, 'F' },
7741edb306SCy Schubert { TH_URG, 'U' },
7841edb306SCy Schubert { TH_PUSH,'P' },
7941edb306SCy Schubert { TH_ECN, 'E' },
80*347dd053SRichard Scheffenegger { TH_CWR, 'W' },
81*347dd053SRichard Scheffenegger { TH_AE, 'e' },
8241edb306SCy Schubert { 0, '\0' }
8341edb306SCy Schubert };
8441edb306SCy Schubert
8541edb306SCy Schubert char *reasons[] = {
8641edb306SCy Schubert "filter-rule",
8741edb306SCy Schubert "log-or-block_1",
8841edb306SCy Schubert "pps-rate",
8941edb306SCy Schubert "jumbogram",
9041edb306SCy Schubert "makefrip-fail",
9141edb306SCy Schubert "state_add-fail",
9241edb306SCy Schubert "updateipid-fail",
9341edb306SCy Schubert "log-or-block_2",
9441edb306SCy Schubert "decap-fail",
9541edb306SCy Schubert "auth_new-fail",
9641edb306SCy Schubert "auth_captured",
9741edb306SCy Schubert "coalesce-fail",
9841edb306SCy Schubert "pullup-fail",
9941edb306SCy Schubert "auth-feedback",
10041edb306SCy Schubert "bad-frag",
10141edb306SCy Schubert "natv4_out-fail",
10241edb306SCy Schubert "natv4_in-fail",
10341edb306SCy Schubert "natv6_out-fail",
10441edb306SCy Schubert "natv6_in-fail",
10541edb306SCy Schubert };
10641edb306SCy Schubert
10741edb306SCy Schubert #if SOLARIS
10841edb306SCy Schubert static char *pidfile = "/etc/opt/ipf/ipmon.pid";
10941edb306SCy Schubert #else
11041edb306SCy Schubert static char *pidfile = "/var/run/ipmon.pid";
11141edb306SCy Schubert #endif
11241edb306SCy Schubert
11341edb306SCy Schubert static char line[2048];
11441edb306SCy Schubert static int donehup = 0;
11541edb306SCy Schubert static void usage(char *);
11641edb306SCy Schubert static void handlehup(int);
11741edb306SCy Schubert static void flushlogs(char *, FILE *);
11841edb306SCy Schubert static void print_log(config_t *, logsource_t *, char *, int);
11941edb306SCy Schubert static void print_ipflog(config_t *, char *, int);
12041edb306SCy Schubert static void print_natlog(config_t *, char *, int);
12141edb306SCy Schubert static void print_statelog(config_t *, char *, int);
12241edb306SCy Schubert static int read_log(int, int *, char *, int);
12341edb306SCy Schubert static void write_pid(char *);
12441edb306SCy Schubert static char *icmpname(u_int, u_int);
12541edb306SCy Schubert static char *icmpname6(u_int, u_int);
12641edb306SCy Schubert static icmp_type_t *find_icmptype(int, icmp_type_t *, size_t);
12741edb306SCy Schubert static icmp_subtype_t *find_icmpsubtype(int, icmp_subtype_t *, size_t);
12841edb306SCy Schubert static struct tm *get_tm(time_t);
12941edb306SCy Schubert
13041edb306SCy Schubert char *portlocalname(int, char *, u_int);
13141edb306SCy Schubert int main(int, char *[]);
13241edb306SCy Schubert
13341edb306SCy Schubert static void logopts(int, char *);
13441edb306SCy Schubert static void init_tabs(void);
13541edb306SCy Schubert static char *getlocalproto(u_int);
13641edb306SCy Schubert static void openlogs(config_t *conf);
13741edb306SCy Schubert static int read_loginfo(config_t *conf);
13841edb306SCy Schubert static void initconfig(config_t *conf);
13941edb306SCy Schubert
14041edb306SCy Schubert static char **protocols = NULL;
14141edb306SCy Schubert static char **udp_ports = NULL;
14241edb306SCy Schubert static char **tcp_ports = NULL;
14341edb306SCy Schubert
14441edb306SCy Schubert
14541edb306SCy Schubert #define HOSTNAMEV4(b) hostname(AF_INET, (u_32_t *)&(b))
14641edb306SCy Schubert
14741edb306SCy Schubert #ifndef LOGFAC
14841edb306SCy Schubert #define LOGFAC LOG_LOCAL0
14941edb306SCy Schubert #endif
15041edb306SCy Schubert int logfac = LOGFAC;
15141edb306SCy Schubert int ipmonopts = 0;
15241edb306SCy Schubert int opts = OPT_NORESOLVE;
15341edb306SCy Schubert int use_inet6 = 0;
15441edb306SCy Schubert
15541edb306SCy Schubert
15641edb306SCy Schubert static icmp_subtype_t icmpunreachnames[] = {
15741edb306SCy Schubert { ICMP_UNREACH_NET, "net" },
15841edb306SCy Schubert { ICMP_UNREACH_HOST, "host" },
15941edb306SCy Schubert { ICMP_UNREACH_PROTOCOL, "protocol" },
16041edb306SCy Schubert { ICMP_UNREACH_PORT, "port" },
16141edb306SCy Schubert { ICMP_UNREACH_NEEDFRAG, "needfrag" },
16241edb306SCy Schubert { ICMP_UNREACH_SRCFAIL, "srcfail" },
16341edb306SCy Schubert { ICMP_UNREACH_NET_UNKNOWN, "net_unknown" },
16441edb306SCy Schubert { ICMP_UNREACH_HOST_UNKNOWN, "host_unknown" },
16541edb306SCy Schubert { ICMP_UNREACH_NET, "isolated" },
16641edb306SCy Schubert { ICMP_UNREACH_NET_PROHIB, "net_prohib" },
16741edb306SCy Schubert { ICMP_UNREACH_NET_PROHIB, "host_prohib" },
16841edb306SCy Schubert { ICMP_UNREACH_TOSNET, "tosnet" },
16941edb306SCy Schubert { ICMP_UNREACH_TOSHOST, "toshost" },
17041edb306SCy Schubert { ICMP_UNREACH_ADMIN_PROHIBIT, "admin_prohibit" },
17141edb306SCy Schubert { -2, NULL }
17241edb306SCy Schubert };
17341edb306SCy Schubert
17441edb306SCy Schubert static icmp_subtype_t redirectnames[] = {
17541edb306SCy Schubert { ICMP_REDIRECT_NET, "net" },
17641edb306SCy Schubert { ICMP_REDIRECT_HOST, "host" },
17741edb306SCy Schubert { ICMP_REDIRECT_TOSNET, "tosnet" },
17841edb306SCy Schubert { ICMP_REDIRECT_TOSHOST, "toshost" },
17941edb306SCy Schubert { -2, NULL }
18041edb306SCy Schubert };
18141edb306SCy Schubert
18241edb306SCy Schubert static icmp_subtype_t timxceednames[] = {
18341edb306SCy Schubert { ICMP_TIMXCEED_INTRANS, "transit" },
18441edb306SCy Schubert { ICMP_TIMXCEED_REASS, "reassem" },
18541edb306SCy Schubert { -2, NULL }
18641edb306SCy Schubert };
18741edb306SCy Schubert
18841edb306SCy Schubert static icmp_subtype_t paramnames[] = {
18941edb306SCy Schubert { ICMP_PARAMPROB_ERRATPTR, "errata_pointer" },
19041edb306SCy Schubert { ICMP_PARAMPROB_OPTABSENT, "optmissing" },
19141edb306SCy Schubert { ICMP_PARAMPROB_LENGTH, "length" },
19241edb306SCy Schubert { -2, NULL }
19341edb306SCy Schubert };
19441edb306SCy Schubert
19541edb306SCy Schubert static icmp_type_t icmptypes4[] = {
19641edb306SCy Schubert { ICMP_ECHOREPLY, NULL, 0, "echoreply" },
19741edb306SCy Schubert { -1, NULL, 0, NULL },
19841edb306SCy Schubert { -1, NULL, 0, NULL },
19941edb306SCy Schubert { ICMP_UNREACH, icmpunreachnames,
20041edb306SCy Schubert IST_SZ(icmpunreachnames),"unreach" },
20141edb306SCy Schubert { ICMP_SOURCEQUENCH, NULL, 0, "sourcequench" },
20241edb306SCy Schubert { ICMP_REDIRECT, redirectnames,
20341edb306SCy Schubert IST_SZ(redirectnames), "redirect" },
20441edb306SCy Schubert { -1, NULL, 0, NULL },
20541edb306SCy Schubert { -1, NULL, 0, NULL },
20641edb306SCy Schubert { ICMP_ECHO, NULL, 0, "echo" },
20741edb306SCy Schubert { ICMP_ROUTERADVERT, NULL, 0, "routeradvert" },
20841edb306SCy Schubert { ICMP_ROUTERSOLICIT, NULL, 0, "routersolicit" },
20941edb306SCy Schubert { ICMP_TIMXCEED, timxceednames,
21041edb306SCy Schubert IST_SZ(timxceednames), "timxceed" },
21141edb306SCy Schubert { ICMP_PARAMPROB, paramnames,
21241edb306SCy Schubert IST_SZ(paramnames), "paramprob" },
21341edb306SCy Schubert { ICMP_TSTAMP, NULL, 0, "timestamp" },
21441edb306SCy Schubert { ICMP_TSTAMPREPLY, NULL, 0, "timestampreply" },
21541edb306SCy Schubert { ICMP_IREQ, NULL, 0, "inforeq" },
21641edb306SCy Schubert { ICMP_IREQREPLY, NULL, 0, "inforeply" },
21741edb306SCy Schubert { ICMP_MASKREQ, NULL, 0, "maskreq" },
21841edb306SCy Schubert { ICMP_MASKREPLY, NULL, 0, "maskreply" },
21941edb306SCy Schubert { -2, NULL, 0, NULL }
22041edb306SCy Schubert };
22141edb306SCy Schubert
22241edb306SCy Schubert static icmp_subtype_t icmpredirect6[] = {
22341edb306SCy Schubert { ICMP6_DST_UNREACH_NOROUTE, "noroute" },
22441edb306SCy Schubert { ICMP6_DST_UNREACH_ADMIN, "admin" },
22541edb306SCy Schubert { ICMP6_DST_UNREACH_NOTNEIGHBOR, "neighbour" },
22641edb306SCy Schubert { ICMP6_DST_UNREACH_ADDR, "address" },
22741edb306SCy Schubert { ICMP6_DST_UNREACH_NOPORT, "noport" },
22841edb306SCy Schubert { -2, NULL }
22941edb306SCy Schubert };
23041edb306SCy Schubert
23141edb306SCy Schubert static icmp_subtype_t icmptimexceed6[] = {
23241edb306SCy Schubert { ICMP6_TIME_EXCEED_TRANSIT, "intransit" },
23341edb306SCy Schubert { ICMP6_TIME_EXCEED_REASSEMBLY, "reassem" },
23441edb306SCy Schubert { -2, NULL }
23541edb306SCy Schubert };
23641edb306SCy Schubert
23741edb306SCy Schubert static icmp_subtype_t icmpparamprob6[] = {
23841edb306SCy Schubert { ICMP6_PARAMPROB_HEADER, "header" },
23941edb306SCy Schubert { ICMP6_PARAMPROB_NEXTHEADER, "nextheader" },
24041edb306SCy Schubert { ICMP6_PARAMPROB_OPTION, "option" },
24141edb306SCy Schubert { -2, NULL }
24241edb306SCy Schubert };
24341edb306SCy Schubert
24441edb306SCy Schubert static icmp_subtype_t icmpquerysubject6[] = {
24541edb306SCy Schubert { ICMP6_NI_SUBJ_IPV6, "ipv6" },
24641edb306SCy Schubert { ICMP6_NI_SUBJ_FQDN, "fqdn" },
24741edb306SCy Schubert { ICMP6_NI_SUBJ_IPV4, "ipv4" },
24841edb306SCy Schubert { -2, NULL },
24941edb306SCy Schubert };
25041edb306SCy Schubert
25141edb306SCy Schubert static icmp_subtype_t icmpnodeinfo6[] = {
25241edb306SCy Schubert { ICMP6_NI_SUCCESS, "success" },
25341edb306SCy Schubert { ICMP6_NI_REFUSED, "refused" },
25441edb306SCy Schubert { ICMP6_NI_UNKNOWN, "unknown" },
25541edb306SCy Schubert { -2, NULL }
25641edb306SCy Schubert };
25741edb306SCy Schubert
25841edb306SCy Schubert static icmp_subtype_t icmprenumber6[] = {
25941edb306SCy Schubert { ICMP6_ROUTER_RENUMBERING_COMMAND, "command" },
26041edb306SCy Schubert { ICMP6_ROUTER_RENUMBERING_RESULT, "result" },
26141edb306SCy Schubert { ICMP6_ROUTER_RENUMBERING_SEQNUM_RESET, "seqnum_reset" },
26241edb306SCy Schubert { -2, NULL }
26341edb306SCy Schubert };
26441edb306SCy Schubert
26541edb306SCy Schubert static icmp_type_t icmptypes6[] = {
26641edb306SCy Schubert { 0, NULL, 0, NULL },
26741edb306SCy Schubert { ICMP6_DST_UNREACH, icmpredirect6,
26841edb306SCy Schubert IST_SZ(icmpredirect6), "unreach" },
26941edb306SCy Schubert { ICMP6_PACKET_TOO_BIG, NULL, 0, "toobig" },
27041edb306SCy Schubert { ICMP6_TIME_EXCEEDED, icmptimexceed6,
27141edb306SCy Schubert IST_SZ(icmptimexceed6), "timxceed" },
27241edb306SCy Schubert { ICMP6_PARAM_PROB, icmpparamprob6,
27341edb306SCy Schubert IST_SZ(icmpparamprob6), "paramprob" },
27441edb306SCy Schubert { ICMP6_ECHO_REQUEST, NULL, 0, "echo" },
27541edb306SCy Schubert { ICMP6_ECHO_REPLY, NULL, 0, "echoreply" },
27641edb306SCy Schubert { ICMP6_MEMBERSHIP_QUERY, icmpquerysubject6,
27741edb306SCy Schubert IST_SZ(icmpquerysubject6), "groupmemberquery" },
27841edb306SCy Schubert { ICMP6_MEMBERSHIP_REPORT,NULL, 0, "groupmemberreport" },
27941edb306SCy Schubert { ICMP6_MEMBERSHIP_REDUCTION,NULL, 0, "groupmemberterm" },
28041edb306SCy Schubert { ND_ROUTER_SOLICIT, NULL, 0, "routersolicit" },
28141edb306SCy Schubert { ND_ROUTER_ADVERT, NULL, 0, "routeradvert" },
28241edb306SCy Schubert { ND_NEIGHBOR_SOLICIT, NULL, 0, "neighborsolicit" },
28341edb306SCy Schubert { ND_NEIGHBOR_ADVERT, NULL, 0, "neighboradvert" },
28441edb306SCy Schubert { ND_REDIRECT, NULL, 0, "redirect" },
28541edb306SCy Schubert { ICMP6_ROUTER_RENUMBERING, icmprenumber6,
28641edb306SCy Schubert IST_SZ(icmprenumber6), "routerrenumber" },
28741edb306SCy Schubert { ICMP6_WRUREQUEST, NULL, 0, "whoareyourequest" },
28841edb306SCy Schubert { ICMP6_WRUREPLY, NULL, 0, "whoareyoureply" },
28941edb306SCy Schubert { ICMP6_FQDN_QUERY, NULL, 0, "fqdnquery" },
29041edb306SCy Schubert { ICMP6_FQDN_REPLY, NULL, 0, "fqdnreply" },
29141edb306SCy Schubert { ICMP6_NI_QUERY, icmpnodeinfo6,
29241edb306SCy Schubert IST_SZ(icmpnodeinfo6), "nodeinforequest" },
29341edb306SCy Schubert { ICMP6_NI_REPLY, NULL, 0, "nodeinforeply" },
29441edb306SCy Schubert { MLD6_MTRACE_RESP, NULL, 0, "mtraceresponse" },
29541edb306SCy Schubert { MLD6_MTRACE, NULL, 0, "mtracerequest" },
29641edb306SCy Schubert { -2, NULL, 0, NULL }
29741edb306SCy Schubert };
29841edb306SCy Schubert
299efeb8bffSCy Schubert static icmp_subtype_t *
find_icmpsubtype(int type,icmp_subtype_t * table,size_t tablesz)300efeb8bffSCy Schubert find_icmpsubtype(int type, icmp_subtype_t *table, size_t tablesz)
30141edb306SCy Schubert {
30241edb306SCy Schubert icmp_subtype_t *ist;
30341edb306SCy Schubert
30441edb306SCy Schubert if (tablesz < 2)
3052582ae57SCy Schubert return (NULL);
30641edb306SCy Schubert
30741edb306SCy Schubert if ((type < 0) || (type > table[tablesz - 2].ist_val))
3082582ae57SCy Schubert return (NULL);
30941edb306SCy Schubert
31041edb306SCy Schubert if (table[type].ist_val == type)
3112582ae57SCy Schubert return (table + type);
31241edb306SCy Schubert
31390f2fee0SJohn Baldwin for (ist = table; ist->ist_val != -2; ist++)
31441edb306SCy Schubert if (ist->ist_val == type)
3152582ae57SCy Schubert return (ist);
3162582ae57SCy Schubert return (NULL);
31741edb306SCy Schubert }
31841edb306SCy Schubert
31941edb306SCy Schubert
320efeb8bffSCy Schubert static icmp_type_t *
find_icmptype(int type,icmp_type_t * table,size_t tablesz)321efeb8bffSCy Schubert find_icmptype(int type, icmp_type_t *table, size_t tablesz)
32241edb306SCy Schubert {
32341edb306SCy Schubert icmp_type_t *it;
32441edb306SCy Schubert
32541edb306SCy Schubert if (tablesz < 2)
3262582ae57SCy Schubert return (NULL);
32741edb306SCy Schubert
32841edb306SCy Schubert if ((type < 0) || (type > table[tablesz - 2].it_val))
3292582ae57SCy Schubert return (NULL);
33041edb306SCy Schubert
33141edb306SCy Schubert if (table[type].it_val == type)
3322582ae57SCy Schubert return (table + type);
33341edb306SCy Schubert
33490f2fee0SJohn Baldwin for (it = table; it->it_val != -2; it++)
33541edb306SCy Schubert if (it->it_val == type)
3362582ae57SCy Schubert return (it);
3372582ae57SCy Schubert return (NULL);
33841edb306SCy Schubert }
33941edb306SCy Schubert
34041edb306SCy Schubert
341efeb8bffSCy Schubert static void
handlehup(int sig)342efeb8bffSCy Schubert handlehup(int sig)
34341edb306SCy Schubert {
34441edb306SCy Schubert signal(SIGHUP, handlehup);
34541edb306SCy Schubert donehup = 1;
34641edb306SCy Schubert }
34741edb306SCy Schubert
34841edb306SCy Schubert
349efeb8bffSCy Schubert static void
init_tabs(void)350efeb8bffSCy Schubert init_tabs(void)
35141edb306SCy Schubert {
35241edb306SCy Schubert struct protoent *p;
35341edb306SCy Schubert struct servent *s;
35441edb306SCy Schubert char *name, **tab;
35541edb306SCy Schubert int port, i;
35641edb306SCy Schubert
35741edb306SCy Schubert if (protocols != NULL) {
35841edb306SCy Schubert for (i = 0; i < 256; i++)
35941edb306SCy Schubert if (protocols[i] != NULL) {
36041edb306SCy Schubert free(protocols[i]);
36141edb306SCy Schubert protocols[i] = NULL;
36241edb306SCy Schubert }
36341edb306SCy Schubert free(protocols);
36441edb306SCy Schubert protocols = NULL;
36541edb306SCy Schubert }
36641edb306SCy Schubert protocols = (char **)malloc(256 * sizeof(*protocols));
36741edb306SCy Schubert if (protocols != NULL) {
36841edb306SCy Schubert bzero((char *)protocols, 256 * sizeof(*protocols));
36941edb306SCy Schubert
37041edb306SCy Schubert setprotoent(1);
37141edb306SCy Schubert while ((p = getprotoent()) != NULL)
37241edb306SCy Schubert if (p->p_proto >= 0 && p->p_proto <= 255 &&
37341edb306SCy Schubert p->p_name != NULL && protocols[p->p_proto] == NULL)
37441edb306SCy Schubert protocols[p->p_proto] = strdup(p->p_name);
37541edb306SCy Schubert endprotoent();
37641edb306SCy Schubert if (protocols[0])
37741edb306SCy Schubert free(protocols[0]);
37841edb306SCy Schubert protocols[0] = strdup("ip");
37941edb306SCy Schubert }
38041edb306SCy Schubert
38141edb306SCy Schubert if (udp_ports != NULL) {
38241edb306SCy Schubert for (i = 0; i < 65536; i++)
38341edb306SCy Schubert if (udp_ports[i] != NULL) {
38441edb306SCy Schubert free(udp_ports[i]);
38541edb306SCy Schubert udp_ports[i] = NULL;
38641edb306SCy Schubert }
38741edb306SCy Schubert free(udp_ports);
38841edb306SCy Schubert udp_ports = NULL;
38941edb306SCy Schubert }
39041edb306SCy Schubert udp_ports = (char **)malloc(65536 * sizeof(*udp_ports));
39141edb306SCy Schubert if (udp_ports != NULL)
39241edb306SCy Schubert bzero((char *)udp_ports, 65536 * sizeof(*udp_ports));
39341edb306SCy Schubert
39441edb306SCy Schubert if (tcp_ports != NULL) {
39541edb306SCy Schubert for (i = 0; i < 65536; i++)
39641edb306SCy Schubert if (tcp_ports[i] != NULL) {
39741edb306SCy Schubert free(tcp_ports[i]);
39841edb306SCy Schubert tcp_ports[i] = NULL;
39941edb306SCy Schubert }
40041edb306SCy Schubert free(tcp_ports);
40141edb306SCy Schubert tcp_ports = NULL;
40241edb306SCy Schubert }
40341edb306SCy Schubert tcp_ports = (char **)malloc(65536 * sizeof(*tcp_ports));
40441edb306SCy Schubert if (tcp_ports != NULL)
40541edb306SCy Schubert bzero((char *)tcp_ports, 65536 * sizeof(*tcp_ports));
40641edb306SCy Schubert
40741edb306SCy Schubert setservent(1);
40841edb306SCy Schubert while ((s = getservent()) != NULL) {
40941edb306SCy Schubert if (s->s_proto == NULL)
41041edb306SCy Schubert continue;
41141edb306SCy Schubert else if (!strcmp(s->s_proto, "tcp")) {
41241edb306SCy Schubert port = ntohs(s->s_port);
41341edb306SCy Schubert name = s->s_name;
41441edb306SCy Schubert tab = tcp_ports;
41541edb306SCy Schubert } else if (!strcmp(s->s_proto, "udp")) {
41641edb306SCy Schubert port = ntohs(s->s_port);
41741edb306SCy Schubert name = s->s_name;
41841edb306SCy Schubert tab = udp_ports;
41941edb306SCy Schubert } else
42041edb306SCy Schubert continue;
42141edb306SCy Schubert if ((port < 0 || port > 65535) || (name == NULL))
42241edb306SCy Schubert continue;
42341edb306SCy Schubert if (tab != NULL)
42441edb306SCy Schubert tab[port] = strdup(name);
42541edb306SCy Schubert }
42641edb306SCy Schubert endservent();
42741edb306SCy Schubert }
42841edb306SCy Schubert
42941edb306SCy Schubert
430efeb8bffSCy Schubert static char *
getlocalproto(u_int p)431efeb8bffSCy Schubert getlocalproto(u_int p)
43241edb306SCy Schubert {
43341edb306SCy Schubert static char pnum[4];
43441edb306SCy Schubert char *s;
43541edb306SCy Schubert
43641edb306SCy Schubert p &= 0xff;
43741edb306SCy Schubert s = protocols ? protocols[p] : NULL;
43841edb306SCy Schubert if (s == NULL) {
439d740faa3SCy Schubert sprintf(pnum, "%u", p);
44041edb306SCy Schubert s = pnum;
44141edb306SCy Schubert }
4422582ae57SCy Schubert return (s);
44341edb306SCy Schubert }
44441edb306SCy Schubert
44541edb306SCy Schubert
446efeb8bffSCy Schubert static int
read_log(int fd,int * lenp,char * buf,int bufsize)447efeb8bffSCy Schubert read_log(int fd, int *lenp, char *buf, int bufsize)
44841edb306SCy Schubert {
44941edb306SCy Schubert int nr;
45041edb306SCy Schubert
45141edb306SCy Schubert if (bufsize > IPFILTER_LOGSIZE)
45241edb306SCy Schubert bufsize = IPFILTER_LOGSIZE;
45341edb306SCy Schubert
45441edb306SCy Schubert nr = read(fd, buf, bufsize);
45541edb306SCy Schubert if (!nr)
4562582ae57SCy Schubert return (2);
45741edb306SCy Schubert if ((nr < 0) && (errno != EINTR))
4582582ae57SCy Schubert return (-1);
45941edb306SCy Schubert *lenp = nr;
4602582ae57SCy Schubert return (0);
46141edb306SCy Schubert }
46241edb306SCy Schubert
46341edb306SCy Schubert
464efeb8bffSCy Schubert char *
portlocalname(int res,char * proto,u_int port)4652ac057ddSJohn Baldwin portlocalname(int res, char *proto, u_int port)
46641edb306SCy Schubert {
46741edb306SCy Schubert static char pname[8];
46841edb306SCy Schubert char *s;
46941edb306SCy Schubert
47041edb306SCy Schubert port = ntohs(port);
47141edb306SCy Schubert port &= 0xffff;
472d740faa3SCy Schubert sprintf(pname, "%u", port);
47341edb306SCy Schubert if (!res || (ipmonopts & IPMON_PORTNUM))
4742582ae57SCy Schubert return (pname);
47541edb306SCy Schubert s = NULL;
47641edb306SCy Schubert if (!strcmp(proto, "tcp"))
47741edb306SCy Schubert s = tcp_ports[port];
47841edb306SCy Schubert else if (!strcmp(proto, "udp"))
47941edb306SCy Schubert s = udp_ports[port];
48041edb306SCy Schubert if (s == NULL)
48141edb306SCy Schubert s = pname;
4822582ae57SCy Schubert return (s);
48341edb306SCy Schubert }
48441edb306SCy Schubert
48541edb306SCy Schubert
486efeb8bffSCy Schubert static char *
icmpname(u_int type,u_int code)487efeb8bffSCy Schubert icmpname(u_int type, u_int code)
48841edb306SCy Schubert {
48941edb306SCy Schubert static char name[80];
49041edb306SCy Schubert icmp_subtype_t *ist;
49141edb306SCy Schubert icmp_type_t *it;
49241edb306SCy Schubert char *s;
49341edb306SCy Schubert
49441edb306SCy Schubert s = NULL;
49541edb306SCy Schubert it = find_icmptype(type, icmptypes4, sizeof(icmptypes4) / sizeof(*it));
49641edb306SCy Schubert if (it != NULL)
49741edb306SCy Schubert s = it->it_name;
49841edb306SCy Schubert
49941edb306SCy Schubert if (s == NULL)
500d740faa3SCy Schubert sprintf(name, "icmptype(%d)/", type);
50141edb306SCy Schubert else
502d740faa3SCy Schubert sprintf(name, "%s/", s);
50341edb306SCy Schubert
50441edb306SCy Schubert ist = NULL;
50541edb306SCy Schubert if (it != NULL && it->it_subtable != NULL)
50641edb306SCy Schubert ist = find_icmpsubtype(code, it->it_subtable, it->it_stsize);
50741edb306SCy Schubert
50841edb306SCy Schubert if (ist != NULL && ist->ist_name != NULL)
50941edb306SCy Schubert strcat(name, ist->ist_name);
510d740faa3SCy Schubert else
511d740faa3SCy Schubert sprintf(name + strlen(name), "%d", code);
51241edb306SCy Schubert
5132582ae57SCy Schubert return (name);
51441edb306SCy Schubert }
51541edb306SCy Schubert
516efeb8bffSCy Schubert static char *
icmpname6(u_int type,u_int code)517efeb8bffSCy Schubert icmpname6(u_int type, u_int code)
51841edb306SCy Schubert {
51941edb306SCy Schubert static char name[80];
52041edb306SCy Schubert icmp_subtype_t *ist;
52141edb306SCy Schubert icmp_type_t *it;
52241edb306SCy Schubert char *s;
52341edb306SCy Schubert
52441edb306SCy Schubert s = NULL;
52541edb306SCy Schubert it = find_icmptype(type, icmptypes6, sizeof(icmptypes6) / sizeof(*it));
52641edb306SCy Schubert if (it != NULL)
52741edb306SCy Schubert s = it->it_name;
52841edb306SCy Schubert
52941edb306SCy Schubert if (s == NULL)
530d740faa3SCy Schubert sprintf(name, "icmpv6type(%d)/", type);
53141edb306SCy Schubert else
532d740faa3SCy Schubert sprintf(name, "%s/", s);
53341edb306SCy Schubert
53441edb306SCy Schubert ist = NULL;
53541edb306SCy Schubert if (it != NULL && it->it_subtable != NULL)
53641edb306SCy Schubert ist = find_icmpsubtype(code, it->it_subtable, it->it_stsize);
53741edb306SCy Schubert
53841edb306SCy Schubert if (ist != NULL && ist->ist_name != NULL)
53941edb306SCy Schubert strcat(name, ist->ist_name);
540d740faa3SCy Schubert else
541d740faa3SCy Schubert sprintf(name + strlen(name), "%d", code);
54241edb306SCy Schubert
5432582ae57SCy Schubert return (name);
54441edb306SCy Schubert }
54541edb306SCy Schubert
54641edb306SCy Schubert
547efeb8bffSCy Schubert void
dumphex(FILE * log,int dopts,char * buf,int len)548efeb8bffSCy Schubert dumphex(FILE *log, int dopts, char *buf, int len)
54941edb306SCy Schubert {
55041edb306SCy Schubert char hline[80];
55141edb306SCy Schubert int i, j, k;
55241edb306SCy Schubert u_char *s = (u_char *)buf, *t = (u_char *)hline;
55341edb306SCy Schubert
55441edb306SCy Schubert if (buf == NULL || len == 0)
55541edb306SCy Schubert return;
55641edb306SCy Schubert
55741edb306SCy Schubert *hline = '\0';
55841edb306SCy Schubert
55941edb306SCy Schubert for (i = len, j = 0; i; i--, j++, s++) {
56041edb306SCy Schubert if (j && !(j & 0xf)) {
56141edb306SCy Schubert *t++ = '\n';
56241edb306SCy Schubert *t = '\0';
56341edb306SCy Schubert if ((dopts & IPMON_SYSLOG))
56441edb306SCy Schubert syslog(LOG_INFO, "%s", hline);
56541edb306SCy Schubert else if (log != NULL)
56641edb306SCy Schubert fputs(hline, log);
56741edb306SCy Schubert t = (u_char *)hline;
56841edb306SCy Schubert *t = '\0';
56941edb306SCy Schubert }
57041edb306SCy Schubert sprintf((char *)t, "%02x", *s & 0xff);
57141edb306SCy Schubert t += 2;
57241edb306SCy Schubert if (!((j + 1) & 0xf)) {
57341edb306SCy Schubert s -= 15;
57441edb306SCy Schubert sprintf((char *)t, " ");
57541edb306SCy Schubert t += 8;
57641edb306SCy Schubert for (k = 16; k; k--, s++)
57741edb306SCy Schubert *t++ = (isprint(*s) ? *s : '.');
57841edb306SCy Schubert s--;
57941edb306SCy Schubert }
58041edb306SCy Schubert
58141edb306SCy Schubert if ((j + 1) & 0xf)
582540be39bSElyes Haouas *t++ = ' ';
58341edb306SCy Schubert }
58441edb306SCy Schubert
58541edb306SCy Schubert if (j & 0xf) {
58641edb306SCy Schubert for (k = 16 - (j & 0xf); k; k--) {
58741edb306SCy Schubert *t++ = ' ';
58841edb306SCy Schubert *t++ = ' ';
58941edb306SCy Schubert *t++ = ' ';
59041edb306SCy Schubert }
59141edb306SCy Schubert sprintf((char *)t, " ");
59241edb306SCy Schubert t += 7;
59341edb306SCy Schubert s -= j & 0xf;
59441edb306SCy Schubert for (k = j & 0xf; k; k--, s++)
59541edb306SCy Schubert *t++ = (isprint(*s) ? *s : '.');
59641edb306SCy Schubert *t++ = '\n';
59741edb306SCy Schubert *t = '\0';
59841edb306SCy Schubert }
59941edb306SCy Schubert if ((dopts & IPMON_SYSLOG) != 0)
60041edb306SCy Schubert syslog(LOG_INFO, "%s", hline);
60141edb306SCy Schubert else if (log != NULL) {
60241edb306SCy Schubert fputs(hline, log);
60341edb306SCy Schubert fflush(log);
60441edb306SCy Schubert }
60541edb306SCy Schubert }
60641edb306SCy Schubert
60741edb306SCy Schubert
608efeb8bffSCy Schubert static struct tm *
get_tm(time_t sec)609efeb8bffSCy Schubert get_tm(time_t sec)
61041edb306SCy Schubert {
61141edb306SCy Schubert struct tm *tm;
61241edb306SCy Schubert time_t t;
61341edb306SCy Schubert
61441edb306SCy Schubert t = sec;
61541edb306SCy Schubert tm = localtime(&t);
6162582ae57SCy Schubert return (tm);
61741edb306SCy Schubert }
61841edb306SCy Schubert
619efeb8bffSCy Schubert static void
print_natlog(config_t * conf,char * buf,int blen)620efeb8bffSCy Schubert print_natlog(config_t *conf, char *buf, int blen)
62141edb306SCy Schubert {
62241edb306SCy Schubert static u_32_t seqnum = 0;
62341edb306SCy Schubert int res, i, len, family;
62441edb306SCy Schubert struct natlog *nl;
62541edb306SCy Schubert struct tm *tm;
62641edb306SCy Schubert iplog_t *ipl;
62741edb306SCy Schubert char *proto;
62841edb306SCy Schubert int simple;
62941edb306SCy Schubert char *t;
63041edb306SCy Schubert
63141edb306SCy Schubert t = line;
63241edb306SCy Schubert simple = 0;
63341edb306SCy Schubert ipl = (iplog_t *)buf;
63441edb306SCy Schubert if (ipl->ipl_seqnum != seqnum) {
63541edb306SCy Schubert if ((ipmonopts & IPMON_SYSLOG) != 0) {
63641edb306SCy Schubert syslog(LOG_WARNING,
63741edb306SCy Schubert "missed %u NAT log entries: %u %u",
63841edb306SCy Schubert ipl->ipl_seqnum - seqnum, seqnum,
63941edb306SCy Schubert ipl->ipl_seqnum);
64041edb306SCy Schubert } else {
64141edb306SCy Schubert (void) fprintf(conf->log,
64241edb306SCy Schubert "missed %u NAT log entries: %u %u\n",
64341edb306SCy Schubert ipl->ipl_seqnum - seqnum, seqnum,
64441edb306SCy Schubert ipl->ipl_seqnum);
64541edb306SCy Schubert }
64641edb306SCy Schubert }
64741edb306SCy Schubert seqnum = ipl->ipl_seqnum + ipl->ipl_count;
64841edb306SCy Schubert
64941edb306SCy Schubert nl = (struct natlog *)((char *)ipl + sizeof(*ipl));
65041edb306SCy Schubert res = (ipmonopts & IPMON_RESOLVE) ? 1 : 0;
65141edb306SCy Schubert tm = get_tm(ipl->ipl_sec);
65241edb306SCy Schubert len = sizeof(line);
65341edb306SCy Schubert
65441edb306SCy Schubert if (!(ipmonopts & IPMON_SYSLOG)) {
65541edb306SCy Schubert (void) strftime(t, len, "%d/%m/%Y ", tm);
65641edb306SCy Schubert i = strlen(t);
65741edb306SCy Schubert len -= i;
65841edb306SCy Schubert t += i;
65941edb306SCy Schubert }
66041edb306SCy Schubert (void) strftime(t, len, "%T", tm);
66141edb306SCy Schubert t += strlen(t);
662d740faa3SCy Schubert sprintf(t, ".%-.6ld @%hd ", (long)ipl->ipl_usec, nl->nl_rule + 1);
66341edb306SCy Schubert t += strlen(t);
66441edb306SCy Schubert
66541edb306SCy Schubert switch (nl->nl_action)
66641edb306SCy Schubert {
66741edb306SCy Schubert case NL_NEW :
66841edb306SCy Schubert strcpy(t, "NAT:NEW");
66941edb306SCy Schubert break;
67041edb306SCy Schubert
67141edb306SCy Schubert case NL_FLUSH :
67241edb306SCy Schubert strcpy(t, "NAT:FLUSH");
67341edb306SCy Schubert break;
67441edb306SCy Schubert
67541edb306SCy Schubert case NL_CLONE :
67641edb306SCy Schubert strcpy(t, "NAT:CLONE");
67741edb306SCy Schubert break;
67841edb306SCy Schubert
67941edb306SCy Schubert case NL_EXPIRE :
68041edb306SCy Schubert strcpy(t, "NAT:EXPIRE");
68141edb306SCy Schubert break;
68241edb306SCy Schubert
68341edb306SCy Schubert case NL_DESTROY :
68441edb306SCy Schubert strcpy(t, "NAT:DESTROY");
68541edb306SCy Schubert break;
68641edb306SCy Schubert
68741edb306SCy Schubert case NL_PURGE :
68841edb306SCy Schubert strcpy(t, "NAT:PURGE");
68941edb306SCy Schubert break;
69041edb306SCy Schubert
69141edb306SCy Schubert default :
692d740faa3SCy Schubert sprintf(t, "NAT:Action(%d)", nl->nl_action);
69341edb306SCy Schubert break;
69441edb306SCy Schubert }
69541edb306SCy Schubert t += strlen(t);
69641edb306SCy Schubert
69741edb306SCy Schubert
69841edb306SCy Schubert switch (nl->nl_type)
69941edb306SCy Schubert {
70041edb306SCy Schubert case NAT_MAP :
70141edb306SCy Schubert strcpy(t, "-MAP ");
70241edb306SCy Schubert simple = 1;
70341edb306SCy Schubert break;
70441edb306SCy Schubert
70541edb306SCy Schubert case NAT_REDIRECT :
70641edb306SCy Schubert strcpy(t, "-RDR ");
70741edb306SCy Schubert simple = 1;
70841edb306SCy Schubert break;
70941edb306SCy Schubert
71041edb306SCy Schubert case NAT_BIMAP :
71141edb306SCy Schubert strcpy(t, "-BIMAP ");
71241edb306SCy Schubert simple = 1;
71341edb306SCy Schubert break;
71441edb306SCy Schubert
71541edb306SCy Schubert case NAT_MAPBLK :
71641edb306SCy Schubert strcpy(t, "-MAPBLOCK ");
71741edb306SCy Schubert simple = 1;
71841edb306SCy Schubert break;
71941edb306SCy Schubert
72041edb306SCy Schubert case NAT_REWRITE|NAT_MAP :
72141edb306SCy Schubert strcpy(t, "-RWR_MAP ");
72241edb306SCy Schubert break;
72341edb306SCy Schubert
72441edb306SCy Schubert case NAT_REWRITE|NAT_REDIRECT :
72541edb306SCy Schubert strcpy(t, "-RWR_RDR ");
72641edb306SCy Schubert break;
72741edb306SCy Schubert
72841edb306SCy Schubert case NAT_ENCAP|NAT_MAP :
72941edb306SCy Schubert strcpy(t, "-ENC_MAP ");
73041edb306SCy Schubert break;
73141edb306SCy Schubert
73241edb306SCy Schubert case NAT_ENCAP|NAT_REDIRECT :
73341edb306SCy Schubert strcpy(t, "-ENC_RDR ");
73441edb306SCy Schubert break;
73541edb306SCy Schubert
73641edb306SCy Schubert case NAT_DIVERTUDP|NAT_MAP :
73741edb306SCy Schubert strcpy(t, "-DIV_MAP ");
73841edb306SCy Schubert break;
73941edb306SCy Schubert
74041edb306SCy Schubert case NAT_DIVERTUDP|NAT_REDIRECT :
74141edb306SCy Schubert strcpy(t, "-DIV_RDR ");
74241edb306SCy Schubert break;
74341edb306SCy Schubert
74441edb306SCy Schubert default :
745d740faa3SCy Schubert sprintf(t, "-Type(%d) ", nl->nl_type);
74641edb306SCy Schubert break;
74741edb306SCy Schubert }
74841edb306SCy Schubert t += strlen(t);
74941edb306SCy Schubert
75041edb306SCy Schubert proto = getlocalproto(nl->nl_p[0]);
75141edb306SCy Schubert
75241edb306SCy Schubert family = vtof(nl->nl_v[0]);
75341edb306SCy Schubert
75441edb306SCy Schubert if (simple == 1) {
755d740faa3SCy Schubert sprintf(t, "%s,%s <- -> ", hostname(family, nl->nl_osrcip.i6),
75641edb306SCy Schubert portlocalname(res, proto, (u_int)nl->nl_osrcport));
75741edb306SCy Schubert t += strlen(t);
758d740faa3SCy Schubert sprintf(t, "%s,%s ", hostname(family, nl->nl_nsrcip.i6),
75941edb306SCy Schubert portlocalname(res, proto, (u_int)nl->nl_nsrcport));
76041edb306SCy Schubert t += strlen(t);
761d740faa3SCy Schubert sprintf(t, "[%s,%s] ", hostname(family, nl->nl_odstip.i6),
76241edb306SCy Schubert portlocalname(res, proto, (u_int)nl->nl_odstport));
76341edb306SCy Schubert } else {
764d740faa3SCy Schubert sprintf(t, "%s,%s ", hostname(family, nl->nl_osrcip.i6),
76541edb306SCy Schubert portlocalname(res, proto, (u_int)nl->nl_osrcport));
76641edb306SCy Schubert t += strlen(t);
767d740faa3SCy Schubert sprintf(t, "%s,%s <- -> ", hostname(family, nl->nl_odstip.i6),
76841edb306SCy Schubert portlocalname(res, proto, (u_int)nl->nl_odstport));
76941edb306SCy Schubert t += strlen(t);
770d740faa3SCy Schubert sprintf(t, "%s,%s ", hostname(family, nl->nl_nsrcip.i6),
77141edb306SCy Schubert portlocalname(res, proto, (u_int)nl->nl_nsrcport));
77241edb306SCy Schubert t += strlen(t);
773d740faa3SCy Schubert sprintf(t, "%s,%s ", hostname(family, nl->nl_ndstip.i6),
77441edb306SCy Schubert portlocalname(res, proto, (u_int)nl->nl_ndstport));
77541edb306SCy Schubert }
77641edb306SCy Schubert t += strlen(t);
77741edb306SCy Schubert
77841edb306SCy Schubert strcpy(t, getlocalproto(nl->nl_p[0]));
77941edb306SCy Schubert t += strlen(t);
78041edb306SCy Schubert
78141edb306SCy Schubert if (nl->nl_action == NL_EXPIRE || nl->nl_action == NL_FLUSH) {
78241edb306SCy Schubert #ifdef USE_QUAD_T
78341edb306SCy Schubert # ifdef PRId64
784d740faa3SCy Schubert sprintf(t, " Pkts %" PRId64 "/%" PRId64 " Bytes %" PRId64 "/%"
78541edb306SCy Schubert PRId64,
78641edb306SCy Schubert # else
787d740faa3SCy Schubert sprintf(t, " Pkts %qd/%qd Bytes %qd/%qd",
78841edb306SCy Schubert # endif
78941edb306SCy Schubert #else
790d740faa3SCy Schubert sprintf(t, " Pkts %ld/%ld Bytes %ld/%ld",
79141edb306SCy Schubert #endif
79241edb306SCy Schubert nl->nl_pkts[0], nl->nl_pkts[1],
79341edb306SCy Schubert nl->nl_bytes[0], nl->nl_bytes[1]);
79441edb306SCy Schubert t += strlen(t);
79541edb306SCy Schubert }
79641edb306SCy Schubert
79741edb306SCy Schubert *t++ = '\n';
79841edb306SCy Schubert *t++ = '\0';
79941edb306SCy Schubert if (ipmonopts & IPMON_SYSLOG)
80041edb306SCy Schubert syslog(LOG_INFO, "%s", line);
80141edb306SCy Schubert else if (conf->log != NULL)
80241edb306SCy Schubert (void) fprintf(conf->log, "%s", line);
80341edb306SCy Schubert }
80441edb306SCy Schubert
80541edb306SCy Schubert
806efeb8bffSCy Schubert static void
print_statelog(config_t * conf,char * buf,int blen)807efeb8bffSCy Schubert print_statelog(config_t *conf, char *buf, int blen)
80841edb306SCy Schubert {
80941edb306SCy Schubert static u_32_t seqnum = 0;
81041edb306SCy Schubert int res, i, len, family;
81141edb306SCy Schubert struct ipslog *sl;
81241edb306SCy Schubert char *t, *proto;
81341edb306SCy Schubert struct tm *tm;
81441edb306SCy Schubert iplog_t *ipl;
81541edb306SCy Schubert
81641edb306SCy Schubert t = line;
81741edb306SCy Schubert ipl = (iplog_t *)buf;
81841edb306SCy Schubert if (ipl->ipl_seqnum != seqnum) {
81941edb306SCy Schubert if ((ipmonopts & IPMON_SYSLOG) != 0) {
82041edb306SCy Schubert syslog(LOG_WARNING,
82141edb306SCy Schubert "missed %u state log entries: %u %u",
82241edb306SCy Schubert ipl->ipl_seqnum - seqnum, seqnum,
82341edb306SCy Schubert ipl->ipl_seqnum);
82441edb306SCy Schubert } else {
82541edb306SCy Schubert (void) fprintf(conf->log,
82641edb306SCy Schubert "missed %u state log entries: %u %u\n",
82741edb306SCy Schubert ipl->ipl_seqnum - seqnum, seqnum,
82841edb306SCy Schubert ipl->ipl_seqnum);
82941edb306SCy Schubert }
83041edb306SCy Schubert }
83141edb306SCy Schubert seqnum = ipl->ipl_seqnum + ipl->ipl_count;
83241edb306SCy Schubert
83341edb306SCy Schubert sl = (struct ipslog *)((char *)ipl + sizeof(*ipl));
83441edb306SCy Schubert res = (ipmonopts & IPMON_RESOLVE) ? 1 : 0;
83541edb306SCy Schubert tm = get_tm(ipl->ipl_sec);
83641edb306SCy Schubert len = sizeof(line);
83741edb306SCy Schubert if (!(ipmonopts & IPMON_SYSLOG)) {
83841edb306SCy Schubert (void) strftime(t, len, "%d/%m/%Y ", tm);
83941edb306SCy Schubert i = strlen(t);
84041edb306SCy Schubert len -= i;
84141edb306SCy Schubert t += i;
84241edb306SCy Schubert }
84341edb306SCy Schubert (void) strftime(t, len, "%T", tm);
84441edb306SCy Schubert t += strlen(t);
845d740faa3SCy Schubert sprintf(t, ".%-.6ld ", (long)ipl->ipl_usec);
84641edb306SCy Schubert t += strlen(t);
84741edb306SCy Schubert
84841edb306SCy Schubert family = vtof(sl->isl_v);
84941edb306SCy Schubert
85041edb306SCy Schubert switch (sl->isl_type)
85141edb306SCy Schubert {
85241edb306SCy Schubert case ISL_NEW :
85341edb306SCy Schubert strcpy(t, "STATE:NEW ");
85441edb306SCy Schubert break;
85541edb306SCy Schubert
85641edb306SCy Schubert case ISL_CLONE :
85741edb306SCy Schubert strcpy(t, "STATE:CLONED ");
85841edb306SCy Schubert break;
85941edb306SCy Schubert
86041edb306SCy Schubert case ISL_EXPIRE :
86141edb306SCy Schubert if ((sl->isl_p == IPPROTO_TCP) &&
86241edb306SCy Schubert (sl->isl_state[0] > IPF_TCPS_ESTABLISHED ||
86341edb306SCy Schubert sl->isl_state[1] > IPF_TCPS_ESTABLISHED))
86441edb306SCy Schubert strcpy(t, "STATE:CLOSE ");
86541edb306SCy Schubert else
86641edb306SCy Schubert strcpy(t, "STATE:EXPIRE ");
86741edb306SCy Schubert break;
86841edb306SCy Schubert
86941edb306SCy Schubert case ISL_FLUSH :
87041edb306SCy Schubert strcpy(t, "STATE:FLUSH ");
87141edb306SCy Schubert break;
87241edb306SCy Schubert
87341edb306SCy Schubert case ISL_INTERMEDIATE :
87441edb306SCy Schubert strcpy(t, "STATE:INTERMEDIATE ");
87541edb306SCy Schubert break;
87641edb306SCy Schubert
87741edb306SCy Schubert case ISL_REMOVE :
87841edb306SCy Schubert strcpy(t, "STATE:REMOVE ");
87941edb306SCy Schubert break;
88041edb306SCy Schubert
88141edb306SCy Schubert case ISL_KILLED :
88241edb306SCy Schubert strcpy(t, "STATE:KILLED ");
88341edb306SCy Schubert break;
88441edb306SCy Schubert
88541edb306SCy Schubert case ISL_UNLOAD :
88641edb306SCy Schubert strcpy(t, "STATE:UNLOAD ");
88741edb306SCy Schubert break;
88841edb306SCy Schubert
88941edb306SCy Schubert default :
890d740faa3SCy Schubert sprintf(t, "Type: %d ", sl->isl_type);
89141edb306SCy Schubert break;
89241edb306SCy Schubert }
89341edb306SCy Schubert t += strlen(t);
89441edb306SCy Schubert
89541edb306SCy Schubert proto = getlocalproto(sl->isl_p);
89641edb306SCy Schubert
89741edb306SCy Schubert if (sl->isl_p == IPPROTO_TCP || sl->isl_p == IPPROTO_UDP) {
898d740faa3SCy Schubert sprintf(t, "%s,%s -> ",
89941edb306SCy Schubert hostname(family, (u_32_t *)&sl->isl_src),
90041edb306SCy Schubert portlocalname(res, proto, (u_int)sl->isl_sport));
90141edb306SCy Schubert t += strlen(t);
902d740faa3SCy Schubert sprintf(t, "%s,%s PR %s",
90341edb306SCy Schubert hostname(family, (u_32_t *)&sl->isl_dst),
90441edb306SCy Schubert portlocalname(res, proto, (u_int)sl->isl_dport), proto);
90541edb306SCy Schubert } else if (sl->isl_p == IPPROTO_ICMP) {
906d740faa3SCy Schubert sprintf(t, "%s -> ", hostname(family, (u_32_t *)&sl->isl_src));
90741edb306SCy Schubert t += strlen(t);
908d740faa3SCy Schubert sprintf(t, "%s PR icmp %d",
90941edb306SCy Schubert hostname(family, (u_32_t *)&sl->isl_dst),
91041edb306SCy Schubert sl->isl_itype);
91141edb306SCy Schubert } else if (sl->isl_p == IPPROTO_ICMPV6) {
912d740faa3SCy Schubert sprintf(t, "%s -> ", hostname(family, (u_32_t *)&sl->isl_src));
91341edb306SCy Schubert t += strlen(t);
914d740faa3SCy Schubert sprintf(t, "%s PR icmpv6 %d",
91541edb306SCy Schubert hostname(family, (u_32_t *)&sl->isl_dst),
91641edb306SCy Schubert sl->isl_itype);
91741edb306SCy Schubert } else {
918d740faa3SCy Schubert sprintf(t, "%s -> ", hostname(family, (u_32_t *)&sl->isl_src));
91941edb306SCy Schubert t += strlen(t);
920d740faa3SCy Schubert sprintf(t, "%s PR %s",
92141edb306SCy Schubert hostname(family, (u_32_t *)&sl->isl_dst), proto);
92241edb306SCy Schubert }
92341edb306SCy Schubert t += strlen(t);
92441edb306SCy Schubert if (sl->isl_tag != FR_NOLOGTAG) {
925d740faa3SCy Schubert sprintf(t, " tag %u", sl->isl_tag);
92641edb306SCy Schubert t += strlen(t);
92741edb306SCy Schubert }
92841edb306SCy Schubert if (sl->isl_type != ISL_NEW) {
929d740faa3SCy Schubert sprintf(t,
93041edb306SCy Schubert #ifdef USE_QUAD_T
93141edb306SCy Schubert #ifdef PRId64
93241edb306SCy Schubert " Forward: Pkts in %" PRId64 " Bytes in %" PRId64
93341edb306SCy Schubert " Pkts out %" PRId64 " Bytes out %" PRId64
93441edb306SCy Schubert " Backward: Pkts in %" PRId64 " Bytes in %" PRId64
93541edb306SCy Schubert " Pkts out %" PRId64 " Bytes out %" PRId64,
93641edb306SCy Schubert #else
93741edb306SCy Schubert " Forward: Pkts in %qd Bytes in %qd Pkts out %qd Bytes out %qd Backward: Pkts in %qd Bytes in %qd Pkts out %qd Bytes out %qd",
93841edb306SCy Schubert #endif /* PRId64 */
93941edb306SCy Schubert #else
94041edb306SCy Schubert " Forward: Pkts in %ld Bytes in %ld Pkts out %ld Bytes out %ld Backward: Pkts in %ld Bytes in %ld Pkts out %ld Bytes out %ld",
94141edb306SCy Schubert #endif
94241edb306SCy Schubert sl->isl_pkts[0], sl->isl_bytes[0],
94341edb306SCy Schubert sl->isl_pkts[1], sl->isl_bytes[1],
94441edb306SCy Schubert sl->isl_pkts[2], sl->isl_bytes[2],
94541edb306SCy Schubert sl->isl_pkts[3], sl->isl_bytes[3]);
94641edb306SCy Schubert
94741edb306SCy Schubert t += strlen(t);
94841edb306SCy Schubert }
94941edb306SCy Schubert
95041edb306SCy Schubert *t++ = '\n';
95141edb306SCy Schubert *t++ = '\0';
95241edb306SCy Schubert if (ipmonopts & IPMON_SYSLOG)
95341edb306SCy Schubert syslog(LOG_INFO, "%s", line);
95441edb306SCy Schubert else if (conf->log != NULL)
95541edb306SCy Schubert (void) fprintf(conf->log, "%s", line);
95641edb306SCy Schubert }
95741edb306SCy Schubert
95841edb306SCy Schubert
959efeb8bffSCy Schubert static void
print_log(config_t * conf,logsource_t * log,char * buf,int blen)960efeb8bffSCy Schubert print_log(config_t *conf, logsource_t *log, char *buf, int blen)
96141edb306SCy Schubert {
96241edb306SCy Schubert char *bp, *bpo;
96341edb306SCy Schubert iplog_t *ipl;
96441edb306SCy Schubert int psize;
96541edb306SCy Schubert
96641edb306SCy Schubert bp = NULL;
96741edb306SCy Schubert bpo = NULL;
96841edb306SCy Schubert
96941edb306SCy Schubert while (blen > 0) {
97041edb306SCy Schubert ipl = (iplog_t *)buf;
97141edb306SCy Schubert if ((u_long)ipl & (sizeof(long)-1)) {
97241edb306SCy Schubert if (bp)
97341edb306SCy Schubert bpo = bp;
97441edb306SCy Schubert bp = (char *)malloc(blen);
97541edb306SCy Schubert bcopy((char *)ipl, bp, blen);
97641edb306SCy Schubert if (bpo) {
97741edb306SCy Schubert free(bpo);
97841edb306SCy Schubert bpo = NULL;
97941edb306SCy Schubert }
98041edb306SCy Schubert buf = bp;
98141edb306SCy Schubert continue;
98241edb306SCy Schubert }
98341edb306SCy Schubert
98441edb306SCy Schubert psize = ipl->ipl_dsize;
98541edb306SCy Schubert if (psize > blen)
98641edb306SCy Schubert break;
98741edb306SCy Schubert
98841edb306SCy Schubert if (conf->blog != NULL) {
98941edb306SCy Schubert fwrite(buf, psize, 1, conf->blog);
99041edb306SCy Schubert fflush(conf->blog);
99141edb306SCy Schubert }
99241edb306SCy Schubert
99341edb306SCy Schubert if (log->logtype == IPL_LOGIPF) {
99441edb306SCy Schubert if (ipl->ipl_magic == IPL_MAGIC)
99541edb306SCy Schubert print_ipflog(conf, buf, psize);
99641edb306SCy Schubert
99741edb306SCy Schubert } else if (log->logtype == IPL_LOGNAT) {
99841edb306SCy Schubert if (ipl->ipl_magic == IPL_MAGIC_NAT)
99941edb306SCy Schubert print_natlog(conf, buf, psize);
100041edb306SCy Schubert
100141edb306SCy Schubert } else if (log->logtype == IPL_LOGSTATE) {
100241edb306SCy Schubert if (ipl->ipl_magic == IPL_MAGIC_STATE)
100341edb306SCy Schubert print_statelog(conf, buf, psize);
100441edb306SCy Schubert }
100541edb306SCy Schubert
100641edb306SCy Schubert blen -= psize;
100741edb306SCy Schubert buf += psize;
100841edb306SCy Schubert }
100941edb306SCy Schubert if (bp)
101041edb306SCy Schubert free(bp);
101141edb306SCy Schubert return;
101241edb306SCy Schubert }
101341edb306SCy Schubert
101441edb306SCy Schubert
1015efeb8bffSCy Schubert static void
print_ipflog(config_t * conf,char * buf,int blen)1016efeb8bffSCy Schubert print_ipflog(config_t *conf, char *buf, int blen)
101741edb306SCy Schubert {
101841edb306SCy Schubert static u_32_t seqnum = 0;
101941edb306SCy Schubert int i, f, lvl, res, len, off, plen, ipoff, defaction;
102041edb306SCy Schubert struct icmp *icmp;
102141edb306SCy Schubert struct icmp *ic;
102241edb306SCy Schubert char *t, *proto;
102341edb306SCy Schubert ip_t *ipc, *ip;
102441edb306SCy Schubert struct tm *tm;
102541edb306SCy Schubert u_32_t *s, *d;
102641edb306SCy Schubert u_short hl, p;
102741edb306SCy Schubert ipflog_t *ipf;
102841edb306SCy Schubert iplog_t *ipl;
102941edb306SCy Schubert tcphdr_t *tp;
103041edb306SCy Schubert #ifdef USE_INET6
103141edb306SCy Schubert struct ip6_ext *ehp;
103241edb306SCy Schubert u_short ehl;
103341edb306SCy Schubert ip6_t *ip6;
103441edb306SCy Schubert int go;
103541edb306SCy Schubert #endif
103641edb306SCy Schubert
103741edb306SCy Schubert ipl = (iplog_t *)buf;
103841edb306SCy Schubert if (ipl->ipl_seqnum != seqnum) {
103941edb306SCy Schubert if ((ipmonopts & IPMON_SYSLOG) != 0) {
104041edb306SCy Schubert syslog(LOG_WARNING,
104141edb306SCy Schubert "missed %u ipf log entries: %u %u",
104241edb306SCy Schubert ipl->ipl_seqnum - seqnum, seqnum,
104341edb306SCy Schubert ipl->ipl_seqnum);
104441edb306SCy Schubert } else {
104541edb306SCy Schubert (void) fprintf(conf->log,
104641edb306SCy Schubert "missed %u ipf log entries: %u %u\n",
104741edb306SCy Schubert ipl->ipl_seqnum - seqnum, seqnum,
104841edb306SCy Schubert ipl->ipl_seqnum);
104941edb306SCy Schubert }
105041edb306SCy Schubert }
105141edb306SCy Schubert seqnum = ipl->ipl_seqnum + ipl->ipl_count;
105241edb306SCy Schubert
105341edb306SCy Schubert ipf = (ipflog_t *)((char *)buf + sizeof(*ipl));
105441edb306SCy Schubert ip = (ip_t *)((char *)ipf + sizeof(*ipf));
105541edb306SCy Schubert f = ipf->fl_family;
105641edb306SCy Schubert res = (ipmonopts & IPMON_RESOLVE) ? 1 : 0;
105741edb306SCy Schubert t = line;
105841edb306SCy Schubert *t = '\0';
105941edb306SCy Schubert tm = get_tm(ipl->ipl_sec);
106041edb306SCy Schubert
106141edb306SCy Schubert len = sizeof(line);
106241edb306SCy Schubert if (!(ipmonopts & IPMON_SYSLOG)) {
106341edb306SCy Schubert (void) strftime(t, len, "%d/%m/%Y ", tm);
106441edb306SCy Schubert i = strlen(t);
106541edb306SCy Schubert len -= i;
106641edb306SCy Schubert t += i;
106741edb306SCy Schubert }
106841edb306SCy Schubert (void) strftime(t, len, "%T", tm);
106941edb306SCy Schubert t += strlen(t);
1070d740faa3SCy Schubert sprintf(t, ".%-.6ld ", (long)ipl->ipl_usec);
107141edb306SCy Schubert t += strlen(t);
107241edb306SCy Schubert if (ipl->ipl_count > 1) {
1073d740faa3SCy Schubert sprintf(t, "%dx ", ipl->ipl_count);
107441edb306SCy Schubert t += strlen(t);
107541edb306SCy Schubert }
107641edb306SCy Schubert {
107741edb306SCy Schubert char ifname[sizeof(ipf->fl_ifname) + 1];
107841edb306SCy Schubert
107941edb306SCy Schubert strncpy(ifname, ipf->fl_ifname, sizeof(ipf->fl_ifname));
108041edb306SCy Schubert ifname[sizeof(ipf->fl_ifname)] = '\0';
1081d740faa3SCy Schubert sprintf(t, "%s", ifname);
108241edb306SCy Schubert t += strlen(t);
108341edb306SCy Schubert # if SOLARIS
108441edb306SCy Schubert if (ISALPHA(*(t - 1))) {
1085d740faa3SCy Schubert sprintf(t, "%d", ipf->fl_unit);
108641edb306SCy Schubert t += strlen(t);
108741edb306SCy Schubert }
108841edb306SCy Schubert # endif
108941edb306SCy Schubert }
109041edb306SCy Schubert if ((ipf->fl_group[0] == (char)~0) && (ipf->fl_group[1] == '\0'))
109141edb306SCy Schubert strcat(t, " @-1:");
109241edb306SCy Schubert else if (ipf->fl_group[0] == '\0')
109341edb306SCy Schubert (void) strcpy(t, " @0:");
109441edb306SCy Schubert else
1095d740faa3SCy Schubert sprintf(t, " @%s:", ipf->fl_group);
109641edb306SCy Schubert t += strlen(t);
109741edb306SCy Schubert if (ipf->fl_rule == 0xffffffff)
109841edb306SCy Schubert strcat(t, "-1 ");
109941edb306SCy Schubert else
1100d740faa3SCy Schubert sprintf(t, "%u ", ipf->fl_rule + 1);
110141edb306SCy Schubert t += strlen(t);
110241edb306SCy Schubert
110341edb306SCy Schubert lvl = LOG_NOTICE;
110441edb306SCy Schubert
110541edb306SCy Schubert if (ipf->fl_lflags & FI_SHORT) {
110641edb306SCy Schubert *t++ = 'S';
110741edb306SCy Schubert lvl = LOG_ERR;
110841edb306SCy Schubert }
110941edb306SCy Schubert
111041edb306SCy Schubert if (FR_ISPASS(ipf->fl_flags)) {
111141edb306SCy Schubert if (ipf->fl_flags & FR_LOGP)
111241edb306SCy Schubert *t++ = 'p';
111341edb306SCy Schubert else
111441edb306SCy Schubert *t++ = 'P';
111541edb306SCy Schubert } else if (FR_ISBLOCK(ipf->fl_flags)) {
111641edb306SCy Schubert if (ipf->fl_flags & FR_LOGB)
111741edb306SCy Schubert *t++ = 'b';
111841edb306SCy Schubert else
111941edb306SCy Schubert *t++ = 'B';
112041edb306SCy Schubert lvl = LOG_WARNING;
112141edb306SCy Schubert } else if ((ipf->fl_flags & FR_LOGMASK) == FR_LOG) {
112241edb306SCy Schubert *t++ = 'L';
112341edb306SCy Schubert lvl = LOG_INFO;
112441edb306SCy Schubert } else if (ipf->fl_flags & FF_LOGNOMATCH) {
112541edb306SCy Schubert *t++ = 'n';
112641edb306SCy Schubert } else {
112741edb306SCy Schubert *t++ = '?';
112841edb306SCy Schubert lvl = LOG_EMERG;
112941edb306SCy Schubert }
113041edb306SCy Schubert if (ipf->fl_loglevel != 0xffff)
113141edb306SCy Schubert lvl = ipf->fl_loglevel;
113241edb306SCy Schubert *t++ = ' ';
113341edb306SCy Schubert *t = '\0';
113441edb306SCy Schubert
113541edb306SCy Schubert if (f == AF_INET) {
113641edb306SCy Schubert hl = IP_HL(ip) << 2;
113741edb306SCy Schubert ipoff = ntohs(ip->ip_off);
113841edb306SCy Schubert off = ipoff & IP_OFFMASK;
113941edb306SCy Schubert p = (u_short)ip->ip_p;
114041edb306SCy Schubert s = (u_32_t *)&ip->ip_src;
114141edb306SCy Schubert d = (u_32_t *)&ip->ip_dst;
114241edb306SCy Schubert plen = ntohs(ip->ip_len);
114341edb306SCy Schubert } else
114441edb306SCy Schubert #ifdef USE_INET6
114541edb306SCy Schubert if (f == AF_INET6) {
114641edb306SCy Schubert off = 0;
114741edb306SCy Schubert ipoff = 0;
114841edb306SCy Schubert hl = sizeof(ip6_t);
114941edb306SCy Schubert ip6 = (ip6_t *)ip;
115041edb306SCy Schubert p = (u_short)ip6->ip6_nxt;
115141edb306SCy Schubert s = (u_32_t *)&ip6->ip6_src;
115241edb306SCy Schubert d = (u_32_t *)&ip6->ip6_dst;
115341edb306SCy Schubert plen = hl + ntohs(ip6->ip6_plen);
115441edb306SCy Schubert go = 1;
115541edb306SCy Schubert ehp = (struct ip6_ext *)((char *)ip6 + hl);
115641edb306SCy Schubert while (go == 1) {
115741edb306SCy Schubert switch (p)
115841edb306SCy Schubert {
115941edb306SCy Schubert case IPPROTO_HOPOPTS :
116041edb306SCy Schubert case IPPROTO_MOBILITY :
116141edb306SCy Schubert case IPPROTO_DSTOPTS :
116241edb306SCy Schubert case IPPROTO_ROUTING :
116341edb306SCy Schubert case IPPROTO_AH :
116441edb306SCy Schubert p = ehp->ip6e_nxt;
116541edb306SCy Schubert ehl = 8 + (ehp->ip6e_len << 3);
116641edb306SCy Schubert hl += ehl;
116741edb306SCy Schubert ehp = (struct ip6_ext *)((char *)ehp + ehl);
116841edb306SCy Schubert break;
116941edb306SCy Schubert case IPPROTO_FRAGMENT :
117041edb306SCy Schubert hl += sizeof(struct ip6_frag);
117141edb306SCy Schubert /* FALLTHROUGH */
117241edb306SCy Schubert default :
117341edb306SCy Schubert go = 0;
117441edb306SCy Schubert break;
117541edb306SCy Schubert }
117641edb306SCy Schubert }
117741edb306SCy Schubert } else
117841edb306SCy Schubert #endif
117941edb306SCy Schubert {
118041edb306SCy Schubert goto printipflog;
118141edb306SCy Schubert }
118241edb306SCy Schubert proto = getlocalproto(p);
118341edb306SCy Schubert
118441edb306SCy Schubert if ((p == IPPROTO_TCP || p == IPPROTO_UDP) && !off) {
118541edb306SCy Schubert tp = (tcphdr_t *)((char *)ip + hl);
118641edb306SCy Schubert if (!(ipf->fl_lflags & FI_SHORT)) {
1187d740faa3SCy Schubert sprintf(t, "%s,%s -> ", hostname(f, s),
118841edb306SCy Schubert portlocalname(res, proto, (u_int)tp->th_sport));
118941edb306SCy Schubert t += strlen(t);
1190d740faa3SCy Schubert sprintf(t, "%s,%s PR %s len %hu %hu",
119141edb306SCy Schubert hostname(f, d),
119241edb306SCy Schubert portlocalname(res, proto, (u_int)tp->th_dport),
119341edb306SCy Schubert proto, hl, plen);
119441edb306SCy Schubert t += strlen(t);
119541edb306SCy Schubert
119641edb306SCy Schubert if (p == IPPROTO_TCP) {
119741edb306SCy Schubert *t++ = ' ';
119841edb306SCy Schubert *t++ = '-';
119941edb306SCy Schubert for (i = 0; tcpfl[i].value; i++)
1200*347dd053SRichard Scheffenegger if (__tcp_get_flags(tp) & tcpfl[i].value)
120141edb306SCy Schubert *t++ = tcpfl[i].flag;
120241edb306SCy Schubert if (ipmonopts & IPMON_VERBOSE) {
1203d740faa3SCy Schubert sprintf(t, " %lu %lu %hu",
120441edb306SCy Schubert (u_long)(ntohl(tp->th_seq)),
120541edb306SCy Schubert (u_long)(ntohl(tp->th_ack)),
120641edb306SCy Schubert ntohs(tp->th_win));
120741edb306SCy Schubert t += strlen(t);
120841edb306SCy Schubert }
120941edb306SCy Schubert }
121041edb306SCy Schubert *t = '\0';
121141edb306SCy Schubert } else {
1212d740faa3SCy Schubert sprintf(t, "%s -> ", hostname(f, s));
121341edb306SCy Schubert t += strlen(t);
121441edb306SCy Schubert sprintf(t, "%s PR %s len %hu %hu",
121541edb306SCy Schubert hostname(f, d), proto, hl, plen);
121641edb306SCy Schubert }
121741edb306SCy Schubert #if defined(AF_INET6) && defined(IPPROTO_ICMPV6)
121841edb306SCy Schubert } else if ((p == IPPROTO_ICMPV6) && !off && (f == AF_INET6)) {
121941edb306SCy Schubert ic = (struct icmp *)((char *)ip + hl);
1220d740faa3SCy Schubert sprintf(t, "%s -> ", hostname(f, s));
122141edb306SCy Schubert t += strlen(t);
1222d740faa3SCy Schubert sprintf(t, "%s PR icmpv6 len %hu %hu icmpv6 %s",
122341edb306SCy Schubert hostname(f, d), hl, plen,
122441edb306SCy Schubert icmpname6(ic->icmp_type, ic->icmp_code));
122541edb306SCy Schubert #endif
122641edb306SCy Schubert } else if ((p == IPPROTO_ICMP) && !off && (f == AF_INET)) {
122741edb306SCy Schubert ic = (struct icmp *)((char *)ip + hl);
1228d740faa3SCy Schubert sprintf(t, "%s -> ", hostname(f, s));
122941edb306SCy Schubert t += strlen(t);
1230d740faa3SCy Schubert sprintf(t, "%s PR icmp len %hu %hu icmp %s",
123141edb306SCy Schubert hostname(f, d), hl, plen,
123241edb306SCy Schubert icmpname(ic->icmp_type, ic->icmp_code));
123341edb306SCy Schubert if (ic->icmp_type == ICMP_UNREACH ||
123441edb306SCy Schubert ic->icmp_type == ICMP_SOURCEQUENCH ||
123541edb306SCy Schubert ic->icmp_type == ICMP_PARAMPROB ||
123641edb306SCy Schubert ic->icmp_type == ICMP_REDIRECT ||
123741edb306SCy Schubert ic->icmp_type == ICMP_TIMXCEED) {
123841edb306SCy Schubert ipc = &ic->icmp_ip;
123941edb306SCy Schubert i = ntohs(ipc->ip_len);
124041edb306SCy Schubert /*
124141edb306SCy Schubert * XXX - try to guess endian of ip_len in ICMP
124241edb306SCy Schubert * returned data.
124341edb306SCy Schubert */
124441edb306SCy Schubert if (i > 1500)
124541edb306SCy Schubert i = ipc->ip_len;
124641edb306SCy Schubert ipoff = ntohs(ipc->ip_off);
124741edb306SCy Schubert proto = getlocalproto(ipc->ip_p);
124841edb306SCy Schubert
124941edb306SCy Schubert if (!(ipoff & IP_OFFMASK) &&
125041edb306SCy Schubert ((ipc->ip_p == IPPROTO_TCP) ||
125141edb306SCy Schubert (ipc->ip_p == IPPROTO_UDP))) {
125241edb306SCy Schubert tp = (tcphdr_t *)((char *)ipc + hl);
125341edb306SCy Schubert t += strlen(t);
1254d740faa3SCy Schubert sprintf(t, " for %s,%s -",
125541edb306SCy Schubert HOSTNAMEV4(ipc->ip_src),
125641edb306SCy Schubert portlocalname(res, proto,
125741edb306SCy Schubert (u_int)tp->th_sport));
125841edb306SCy Schubert t += strlen(t);
1259d740faa3SCy Schubert sprintf(t, " %s,%s PR %s len %hu %hu",
126041edb306SCy Schubert HOSTNAMEV4(ipc->ip_dst),
126141edb306SCy Schubert portlocalname(res, proto,
126241edb306SCy Schubert (u_int)tp->th_dport),
126341edb306SCy Schubert proto, IP_HL(ipc) << 2, i);
126441edb306SCy Schubert } else if (!(ipoff & IP_OFFMASK) &&
126541edb306SCy Schubert (ipc->ip_p == IPPROTO_ICMP)) {
126641edb306SCy Schubert icmp = (icmphdr_t *)((char *)ipc + hl);
126741edb306SCy Schubert
126841edb306SCy Schubert t += strlen(t);
1269d740faa3SCy Schubert sprintf(t, " for %s -",
127041edb306SCy Schubert HOSTNAMEV4(ipc->ip_src));
127141edb306SCy Schubert t += strlen(t);
1272d740faa3SCy Schubert sprintf(t,
127341edb306SCy Schubert " %s PR icmp len %hu %hu icmp %d/%d",
127441edb306SCy Schubert HOSTNAMEV4(ipc->ip_dst),
127541edb306SCy Schubert IP_HL(ipc) << 2, i,
127641edb306SCy Schubert icmp->icmp_type, icmp->icmp_code);
127741edb306SCy Schubert } else {
127841edb306SCy Schubert t += strlen(t);
1279d740faa3SCy Schubert sprintf(t, " for %s -",
128041edb306SCy Schubert HOSTNAMEV4(ipc->ip_src));
128141edb306SCy Schubert t += strlen(t);
1282d740faa3SCy Schubert sprintf(t, " %s PR %s len %hu (%hu)",
128341edb306SCy Schubert HOSTNAMEV4(ipc->ip_dst), proto,
128441edb306SCy Schubert IP_HL(ipc) << 2, i);
128541edb306SCy Schubert t += strlen(t);
128641edb306SCy Schubert if (ipoff & IP_OFFMASK) {
1287d740faa3SCy Schubert sprintf(t, "(frag %d:%hu@%hu%s%s)",
128841edb306SCy Schubert ntohs(ipc->ip_id),
128941edb306SCy Schubert i - (IP_HL(ipc) << 2),
129041edb306SCy Schubert (ipoff & IP_OFFMASK) << 3,
129141edb306SCy Schubert ipoff & IP_MF ? "+" : "",
129241edb306SCy Schubert ipoff & IP_DF ? "-" : "");
129341edb306SCy Schubert }
129441edb306SCy Schubert }
129541edb306SCy Schubert
129641edb306SCy Schubert }
129741edb306SCy Schubert } else {
1298d740faa3SCy Schubert sprintf(t, "%s -> ", hostname(f, s));
129941edb306SCy Schubert t += strlen(t);
1300d740faa3SCy Schubert sprintf(t, "%s PR %s len %hu (%hu)",
130141edb306SCy Schubert hostname(f, d), proto, hl, plen);
130241edb306SCy Schubert t += strlen(t);
130341edb306SCy Schubert if (off & IP_OFFMASK)
1304d740faa3SCy Schubert sprintf(t, " (frag %d:%hu@%hu%s%s)",
130541edb306SCy Schubert ntohs(ip->ip_id),
130641edb306SCy Schubert plen - hl, (off & IP_OFFMASK) << 3,
130741edb306SCy Schubert ipoff & IP_MF ? "+" : "",
130841edb306SCy Schubert ipoff & IP_DF ? "-" : "");
130941edb306SCy Schubert }
131041edb306SCy Schubert t += strlen(t);
131141edb306SCy Schubert
131241edb306SCy Schubert printipflog:
131341edb306SCy Schubert if (ipf->fl_flags & FR_KEEPSTATE) {
131441edb306SCy Schubert (void) strcpy(t, " K-S");
131541edb306SCy Schubert t += strlen(t);
131641edb306SCy Schubert }
131741edb306SCy Schubert
131841edb306SCy Schubert if (ipf->fl_flags & FR_KEEPFRAG) {
131941edb306SCy Schubert (void) strcpy(t, " K-F");
132041edb306SCy Schubert t += strlen(t);
132141edb306SCy Schubert }
132241edb306SCy Schubert
132341edb306SCy Schubert if (ipf->fl_dir == 0)
132441edb306SCy Schubert strcpy(t, " IN");
132541edb306SCy Schubert else if (ipf->fl_dir == 1)
132641edb306SCy Schubert strcpy(t, " OUT");
132741edb306SCy Schubert t += strlen(t);
132841edb306SCy Schubert if (ipf->fl_logtag != 0) {
1329d740faa3SCy Schubert sprintf(t, " log-tag %d", ipf->fl_logtag);
133041edb306SCy Schubert t += strlen(t);
133141edb306SCy Schubert }
133241edb306SCy Schubert if (ipf->fl_nattag.ipt_num[0] != 0) {
133341edb306SCy Schubert strcpy(t, " nat-tag ");
133441edb306SCy Schubert t += strlen(t);
133541edb306SCy Schubert strncpy(t, ipf->fl_nattag.ipt_tag, sizeof(ipf->fl_nattag));
133641edb306SCy Schubert t += strlen(t);
133741edb306SCy Schubert }
133841edb306SCy Schubert if ((ipf->fl_lflags & FI_LOWTTL) != 0) {
133941edb306SCy Schubert strcpy(t, " low-ttl");
134041edb306SCy Schubert t += 8;
134141edb306SCy Schubert }
134241edb306SCy Schubert if ((ipf->fl_lflags & FI_OOW) != 0) {
134341edb306SCy Schubert strcpy(t, " OOW");
134441edb306SCy Schubert t += 4;
134541edb306SCy Schubert }
134641edb306SCy Schubert if ((ipf->fl_lflags & FI_BAD) != 0) {
134741edb306SCy Schubert strcpy(t, " bad");
134841edb306SCy Schubert t += 4;
134941edb306SCy Schubert }
135041edb306SCy Schubert if ((ipf->fl_lflags & FI_NATED) != 0) {
135141edb306SCy Schubert strcpy(t, " NAT");
135241edb306SCy Schubert t += 4;
135341edb306SCy Schubert }
135441edb306SCy Schubert if ((ipf->fl_lflags & FI_BADNAT) != 0) {
135541edb306SCy Schubert strcpy(t, " bad-NAT");
135641edb306SCy Schubert t += 8;
135741edb306SCy Schubert }
135841edb306SCy Schubert if ((ipf->fl_lflags & FI_BADSRC) != 0) {
135941edb306SCy Schubert strcpy(t, " bad-src");
136041edb306SCy Schubert t += 8;
136141edb306SCy Schubert }
136241edb306SCy Schubert if ((ipf->fl_lflags & FI_MULTICAST) != 0) {
136341edb306SCy Schubert strcpy(t, " multicast");
136441edb306SCy Schubert t += 10;
136541edb306SCy Schubert }
136641edb306SCy Schubert if ((ipf->fl_lflags & FI_BROADCAST) != 0) {
136741edb306SCy Schubert strcpy(t, " broadcast");
136841edb306SCy Schubert t += 10;
136941edb306SCy Schubert }
137041edb306SCy Schubert if ((ipf->fl_lflags & (FI_MULTICAST|FI_BROADCAST|FI_MBCAST)) ==
137141edb306SCy Schubert FI_MBCAST) {
137241edb306SCy Schubert strcpy(t, " mbcast");
137341edb306SCy Schubert t += 7;
137441edb306SCy Schubert }
137541edb306SCy Schubert if (ipf->fl_breason != 0) {
137641edb306SCy Schubert strcpy(t, " reason:");
137741edb306SCy Schubert t += 8;
137841edb306SCy Schubert strcpy(t, reasons[ipf->fl_breason]);
137941edb306SCy Schubert t += strlen(reasons[ipf->fl_breason]);
138041edb306SCy Schubert }
138141edb306SCy Schubert *t++ = '\n';
138241edb306SCy Schubert *t++ = '\0';
138341edb306SCy Schubert defaction = 0;
138441edb306SCy Schubert if (conf->cfile != NULL)
138541edb306SCy Schubert defaction = check_action(buf, line, ipmonopts, lvl);
138641edb306SCy Schubert
138741edb306SCy Schubert if (defaction == 0) {
138841edb306SCy Schubert if (ipmonopts & IPMON_SYSLOG) {
138941edb306SCy Schubert syslog(lvl, "%s", line);
139041edb306SCy Schubert } else if (conf->log != NULL) {
139141edb306SCy Schubert (void) fprintf(conf->log, "%s", line);
139241edb306SCy Schubert }
139341edb306SCy Schubert
139441edb306SCy Schubert if (ipmonopts & IPMON_HEXHDR) {
139541edb306SCy Schubert dumphex(conf->log, ipmonopts, buf,
139641edb306SCy Schubert sizeof(iplog_t) + sizeof(*ipf));
139741edb306SCy Schubert }
139841edb306SCy Schubert if (ipmonopts & IPMON_HEXBODY) {
139941edb306SCy Schubert dumphex(conf->log, ipmonopts, (char *)ip,
140041edb306SCy Schubert ipf->fl_plen + ipf->fl_hlen);
140141edb306SCy Schubert } else if ((ipmonopts & IPMON_LOGBODY) &&
140241edb306SCy Schubert (ipf->fl_flags & FR_LOGBODY)) {
140341edb306SCy Schubert dumphex(conf->log, ipmonopts, (char *)ip + ipf->fl_hlen,
140441edb306SCy Schubert ipf->fl_plen);
140541edb306SCy Schubert }
140641edb306SCy Schubert }
140741edb306SCy Schubert }
140841edb306SCy Schubert
140941edb306SCy Schubert
1410efeb8bffSCy Schubert static void
usage(char * prog)1411efeb8bffSCy Schubert usage(char *prog)
141241edb306SCy Schubert {
141341edb306SCy Schubert fprintf(stderr, "Usage: %s [ -abDFhnpstvxX ] [ -B <binary-logfile> ] [ -C <config-file> ]\n"
141441edb306SCy Schubert "\t[ -f <device> ] [ -L <facility> ] [ -N <device> ]\n"
141541edb306SCy Schubert "\t[ -o [NSI] ] [ -O [NSI] ] [ -P <pidfile> ] [ -S <device> ]\n"
141641edb306SCy Schubert "\t[ <filename> ]\n", prog);
141741edb306SCy Schubert exit(1);
141841edb306SCy Schubert }
141941edb306SCy Schubert
142041edb306SCy Schubert
1421efeb8bffSCy Schubert static void
write_pid(char * file)1422efeb8bffSCy Schubert write_pid(char *file)
142341edb306SCy Schubert {
142441edb306SCy Schubert FILE *fp = NULL;
142541edb306SCy Schubert int fd;
142641edb306SCy Schubert
142741edb306SCy Schubert if ((fd = open(file, O_CREAT|O_TRUNC|O_WRONLY, 0644)) >= 0) {
142841edb306SCy Schubert fp = fdopen(fd, "w");
142941edb306SCy Schubert if (fp == NULL) {
143041edb306SCy Schubert close(fd);
143141edb306SCy Schubert fprintf(stderr,
143241edb306SCy Schubert "unable to open/create pid file: %s\n", file);
143341edb306SCy Schubert return;
143441edb306SCy Schubert }
143541edb306SCy Schubert fprintf(fp, "%d", getpid());
143641edb306SCy Schubert fclose(fp);
143741edb306SCy Schubert }
143841edb306SCy Schubert }
143941edb306SCy Schubert
144041edb306SCy Schubert
1441efeb8bffSCy Schubert static void
flushlogs(char * file,FILE * log)1442efeb8bffSCy Schubert flushlogs(char *file, FILE *log)
144341edb306SCy Schubert {
144441edb306SCy Schubert int fd, flushed = 0;
144541edb306SCy Schubert
144641edb306SCy Schubert if ((fd = open(file, O_RDWR)) == -1) {
144741edb306SCy Schubert (void) fprintf(stderr, "%s: open: %s\n",
144841edb306SCy Schubert file, STRERROR(errno));
144941edb306SCy Schubert exit(1);
145041edb306SCy Schubert }
145141edb306SCy Schubert
145241edb306SCy Schubert if (ioctl(fd, SIOCIPFFB, &flushed) == 0) {
145341edb306SCy Schubert printf("%d bytes flushed from log buffer\n",
145441edb306SCy Schubert flushed);
145541edb306SCy Schubert fflush(stdout);
145641edb306SCy Schubert } else
145741edb306SCy Schubert ipferror(fd, "SIOCIPFFB");
145841edb306SCy Schubert (void) close(fd);
145941edb306SCy Schubert
146041edb306SCy Schubert if (flushed) {
146141edb306SCy Schubert if (ipmonopts & IPMON_SYSLOG) {
146241edb306SCy Schubert syslog(LOG_INFO, "%d bytes flushed from log\n",
146341edb306SCy Schubert flushed);
146441edb306SCy Schubert } else if ((log != stdout) && (log != NULL)) {
146541edb306SCy Schubert fprintf(log, "%d bytes flushed from log\n", flushed);
146641edb306SCy Schubert }
146741edb306SCy Schubert }
146841edb306SCy Schubert }
146941edb306SCy Schubert
147041edb306SCy Schubert
1471efeb8bffSCy Schubert static void
logopts(int turnon,char * options)1472efeb8bffSCy Schubert logopts(int turnon, char *options)
147341edb306SCy Schubert {
147441edb306SCy Schubert int flags = 0;
147541edb306SCy Schubert char *s;
147641edb306SCy Schubert
147741edb306SCy Schubert for (s = options; *s; s++)
147841edb306SCy Schubert {
147941edb306SCy Schubert switch (*s)
148041edb306SCy Schubert {
148141edb306SCy Schubert case 'N' :
148241edb306SCy Schubert flags |= IPMON_NAT;
148341edb306SCy Schubert break;
148441edb306SCy Schubert case 'S' :
148541edb306SCy Schubert flags |= IPMON_STATE;
148641edb306SCy Schubert break;
148741edb306SCy Schubert case 'I' :
148841edb306SCy Schubert flags |= IPMON_FILTER;
148941edb306SCy Schubert break;
149041edb306SCy Schubert default :
149141edb306SCy Schubert fprintf(stderr, "Unknown log option %c\n", *s);
149241edb306SCy Schubert exit(1);
149341edb306SCy Schubert }
149441edb306SCy Schubert }
149541edb306SCy Schubert
149641edb306SCy Schubert if (turnon)
149741edb306SCy Schubert ipmonopts |= flags;
149841edb306SCy Schubert else
149941edb306SCy Schubert ipmonopts &= ~(flags);
150041edb306SCy Schubert }
150141edb306SCy Schubert
1502efeb8bffSCy Schubert static void
initconfig(config_t * conf)1503efeb8bffSCy Schubert initconfig(config_t *conf)
150441edb306SCy Schubert {
150541edb306SCy Schubert int i;
150641edb306SCy Schubert
150741edb306SCy Schubert memset(conf, 0, sizeof(*conf));
150841edb306SCy Schubert
150941edb306SCy Schubert conf->log = stdout;
151041edb306SCy Schubert conf->maxfd = -1;
151141edb306SCy Schubert
151241edb306SCy Schubert for (i = 0; i < 3; i++) {
151341edb306SCy Schubert conf->logsrc[i].fd = -1;
151441edb306SCy Schubert conf->logsrc[i].logtype = -1;
151541edb306SCy Schubert conf->logsrc[i].regular = -1;
151641edb306SCy Schubert }
151741edb306SCy Schubert
151841edb306SCy Schubert conf->logsrc[0].file = IPL_NAME;
151941edb306SCy Schubert conf->logsrc[1].file = IPNAT_NAME;
152041edb306SCy Schubert conf->logsrc[2].file = IPSTATE_NAME;
152141edb306SCy Schubert
152241edb306SCy Schubert add_doing(&executesaver);
152341edb306SCy Schubert add_doing(&snmpv1saver);
152441edb306SCy Schubert add_doing(&snmpv2saver);
152541edb306SCy Schubert add_doing(&syslogsaver);
152641edb306SCy Schubert add_doing(&filesaver);
152741edb306SCy Schubert add_doing(¬hingsaver);
152841edb306SCy Schubert }
152941edb306SCy Schubert
153041edb306SCy Schubert
1531efeb8bffSCy Schubert int
main(int argc,char * argv[])1532efeb8bffSCy Schubert main(int argc, char *argv[])
153341edb306SCy Schubert {
153441edb306SCy Schubert int doread, c, make_daemon = 0;
153541edb306SCy Schubert char *prog;
153641edb306SCy Schubert config_t config;
153741edb306SCy Schubert
153841edb306SCy Schubert prog = strrchr(argv[0], '/');
153941edb306SCy Schubert if (prog == NULL)
154041edb306SCy Schubert prog = argv[0];
154141edb306SCy Schubert else
154241edb306SCy Schubert prog++;
154341edb306SCy Schubert
154441edb306SCy Schubert initconfig(&config);
154541edb306SCy Schubert
154641edb306SCy Schubert while ((c = getopt(argc, argv,
154741edb306SCy Schubert "?abB:C:Df:FhL:nN:o:O:pP:sS:tvxX")) != -1)
154841edb306SCy Schubert switch (c)
154941edb306SCy Schubert {
155041edb306SCy Schubert case 'a' :
155141edb306SCy Schubert ipmonopts |= IPMON_LOGALL;
155241edb306SCy Schubert config.logsrc[0].logtype = IPL_LOGIPF;
155341edb306SCy Schubert config.logsrc[1].logtype = IPL_LOGNAT;
155441edb306SCy Schubert config.logsrc[2].logtype = IPL_LOGSTATE;
155541edb306SCy Schubert break;
155641edb306SCy Schubert case 'b' :
155741edb306SCy Schubert ipmonopts |= IPMON_LOGBODY;
155841edb306SCy Schubert break;
155941edb306SCy Schubert case 'B' :
156041edb306SCy Schubert config.bfile = optarg;
156141edb306SCy Schubert config.blog = fopen(optarg, "a");
156241edb306SCy Schubert break;
156341edb306SCy Schubert case 'C' :
156441edb306SCy Schubert config.cfile = optarg;
156541edb306SCy Schubert break;
156641edb306SCy Schubert case 'D' :
156741edb306SCy Schubert make_daemon = 1;
156841edb306SCy Schubert break;
156941edb306SCy Schubert case 'f' : case 'I' :
157041edb306SCy Schubert ipmonopts |= IPMON_FILTER;
157141edb306SCy Schubert config.logsrc[0].logtype = IPL_LOGIPF;
157241edb306SCy Schubert config.logsrc[0].file = optarg;
157341edb306SCy Schubert break;
157441edb306SCy Schubert case 'F' :
157541edb306SCy Schubert flushlogs(config.logsrc[0].file, config.log);
157641edb306SCy Schubert flushlogs(config.logsrc[1].file, config.log);
157741edb306SCy Schubert flushlogs(config.logsrc[2].file, config.log);
157841edb306SCy Schubert break;
157941edb306SCy Schubert case 'L' :
158041edb306SCy Schubert logfac = fac_findname(optarg);
158141edb306SCy Schubert if (logfac == -1) {
158241edb306SCy Schubert fprintf(stderr,
158341edb306SCy Schubert "Unknown syslog facility '%s'\n",
158441edb306SCy Schubert optarg);
158541edb306SCy Schubert exit(1);
158641edb306SCy Schubert }
158741edb306SCy Schubert break;
158841edb306SCy Schubert case 'n' :
158941edb306SCy Schubert ipmonopts |= IPMON_RESOLVE;
159041edb306SCy Schubert opts &= ~OPT_NORESOLVE;
159141edb306SCy Schubert break;
159241edb306SCy Schubert case 'N' :
159341edb306SCy Schubert ipmonopts |= IPMON_NAT;
159441edb306SCy Schubert config.logsrc[1].logtype = IPL_LOGNAT;
159541edb306SCy Schubert config.logsrc[1].file = optarg;
159641edb306SCy Schubert break;
159741edb306SCy Schubert case 'o' : case 'O' :
159841edb306SCy Schubert logopts(c == 'o', optarg);
159941edb306SCy Schubert if (ipmonopts & IPMON_FILTER)
160041edb306SCy Schubert config.logsrc[0].logtype = IPL_LOGIPF;
160141edb306SCy Schubert if (ipmonopts & IPMON_NAT)
160241edb306SCy Schubert config.logsrc[1].logtype = IPL_LOGNAT;
160341edb306SCy Schubert if (ipmonopts & IPMON_STATE)
160441edb306SCy Schubert config.logsrc[2].logtype = IPL_LOGSTATE;
160541edb306SCy Schubert break;
160641edb306SCy Schubert case 'p' :
160741edb306SCy Schubert ipmonopts |= IPMON_PORTNUM;
160841edb306SCy Schubert break;
160941edb306SCy Schubert case 'P' :
161041edb306SCy Schubert pidfile = optarg;
161141edb306SCy Schubert break;
161241edb306SCy Schubert case 's' :
161341edb306SCy Schubert ipmonopts |= IPMON_SYSLOG;
161441edb306SCy Schubert config.log = NULL;
161541edb306SCy Schubert break;
161641edb306SCy Schubert case 'S' :
161741edb306SCy Schubert ipmonopts |= IPMON_STATE;
161841edb306SCy Schubert config.logsrc[2].logtype = IPL_LOGSTATE;
161941edb306SCy Schubert config.logsrc[2].file = optarg;
162041edb306SCy Schubert break;
162141edb306SCy Schubert case 't' :
162241edb306SCy Schubert ipmonopts |= IPMON_TAIL;
162341edb306SCy Schubert break;
162441edb306SCy Schubert case 'v' :
162541edb306SCy Schubert ipmonopts |= IPMON_VERBOSE;
162641edb306SCy Schubert break;
162741edb306SCy Schubert case 'x' :
162841edb306SCy Schubert ipmonopts |= IPMON_HEXBODY;
162941edb306SCy Schubert break;
163041edb306SCy Schubert case 'X' :
163141edb306SCy Schubert ipmonopts |= IPMON_HEXHDR;
163241edb306SCy Schubert break;
163341edb306SCy Schubert default :
163441edb306SCy Schubert case 'h' :
163541edb306SCy Schubert case '?' :
163641edb306SCy Schubert usage(argv[0]);
163741edb306SCy Schubert }
163841edb306SCy Schubert
163941edb306SCy Schubert if (ipmonopts & IPMON_SYSLOG)
164041edb306SCy Schubert openlog(prog, LOG_NDELAY|LOG_PID, logfac);
164141edb306SCy Schubert
164241edb306SCy Schubert init_tabs();
164341edb306SCy Schubert if (config.cfile)
164441edb306SCy Schubert if (load_config(config.cfile) == -1) {
164541edb306SCy Schubert unload_config();
164641edb306SCy Schubert exit(1);
164741edb306SCy Schubert }
164841edb306SCy Schubert
164941edb306SCy Schubert /*
165041edb306SCy Schubert * Default action is to only open the filter log file.
165141edb306SCy Schubert */
165241edb306SCy Schubert if ((config.logsrc[0].logtype == -1) &&
165341edb306SCy Schubert (config.logsrc[0].logtype == -1) &&
165441edb306SCy Schubert (config.logsrc[0].logtype == -1))
165541edb306SCy Schubert config.logsrc[0].logtype = IPL_LOGIPF;
165641edb306SCy Schubert
165741edb306SCy Schubert openlogs(&config);
165841edb306SCy Schubert
165941edb306SCy Schubert if (!(ipmonopts & IPMON_SYSLOG)) {
166041edb306SCy Schubert config.file = argv[optind];
166141edb306SCy Schubert config.log = config.file ? fopen(config.file, "a") : stdout;
166241edb306SCy Schubert if (config.log == NULL) {
166341edb306SCy Schubert (void) fprintf(stderr, "%s: fopen: %s\n",
166441edb306SCy Schubert argv[optind], STRERROR(errno));
166541edb306SCy Schubert exit(1);
166641edb306SCy Schubert /* NOTREACHED */
166741edb306SCy Schubert }
166841edb306SCy Schubert setvbuf(config.log, NULL, _IONBF, 0);
166941edb306SCy Schubert } else {
167041edb306SCy Schubert config.log = NULL;
167141edb306SCy Schubert }
167241edb306SCy Schubert
167341edb306SCy Schubert if (make_daemon &&
167441edb306SCy Schubert ((config.log != stdout) || (ipmonopts & IPMON_SYSLOG))) {
167541edb306SCy Schubert #ifdef BSD
167641edb306SCy Schubert daemon(0, !(ipmonopts & IPMON_SYSLOG));
167741edb306SCy Schubert #else
167841edb306SCy Schubert int pid;
167941edb306SCy Schubert
168041edb306SCy Schubert switch (fork())
168141edb306SCy Schubert {
168241edb306SCy Schubert case -1 :
168341edb306SCy Schubert (void) fprintf(stderr, "%s: fork() failed: %s\n",
168441edb306SCy Schubert argv[0], STRERROR(errno));
168541edb306SCy Schubert exit(1);
168641edb306SCy Schubert /* NOTREACHED */
168741edb306SCy Schubert case 0 :
168841edb306SCy Schubert break;
168941edb306SCy Schubert default :
169041edb306SCy Schubert exit(0);
169141edb306SCy Schubert }
169241edb306SCy Schubert
169341edb306SCy Schubert setsid();
169441edb306SCy Schubert if ((ipmonopts & IPMON_SYSLOG))
169541edb306SCy Schubert close(2);
169641edb306SCy Schubert #endif /* !BSD */
169741edb306SCy Schubert close(0);
169841edb306SCy Schubert close(1);
169941edb306SCy Schubert write_pid(pidfile);
170041edb306SCy Schubert }
170141edb306SCy Schubert
170241edb306SCy Schubert signal(SIGHUP, handlehup);
170341edb306SCy Schubert
170441edb306SCy Schubert for (doread = 1; doread; )
170541edb306SCy Schubert doread = read_loginfo(&config);
170641edb306SCy Schubert
170741edb306SCy Schubert unload_config();
170841edb306SCy Schubert
170941edb306SCy Schubert return (0);
171041edb306SCy Schubert /* NOTREACHED */
171141edb306SCy Schubert }
171241edb306SCy Schubert
171341edb306SCy Schubert
1714efeb8bffSCy Schubert static void
openlogs(config_t * conf)1715efeb8bffSCy Schubert openlogs(config_t *conf)
171641edb306SCy Schubert {
171741edb306SCy Schubert logsource_t *l;
171841edb306SCy Schubert struct stat sb;
171941edb306SCy Schubert int i;
172041edb306SCy Schubert
172141edb306SCy Schubert for (i = 0; i < 3; i++) {
172241edb306SCy Schubert l = &conf->logsrc[i];
172341edb306SCy Schubert if (l->logtype == -1)
172441edb306SCy Schubert continue;
172541edb306SCy Schubert if (!strcmp(l->file, "-"))
172641edb306SCy Schubert l->fd = 0;
172741edb306SCy Schubert else {
172841edb306SCy Schubert if ((l->fd= open(l->file, O_RDONLY)) == -1) {
172941edb306SCy Schubert (void) fprintf(stderr,
173041edb306SCy Schubert "%s: open: %s\n", l->file,
173141edb306SCy Schubert STRERROR(errno));
173241edb306SCy Schubert exit(1);
173341edb306SCy Schubert /* NOTREACHED */
173441edb306SCy Schubert }
173541edb306SCy Schubert
173641edb306SCy Schubert if (fstat(l->fd, &sb) == -1) {
173741edb306SCy Schubert (void) fprintf(stderr, "%d: fstat: %s\n",
173841edb306SCy Schubert l->fd, STRERROR(errno));
173941edb306SCy Schubert exit(1);
174041edb306SCy Schubert /* NOTREACHED */
174141edb306SCy Schubert }
174241edb306SCy Schubert
174341edb306SCy Schubert l->regular = !S_ISCHR(sb.st_mode);
174441edb306SCy Schubert if (l->regular)
174541edb306SCy Schubert l->size = sb.st_size;
174641edb306SCy Schubert
174741edb306SCy Schubert FD_SET(l->fd, &conf->fdmr);
174841edb306SCy Schubert if (l->fd > conf->maxfd)
174941edb306SCy Schubert conf->maxfd = l->fd;
175041edb306SCy Schubert }
175141edb306SCy Schubert }
175241edb306SCy Schubert }
175341edb306SCy Schubert
175441edb306SCy Schubert
1755efeb8bffSCy Schubert static int
read_loginfo(config_t * conf)1756efeb8bffSCy Schubert read_loginfo(config_t *conf)
175741edb306SCy Schubert {
175841edb306SCy Schubert iplog_t buf[DEFAULT_IPFLOGSIZE/sizeof(iplog_t)+1];
175941edb306SCy Schubert int n, tr, nr, i;
176041edb306SCy Schubert logsource_t *l;
176141edb306SCy Schubert fd_set fdr;
176241edb306SCy Schubert
176341edb306SCy Schubert fdr = conf->fdmr;
176441edb306SCy Schubert
176541edb306SCy Schubert n = select(conf->maxfd + 1, &fdr, NULL, NULL, NULL);
176641edb306SCy Schubert if (n == 0)
17672582ae57SCy Schubert return (1);
176841edb306SCy Schubert if (n == -1) {
176941edb306SCy Schubert if (errno == EINTR)
17702582ae57SCy Schubert return (1);
17712582ae57SCy Schubert return (-1);
177241edb306SCy Schubert }
177341edb306SCy Schubert
177441edb306SCy Schubert for (i = 0, nr = 0; i < 3; i++) {
177541edb306SCy Schubert l = &conf->logsrc[i];
177641edb306SCy Schubert
177741edb306SCy Schubert if ((l->logtype == -1) || !FD_ISSET(l->fd, &fdr))
177841edb306SCy Schubert continue;
177941edb306SCy Schubert
178041edb306SCy Schubert tr = 0;
178141edb306SCy Schubert if (l->regular) {
178241edb306SCy Schubert tr = (lseek(l->fd, 0, SEEK_CUR) < l->size);
178341edb306SCy Schubert if (!tr && !(ipmonopts & IPMON_TAIL))
17842582ae57SCy Schubert return (0);
178541edb306SCy Schubert }
178641edb306SCy Schubert
178741edb306SCy Schubert n = 0;
178841edb306SCy Schubert tr = read_log(l->fd, &n, (char *)buf, sizeof(buf));
178941edb306SCy Schubert if (donehup) {
179041edb306SCy Schubert if (conf->file != NULL) {
179141edb306SCy Schubert if (conf->log != NULL) {
179241edb306SCy Schubert fclose(conf->log);
179341edb306SCy Schubert conf->log = NULL;
179441edb306SCy Schubert }
179541edb306SCy Schubert conf->log = fopen(conf->file, "a");
179641edb306SCy Schubert }
179741edb306SCy Schubert
179841edb306SCy Schubert if (conf->bfile != NULL) {
179941edb306SCy Schubert if (conf->blog != NULL) {
180041edb306SCy Schubert fclose(conf->blog);
180141edb306SCy Schubert conf->blog = NULL;
180241edb306SCy Schubert }
180341edb306SCy Schubert conf->blog = fopen(conf->bfile, "a");
180441edb306SCy Schubert }
180541edb306SCy Schubert
180641edb306SCy Schubert init_tabs();
180741edb306SCy Schubert if (conf->cfile != NULL)
180841edb306SCy Schubert load_config(conf->cfile);
180941edb306SCy Schubert donehup = 0;
181041edb306SCy Schubert }
181141edb306SCy Schubert
181241edb306SCy Schubert switch (tr)
181341edb306SCy Schubert {
181441edb306SCy Schubert case -1 :
181541edb306SCy Schubert if (ipmonopts & IPMON_SYSLOG)
181641edb306SCy Schubert syslog(LOG_CRIT, "read: %m\n");
181741edb306SCy Schubert else {
181841edb306SCy Schubert ipferror(l->fd, "read");
181941edb306SCy Schubert }
18202582ae57SCy Schubert return (0);
182141edb306SCy Schubert case 1 :
182241edb306SCy Schubert if (ipmonopts & IPMON_SYSLOG)
182341edb306SCy Schubert syslog(LOG_CRIT, "aborting logging\n");
182441edb306SCy Schubert else if (conf->log != NULL)
182541edb306SCy Schubert fprintf(conf->log, "aborting logging\n");
18262582ae57SCy Schubert return (0);
182741edb306SCy Schubert case 2 :
182841edb306SCy Schubert break;
182941edb306SCy Schubert case 0 :
183041edb306SCy Schubert nr += tr;
183141edb306SCy Schubert if (n > 0) {
183241edb306SCy Schubert print_log(conf, l, (char *)buf, n);
183341edb306SCy Schubert if (!(ipmonopts & IPMON_SYSLOG))
183441edb306SCy Schubert fflush(conf->log);
183541edb306SCy Schubert }
183641edb306SCy Schubert break;
183741edb306SCy Schubert }
183841edb306SCy Schubert }
183941edb306SCy Schubert
184041edb306SCy Schubert if (!nr && (ipmonopts & IPMON_TAIL))
184141edb306SCy Schubert sleep(1);
184241edb306SCy Schubert
18432582ae57SCy Schubert return (1);
184441edb306SCy Schubert }
1845