1 1.1 christos /* $NetBSD: snit.c,v 1.1.1.1 2012/03/23 21:20:06 christos Exp $ */ 2 1.1 christos 3 1.1 christos /* 4 1.1 christos * (C)opyright 1992-1998 Darren Reed. (from tcplog) 5 1.1 christos * 6 1.1 christos * See the IPFILTER.LICENCE file for details on licencing. 7 1.1 christos * 8 1.1 christos */ 9 1.1 christos 10 1.1 christos #include <stdio.h> 11 1.1 christos #include <netdb.h> 12 1.1 christos #include <ctype.h> 13 1.1 christos #include <signal.h> 14 1.1 christos #include <errno.h> 15 1.1 christos #include <sys/types.h> 16 1.1 christos #include <sys/time.h> 17 1.1 christos #include <sys/timeb.h> 18 1.1 christos #include <sys/socket.h> 19 1.1 christos #include <sys/file.h> 20 1.1 christos #include <sys/ioctl.h> 21 1.1 christos #include <net/nit.h> 22 1.1 christos #include <sys/fcntlcom.h> 23 1.1 christos #include <sys/dir.h> 24 1.1 christos #include <net/nit_if.h> 25 1.1 christos #include <net/nit_pf.h> 26 1.1 christos #include <net/nit_buf.h> 27 1.1 christos #include <net/packetfilt.h> 28 1.1 christos #include <sys/stropts.h> 29 1.1 christos 30 1.1 christos #include <net/if.h> 31 1.1 christos #include <netinet/in.h> 32 1.1 christos #include <netinet/in_systm.h> 33 1.1 christos #include <netinet/ip.h> 34 1.1 christos #include <netinet/if_ether.h> 35 1.1 christos #include <netinet/ip_var.h> 36 1.1 christos #include <netinet/udp.h> 37 1.1 christos #include <netinet/udp_var.h> 38 1.1 christos #include <netinet/tcp.h> 39 1.1 christos #include <netinet/tcpip.h> 40 1.1 christos 41 1.1 christos #ifndef lint 42 1.1 christos static char snitid[] = "@(#)snit.c 1.2 12/3/95 (C)1995 Darren Reed"; 43 1.1 christos #endif 44 1.1 christos 45 1.1 christos #define BUFSPACE 32768 46 1.1 christos 47 1.1 christos /* 48 1.1 christos * Be careful to only include those defined in the flags option for the 49 1.1 christos * interface are included in the header size. 50 1.1 christos */ 51 1.1 christos #define BUFHDR_SIZE (sizeof(struct nit_bufhdr)) 52 1.1 christos #define NIT_HDRSIZE (BUFHDR_SIZE) 53 1.1 christos 54 1.1 christos static int timeout; 55 1.1 christos 56 1.1 christos 57 1.1 christos int ack_recv(ep) 58 1.1 christos char *ep; 59 1.1 christos { 60 1.1 christos struct tcpiphdr tip; 61 1.1 christos struct tcphdr *tcp; 62 1.1 christos struct ip *ip; 63 1.1 christos 64 1.1 christos ip = (struct ip *)&tip; 65 1.1 christos tcp = (struct tcphdr *)(ip + 1); 66 1.1 christos bcopy(ep + 14, (char *)ip, sizeof(*ip)); 67 1.1 christos bcopy(ep + 14 + (ip->ip_hl << 2), (char *)tcp, sizeof(*tcp)); 68 1.1 christos if (ip->ip_off & 0x1fff != 0) 69 1.1 christos return 0; 70 1.1 christos if (0 == detect(ip, tcp)) 71 1.1 christos return 1; 72 1.1 christos return 0; 73 1.1 christos } 74 1.1 christos 75 1.1 christos 76 1.1 christos int readloop(fd, dst) 77 1.1 christos int fd; 78 1.1 christos struct in_addr dst; 79 1.1 christos { 80 1.1 christos static u_char buf[BUFSPACE]; 81 1.1 christos register u_char *bp, *cp, *bufend; 82 1.1 christos register struct nit_bufhdr *hp; 83 1.1 christos register int cc; 84 1.1 christos time_t now = time(NULL); 85 1.1 christos int done = 0; 86 1.1 christos 87 1.1 christos while ((cc = read(fd, buf, BUFSPACE-1)) >= 0) { 88 1.1 christos if (!cc) 89 1.1 christos if ((time(NULL) - now) > timeout) 90 1.1 christos return done; 91 1.1 christos else 92 1.1 christos continue; 93 1.1 christos bp = buf; 94 1.1 christos bufend = buf + cc; 95 1.1 christos /* 96 1.1 christos * loop through each snapshot in the chunk 97 1.1 christos */ 98 1.1 christos while (bp < bufend) { 99 1.1 christos cp = (u_char *)((char *)bp + NIT_HDRSIZE); 100 1.1 christos /* 101 1.1 christos * get past NIT buffer 102 1.1 christos */ 103 1.1 christos hp = (struct nit_bufhdr *)bp; 104 1.1 christos /* 105 1.1 christos * next snapshot 106 1.1 christos */ 107 1.1 christos bp += hp->nhb_totlen; 108 1.1 christos done += ack_recv(cp); 109 1.1 christos } 110 1.1 christos return done; 111 1.1 christos } 112 1.1 christos perror("read"); 113 1.1 christos exit(-1); 114 1.1 christos } 115 1.1 christos 116 1.1 christos int initdevice(device, tout) 117 1.1 christos char *device; 118 1.1 christos int tout; 119 1.1 christos { 120 1.1 christos struct strioctl si; 121 1.1 christos struct timeval to; 122 1.1 christos struct ifreq ifr; 123 1.1 christos struct packetfilt pfil; 124 1.1 christos u_long if_flags; 125 1.1 christos u_short *fwp = pfil.Pf_Filter; 126 1.1 christos int ret, offset, fd, snaplen= 76, chunksize = BUFSPACE; 127 1.1 christos 128 1.1 christos if ((fd = open("/dev/nit", O_RDWR)) < 0) 129 1.1 christos { 130 1.1 christos perror("/dev/nit"); 131 1.1 christos exit(-1); 132 1.1 christos } 133 1.1 christos 134 1.1 christos /* 135 1.1 christos * Create some filter rules for our TCP watcher. We only want ethernet 136 1.1 christos * pacets which are IP protocol and only the TCP packets from IP. 137 1.1 christos */ 138 1.1 christos offset = 6; 139 1.1 christos *fwp++ = ENF_PUSHWORD + offset; 140 1.1 christos *fwp++ = ENF_PUSHLIT | ENF_CAND; 141 1.1 christos *fwp++ = htons(ETHERTYPE_IP); 142 1.1 christos *fwp++ = ENF_PUSHWORD + sizeof(struct ether_header)/sizeof(short)+4; 143 1.1 christos *fwp++ = ENF_PUSHLIT | ENF_AND; 144 1.1 christos *fwp++ = htons(0x00ff); 145 1.1 christos *fwp++ = ENF_PUSHLIT | ENF_COR; 146 1.1 christos *fwp++ = htons(IPPROTO_TCP); 147 1.1 christos *fwp++ = ENF_PUSHWORD + sizeof(struct ether_header)/sizeof(short)+4; 148 1.1 christos *fwp++ = ENF_PUSHLIT | ENF_AND; 149 1.1 christos *fwp++ = htons(0x00ff); 150 1.1 christos *fwp++ = ENF_PUSHLIT | ENF_CAND; 151 1.1 christos *fwp++ = htons(IPPROTO_UDP); 152 1.1 christos pfil.Pf_FilterLen = fwp - &pfil.Pf_Filter[0]; 153 1.1 christos /* 154 1.1 christos * put filter in place. 155 1.1 christos */ 156 1.1 christos if (ioctl(fd, I_PUSH, "pf") == -1) 157 1.1 christos { 158 1.1 christos perror("ioctl: I_PUSH pf"); 159 1.1 christos exit(1); 160 1.1 christos } 161 1.1 christos if (ioctl(fd, NIOCSETF, &pfil) == -1) 162 1.1 christos { 163 1.1 christos perror("ioctl: NIOCSETF"); 164 1.1 christos exit(1); 165 1.1 christos } 166 1.1 christos /* 167 1.1 christos * arrange to get messages from the NIT STREAM and use NIT_BUF option 168 1.1 christos */ 169 1.1 christos ioctl(fd, I_SRDOPT, (char*)RMSGD); 170 1.1 christos ioctl(fd, I_PUSH, "nbuf"); 171 1.1 christos /* 172 1.1 christos * set the timeout 173 1.1 christos */ 174 1.1 christos timeout = tout; 175 1.1 christos si.ic_timout = 1; 176 1.1 christos to.tv_sec = 1; 177 1.1 christos to.tv_usec = 0; 178 1.1 christos si.ic_cmd = NIOCSTIME; 179 1.1 christos si.ic_len = sizeof(to); 180 1.1 christos si.ic_dp = (char*)&to; 181 1.1 christos if (ioctl(fd, I_STR, (char*)&si) == -1) 182 1.1 christos { 183 1.1 christos perror("ioctl: NIT timeout"); 184 1.1 christos exit(-1); 185 1.1 christos } 186 1.1 christos /* 187 1.1 christos * set the chunksize 188 1.1 christos */ 189 1.1 christos si.ic_cmd = NIOCSCHUNK; 190 1.1 christos si.ic_len = sizeof(chunksize); 191 1.1 christos si.ic_dp = (char*)&chunksize; 192 1.1 christos if (ioctl(fd, I_STR, (char*)&si) == -1) 193 1.1 christos perror("ioctl: NIT chunksize"); 194 1.1 christos if (ioctl(fd, NIOCGCHUNK, (char*)&chunksize) == -1) 195 1.1 christos { 196 1.1 christos perror("ioctl: NIT chunksize"); 197 1.1 christos exit(-1); 198 1.1 christos } 199 1.1 christos printf("NIT buffer size: %d\n", chunksize); 200 1.1 christos 201 1.1 christos /* 202 1.1 christos * request the interface 203 1.1 christos */ 204 1.1 christos strncpy(ifr.ifr_name, device, sizeof(ifr.ifr_name)); 205 1.1 christos ifr.ifr_name[sizeof(ifr.ifr_name) - 1] = ' '; 206 1.1 christos si.ic_cmd = NIOCBIND; 207 1.1 christos si.ic_len = sizeof(ifr); 208 1.1 christos si.ic_dp = (char*)𝔦 209 1.1 christos if (ioctl(fd, I_STR, (char*)&si) == -1) 210 1.1 christos { 211 1.1 christos perror(ifr.ifr_name); 212 1.1 christos exit(1); 213 1.1 christos } 214 1.1 christos 215 1.1 christos /* 216 1.1 christos * set the snapshot length 217 1.1 christos */ 218 1.1 christos si.ic_cmd = NIOCSSNAP; 219 1.1 christos si.ic_len = sizeof(snaplen); 220 1.1 christos si.ic_dp = (char*)&snaplen; 221 1.1 christos if (ioctl(fd, I_STR, (char*)&si) == -1) 222 1.1 christos { 223 1.1 christos perror("ioctl: NIT snaplen"); 224 1.1 christos exit(1); 225 1.1 christos } 226 1.1 christos (void) ioctl(fd, I_FLUSH, (char*)FLUSHR); 227 1.1 christos return fd; 228 1.1 christos } 229