diff options
author | rsc <devnull@localhost> | 2005-12-26 04:48:52 +0000 |
---|---|---|
committer | rsc <devnull@localhost> | 2005-12-26 04:48:52 +0000 |
commit | 87a52e0485d3281ebea6bf4b725aa8023690e96f (patch) | |
tree | 0abc2d2ddb875196177231639d3cb4519e814b9d /src/cmd/ip/snoopy | |
parent | 35d26aa32167e84326cdb745c0e906393b8de71d (diff) | |
download | plan9port-87a52e0485d3281ebea6bf4b725aa8023690e96f.tar.gz plan9port-87a52e0485d3281ebea6bf4b725aa8023690e96f.tar.bz2 plan9port-87a52e0485d3281ebea6bf4b725aa8023690e96f.zip |
new goodies
Diffstat (limited to 'src/cmd/ip/snoopy')
33 files changed, 5496 insertions, 0 deletions
diff --git a/src/cmd/ip/snoopy/Linux.c b/src/cmd/ip/snoopy/Linux.c new file mode 100644 index 00000000..20bc899c --- /dev/null +++ b/src/cmd/ip/snoopy/Linux.c @@ -0,0 +1,58 @@ +#include <u.h> +#include <sys/socket.h> +#include <net/if.h> +#include <netpacket/packet.h> +#include <net/ethernet.h> +#include <netinet/in.h> +#include <sys/ioctl.h> +#include <libc.h> +#include <ip.h> +#include <bio.h> +#include <fcall.h> +#include <libsec.h> +#include "dat.h" +#include "protos.h" +#include "y.tab.h" + +int +opendevice(char *dev, int promisc) +{ + int fd; + struct ifreq ifr; + struct sockaddr_ll sa; + + if((fd = socket(PF_PACKET, SOCK_RAW, htons(ETH_P_ALL))) < 0) + return -1; + + if(dev){ + memset(&ifr, 0, sizeof ifr); + strncpy(ifr.ifr_name, dev, sizeof ifr.ifr_name); + if(ioctl(fd, SIOCGIFINDEX, &ifr) < 0){ + close(fd); + return -1; + } + memset(&sa, 0, sizeof sa); + sa.sll_family = AF_PACKET; + sa.sll_protocol = htons(ETH_P_ALL); + sa.sll_ifindex = ifr.ifr_ifindex; + if(bind(fd, (struct sockaddr*)&sa, sizeof sa) < 0){ + close(fd); + return -1; + } + } + + if(promisc){ + memset(&ifr, 0, sizeof ifr); + strncpy(ifr.ifr_name, dev, sizeof ifr.ifr_name); + if(ioctl(fd, SIOCGIFFLAGS, &ifr) < 0){ + close(fd); + return -1; + } + ifr.ifr_flags |= IFF_PROMISC; + if(ioctl(fd, SIOCSIFFLAGS, &ifr) < 0){ + close(fd); + return -1; + } + } + return fd; +} diff --git a/src/cmd/ip/snoopy/arp.c b/src/cmd/ip/snoopy/arp.c new file mode 100755 index 00000000..3433ebc6 --- /dev/null +++ b/src/cmd/ip/snoopy/arp.c @@ -0,0 +1,128 @@ +#include <u.h> +#include <libc.h> +#include <ip.h> +#include "dat.h" +#include "protos.h" + +typedef struct Hdr Hdr; +struct Hdr +{ + uchar hrd[2]; + uchar pro[2]; + uchar hln; + uchar pln; + uchar op[2]; + uchar sha[6]; + uchar spa[4]; + uchar tha[6]; + uchar tpa[4]; +}; + +enum +{ + ARPLEN= 28, +}; + +enum +{ + Ospa, + Otpa, + Ostpa, + Osha, + Otha, + Ostha, + Opa, +}; + +static Field p_fields[] = +{ + {"spa", Fv4ip, Ospa, "protocol source", } , + {"tpa", Fv4ip, Otpa, "protocol target", } , + {"a", Fv4ip, Ostpa, "protocol source/target", } , + {"sha", Fba, Osha, "hardware source", } , + {"tha", Fba, Otha, "hardware target", } , + {"ah", Fba, Ostha, "hardware source/target", } , + {0} +}; + +static void +p_compile(Filter *f) +{ + if(f->op == '='){ + compile_cmp(arp.name, f, p_fields); + return; + } + sysfatal("unknown arp field: %s", f->s); +} + +static int +p_filter(Filter *f, Msg *m) +{ + Hdr *h; + + if(m->pe - m->ps < ARPLEN) + return 0; + + h = (Hdr*)m->ps; + m->ps += ARPLEN; + + switch(f->subop){ + case Ospa: + return h->pln == 4 && NetL(h->spa) == f->ulv; + case Otpa: + return h->pln == 4 && NetL(h->tpa) == f->ulv; + case Ostpa: + return h->pln == 4 && (NetL(h->tpa) == f->ulv || + NetL(h->spa) == f->ulv); + case Osha: + return memcmp(h->sha, f->a, h->hln) == 0; + case Otha: + return memcmp(h->tha, f->a, h->hln) == 0; + case Ostha: + return memcmp(h->sha, f->a, h->hln)==0 + ||memcmp(h->tha, f->a, h->hln)==0; + } + return 0; +} + +static int +p_seprint(Msg *m) +{ + Hdr *h; + + if(m->pe - m->ps < ARPLEN) + return -1; + + h = (Hdr*)m->ps; + m->ps += ARPLEN; + + /* no next protocol */ + m->pr = nil; + + m->p = seprint(m->p, m->e, "op=%1d len=%1d/%1d spa=%V sha=%E tpa=%V tha=%E", + NetS(h->op), h->pln, h->hln, + h->spa, h->sha, h->tpa, h->tha); + return 0; +} + +Proto arp = +{ + "arp", + p_compile, + p_filter, + p_seprint, + nil, + p_fields, + defaultframer, +}; + +Proto rarp = +{ + "rarp", + p_compile, + p_filter, + p_seprint, + nil, + p_fields, + defaultframer, +}; diff --git a/src/cmd/ip/snoopy/bootp.c b/src/cmd/ip/snoopy/bootp.c new file mode 100755 index 00000000..1adfdad1 --- /dev/null +++ b/src/cmd/ip/snoopy/bootp.c @@ -0,0 +1,176 @@ +#include <u.h> +#include <libc.h> +#include <ip.h> +#include "dat.h" +#include "protos.h" + +enum +{ + OfferTimeout= 60, /* when an offer times out */ + MaxLease= 60*60, /* longest lease for dynamic binding */ + MinLease= 15*60, /* shortest lease for dynamic binding */ + StaticLease= 30*60, /* lease for static binding */ + + IPUDPHDRSIZE= 28, /* size of an IP plus UDP header */ + MINSUPPORTED= 576, /* biggest IP message the client must support */ + + /* lengths of some bootp fields */ + Maxhwlen= 16, + Maxfilelen= 128, + Maxoptlen= 312-4, + + /* bootp types */ + Bootrequest= 1, + Bootreply= 2, + + /* bootp flags */ + Fbroadcast= 1<<15, +}; + +typedef struct Hdr Hdr; +struct Hdr +{ + uchar op; /* opcode */ + uchar htype; /* hardware type */ + uchar hlen; /* hardware address len */ + uchar hops; /* hops */ + uchar xid[4]; /* a random number */ + uchar secs[2]; /* elapsed since client started booting */ + uchar flags[2]; + uchar ciaddr[IPv4addrlen]; /* client IP address (client tells server) */ + uchar yiaddr[IPv4addrlen]; /* client IP address (server tells client) */ + uchar siaddr[IPv4addrlen]; /* server IP address */ + uchar giaddr[IPv4addrlen]; /* gateway IP address */ + uchar chaddr[Maxhwlen]; /* client hardware address */ + char sname[64]; /* server host name (optional) */ + char file[Maxfilelen]; /* boot file name */ + uchar optmagic[4]; + uchar optdata[Maxoptlen]; +}; + +enum +{ + Oca, + Osa, + Ot, +}; + +static Field p_fields[] = +{ + {"ca", Fv4ip, Oca, "client IP addr", } , + {"sa", Fv4ip, Osa, "server IP addr", } , + {0} +}; + +#define plan9opt ((ulong)(('p'<<24) | ('9'<<16) | (' '<<8) | ' ')) +#define genericopt (0x63825363UL) + +static Mux p_mux[] = +{ + {"dhcp", genericopt,}, + {"plan9bootp", plan9opt,}, + {"dump", 0,}, + {0} +}; + +static void +p_compile(Filter *f) +{ + Mux *m; + + if(f->op == '='){ + compile_cmp(arp.name, f, p_fields); + return; + } + for(m = p_mux; m->name != nil; m++) + if(strcmp(f->s, m->name) == 0){ + f->pr = m->pr; + f->ulv = m->val; + f->subop = Ot; + return; + } + sysfatal("unknown bootp field: %s", f->s); +} + +static int +p_filter(Filter *f, Msg *m) +{ + Hdr *h; + + h = (Hdr*)m->ps; + + if(m->pe < (uchar*)h->sname) + return 0; + m->ps = h->optdata; + + switch(f->subop){ + case Oca: + return NetL(h->ciaddr) == f->ulv || NetL(h->yiaddr) == f->ulv; + case Osa: + return NetL(h->siaddr) == f->ulv; + case Ot: + return NetL(h->optmagic) == f->ulv; + } + return 0; +} + +static char* +op(int i) +{ + static char x[20]; + + switch(i){ + case Bootrequest: + return "Req"; + case Bootreply: + return "Rep"; + default: + sprint(x, "%d", i); + return x; + } +} + + +static int +p_seprint(Msg *m) +{ + Hdr *h; + ulong x; + + h = (Hdr*)m->ps; + + if(m->pe < (uchar*)h->sname) + return -1; + + /* point past data */ + m->ps = h->optdata; + + /* next protocol */ + m->pr = nil; + if(m->pe >= (uchar*)h->optdata){ + x = NetL(h->optmagic); + demux(p_mux, x, x, m, &dump); + } + + m->p = seprint(m->p, m->e, "t=%s ht=%d hl=%d hp=%d xid=%ux sec=%d fl=%4.4ux ca=%V ya=%V sa=%V ga=%V cha=%E magic=%lux", + op(h->op), h->htype, h->hlen, h->hops, + NetL(h->xid), NetS(h->secs), NetS(h->flags), + h->ciaddr, h->yiaddr, h->siaddr, h->giaddr, h->chaddr, + (ulong)NetL(h->optmagic)); + if(m->pe > (uchar*)h->sname && *h->sname) + m->p = seprint(m->p, m->e, " snam=%s", h->sname); + if(m->pe > (uchar*)h->file && *h->file) + m->p = seprint(m->p, m->e, " file=%s", h->file); + return 0; +} + +Proto bootp = +{ + "bootp", + p_compile, + p_filter, + p_seprint, + p_mux, + p_fields, + defaultframer, +}; diff --git a/src/cmd/ip/snoopy/dat.h b/src/cmd/ip/snoopy/dat.h new file mode 100755 index 00000000..a4258c25 --- /dev/null +++ b/src/cmd/ip/snoopy/dat.h @@ -0,0 +1,106 @@ +typedef struct Field Field; +typedef struct Filter Filter; +typedef struct Msg Msg; +typedef struct Mux Mux; +typedef struct Proto Proto; + +#define NetS(x) ((((uchar*)x)[0]<<8) | ((uchar*)x)[1]) +#define Net3(x) ((((uchar*)x)[0]<<16) | (((uchar*)x)[1]<<8) | ((uchar*)x)[2]) +#define NetL(x) ((((uchar*)x)[0]<<24) | (((uchar*)x)[1]<<16) | (((uchar*)x)[2]<<8) | ((uchar*)x)[3]) + +/* + * one per protocol module + */ +struct Proto +{ + char* name; + void (*compile)(Filter*); + int (*filter)(Filter*, Msg*); + int (*seprint)(Msg*); + Mux* mux; + Field* field; + int (*framer)(int, uchar*, int); +}; +extern Proto *protos[]; + +/* + * one per protocol module, pointed to by Proto.mux + */ +struct Mux +{ + char* name; + ulong val; + Proto* pr; +}; + +/* + * a field defining a comparison filter + */ +struct Field +{ + char* name; + int ftype; + int subop; + char* help; +}; + +/* + * the status of the current message walk + */ +struct Msg +{ + uchar *ps; /* packet ptr */ + uchar *pe; /* packet end */ + + char *p; /* buffer start */ + char *e; /* buffer end */ + + int needroot; /* pr is root, need to see in expression */ + Proto *pr; /* current/next protocol */ +}; + +enum +{ + Fnum, /* just a number */ + Fether, /* ethernet address */ + Fv4ip, /* v4 ip address */ + Fv6ip, /* v6 ip address */ + Fba, /* byte array */ +}; + +/* + * a node in the filter tree + */ +struct Filter { + int op; /* token type */ + char *s; /* string */ + Filter *l; + Filter *r; + + Proto *pr; /* next protocol; + + /* protocol specific */ + int subop; + ulong param; + union { + ulong ulv; + vlong vlv; + uchar a[32]; + }; +}; + +extern void yyinit(char*); +extern int yyparse(void); +extern Filter* newfilter(void); +extern void compile_cmp(char*, Filter*, Field*); +extern void demux(Mux*, ulong, ulong, Msg*, Proto*); +extern int defaultframer(int, uchar*, int); +extern int opendevice(char*, int); + +extern int Nflag; +extern int dflag; +extern int Cflag; + +typedef Filter *Filterptr; +#define YYSTYPE Filterptr +extern Filter *filter; diff --git a/src/cmd/ip/snoopy/dhcp.c b/src/cmd/ip/snoopy/dhcp.c new file mode 100755 index 00000000..8b7649cf --- /dev/null +++ b/src/cmd/ip/snoopy/dhcp.c @@ -0,0 +1,483 @@ +#include <u.h> +#include <libc.h> +#include <ip.h> +#include "dat.h" +#include "protos.h" + +enum +{ + Maxoptlen= 312-4, + + /* dhcp types */ + Discover= 1, + Offer= 2, + Request= 3, + Decline= 4, + Ack= 5, + Nak= 6, + Release= 7, + Inform= 8, + + /* bootp option types */ + OBend= 255, + OBpad= 0, + OBmask= 1, + OBtimeoff= 2, + OBrouter= 3, + OBtimeserver= 4, + OBnameserver= 5, + OBdnserver= 6, + OBlogserver= 7, + OBcookieserver= 8, + OBlprserver= 9, + OBimpressserver= 10, + OBrlserver= 11, + OBhostname= 12, /* 0xc0 */ + OBbflen= 13, + OBdumpfile= 14, + OBdomainname= 15, + OBswapserver= 16, /* 0x10 */ + OBrootpath= 17, + OBextpath= 18, + OBipforward= 19, + OBnonlocal= 20, + OBpolicyfilter= 21, + OBmaxdatagram= 22, + OBttl= 23, + OBpathtimeout= 24, + OBpathplateau= 25, + OBmtu= 26, + OBsubnetslocal= 27, + OBbaddr= 28, + OBdiscovermask= 29, + OBsupplymask= 30, + OBdiscoverrouter= 31, + OBrsserver= 32, /* 0x20 */ + OBstaticroutes= 33, + OBtrailerencap= 34, + OBarptimeout= 35, + OBetherencap= 36, + OBtcpttl= 37, + OBtcpka= 38, + OBtcpkag= 39, + OBnisdomain= 40, + OBniserver= 41, + OBntpserver= 42, + OBvendorinfo= 43, /* 0x2b */ + OBnetbiosns= 44, + OBnetbiosdds= 45, + OBnetbiostype= 46, + OBnetbiosscope= 47, + OBxfontserver= 48, /* 0x30 */ + OBxdispmanager= 49, + OBnisplusdomain= 64, /* 0x40 */ + OBnisplusserver= 65, + OBhomeagent= 68, + OBsmtpserver= 69, + OBpop3server= 70, + OBnntpserver= 71, + OBwwwserver= 72, + OBfingerserver= 73, + OBircserver= 74, + OBstserver= 75, + OBstdaserver= 76, + + /* dhcp options */ + ODipaddr= 50, /* 0x32 */ + ODlease= 51, + ODoverload= 52, + ODtype= 53, /* 0x35 */ + ODserverid= 54, /* 0x36 */ + ODparams= 55, /* 0x37 */ + ODmessage= 56, + ODmaxmsg= 57, + ODrenewaltime= 58, + ODrebindingtime= 59, + ODvendorclass= 60, + ODclientid= 61, /* 0x3d */ + ODtftpserver= 66, + ODbootfile= 67, + + /* plan9 vendor info options */ + OP9fs= 128, // plan9 file servers + OP9auth= 129, // plan9 auth servers +}; + +static void +p_compile(Filter *f) +{ + sysfatal("unknown bootp field: %s", f->s); +} + +static int +p_filter(Filter *f, Msg *m) +{ + USED(f); + USED(m); + return 0; +} + +/* + * convert a byte array to hex + */ +static char +hex(int x) +{ + if(x < 10) + return x + '0'; + return x - 10 + 'a'; +} +static char* +phex(char *p, char *e, char *tag, uchar *o, int n) +{ + p = seprint(p, e, "%s=", tag); + + for(; p+2 < e && n > 0; n--){ + *p++ = hex(*o>>4); + *p++ = hex(*o & 0xf); + o++; + } + return p; +} + +static char* +pstring(char *p, char *e, char *tag, uchar *o, int n) +{ + char msg[256]; + + if(n > sizeof msg - 1) + n = sizeof msg - 1; + memmove(msg, o, n); + msg[n] = 0; + return seprint(p, e, "%s=%s", tag, msg); +} + +static char* +pint(char *p, char *e, char *tag, uchar *o, int n) +{ + int x; + + x = *(char*)o++; + for(; n > 1; n--) + x = (x<<8)|*o++; + return seprint(p, e, "%s=%d", tag, x); +} + +static char* +puint(char *p, char *e, char *tag, uchar *o, int n) +{ + uint x; + + x = *o++; + for(; n > 1; n--) + x = (x<<8)|*o++; + return seprint(p, e, "%s=%ud", tag, x); +} + +static char* +pserver(char *p, char *e, char *tag, uchar *o, int n) +{ + p = seprint(p, e, "%s=(", tag); + while(n >= 4){ + p = seprint(p, e, " %V", o); + n -= 4; + o += 4; + } + p = seprint(p, e, ")"); + return p; +} + +static char *dhcptype[256] = +{ +[Discover] "Discover", +[Offer] "Offer", +[Request] "Request", +[Decline] "Decline", +[Ack] "Ack", +[Nak] "Nak", +[Release] "Release", +[Inform] "Inform", +}; + + +static char* +ptype(char *p, char *e, uchar val) +{ + char *x; + + x = dhcptype[val]; + if(x != nil) + return seprint(p, e, "t=%s", x); + else + return seprint(p, e, "t=%d", val); +} + +static int +p_seprint(Msg *m) +{ + int i, n, code; + uchar *o, *ps; + char *p, *e; + char msg[64]; + + /* no next proto */ + m->pr = nil; + + p = m->p; + e = m->e; + ps = m->ps; + + while(ps < m->pe){ + code = *ps++; + if(code == 255) + break; + if(code == 0) + continue; + + /* ignore anything that's too long */ + n = *ps++; + o = ps; + ps += n; + if(ps > m->pe) + break; + + switch(code){ + case ODipaddr: /* requested ip address */ + p = pserver(p, e, "ipaddr", o, n); + break; + case ODlease: /* requested lease time */ + p = pint(p, e, "lease", o, n); + break; + case ODtype: + p = ptype(p, e, *o); + break; + case ODserverid: + p = pserver(p, e, "serverid", o, n); + break; + case ODmessage: + p = pstring(p, e, "message", o, n); + break; + case ODmaxmsg: + p = puint(p, e, "maxmsg", o, n); + break; + case ODclientid: + p = phex(p, e, "clientid", o, n); + break; + case ODparams: + p = seprint(p, e, " requested=("); + for(i = 0; i < n; i++){ + if(i != 0) + p = seprint(p, e, " "); + p = seprint(p, e, "%ud", o[i]); + } + p = seprint(p, e, ")"); + break; + case ODvendorclass: + p = pstring(p, e, "vendorclass", o, n); + break; + case OBmask: + p = pserver(p, e, "mask", o, n); + break; + case OBtimeoff: + p = pint(p, e, "timeoff", o, n); + break; + case OBrouter: + p = pserver(p, e, "router", o, n); + break; + case OBtimeserver: + p = pserver(p, e, "timesrv", o, n); + break; + case OBnameserver: + p = pserver(p, e, "namesrv", o, n); + break; + case OBdnserver: + p = pserver(p, e, "dnssrv", o, n); + break; + case OBlogserver: + p = pserver(p, e, "logsrv", o, n); + break; + case OBcookieserver: + p = pserver(p, e, "cookiesrv", o, n); + break; + case OBlprserver: + p = pserver(p, e, "lprsrv", o, n); + break; + case OBimpressserver: + p = pserver(p, e, "impresssrv", o, n); + break; + case OBrlserver: + p = pserver(p, e, "rlsrv", o, n); + break; + case OBhostname: + p = pstring(p, e, "hostname", o, n); + break; + case OBbflen: + break; + case OBdumpfile: + p = pstring(p, e, "dumpfile", o, n); + break; + case OBdomainname: + p = pstring(p, e, "domname", o, n); + break; + case OBswapserver: + p = pserver(p, e, "swapsrv", o, n); + break; + case OBrootpath: + p = pstring(p, e, "rootpath", o, n); + break; + case OBextpath: + p = pstring(p, e, "extpath", o, n); + break; + case OBipforward: + p = phex(p, e, "ipforward", o, n); + break; + case OBnonlocal: + p = phex(p, e, "nonlocal", o, n); + break; + case OBpolicyfilter: + p = phex(p, e, "policyfilter", o, n); + break; + case OBmaxdatagram: + p = phex(p, e, "maxdatagram", o, n); + break; + case OBttl: + p = puint(p, e, "ttl", o, n); + break; + case OBpathtimeout: + p = puint(p, e, "pathtimeout", o, n); + break; + case OBpathplateau: + p = phex(p, e, "pathplateau", o, n); + break; + case OBmtu: + p = puint(p, e, "mtu", o, n); + break; + case OBsubnetslocal: + p = pserver(p, e, "subnet", o, n); + break; + case OBbaddr: + p = pserver(p, e, "baddr", o, n); + break; + case OBdiscovermask: + p = pserver(p, e, "discovermsak", o, n); + break; + case OBsupplymask: + p = pserver(p, e, "rousupplymaskter", o, n); + break; + case OBdiscoverrouter: + p = pserver(p, e, "discoverrouter", o, n); + break; + case OBrsserver: + p = pserver(p, e, "rsrouter", o, n); + break; + case OBstaticroutes: + p = phex(p, e, "staticroutes", o, n); + break; + case OBtrailerencap: + p = phex(p, e, "trailerencap", o, n); + break; + case OBarptimeout: + p = puint(p, e, "arptimeout", o, n); + break; + case OBetherencap: + p = phex(p, e, "etherencap", o, n); + break; + case OBtcpttl: + p = puint(p, e, "tcpttl", o, n); + break; + case OBtcpka: + p = puint(p, e, "tcpka", o, n); + break; + case OBtcpkag: + p = phex(p, e, "tcpkag", o, n); + break; + case OBnisdomain: + p = pstring(p, e, "nisdomain", o, n); + break; + case OBniserver: + p = pserver(p, e, "nisrv", o, n); + break; + case OBntpserver: + p = pserver(p, e, "ntpsrv", o, n); + break; + case OBvendorinfo: + p = phex(p, e, "vendorinfo", o, n); + break; + case OBnetbiosns: + p = pserver(p, e, "biosns", o, n); + break; + case OBnetbiosdds: + p = phex(p, e, "biosdds", o, n); + break; + case OBnetbiostype: + p = phex(p, e, "biostype", o, n); + break; + case OBnetbiosscope: + p = phex(p, e, "biosscope", o, n); + break; + case OBxfontserver: + p = pserver(p, e, "fontsrv", o, n); + break; + case OBxdispmanager: + p = pserver(p, e, "xdispmgr", o, n); + break; + case OBnisplusdomain: + p = pstring(p, e, "nisplusdomain", o, n); + break; + case OBnisplusserver: + p = pserver(p, e, "nisplussrv", o, n); + break; + case OBhomeagent: + p = pserver(p, e, "homeagent", o, n); + break; + case OBsmtpserver: + p = pserver(p, e, "smtpsrv", o, n); + break; + case OBpop3server: + p = pserver(p, e, "pop3srv", o, n); + break; + case OBnntpserver: + p = pserver(p, e, "ntpsrv", o, n); + break; + case OBwwwserver: + p = pserver(p, e, "wwwsrv", o, n); + break; + case OBfingerserver: + p = pserver(p, e, "fingersrv", o, n); + break; + case OBircserver: + p = pserver(p, e, "ircsrv", o, n); + break; + case OBstserver: + p = pserver(p, e, "stsrv", o, n); + break; + case OBstdaserver: + p = pserver(p, e, "stdasrv", o, n); + break; + case OBend: + goto out; + default: + snprint(msg, sizeof msg, " T%ud", code); + p = phex(p, e, msg, o, n); + break; + } + if(*ps != OBend) + p = seprint(p, e, " "); + } +out: + m->p = p; + m->ps = ps; + return 0; +} + +Proto dhcp = +{ + "dhcp", + p_compile, + p_filter, + p_seprint, + nil, + nil, + defaultframer, +}; + diff --git a/src/cmd/ip/snoopy/dump.c b/src/cmd/ip/snoopy/dump.c new file mode 100755 index 00000000..5174e4fc --- /dev/null +++ b/src/cmd/ip/snoopy/dump.c @@ -0,0 +1,92 @@ +#include <u.h> +#include <libc.h> +#include <ip.h> +#include <ctype.h> +#include "dat.h" +#include "protos.h" + +static void +p_compile(Filter *f) +{ + USED(f); +} + +static int +p_filter(Filter *f, Msg *m) +{ + USED(f); + USED(m); + return 0; +} + +static char tohex[16] = { + '0', '1', '2', '3', '4', '5', '6', '7', '8', '9', + 'a', 'b', 'c', 'd', 'e', 'f' +}; + +static int +p_seprint(Msg *m) +{ + int c, i, n, isstring; + uchar *ps = m->ps; + char *p = m->p; + char *e = m->e; + + n = m->pe - ps; + if(n > Nflag) + n = Nflag; + + isstring = 1; + for(i = 0; i < n; i++){ + c = ps[i]; + if(!isprint(c) && !isspace(c)){ + isstring = 0; + break; + } + } + + if(isstring){ + for(i = 0; i < n && p+1<e; i++){ + c = ps[i]; + switch(c){ + case '\t': + *p++ = '\\'; + *p++ = 't'; + break; + case '\r': + *p++ = '\\'; + *p++ = 'r'; + break; + case '\n': + *p++ = '\\'; + *p++ = 'n'; + break; + default: + *p++ = c; + } + } + } else { + for(i = 0; i < n && p+1<e; i++){ + c = ps[i]; + *p++ = tohex[c>>4]; + *p++ = tohex[c&0xf]; + } + } + + m->pr = nil; + m->p = p; + m->ps = ps; + + return 0; +} + +Proto dump = +{ + "dump", + p_compile, + p_filter, + p_seprint, + nil, + nil, + defaultframer, +}; diff --git a/src/cmd/ip/snoopy/ether.c b/src/cmd/ip/snoopy/ether.c new file mode 100755 index 00000000..62ec485d --- /dev/null +++ b/src/cmd/ip/snoopy/ether.c @@ -0,0 +1,121 @@ +#include <u.h> +#include <libc.h> +#include <ip.h> +#include "dat.h" +#include "protos.h" + +typedef struct Hdr Hdr; +struct Hdr { + uchar d[6]; + uchar s[6]; + uchar type[2]; + char data[1500]; +}; +#define ETHERMINTU 60 /* minimum transmit size */ +#define ETHERMAXTU 1514 /* maximum transmit size */ +#define ETHERHDRSIZE 14 /* size of an ethernet header */ + +static Mux p_mux[] = +{ + {"ip", 0x0800, } , + {"arp", 0x0806, } , + {"rarp", 0x0806, } , + {"ip6", 0x86dd, } , + {"pppoe_disc", 0x8863, }, + {"pppoe_sess", 0x8864, }, + {0} +}; + +enum +{ + Os, // source + Od, // destination + Oa, // source or destination + Ot, // type +}; + +static Field p_fields[] = +{ + {"s", Fether, Os, "source address", } , + {"d", Fether, Od, "destination address", } , + {"a", Fether, Oa, "source|destination address" } , + {"sd", Fether, Oa, "source|destination address" } , + {"t", Fnum, Ot, "type" } , + {0} +}; + +static void +p_compile(Filter *f) +{ + Mux *m; + + if(f->op == '='){ + compile_cmp(ether.name, f, p_fields); + return; + } + for(m = p_mux; m->name != nil; m++) + if(strcmp(f->s, m->name) == 0){ + f->pr = m->pr; + f->ulv = m->val; + f->subop = Ot; + return; + } + sysfatal("unknown ethernet field or protocol: %s", f->s); +} + +static int +p_filter(Filter *f, Msg *m) +{ + Hdr *h; + + if(m->pe - m->ps < ETHERHDRSIZE) + return 0; + + h = (Hdr*)m->ps; + m->ps += ETHERHDRSIZE; + + switch(f->subop){ + case Os: + return !memcmp(h->s, f->a, 6); + case Od: + return !memcmp(h->d, f->a, 6); + case Oa: + return memcmp(h->s, f->a, 6) == 0 || memcmp(h->d, f->a, 6) == 0; + case Ot: + return NetS(h->type) == f->ulv; + } + return 0; +} + +static int +p_seprint(Msg *m) +{ + Hdr *h; + uint t; + int len; + + len = m->pe - m->ps; + if(len < ETHERHDRSIZE) + return -1; + + h = (Hdr*)m->ps; + m->ps += ETHERHDRSIZE; + + t = NetS(h->type); + demux(p_mux, t, t, m, &dump); + + m->p = seprint(m->p, m->e, "s=%E d=%E pr=%4.4ux ln=%d", h->s, h->d, + t, len); + return 0; +} + +Proto ether = +{ + "ether", + p_compile, + p_filter, + p_seprint, + p_mux, + p_fields, + defaultframer +}; diff --git a/src/cmd/ip/snoopy/gre.c b/src/cmd/ip/snoopy/gre.c new file mode 100755 index 00000000..908d1553 --- /dev/null +++ b/src/cmd/ip/snoopy/gre.c @@ -0,0 +1,83 @@ + +/* GRE flag bits */ +enum { + GRE_chksum = (1<<15), + GRE_routing = (1<<14), + GRE_key = (1<<13), + GRE_seq = (1<<12), + GRE_srcrt = (1<<11), + GRE_recur = (7<<8), + GRE_ack = (1<<7), + GRE_ver = 0x7, +}; + +/* GRE protocols */ +enum { + GRE_sna = 0x0004, + GRE_osi = 0x00fe, + GRE_pup = 0x0200, + GRE_xns = 0x0600, + GRE_ip = 0x0800, + GRE_chaos = 0x0804, + GRE_rfc826 = 0x0806, + GRE_frarp = 0x0808, + GRE_vines = 0x0bad, + GRE_vinesecho = 0x0bae, + GRE_vinesloop = 0x0baf, + GRE_decnetIV = 0x6003, + GRE_ppp = 0x880b, +}; + +int +sprintgre(void *a, char *buf, int len) +{ + int flag, prot, chksum, offset, key, seq, ack; + int n; + uchar *p = a; + + chksum = offset = key = seq = ack = 0; + + flag = NetS(p); + prot = NetS(p+2); + p += 4; len -= 4; + if(flag & (GRE_chksum|GRE_routing)){ + chksum = NetS(p); + offset = NetS(p+2); + p += 4; len -= 4; + } + if(flag&GRE_key){ + key = NetL(p); + p += 4; len -= 4; + } + if(flag&GRE_seq){ + seq = NetL(p); + p += 4; len -= 4; + } + if(flag&GRE_ack){ + ack = NetL(p); + p += 4; len -= 4; + } + /* skip routing if present */ + if(flag&GRE_routing) { + while(len >= 4 && (n=p[3]) != 0) { + len -= n; + p += n; + } + } + + USED(offset); + USED(chksum); + + n = sprint(buf, "GRE(f %4.4ux p %ux k %ux", flag, prot, key); + if(flag&GRE_seq) + n += sprint(buf+n, " s %ux", seq); + if(flag&GRE_ack) + n += sprint(buf+n, " a %ux", ack); + n += sprint(buf+n, " len = %d/%d) ", len, key>>16); + if(prot == GRE_ppp && len > 0) + n += sprintppp(p, buf+n, len); + else + n += sprintx(p, buf+n, len); + + return n; +} diff --git a/src/cmd/ip/snoopy/hdlc.c b/src/cmd/ip/snoopy/hdlc.c new file mode 100755 index 00000000..bed0c35a --- /dev/null +++ b/src/cmd/ip/snoopy/hdlc.c @@ -0,0 +1,174 @@ +#include <u.h> +#include <libc.h> +#include <ip.h> +#include "dat.h" +#include "protos.h" + +enum { + HDLC_frame= 0x7e, + HDLC_esc= 0x7d, + + /* PPP frame fields */ + PPP_addr= 0xff, + PPP_ctl= 0x3, + PPP_initfcs= 0xffff, + PPP_goodfcs= 0xf0b8, +}; + +/* + * Calculate FCS - rfc 1331 + */ +ushort fcstab[256] = +{ + 0x0000, 0x1189, 0x2312, 0x329b, 0x4624, 0x57ad, 0x6536, 0x74bf, + 0x8c48, 0x9dc1, 0xaf5a, 0xbed3, 0xca6c, 0xdbe5, 0xe97e, 0xf8f7, + 0x1081, 0x0108, 0x3393, 0x221a, 0x56a5, 0x472c, 0x75b7, 0x643e, + 0x9cc9, 0x8d40, 0xbfdb, 0xae52, 0xdaed, 0xcb64, 0xf9ff, 0xe876, + 0x2102, 0x308b, 0x0210, 0x1399, 0x6726, 0x76af, 0x4434, 0x55bd, + 0xad4a, 0xbcc3, 0x8e58, 0x9fd1, 0xeb6e, 0xfae7, 0xc87c, 0xd9f5, + 0x3183, 0x200a, 0x1291, 0x0318, 0x77a7, 0x662e, 0x54b5, 0x453c, + 0xbdcb, 0xac42, 0x9ed9, 0x8f50, 0xfbef, 0xea66, 0xd8fd, 0xc974, + 0x4204, 0x538d, 0x6116, 0x709f, 0x0420, 0x15a9, 0x2732, 0x36bb, + 0xce4c, 0xdfc5, 0xed5e, 0xfcd7, 0x8868, 0x99e1, 0xab7a, 0xbaf3, + 0x5285, 0x430c, 0x7197, 0x601e, 0x14a1, 0x0528, 0x37b3, 0x263a, + 0xdecd, 0xcf44, 0xfddf, 0xec56, 0x98e9, 0x8960, 0xbbfb, 0xaa72, + 0x6306, 0x728f, 0x4014, 0x519d, 0x2522, 0x34ab, 0x0630, 0x17b9, + 0xef4e, 0xfec7, 0xcc5c, 0xddd5, 0xa96a, 0xb8e3, 0x8a78, 0x9bf1, + 0x7387, 0x620e, 0x5095, 0x411c, 0x35a3, 0x242a, 0x16b1, 0x0738, + 0xffcf, 0xee46, 0xdcdd, 0xcd54, 0xb9eb, 0xa862, 0x9af9, 0x8b70, + 0x8408, 0x9581, 0xa71a, 0xb693, 0xc22c, 0xd3a5, 0xe13e, 0xf0b7, + 0x0840, 0x19c9, 0x2b52, 0x3adb, 0x4e64, 0x5fed, 0x6d76, 0x7cff, + 0x9489, 0x8500, 0xb79b, 0xa612, 0xd2ad, 0xc324, 0xf1bf, 0xe036, + 0x18c1, 0x0948, 0x3bd3, 0x2a5a, 0x5ee5, 0x4f6c, 0x7df7, 0x6c7e, + 0xa50a, 0xb483, 0x8618, 0x9791, 0xe32e, 0xf2a7, 0xc03c, 0xd1b5, + 0x2942, 0x38cb, 0x0a50, 0x1bd9, 0x6f66, 0x7eef, 0x4c74, 0x5dfd, + 0xb58b, 0xa402, 0x9699, 0x8710, 0xf3af, 0xe226, 0xd0bd, 0xc134, + 0x39c3, 0x284a, 0x1ad1, 0x0b58, 0x7fe7, 0x6e6e, 0x5cf5, 0x4d7c, + 0xc60c, 0xd785, 0xe51e, 0xf497, 0x8028, 0x91a1, 0xa33a, 0xb2b3, + 0x4a44, 0x5bcd, 0x6956, 0x78df, 0x0c60, 0x1de9, 0x2f72, 0x3efb, + 0xd68d, 0xc704, 0xf59f, 0xe416, 0x90a9, 0x8120, 0xb3bb, 0xa232, + 0x5ac5, 0x4b4c, 0x79d7, 0x685e, 0x1ce1, 0x0d68, 0x3ff3, 0x2e7a, + 0xe70e, 0xf687, 0xc41c, 0xd595, 0xa12a, 0xb0a3, 0x8238, 0x93b1, + 0x6b46, 0x7acf, 0x4854, 0x59dd, 0x2d62, 0x3ceb, 0x0e70, 0x1ff9, + 0xf78f, 0xe606, 0xd49d, 0xc514, 0xb1ab, 0xa022, 0x92b9, 0x8330, + 0x7bc7, 0x6a4e, 0x58d5, 0x495c, 0x3de3, 0x2c6a, 0x1ef1, 0x0f78 +}; + +static uchar inbuf[16*1024]; +static int inlen; + +static Mux p_mux[] = +{ + {"ppp", (PPP_addr<<8)|PPP_ctl, } , + {0} +}; + +enum +{ + Ot = 1 +}; + +static void +p_compile(Filter *f) +{ + Mux *m; + + for(m = p_mux; m->name != nil; m++) + if(strcmp(f->s, m->name) == 0){ + f->pr = m->pr; + f->ulv = m->val; + f->subop = Ot; + return; + } + sysfatal("unknown ethernet field or protocol: %s", f->s); +} + +static int +p_filter(Filter *f, Msg *m) +{ + ulong t; + + if(m->pe-m->ps < 2) + return 0; + + switch(f->subop){ + case Ot: + t = (m->ps[0]<<8)|m->ps[1]; + if(t != f->ulv) + return 0; + break; + } + return 1; +} + +static int +p_seprint(Msg *m) +{ + ulong t; + + if(m->pe-m->ps < 2) + return -1; + + t = (m->ps[0]<<8)|m->ps[1]; + m->ps += 2; + demux(p_mux, t, t, m, &dump); + + return 0; +} + +static int +p_framer(int fd, uchar *pkt, int pktlen) +{ + ushort fcs; + uchar *from, *efrom, *to, *eto; + int n; + ulong c; + + eto = pkt+pktlen; + for(;;){ + efrom = memchr(inbuf, HDLC_frame, inlen); + if(efrom == nil){ + if(inlen >= sizeof(inbuf)) + inlen = 0; + n = read(fd, inbuf+inlen, sizeof(inbuf)-inlen); + if(n <= 0) + break; + inlen += n; + continue; + } + + /* checksum and unescape the frame */ + fcs = PPP_initfcs; + to = pkt; + for(from = inbuf; from < efrom;){ + c = *from++; + if(c == HDLC_esc) + c = (*from++) ^ 0x20; + if(to < eto) + *to++ = c; + fcs = (fcs >> 8) ^ fcstab[(fcs ^ c) & 0xff]; + } + + /* move down anything that's left */ + inlen -= efrom+1-inbuf; + memmove(inbuf, efrom+1, inlen); + + /* accept if this is a good packet */ + if(fcs != PPP_goodfcs) + print("bad frame %ld %2.2ux %2.2ux!\n", to-pkt, pkt[0], pkt[1]); + else + return to-pkt; + } + return -1; +} + +Proto hdlc = +{ + "hdlc", + p_compile, + p_filter, + p_seprint, + p_mux, + nil, + p_framer, +}; diff --git a/src/cmd/ip/snoopy/icmp.c b/src/cmd/ip/snoopy/icmp.c new file mode 100755 index 00000000..cf286d87 --- /dev/null +++ b/src/cmd/ip/snoopy/icmp.c @@ -0,0 +1,196 @@ +#include <u.h> +#include <libc.h> +#include <ip.h> +#include "dat.h" +#include "protos.h" + +typedef struct Hdr Hdr; +struct Hdr +{ uchar type; + uchar code; + uchar cksum[2]; /* Checksum */ + uchar data[1]; +}; + +enum +{ + ICMPLEN= 4, +}; + +enum +{ + Ot, /* type */ + Op, /* next protocol */ +}; + +static Field p_fields[] = +{ + {"t", Fnum, Ot, "type", } , + {0} +}; + +enum +{ + EchoRep= 0, + Unreachable= 3, + SrcQuench= 4, + Redirect= 5, + EchoReq= 8, + TimeExceed= 11, + ParamProb= 12, + TSreq= 13, + TSrep= 14, + InfoReq= 15, + InfoRep= 16, +}; + +static Mux p_mux[] = +{ + {"ip", Unreachable, }, + {"ip", SrcQuench, }, + {"ip", Redirect, }, + {"ip", TimeExceed, }, + {"ip", ParamProb, }, + {0}, +}; + +char *icmpmsg[236] = +{ +[EchoRep] "EchoRep", +[Unreachable] "Unreachable", +[SrcQuench] "SrcQuench", +[Redirect] "Redirect", +[EchoReq] "EchoReq", +[TimeExceed] "TimeExceed", +[ParamProb] "ParamProb", +[TSreq] "TSreq", +[TSrep] "TSrep", +[InfoReq] "InfoReq", +[InfoRep] "InfoRep", +}; + +static void +p_compile(Filter *f) +{ + if(f->op == '='){ + compile_cmp(udp.name, f, p_fields); + return; + } + if(strcmp(f->s, "ip") == 0){ + f->pr = p_mux->pr; + f->subop = Op; + return; + } + sysfatal("unknown icmp field or protocol: %s", f->s); +} + +static int +p_filter(Filter *f, Msg *m) +{ + Hdr *h; + + if(m->pe - m->ps < ICMPLEN) + return 0; + + h = (Hdr*)m->ps; + m->ps += ICMPLEN; + + switch(f->subop){ + case Ot: + if(h->type == f->ulv) + return 1; + break; + case Op: + switch(h->type){ + case Unreachable: + case TimeExceed: + case SrcQuench: + case Redirect: + case ParamProb: + m->ps += 4; + return 1; + } + } + return 0; +} + +static int +p_seprint(Msg *m) +{ + Hdr *h; + char *tn; + char *p = m->p; + char *e = m->e; + ushort cksum2, cksum; + + h = (Hdr*)m->ps; + m->ps += ICMPLEN; + m->pr = &dump; + + if(m->pe - m->ps < ICMPLEN) + return -1; + + tn = icmpmsg[h->type]; + if(tn == nil) + p = seprint(p, e, "t=%ud c=%d ck=%4.4ux", h->type, + h->code, (ushort)NetS(h->cksum)); + else + p = seprint(p, e, "t=%s c=%d ck=%4.4ux", tn, + h->code, (ushort)NetS(h->cksum)); + if(Cflag){ + cksum = NetS(h->cksum); + h->cksum[0] = 0; + h->cksum[1] = 0; + cksum2 = ~ptclbsum((uchar*)h, m->pe - m->ps + ICMPLEN) & 0xffff; + if(cksum != cksum2) + p = seprint(p,e, " !ck=%4.4ux", cksum2); + } + switch(h->type){ + case EchoRep: + case EchoReq: + m->ps += 4; + p = seprint(p, e, " id=%ux seq=%ux", + NetS(h->data), NetS(h->data+2)); + break; + case TSreq: + case TSrep: + m->ps += 12; + p = seprint(p, e, " orig=%ud rcv=%ux xmt=%ux", + NetL(h->data), NetL(h->data+4), + NetL(h->data+8)); + m->pr = nil; + break; + case InfoReq: + case InfoRep: + break; + case Unreachable: + case TimeExceed: + case SrcQuench: + m->ps += 4; + m->pr = &ip; + break; + case Redirect: + m->ps += 4; + m->pr = &ip; + p = seprint(p, e, "gw=%V", h->data); + break; + case ParamProb: + m->ps += 4; + m->pr = &ip; + p = seprint(p, e, "ptr=%2.2ux", h->data[0]); + break; + } + m->p = p; + return 0; +} + +Proto icmp = +{ + "icmp", + p_compile, + p_filter, + p_seprint, + p_mux, + p_fields, + defaultframer, +}; diff --git a/src/cmd/ip/snoopy/icmp6.c b/src/cmd/ip/snoopy/icmp6.c new file mode 100755 index 00000000..9c154f88 --- /dev/null +++ b/src/cmd/ip/snoopy/icmp6.c @@ -0,0 +1,428 @@ +#include <u.h> +#include <libc.h> +#include <ip.h> +#include "dat.h" +#include "protos.h" + +typedef struct Hdr Hdr; +struct Hdr +{ uchar type; + uchar code; + uchar cksum[2]; /* Checksum */ + uchar data[1]; +}; + +enum +{ + ICMP6LEN= 4, +}; + +enum +{ + Ot, /* type */ + Op, /* next protocol */}; + +static Field p_fields[] = +{ + {"t", Fnum, Ot, "type", } , + {0} +}; + +enum +{ + // ICMPv6 types + EchoReply = 0, + UnreachableV6 = 1, + PacketTooBigV6 = 2, + TimeExceedV6 = 3, + ParamProblemV6 = 4, + Redirect = 5, + EchoRequest = 8, + TimeExceed = 11, + InParmProblem = 12, + Timestamp = 13, + TimestampReply = 14, + InfoRequest = 15, + InfoReply = 16, + AddrMaskRequest = 17, + AddrMaskReply = 18, + EchoRequestV6 = 128, + EchoReplyV6 = 129, + RouterSolicit = 133, + RouterAdvert = 134, + NbrSolicit = 135, + NbrAdvert = 136, + RedirectV6 = 137, + + Maxtype6 = 137, +}; + +static Mux p_mux[] = +{ + {"ip6", UnreachableV6, }, + {"ip6", RedirectV6, }, + {"ip6", TimeExceedV6, }, + {0}, +}; + +char *icmpmsg6[256] = +{ +[EchoReply] "EchoReply", +[UnreachableV6] "UnreachableV6", +[PacketTooBigV6] "PacketTooBigV6", +[TimeExceedV6] "TimeExceedV6", +[Redirect] "Redirect", +[EchoRequest] "EchoRequest", +[TimeExceed] "TimeExceed", +[InParmProblem] "InParmProblem", +[Timestamp] "Timestamp", +[TimestampReply] "TimestampReply", +[InfoRequest] "InfoRequest", +[InfoReply] "InfoReply", +[AddrMaskRequest] "AddrMaskRequest", +[AddrMaskReply] "AddrMaskReply", +[EchoRequestV6] "EchoRequestV6", +[EchoReplyV6] "EchoReplyV6", +[RouterSolicit] "RouterSolicit", +[RouterAdvert] "RouterAdvert", +[NbrSolicit] "NbrSolicit", +[NbrAdvert] "NbrAdvert", +[RedirectV6] "RedirectV6", +}; + +static char *unreachcode[] = +{ +[0] "no route to destination", +[1] "comm with destination administratively prohibited", +[2] "icmp unreachable: unassigned error code (2)", +[3] "address unreachable", +[4] "port unreachable", +[5] "icmp unreachable: unknown code", +}; + +static char *timexcode[] = +{ +[0] "hop limit exc", +[1] "reassmbl time exc", +[2] "icmp time exc: unknown code", +}; + +static char *parpcode[] = +{ +[0] "erroneous header field encountered", +[1] "unrecognized Next Header type encountered", +[2] "unrecognized IPv6 option encountered", +[3] "icmp par prob: unknown code", +}; +enum +{ + sll = 1, + tll = 2, + pref = 3, + redir = 4, + mtu = 5, +}; + +static char *icmp6opts[256] = +{ +[0] "unknown opt", +[1] "sll_addr", +[2] "tll_addr", +[3] "pref_opt", +[4] "redirect", +[5] "mtu_opt", +}; + +static void +p_compile(Filter *f) +{ + if(f->op == '='){ + compile_cmp(icmp6.name, f, p_fields); + return; + } + if(strcmp(f->s, "ip6") == 0){ + f->pr = p_mux->pr; + f->subop = Op; + return; + } + sysfatal("unknown icmp field or protocol: %s", f->s); +} + +static int +p_filter(Filter *f, Msg *m) +{ + Hdr *h; + + if(m->pe - m->ps < ICMP6LEN) + return 0; + + h = (Hdr*)m->ps; + m->ps += ICMP6LEN; + + switch(f->subop){ + + case Ot: + if(h->type == f->ulv) + return 1; + break; + case Op: + switch(h->type){ + case UnreachableV6: + case RedirectV6: + case TimeExceedV6: + m->ps += 4; + return 1; + } + } + return 0; +} + +static char* +opt_seprint(Msg *m) +{ + int otype, osz, pktsz; + uchar *a; + char *p = m->p; + char *e = m->e; + char *opt; + char optbuf[12]; + + pktsz = m->pe - m->ps; + a = m->ps; + while (pktsz > 0) { + otype = *a; + opt = icmp6opts[otype]; + if(opt == nil){ + sprint(optbuf, "0x%ux", otype); + opt = optbuf; + } + osz = (*(a+1)) * 8; + + switch (otype) { + default: + p = seprint(p, e, "\n option=%s ", opt); + m->pr = &dump; + return p; + + case sll: + case tll: + if ((pktsz < osz) || (osz != 8)) { + p = seprint(p, e, "\n option=%s bad size=%d", opt, osz); + m->pr = &dump; + return p; + } + p = seprint(p, e, "\n option=%s maddr=%E", opt, a+2); + pktsz -= osz; + a += osz; + break; + + case pref: + if ((pktsz < osz) || (osz != 32)) { + p = seprint(p, e, "\n option=%s: bad size=%d", opt, osz); + m->pr = &dump; + return p; + } + + p = seprint(p, e, "\n option=%s pref=%I preflen=%3.3d lflag=%1.1d aflag=%1.1d unused1=%1.1d validlt=%d preflt=%d unused2=%1.1d", + opt, + a+16, + (int) (*(a+2)), + (*(a+3) & (1 << 7))!=0, + (*(a+3) & (1 << 6))!=0, + (*(a+3) & 63) != 0, + NetL(a+4), + NetL(a+8), + NetL(a+12)!=0); + + pktsz -= osz; + a += osz; + break; + + case redir: + if (pktsz < osz) { + p = seprint(p, e, "\n option=%s: bad size=%d", opt, osz); + m->pr = &dump; + return p; + } + + p = seprint(p, e, "\n option=%s len %d", opt, osz); + a += osz; + m->ps = a; + return p; + break; + + case mtu: + if ((pktsz < osz) || (osz != 8)) { + p = seprint(p, e, "\n option=%s: bad size=%d", opt, osz); + m->pr = &dump; + return p; + } + + p = seprint(p, e, "\n option=%s unused=%1.1d mtu=%d", opt, NetL(a+2)!=0, NetL(a+4)); + pktsz -= osz; + a += osz; + break; + } + } + + m->ps = a; + return p; +} + +static int +p_seprint(Msg *m) +{ + Hdr *h; + char *tn; + char *p = m->p; + char *e = m->e; + int i; + uchar *a; +// ushort cksum2, cksum; + + h = (Hdr*)m->ps; + m->ps += ICMP6LEN; + m->pr = &dump; + a = m->ps; + + if(m->pe - m->ps < ICMP6LEN) + return -1; + + tn = icmpmsg6[h->type]; + if(tn == nil) + p = seprint(p, e, "t=%ud c=%d ck=%4.4ux", h->type, + h->code, (ushort)NetS(h->cksum)); + else + p = seprint(p, e, "t=%s c=%d ck=%4.4ux", tn, + h->code, (ushort)NetS(h->cksum)); + + /* + if(Cflag){ + cksum = NetS(h->cksum); + h->cksum[0] = 0; + h->cksum[1] = 0; + cksum2 = ~ptclbsum((uchar*)h, m->pe - m->ps + ICMP6LEN) & 0xffff; + if(cksum != cksum2) + p = seprint(p,e, " !ck=%4.4ux", cksum2); + } + */ + + switch(h->type){ + + case UnreachableV6: + m->ps += 4; + m->pr = &ip6; + if (h->code >= nelem(unreachcode)) + i = nelem(unreachcode)-1; + else + i = h->code; + p = seprint(p, e, " code=%s unused=%1.1d ", unreachcode[i], NetL(a)!=0); + break; + + case PacketTooBigV6: + m->ps += 4; + m->pr = &ip6; + p = seprint(p, e, " mtu=%4.4d ", NetL(a)); + break; + + case TimeExceedV6: + m->ps += 4; + m->pr = &ip6; + if (h->code >= nelem(timexcode)) + i = nelem(timexcode)-1; + else + i = h->code; + p = seprint(p, e, " code=%s unused=%1.1d ", timexcode[i], NetL(a)!=0); + break; + + case ParamProblemV6: + m->ps += 4; + m->pr = &ip6; + if (h->code > nelem(parpcode)) + i = nelem(parpcode)-1; + else + i = h->code; + p = seprint(p, e, " code=%s ptr=%2.2ux", parpcode[i], h->data[0]); + break; + + case EchoReplyV6: + case EchoRequestV6: + m->ps += 4; + p = seprint(p, e, " id=%ux seq=%ux", + NetS(h->data), NetS(h->data+2)); + break; + + case RouterSolicit: + m->ps += 4; + m->pr = nil; + m->p = seprint(p, e, " unused=%1.1d ", NetL(a)!=0); + p = opt_seprint(m); + break; + + case RouterAdvert: + m->ps += 12; + m->pr = nil; + m->p = seprint(p, e, " hoplim=%3.3d mflag=%1.1d oflag=%1.1d unused=%1.1d routerlt=%8.8d reachtime=%d rxmtimer=%d", + (int) *a, + (*(a+1) & (1 << 7)) != 0, + (*(a+1) & (1 << 6)) != 0, + (*(a+1) & 63) != 0, + NetS(a+2), + NetL(a+4), + NetL(a+8)); + p = opt_seprint(m); + break; + + case NbrSolicit: + m->ps += 20; + m->pr = nil; + m->p = seprint(p, e, " unused=%1.1d targ %I", NetL(a)!=0, a+4); + p = opt_seprint(m); + break; + + case NbrAdvert: + m->ps += 20; + m->pr = nil; + m->p = seprint(p, e, " rflag=%1.1d sflag=%1.1d oflag=%1.1d targ=%I", + (*a & (1 << 7)) != 0, + (*a & (1 << 6)) != 0, + (*a & (1 << 5)) != 0, + a+4); + p = opt_seprint(m); + break; + + case RedirectV6: + m->ps += 36; + m->pr = &ip6; + m->p = seprint(p, e, " unused=%1.1d targ=%I dest=%I", NetL(a)!=0, a+4, a+20); + p = opt_seprint(m); + break; + + case Timestamp: + case TimestampReply: + m->ps += 12; + p = seprint(p, e, " orig=%ud rcv=%ux xmt=%ux", + NetL(h->data), NetL(h->data+4), + NetL(h->data+8)); + m->pr = nil; + break; + + case InfoRequest: + case InfoReply: + break; + + } + m->p = p; + return 0; +} + +Proto icmp6 = +{ + "icmp6", + p_compile, + p_filter, + p_seprint, + p_mux, + p_fields, + defaultframer, +}; diff --git a/src/cmd/ip/snoopy/il.c b/src/cmd/ip/snoopy/il.c new file mode 100755 index 00000000..d75a80cd --- /dev/null +++ b/src/cmd/ip/snoopy/il.c @@ -0,0 +1,146 @@ +#include <u.h> +#include <libc.h> +#include <ip.h> +#include "dat.h" +#include "protos.h" + +typedef struct Hdr Hdr; +struct Hdr +{ + uchar sum[2]; /* Checksum including header */ + uchar len[2]; /* Packet length */ + uchar type; /* Packet type */ + uchar spec; /* Special */ + uchar sport[2]; /* Src port */ + uchar dport[2]; /* Dst port */ + uchar id[4]; /* Sequence id */ + uchar ack[4]; /* Acked sequence */ +}; + +enum +{ + ILLEN= 18, +}; + +enum +{ + Os, + Od, + Osd, +}; + +static Field p_fields[] = +{ + {"s", Fnum, Os, "source port", } , + {"d", Fnum, Od, "dest port", } , + {"a", Fnum, Osd, "source/dest port", } , + {"sd", Fnum, Osd, "source/dest port", } , + {0} +}; + +static Mux p_mux[] = +{ + {"ninep", 17007, }, /* exportfs */ + {"ninep", 17008, }, /* 9fs */ + {"ninep", 17005, }, /* ocpu */ + {"ninep", 17010, }, /* ncpu */ + {"ninep", 17013, }, /* cpu */ + {0}, +}; + +static void +p_compile(Filter *f) +{ + Mux *m; + + if(f->op == '='){ + compile_cmp(udp.name, f, p_fields); + return; + } + for(m = p_mux; m->name != nil; m++) + if(strcmp(f->s, m->name) == 0){ + f->pr = m->pr; + f->ulv = m->val; + f->subop = Osd; + return; + } + sysfatal("unknown il field or protocol: %s", f->s); +} + +static int +p_filter(Filter *f, Msg *m) +{ + Hdr *h; + + if(m->pe - m->ps < ILLEN) + return 0; + h = (Hdr*)m->ps; + m->ps += ILLEN; + + switch(f->subop){ + case Os: + return NetS(h->sport) == f->ulv; + case Od: + return NetS(h->dport) == f->ulv; + case Osd: + return NetS(h->sport) == f->ulv || NetS(h->dport) == f->ulv; + } + return 0; +} + +char *pktnames[] = +{ + "Sync", + "Data", + "Dataquery", + "Ack", + "Query", + "State", + "Close" +}; + +static char* +pkttype(int t) +{ + static char b[10]; + + if(t > 6){ + sprint(b, "%d", t); + return b; + } + return pktnames[t]; +} + +static int +p_seprint(Msg *m) +{ + Hdr *h; + int dport, sport; + + if(m->pe - m->ps < ILLEN) + return -1; + h = (Hdr*)m->ps; + m->ps += ILLEN; + + dport = NetS(h->dport); + sport = NetS(h->sport); + demux(p_mux, sport, dport, m, &dump); + + m->p = seprint(m->p, m->e, "s=%d d=%d t=%s id=%lud ack=%lud spec=%d ck=%4.4ux ln=%d", + sport, dport, pkttype(h->type), + (ulong)NetL(h->id), (ulong)NetL(h->ack), + h->spec, + NetS(h->sum), NetS(h->len)); + return 0; +} + +Proto il = +{ + "il", + p_compile, + p_filter, + p_seprint, + p_mux, + p_fields, + defaultframer, +}; diff --git a/src/cmd/ip/snoopy/ip.c b/src/cmd/ip/snoopy/ip.c new file mode 100755 index 00000000..e321e2fe --- /dev/null +++ b/src/cmd/ip/snoopy/ip.c @@ -0,0 +1,236 @@ +#include <u.h> +#include <libc.h> +#include <ip.h> +#include "dat.h" +#include "protos.h" + +typedef struct Hdr Hdr; +struct Hdr +{ + uchar vihl; /* Version and header length */ + uchar tos; /* Type of service */ + uchar length[2]; /* packet length */ + uchar id[2]; /* ip->identification */ + uchar frag[2]; /* Fragment information */ + uchar ttl; /* Time to live */ + uchar proto; /* Protocol */ + uchar cksum[2]; /* Header checksum */ + uchar src[4]; /* IP source */ + uchar dst[4]; /* IP destination */ +}; + +enum +{ + IPHDR = 20, /* sizeof(Iphdr) */ + IP_VER = 0x40, /* Using IP version 4 */ + IP_DF = 0x4000, /* Don't fragment */ + IP_MF = 0x2000, /* More fragments */ +}; + +static Mux p_mux[] = +{ + { "icmp", 1, }, + { "igmp", 2, }, + { "ggp", 3, }, + { "ip", 4, }, + { "st", 5, }, + { "tcp", 6, }, + { "ucl", 7, }, + { "egp", 8, }, + { "igp", 9, }, + { "bbn-rcc-mon", 10, }, + { "nvp-ii", 11, }, + { "pup", 12, }, + { "argus", 13, }, + { "emcon", 14, }, + { "xnet", 15, }, + { "chaos", 16, }, + { "udp", 17, }, + { "mux", 18, }, + { "dcn-meas", 19, }, + { "hmp", 20, }, + { "prm", 21, }, + { "xns-idp", 22, }, + { "trunk-1", 23, }, + { "trunk-2", 24, }, + { "leaf-1", 25, }, + { "leaf-2", 26, }, + { "rdp", 27, }, + { "irtp", 28, }, + { "iso-tp4", 29, }, + { "netblt", 30, }, + { "mfe-nsp", 31, }, + { "merit-inp", 32, }, + { "sep", 33, }, + { "3pc", 34, }, + { "idpr", 35, }, + { "xtp", 36, }, + { "ddp", 37, }, + { "idpr-cmtp", 38, }, + { "tp++", 39, }, + { "il", 40, }, + { "sip", 41, }, + { "sdrp", 42, }, + { "sip-sr", 43, }, + { "sip-frag", 44, }, + { "idrp", 45, }, + { "rsvp", 46, }, + { "gre", 47, }, + { "mhrp", 48, }, + { "bna", 49, }, + { "sipp-esp", 50, }, + { "sipp-ah", 51, }, + { "i-nlsp", 52, }, + { "swipe", 53, }, + { "nhrp", 54, }, + { "any", 61, }, + { "cftp", 62, }, + { "any", 63, }, + { "sat-expak", 64, }, + { "kryptolan", 65, }, + { "rvd", 66, }, + { "ippc", 67, }, + { "any", 68, }, + { "sat-mon", 69, }, + { "visa", 70, }, + { "ipcv", 71, }, + { "cpnx", 72, }, + { "cphb", 73, }, + { "wsn", 74, }, + { "pvp", 75, }, + { "br-sat-mon", 76, }, + { "sun-nd", 77, }, + { "wb-mon", 78, }, + { "wb-expak", 79, }, + { "iso-ip", 80, }, + { "vmtp", 81, }, + { "secure-vmtp", 82, }, + { "vines", 83, }, + { "ttp", 84, }, + { "nsfnet-igp", 85, }, + { "dgp", 86, }, + { "tcf", 87, }, + { "igrp", 88, }, + { "ospf", 89, }, + { "sprite-rpc", 90, }, + { "larp", 91, }, + { "mtp", 92, }, + { "ax.25", 93, }, + { "ipip", 94, }, + { "micp", 95, }, + { "scc-sp", 96, }, + { "etherip", 97, }, + { "encap", 98, }, + { "any", 99, }, + { "gmtp", 100, }, + { "rudp", 254, }, + { 0 } +}; + +enum +{ + Os, // source + Od, // destination + Osd, // source or destination + Ot, // type +}; + +static Field p_fields[] = +{ + {"s", Fv4ip, Os, "source address", } , + {"d", Fv4ip, Od, "destination address", } , + {"a", Fv4ip, Osd, "source|destination address",} , + {"sd", Fv4ip, Osd, "source|destination address",} , + {"t", Fnum, Ot, "sub protocol number", } , + {0} +}; + +static void +p_compile(Filter *f) +{ + Mux *m; + + if(f->op == '='){ + compile_cmp(ip.name, f, p_fields); + return; + } + for(m = p_mux; m->name != nil; m++) + if(strcmp(f->s, m->name) == 0){ + f->pr = m->pr; + f->ulv = m->val; + f->subop = Ot; + return; + } + sysfatal("unknown ip field or protocol: %s", f->s); +} + +static int +p_filter(Filter *f, Msg *m) +{ + Hdr *h; + + if(m->pe - m->ps < IPHDR) + return 0; + + h = (Hdr*)m->ps; + m->ps += ((h->vihl&0xf)<<2); + + switch(f->subop){ + case Os: + return NetL(h->src) == f->ulv; + case Od: + return NetL(h->dst) == f->ulv; + case Osd: + return NetL(h->src) == f->ulv || NetL(h->dst) == f->ulv; + case Ot: + return h->proto == f->ulv; + } + return 0; +} + +static int +p_seprint(Msg *m) +{ + Hdr *h; + int f; + int len; + + if(m->pe - m->ps < IPHDR) + return -1; + h = (Hdr*)m->ps; + + /* next protocol, just dump unless this is the first fragment */ + m->pr = &dump; + f = NetS(h->frag); + if((f & ~(IP_DF|IP_MF)) == 0) + demux(p_mux, h->proto, h->proto, m, &dump); + + /* truncate the message if there's extra */ + len = NetS(h->length); + if(len < m->pe - m->ps) + m->pe = m->ps + len; + + /* next header */ + m->ps += ((h->vihl&0xf)<<2); + + m->p = seprint(m->p, m->e, "s=%V d=%V id=%4.4ux frag=%4.4ux ttl=%3d pr=%d ln=%d", + h->src, h->dst, + NetS(h->id), + NetS(h->frag), + h->ttl, + h->proto, + NetS(h->length) + ); + return 0; +} + +Proto ip = +{ + "ip", + p_compile, + p_filter, + p_seprint, + p_mux, + p_fields, + defaultframer, +}; diff --git a/src/cmd/ip/snoopy/ip6.c b/src/cmd/ip/snoopy/ip6.c new file mode 100755 index 00000000..7fdc9679 --- /dev/null +++ b/src/cmd/ip/snoopy/ip6.c @@ -0,0 +1,309 @@ +#include <u.h> +#include <libc.h> +#include <ip.h> +#include "dat.h" +#include "protos.h" + +typedef struct Hdr Hdr; +struct Hdr +{ + uchar vcf[4]; /* Version and header length */ + uchar length[2]; /* packet length */ + uchar proto; /* Protocol */ + uchar ttl; /* Time to live */ + uchar src[IPaddrlen]; /* IP source */ + uchar dst[IPaddrlen]; /* IP destination */ +}; + +enum +{ + IP6HDR = 40, /* sizeof(Iphdr) */ + IP_VER = 0x60, /* Using IP version 4 */ + HBH_HDR = 0, + ROUT_HDR = 43, + FRAG_HDR = 44, + FRAG_HSZ = 8, /* in bytes */ + DEST_HDR = 60, +}; + +static Mux p_mux[] = +{ + { "igmp", 2, }, + { "ggp", 3, }, + { "ip", 4, }, + { "st", 5, }, + { "tcp", 6, }, + { "ucl", 7, }, + { "egp", 8, }, + { "igp", 9, }, + { "bbn-rcc-mon", 10, }, + { "nvp-ii", 11, }, + { "pup", 12, }, + { "argus", 13, }, + { "emcon", 14, }, + { "xnet", 15, }, + { "chaos", 16, }, + { "udp", 17, }, + { "mux", 18, }, + { "dcn-meas", 19, }, + { "hmp", 20, }, + { "prm", 21, }, + { "xns-idp", 22, }, + { "trunk-1", 23, }, + { "trunk-2", 24, }, + { "leaf-1", 25, }, + { "leaf-2", 26, }, + { "rdp", 27, }, + { "irtp", 28, }, + { "iso-tp4", 29, }, + { "netblt", 30, }, + { "mfe-nsp", 31, }, + { "merit-inp", 32, }, + { "sep", 33, }, + { "3pc", 34, }, + { "idpr", 35, }, + { "xtp", 36, }, + { "ddp", 37, }, + { "idpr-cmtp", 38, }, + { "tp++", 39, }, + { "il", 40, }, + { "sip", 41, }, + { "sdrp", 42, }, + { "idrp", 45, }, + { "rsvp", 46, }, + { "gre", 47, }, + { "mhrp", 48, }, + { "bna", 49, }, + { "sipp-esp", 50, }, + { "sipp-ah", 51, }, + { "i-nlsp", 52, }, + { "swipe", 53, }, + { "nhrp", 54, }, + { "icmp6", 58, }, + { "any", 61, }, + { "cftp", 62, }, + { "any", 63, }, + { "sat-expak", 64, }, + { "kryptolan", 65, }, + { "rvd", 66, }, + { "ippc", 67, }, + { "any", 68, }, + { "sat-mon", 69, }, + { "visa", 70, }, + { "ipcv", 71, }, + { "cpnx", 72, }, + { "cphb", 73, }, + { "wsn", 74, }, + { "pvp", 75, }, + { "br-sat-mon", 76, }, + { "sun-nd", 77, }, + { "wb-mon", 78, }, + { "wb-expak", 79, }, + { "iso-ip", 80, }, + { "vmtp", 81, }, + { "secure-vmtp", 82, }, + { "vines", 83, }, + { "ttp", 84, }, + { "nsfnet-igp", 85, }, + { "dgp", 86, }, + { "tcf", 87, }, + { "igrp", 88, }, + { "ospf", 89, }, + { "sprite-rpc", 90, }, + { "larp", 91, }, + { "mtp", 92, }, + { "ax.25", 93, }, + { "ipip", 94, }, + { "micp", 95, }, + { "scc-sp", 96, }, + { "etherip", 97, }, + { "encap", 98, }, + { "any", 99, }, + { "gmtp", 100, }, + { "rudp", 254, }, + { 0 } +}; + +enum +{ + Os, // source + Od, // destination + Osd, // source or destination + Ot, // type +}; + +static Field p_fields[] = +{ + {"s", Fv6ip, Os, "source address", } , + {"d", Fv6ip, Od, "destination address", } , + {"a", Fv6ip, Osd, "source|destination address",} , + {"t", Fnum, Ot, "sub protocol number", } , + {0} +}; + +static void +p_compile(Filter *f) +{ + Mux *m; + + if(f->op == '='){ + compile_cmp(ip6.name, f, p_fields); + return; + } + for(m = p_mux; m->name != nil; m++) + if(strcmp(f->s, m->name) == 0){ + f->pr = m->pr; + f->ulv = m->val; + f->subop = Ot; + return; + } + sysfatal("unknown ip6 field or protocol: %s", f->s); +} + +static int +v6hdrlen(Hdr *h) +{ + int plen, len = IP6HDR; + int pktlen = IP6HDR + NetS(h->length); + uchar nexthdr = h->proto; + uchar *pkt = (uchar*) h; + + pkt += len; + plen = len; + + while ( (nexthdr == HBH_HDR) || (nexthdr == ROUT_HDR) || + (nexthdr == FRAG_HDR) || (nexthdr == DEST_HDR) ) { + + if (nexthdr == FRAG_HDR) + len = FRAG_HSZ; + else + len = ( ((int) *(pkt+1)) + 1) * 8; + + if (plen + len > pktlen) + return -1; + + pkt += len; + nexthdr = *pkt; + plen += len; + } + return plen; +} + +static int +p_filter(Filter *f, Msg *m) +{ + Hdr *h; + int hlen; + + if(m->pe - m->ps < IP6HDR) + return 0; + + h = (Hdr*)m->ps; + + if ((hlen = v6hdrlen(h)) < 0) + return 0; + else + m->ps += hlen; + switch(f->subop){ + case Os: + return !memcmp(h->src, f->a, IPaddrlen); + case Od: + return !memcmp(h->dst, f->a, IPaddrlen); + case Osd: + return !memcmp(h->src, f->a, IPaddrlen) || !memcmp(h->dst, f->a, IPaddrlen); + case Ot: + return h->proto == f->ulv; + } + return 0; +} + +static int +v6hdr_seprint(Msg *m) +{ + int len = IP6HDR; + uchar *pkt = m->ps; + Hdr *h = (Hdr *) pkt; + int pktlen = IP6HDR + NetS(h->length); + uchar nexthdr = h->proto; + int plen; + + pkt += len; + plen = len; + + while ( (nexthdr == HBH_HDR) || (nexthdr == ROUT_HDR) || + (nexthdr == FRAG_HDR) || (nexthdr == DEST_HDR) ) { + + switch (nexthdr) { + case FRAG_HDR: + m->p = seprint(m->p, m->e, "\n xthdr=frag id=%d offset=%d pr=%d more=%d res1=%d res2=%d", + NetL(pkt+4), + NetS(pkt+2) & ~7, + (int) (*pkt), + (int) (*(pkt+3) & 0x1), + (int) *(pkt+1), + (int) (*(pkt+3) & 0x6) + ); + len = FRAG_HSZ; + break; + + case HBH_HDR: + case ROUT_HDR: + case DEST_HDR: + len = ( ((int) *(pkt+1)) + 1) * 8; + break; + } + + if (plen + len > pktlen) { + m->p = seprint(m->p, m->e, "bad pkt"); + m->pr = &dump; + return -1; + } + plen += len; + pkt += len; + nexthdr = *pkt; + } + + m->ps = pkt; + return 1; + +} + +static int +p_seprint(Msg *m) +{ + Hdr *h; + int len; + + if(m->pe - m->ps < IP6HDR) + return -1; + h = (Hdr*)m->ps; + + demux(p_mux, h->proto, h->proto, m, &dump); + + /* truncate the message if there's extra */ + len = NetS(h->length) + IP6HDR; + if(len < m->pe - m->ps) + m->pe = m->ps + len; + + m->p = seprint(m->p, m->e, "s=%I d=%I ttl=%3d pr=%d ln=%d", + h->src, h->dst, + h->ttl, + h->proto, + NetS(h->length) + ); + + v6hdr_seprint(m); + + return 0; +} + +Proto ip6 = +{ + "ip6", + p_compile, + p_filter, + p_seprint, + p_mux, + p_fields, + defaultframer, +}; diff --git a/src/cmd/ip/snoopy/main.c b/src/cmd/ip/snoopy/main.c new file mode 100755 index 00000000..73c1e27b --- /dev/null +++ b/src/cmd/ip/snoopy/main.c @@ -0,0 +1,841 @@ +#include <u.h> +#include <libc.h> +#include <ip.h> +#include <bio.h> +#include <fcall.h> +#include <libsec.h> +#include "dat.h" +#include "protos.h" +#include "y.tab.h" + +int Cflag; +int pflag; +int Nflag; +int sflag; +int tiflag; +int toflag; + +char *prom = "promiscuous"; + +enum +{ + Pktlen= 64*1024, + Blen= 16*1024, +}; + +Filter *filter; +Proto *root; +Biobuf out; +vlong starttime, pkttime; +int pcap; + +int filterpkt(Filter *f, uchar *ps, uchar *pe, Proto *pr, int); +void printpkt(char *p, char *e, uchar *ps, uchar *pe); +void mkprotograph(void); +Proto* findproto(char *name); +Filter* compile(Filter *f); +void printfilter(Filter *f, char *tag); +void printhelp(void); +void tracepkt(uchar*, int); +void pcaphdr(void); + +void +usage(void) +{ + fprint(2, "usage: %s [-std?] [-c] [-N n] [-f filter] [-h first-header] path", argv0); + exits("usage"); +} + +void +main(int argc, char **argv) +{ + uchar *pkt; + char *buf, *file, *p, *e; + int fd; + int n; + + Binit(&out, 1, OWRITE); + + fmtinstall('E', eipfmt); + fmtinstall('V', eipfmt); + fmtinstall('I', eipfmt); + fmtinstall('H', encodefmt); + fmtinstall('F', fcallfmt); + + pkt = malloc(Pktlen+16); + pkt += 16; + buf = malloc(Blen); + e = buf+Blen-1; + + pflag = 1; + Nflag = 32; + sflag = 0; + + mkprotograph(); + + ARGBEGIN{ + case '?': + printhelp(); + exits(0); + break; + case 'N': + p = ARGF(); + if(p == nil) + usage(); + Nflag = atoi(p); + break; + case 'f': + p = ARGF(); + if(p == nil) + usage(); + yyinit(p); + yyparse(); + break; + case 's': + sflag = 1; + break; + case 'h': + p = ARGF(); + if(p == nil) + usage(); + root = findproto(p); + if(root == nil) + sysfatal("unknown protocol: %s", p); + break; + case 'd': + toflag = 1; + break; + case 'D': + toflag = 1; + pcap = 1; + break; + case 't': + tiflag = 1; + break; + case 'C': + Cflag = 1; + break; + case 'p': + pflag = 0; + break; + }ARGEND; + + if(pcap) + pcaphdr(); + + if(argc == 0) + file = nil; + else + file = argv[0]; + + if(tiflag){ + fd = open(file, OREAD); + if(fd < 0) + sysfatal("opening %s: %r", file); + }else{ + fd = opendevice(file, pflag); + if(fd < 0) + sysfatal("opening device %s: %r", file ? file : "(all)"); + } + if(root == nil) + root = ðer; + filter = compile(filter); + + if(tiflag){ + /* read a trace file */ + for(;;){ + n = read(fd, pkt, 10); + if(n != 10) + break; + pkttime = NetL(pkt+2); + pkttime = (pkttime<<32) | NetL(pkt+6); + if(starttime == 0LL) + starttime = pkttime; + n = NetS(pkt); + if(readn(fd, pkt, n) != n) + break; + if(filterpkt(filter, pkt, pkt+n, root, 1)) + if(toflag) + tracepkt(pkt, n); + else + printpkt(buf, e, pkt, pkt+n); + } + } else { + /* read a real time stream */ + starttime = nsec(); + for(;;){ + n = root->framer(fd, pkt, Pktlen); + if(n <= 0) + break; + pkttime = nsec(); + if(filterpkt(filter, pkt, pkt+n, root, 1)) + if(toflag) + tracepkt(pkt, n); + else + printpkt(buf, e, pkt, pkt+n); + } + } +} + +/* create a new filter node */ +Filter* +newfilter(void) +{ + Filter *f; + + f = mallocz(sizeof(*f), 1); + if(f == nil) + sysfatal("newfilter: %r"); + return f; +} + +/* + * apply filter to packet + */ +int +_filterpkt(Filter *f, Msg *m) +{ + Msg ma; + + if(f == nil) + return 1; + + switch(f->op){ + case '!': + return !_filterpkt(f->l, m); + case LAND: + ma = *m; + return _filterpkt(f->l, &ma) && _filterpkt(f->r, m); + case LOR: + ma = *m; + return _filterpkt(f->l, &ma) || _filterpkt(f->r, m); + case WORD: + if(m->needroot){ + if(m->pr != f->pr) + return 0; + m->needroot = 0; + }else{ + if(m->pr != nil && !(m->pr->filter)(f, m)) + return 0; + } + if(f->l == nil) + return 1; + m->pr = f->pr; + return _filterpkt(f->l, m); + } + sysfatal("internal error: filterpkt op: %d", f->op); + return 0; +} +int +filterpkt(Filter *f, uchar *ps, uchar *pe, Proto *pr, int needroot) +{ + Msg m; + + if(f == nil) + return 1; + + m.needroot = needroot; + m.ps = ps; + m.pe = pe; + m.pr = pr; + return _filterpkt(f, &m); +} + +/* + * from the Unix world + */ +#define PCAP_VERSION_MAJOR 2 +#define PCAP_VERSION_MINOR 4 +#define TCPDUMP_MAGIC 0xa1b2c3d4 + +struct pcap_file_header { + ulong magic; + ushort version_major; + ushort version_minor; + long thiszone; /* gmt to local correction */ + ulong sigfigs; /* accuracy of timestamps */ + ulong snaplen; /* max length saved portion of each pkt */ + ulong linktype; /* data link type (DLT_*) */ +}; + +struct pcap_pkthdr { + uvlong ts; /* time stamp */ + ulong caplen; /* length of portion present */ + ulong len; /* length this packet (off wire) */ +}; + +/* + * pcap trace header + */ +void +pcaphdr(void) +{ + struct pcap_file_header hdr; + + hdr.magic = TCPDUMP_MAGIC; + hdr.version_major = PCAP_VERSION_MAJOR; + hdr.version_minor = PCAP_VERSION_MINOR; + + hdr.thiszone = 0; + hdr.snaplen = 1500; + hdr.sigfigs = 0; + hdr.linktype = 1; + + write(1, &hdr, sizeof(hdr)); +} + +/* + * write out a packet trace + */ +void +tracepkt(uchar *ps, int len) +{ + struct pcap_pkthdr *goo; + + if(pcap){ + goo = (struct pcap_pkthdr*)(ps-16); + goo->ts = pkttime; + goo->caplen = len; + goo->len = len; + write(1, goo, len+16); + } else { + hnputs(ps-10, len); + hnputl(ps-8, pkttime>>32); + hnputl(ps-4, pkttime); + write(1, ps-10, len+10); + } +} + +/* + * format and print a packet + */ +void +printpkt(char *p, char *e, uchar *ps, uchar *pe) +{ + Msg m; + uvlong dt; + Tm tm; + + tm = *localtime(pkttime/1000000000LL); + m.p = seprint(p, e, "%02d/%02d/%04d %02d:%02d:%02d.%09lld", + tm.mon+1, tm.mday, tm.year+1900, tm.hour, tm.min, tm.sec, + pkttime%1000000000LL); + m.ps = ps; + m.pe = pe; + m.e = e; + m.pr = root; + while(m.p < m.e){ + if(!sflag) + m.p = seprint(m.p, m.e, "\n\t"); + m.p = seprint(m.p, m.e, "%s(", m.pr->name); + if((*m.pr->seprint)(&m) < 0){ + m.p = seprint(m.p, m.e, "TOO SHORT"); + m.ps = m.pe; + } + m.p = seprint(m.p, m.e, ")"); + if(m.pr == nil || m.ps >= m.pe) + break; + } + *m.p++ = '\n'; + + if(write(1, p, m.p - p) < 0) + sysfatal("stdout: %r"); +} + +Proto **xprotos; +int nprotos; + +/* look up a protocol by its name */ +Proto* +findproto(char *name) +{ + int i; + + for(i = 0; i < nprotos; i++) + if(strcmp(xprotos[i]->name, name) == 0) + return xprotos[i]; + return nil; +} + +/* + * add an undefined protocol to protos[] + */ +Proto* +addproto(char *name) +{ + Proto *pr; + + xprotos = realloc(xprotos, (nprotos+1)*sizeof(Proto*)); + pr = malloc(sizeof *pr); + *pr = dump; + pr->name = name; + xprotos[nprotos++] = pr; + return pr; +} + +/* + * build a graph of protocols, this could easily be circular. This + * links together all the multiplexing in the protocol modules. + */ +void +mkprotograph(void) +{ + Proto **l; + Proto *pr; + Mux *m; + + /* copy protos into a reallocable area */ + for(nprotos = 0; protos[nprotos] != nil; nprotos++) + ; + xprotos = malloc(nprotos*sizeof(Proto*)); + memmove(xprotos, protos, nprotos*sizeof(Proto*)); + + for(l = protos; *l != nil; l++){ + pr = *l; + for(m = pr->mux; m != nil && m->name != nil; m++){ + m->pr = findproto(m->name); + if(m->pr == nil) + m->pr = addproto(m->name); + } + } +} + +/* + * add in a protocol node + */ +static Filter* +addnode(Filter *f, Proto *pr) +{ + Filter *nf; + nf = newfilter(); + nf->pr = pr; + nf->s = pr->name; + nf->l = f; + nf->op = WORD; + return nf; +} + +/* + * recurse through the protocol graph adding missing nodes + * to the filter if we reach the filter's protocol + */ +static Filter* +_fillin(Filter *f, Proto *last, int depth) +{ + Mux *m; + Filter *nf; + + if(depth-- <= 0) + return nil; + + for(m = last->mux; m != nil && m->name != nil; m++){ + if(m->pr == nil) + continue; + if(f->pr == m->pr) + return f; + nf = _fillin(f, m->pr, depth); + if(nf != nil) + return addnode(nf, m->pr); + } + return nil; +} + +static Filter* +fillin(Filter *f, Proto *last) +{ + int i; + Filter *nf; + + /* hack to make sure top level node is the root */ + if(last == nil){ + if(f->pr == root) + return f; + f = fillin(f, root); + if(f == nil) + return nil; + return addnode(f, root); + } + + /* breadth first search though the protocol graph */ + nf = f; + for(i = 1; i < 20; i++){ + nf = _fillin(f, last, i); + if(nf != nil) + break; + } + return nf; +} + +/* + * massage tree so that all paths from the root to a leaf + * contain a filter node for each header. + * + * also, set f->pr where possible + */ +Filter* +complete(Filter *f, Proto *last) +{ + Proto *pr; + + if(f == nil) + return f; + + /* do a depth first traversal of the filter tree */ + switch(f->op){ + case '!': + f->l = complete(f->l, last); + break; + case LAND: + case LOR: + f->l = complete(f->l, last); + f->r = complete(f->r, last); + break; + case '=': + break; + case WORD: + pr = findproto(f->s); + f->pr = pr; + if(pr == nil){ + if(f->l != nil){ + fprint(2, "%s unknown proto, ignoring params\n", + f->s); + f->l = nil; + } + } else { + f->l = complete(f->l, pr); + f = fillin(f, last); + if(f == nil) + sysfatal("internal error: can't get to %s", pr->name); + } + break; + } + return f; +} + +/* + * merge common nodes under | and & moving the merged node + * above the | or &. + * + * do some constant foldong, e.g. `true & x' becomes x and + * 'true | x' becomes true. + */ +static int changed; + +static Filter* +_optimize(Filter *f) +{ + Filter *l; + + if(f == nil) + return f; + + switch(f->op){ + case '!': + /* is child also a not */ + if(f->l->op == '!'){ + changed = 1; + return f->l->l; + } + break; + case LOR: + /* are two children the same protocol? */ + if(f->l->op != f->r->op || f->r->op != WORD + || f->l->pr != f->r->pr || f->l->pr == nil) + break; /* no optimization */ + + changed = 1; + + /* constant folding */ + /* if either child is childless, just return that */ + if(f->l->l == nil) + return f->l; + else if(f->r->l == nil) + return f->r; + + /* move the common node up, thow away one node */ + l = f->l; + f->l = l->l; + f->r = f->r->l; + l->l = f; + return l; + case LAND: + /* are two children the same protocol? */ + if(f->l->op != f->r->op || f->r->op != WORD + || f->l->pr != f->r->pr || f->l->pr == nil) + break; /* no optimization */ + + changed = 1; + + /* constant folding */ + /* if either child is childless, ignore it */ + if(f->l->l == nil) + return f->r; + else if(f->r->l == nil) + return f->l; + + /* move the common node up, thow away one node */ + l = f->l; + f->l = _optimize(l->l); + f->r = _optimize(f->r->l); + l->l = f; + return l; + } + f->l = _optimize(f->l); + f->r = _optimize(f->r); + return f; +} + +Filter* +optimize(Filter *f) +{ + do{ + changed = 0; + f = _optimize(f); + }while(changed); + + return f; +} + +/* + * find any top level nodes that aren't the root + */ +int +findbogus(Filter *f) +{ + int rv; + + if(f->op != WORD){ + rv = findbogus(f->l); + if(f->r) + rv |= findbogus(f->r); + return rv; + } else if(f->pr != root){ + fprint(2, "bad top-level protocol: %s\n", f->s); + return 1; + } + return 0; +} + +/* + * compile the filter + */ +static void +_compile(Filter *f, Proto *last) +{ + if(f == nil) + return; + + switch(f->op){ + case '!': + _compile(f->l, last); + break; + case LOR: + case LAND: + _compile(f->l, last); + _compile(f->r, last); + break; + case WORD: + if(last != nil) + (*last->compile)(f); + if(f->l) + _compile(f->l, f->pr); + break; + case '=': + if(last == nil) + sysfatal("internal error: compilewalk: badly formed tree"); + (*last->compile)(f); + break; + default: + sysfatal("internal error: compilewalk op: %d", f->op); + } +} + +Filter* +compile(Filter *f) +{ + if(f == nil) + return f; + + /* fill in the missing header filters */ + f = complete(f, nil); + + /* constant folding */ + f = optimize(f); + if(!toflag) + printfilter(f, "after optimize"); + + /* protocol specific compilations */ + _compile(f, nil); + + /* at this point, the root had better be the root proto */ + if(findbogus(f)){ + fprint(2, "bogus filter\n"); + exits("bad filter"); + } + + return f; +} + +/* + * parse a byte array + */ +int +parseba(uchar *to, char *from) +{ + char nip[4]; + char *p; + int i; + + p = from; + for(i = 0; i < 16; i++){ + if(*p == 0) + return -1; + nip[0] = *p++; + if(*p == 0) + return -1; + nip[1] = *p++; + nip[2] = 0; + to[i] = strtoul(nip, 0, 16); + } + return i; +} + +/* + * compile WORD = WORD, becomes a single node with a subop + */ +void +compile_cmp(char *proto, Filter *f, Field *fld) +{ + uchar x[IPaddrlen]; + + if(f->op != '=') + sysfatal("internal error: compile_cmp %s: not a cmp", proto); + + for(; fld->name != nil; fld++){ + if(strcmp(f->l->s, fld->name) == 0){ + f->op = WORD; + f->subop = fld->subop; + switch(fld->ftype){ + case Fnum: + f->ulv = atoi(f->r->s); + break; + case Fether: + parseether(f->a, f->r->s); + break; + case Fv4ip: + f->ulv = parseip(x, f->r->s); + break; + case Fv6ip: + parseip(f->a, f->r->s); + break; + case Fba: + parseba(f->a, f->r->s); + break; + default: + sysfatal("internal error: compile_cmp %s: %d", + proto, fld->ftype); + } + f->l = f->r = nil; + return; + } + } + sysfatal("unknown %s field in: %s = %s", proto, f->l->s, f->r->s); +} + +void +_pf(Filter *f) +{ + char *s; + + if(f == nil) + return; + + s = nil; + switch(f->op){ + case '!': + fprint(2, "!"); + _pf(f->l); + break; + case WORD: + fprint(2, "%s", f->s); + if(f->l != nil){ + fprint(2, "( "); + _pf(f->l); + fprint(2, " )"); + } + break; + case LAND: + s = "&&"; + goto print; + case LOR: + s = "||"; + goto print; + case '=': + print: + _pf(f->l); + if(s) + fprint(2, " %s ", s); + else + fprint(2, " %c ", f->op); + _pf(f->r); + break; + default: + fprint(2, "???"); + break; + } +} + +void +printfilter(Filter *f, char *tag) +{ + fprint(2, "%s: ", tag); + _pf(f); + fprint(2, "\n"); +} + +void +printhelp(void) +{ + Proto *pr, **l; + Mux *m; + Field *f; + + for(l = protos; *l != nil; l++){ + pr = *l; + if(pr->field != nil){ + print("%s's filter attr:\n", pr->name); + for(f = pr->field; f->name != nil; f++) + print("\t%s\t- %s\n", f->name, f->help); + } + if(pr->mux != nil){ + print("%s's subprotos:\n", pr->name); + for(m = pr->mux; m->name != nil; m++) + print("\t%s\n", m->name); + } + } +} + +/* + * demultiplex to next prototol header + */ +void +demux(Mux *mx, ulong val1, ulong val2, Msg *m, Proto *def) +{ + m->pr = def; + for(mx = mx; mx->name != nil; mx++){ + if(val1 == mx->val || val2 == mx->val){ + m->pr = mx->pr; + break; + } + } +} + +/* + * default framer just assumes the input packet is + * a single read + */ +int +defaultframer(int fd, uchar *pkt, int pktlen) +{ + return read(fd, pkt, pktlen); +} diff --git a/src/cmd/ip/snoopy/mkfile b/src/cmd/ip/snoopy/mkfile new file mode 100755 index 00000000..5a76591f --- /dev/null +++ b/src/cmd/ip/snoopy/mkfile @@ -0,0 +1,69 @@ +<$PLAN9/src/mkhdr + +TARG=snoopy +PROTOS=\ + ether\ + ip\ + ip6\ + dump\ + arp\ + rarp\ + udp\ + bootp\ + dhcp\ + hdlc\ + rtp\ + rtcp\ + tcp\ + il\ + icmp\ + icmp6\ + ninep\ + ospf\ + ppp\ + ppp_ccp\ + ppp_lcp\ + ppp_chap\ + ppp_ipcp\ + pppoe_sess\ + pppoe_disc\ + +POBJS=${PROTOS:%=%.$O} + +OFILES= main.$O\ + y.tab.$O\ + protos.$O\ + $SYSNAME.$O\ + $POBJS + +HFILES=dat.h\ + protos.h\ + y.tab.h\ + +<$PLAN9/src/mkone + +protos.h: mkfile + ( + for i in $PROTOS + do + echo extern Proto $i';' + done + ) > protos.h + +protos.c: mkfile + ( + echo '#include <u.h>' + echo '#include <libc.h>' + echo '#include "dat.h"' + echo '#include "protos.h"' + echo 'Proto *protos[] =' + echo '{' + for i in $PROTOS + do + echo ' &'$i',' + done + echo ' 0,' + echo '};' + ) > protos.c + +y.tab.c: filter.y diff --git a/src/cmd/ip/snoopy/ninep.c b/src/cmd/ip/snoopy/ninep.c new file mode 100755 index 00000000..8e7a8d4e --- /dev/null +++ b/src/cmd/ip/snoopy/ninep.c @@ -0,0 +1,55 @@ +#include <u.h> +#include <libc.h> +#include <ip.h> +#include <fcall.h> +#include "dat.h" +#include "protos.h" + +static void +p_compile(Filter *f) +{ + sysfatal("unknown ninep field: %s", f->s); +} + +static int +p_filter(Filter *f, Msg *m) +{ + USED(f); + USED(m); + return 0; +} + +static int +p_seprint(Msg *m) +{ + Fcall f; + char *p; + + memset(&f, 0, sizeof(f)); + f.type = 0; + f.data = 0; /* protection for %F */ + if(convM2S(m->ps, m->pe-m->ps, &f)){ + p = m->p; + m->p = seprint(m->p, m->e, "%F", &f); + while(p < m->p){ + p = strchr(p, '\n'); + if(p == nil) + break; + *p = '\\'; + } + } else + dump.seprint(m); + m->pr = nil; + return 0; +} + +Proto ninep = +{ + "ninep", + p_compile, + p_filter, + p_seprint, + nil, + nil, + defaultframer, +}; diff --git a/src/cmd/ip/snoopy/ospf.c b/src/cmd/ip/snoopy/ospf.c new file mode 100755 index 00000000..35ed55e1 --- /dev/null +++ b/src/cmd/ip/snoopy/ospf.c @@ -0,0 +1,404 @@ +#include <u.h> +#include <libc.h> +#include <ip.h> +#include <libsec.h> +#include "dat.h" +#include "protos.h" + + +/* + * OSPF packets + */ +typedef struct Ospfpkt Ospfpkt; +struct Ospfpkt +{ + uchar version; + uchar type; + uchar length[2]; + uchar router[4]; + uchar area[4]; + uchar sum[2]; + uchar autype[2]; + uchar auth[8]; + uchar data[1]; +}; +#define OSPF_HDRSIZE 24 + +enum +{ + OSPFhello= 1, + OSPFdd= 2, + OSPFlsrequest= 3, + OSPFlsupdate= 4, + OSPFlsack= 5, +}; + + +char *ospftype[] = { + [OSPFhello] "hello", + [OSPFdd] "data definition", + [OSPFlsrequest] "link state request", + [OSPFlsupdate] "link state update", + [OSPFlsack] "link state ack", +}; + +char* +ospfpkttype(int x) +{ + static char type[16]; + + if(x > 0 && x <= OSPFlsack) + return ospftype[x]; + sprint(type, "type %d", x); + return type; +} + +char* +ospfauth(Ospfpkt *ospf) +{ + static char auth[100]; + + switch(ospf->type){ + case 0: + return "no authentication"; + case 1: + sprint(auth, "password(%8.8ux %8.8ux)", NetL(ospf->auth), + NetL(ospf->auth+4)); + break; + case 2: + sprint(auth, "crypto(plen %d id %d dlen %d)", NetS(ospf->auth), + ospf->auth[2], ospf->auth[3]); + break; + default: + sprint(auth, "auth%d(%8.8ux %8.8ux)", NetS(ospf->autype), NetL(ospf->auth), + NetL(ospf->auth+4)); + } + return auth; +} + +typedef struct Ospfhello Ospfhello; +struct Ospfhello +{ + uchar mask[4]; + uchar interval[2]; + uchar options; + uchar pri; + uchar deadint[4]; + uchar designated[4]; + uchar bdesignated[4]; + uchar neighbor[1]; +}; + +char* +seprintospfhello(char *p, char *e, void *a, int x) +{ + Ospfhello *h = a; + + USED(x); + return seprint(p, e, "%s(mask %V interval %d opt %ux pri %ux deadt %d designated %V bdesignated %V)", + ospftype[OSPFhello], + h->mask, NetS(h->interval), h->options, h->pri, + NetL(h->deadint), h->designated, h->bdesignated); +} + +enum +{ + LSARouter= 1, + LSANetwork= 2, + LSASummN= 3, + LSASummR= 4, + LSAASext= 5 +}; + + +char *lsatype[] = { + [LSARouter] "Router LSA", + [LSANetwork] "Network LSA", + [LSASummN] "Summary LSA (Network)", + [LSASummR] "Summary LSA (Router)", + [LSAASext] "LSA AS external", +}; + +char* +lsapkttype(int x) +{ + static char type[16]; + + if(x > 0 && x <= LSAASext) + return lsatype[x]; + sprint(type, "type %d", x); + return type; +} + +/* OSPF Link State Advertisement Header */ +/* rfc2178 section 12.1 */ +/* data of Ospfpkt point to a 4-uchar value that is the # of LSAs */ +struct OspfLSAhdr { + uchar lsage[2]; + uchar options; /* 0x2=stub area, 0x1=TOS routing capable */ + + uchar lstype; /* 1=Router-LSAs + * 2=Network-LSAs + * 3=Summary-LSAs (to network) + * 4=Summary-LSAs (to AS boundary routers) + * 5=AS-External-LSAs + */ + uchar lsid[4]; + uchar advtrt[4]; + + uchar lsseqno[4]; + uchar lscksum[2]; + uchar lsalen[2]; /* includes the 20 byte lsa header */ +}; + +struct Ospfrt { + uchar linkid[4]; + uchar linkdata[4]; + uchar typ; + uchar numtos; + uchar metric[2]; + +}; + +struct OspfrtLSA { + struct OspfLSAhdr hdr; + uchar netmask[4]; +}; + +struct OspfntLSA { + struct OspfLSAhdr hdr; + uchar netmask[4]; + uchar attrt[4]; +}; + +/* Summary Link State Advertisement info */ +struct Ospfsumm { + uchar flag; /* always zero */ + uchar metric[3]; +}; + +struct OspfsummLSA { + struct OspfLSAhdr hdr; + uchar netmask[4]; + struct Ospfsumm lsa; +}; + +/* AS external Link State Advertisement info */ +struct OspfASext { + uchar flag; /* external */ + uchar metric[3]; + uchar fwdaddr[4]; + uchar exrttag[4]; +}; + +struct OspfASextLSA { + struct OspfLSAhdr hdr; + uchar netmask[4]; + struct OspfASext lsa; +}; + +/* OSPF Link State Update Packet */ +struct OspfLSupdpkt { + uchar lsacnt[4]; + union { + uchar hdr[1]; + struct OspfrtLSA rt[1]; + struct OspfntLSA nt[1]; + struct OspfsummLSA sum[1]; + struct OspfASextLSA as[1]; + }; +}; + +char* +seprintospflsaheader(char *p, char *e, struct OspfLSAhdr *h) +{ + return seprint(p, e, "age %d opt %ux type %ux lsid %V adv_rt %V seqno %ux c %4.4ux l %d", + NetS(h->lsage), h->options&0xff, h->lstype, + h->lsid, h->advtrt, NetL(h->lsseqno), NetS(h->lscksum), + NetS(h->lsalen)); +} + +/* OSPF Database Description Packet */ +struct OspfDDpkt { + uchar intMTU[2]; + uchar options; + uchar bits; + uchar DDseqno[4]; + struct OspfLSAhdr hdr[1]; /* LSA headers... */ +}; + +char* +seprintospfdatadesc(char *p, char *e, void *a, int len) +{ + int nlsa, i; + struct OspfDDpkt *g; + + g = (struct OspfDDpkt *)a; + nlsa = len/sizeof(struct OspfLSAhdr); + for (i=0; i<nlsa; i++) { + p = seprint(p, e, "lsa%d(", i); + p = seprintospflsaheader(p, e, &(g->hdr[i])); + p = seprint(p, e, ")"); + } + return seprint(p, e, ")"); +} + +char* +seprintospflsupdate(char *p, char *e, void *a, int len) +{ + int nlsa, i; + struct OspfLSupdpkt *g; + struct OspfLSAhdr *h; + + g = (struct OspfLSupdpkt *)a; + nlsa = NetL(g->lsacnt); + h = (struct OspfLSAhdr *)(g->hdr); + p = seprint(p, e, "%d-%s(", nlsa, ospfpkttype(OSPFlsupdate)); + + switch(h->lstype) { + case LSARouter: + { +/* struct OspfrtLSA *h; + */ + } + break; + case LSANetwork: + { + struct OspfntLSA *h; + + for (i=0; i<nlsa; i++) { + h = &(g->nt[i]); + p = seprint(p, e, "lsa%d(", i); + p = seprintospflsaheader(p, e, &(h->hdr)); + p = seprint(p, e, " mask %V attrt %V)", + h->netmask, h->attrt); + } + } + break; + case LSASummN: + case LSASummR: + { + struct OspfsummLSA *h; + + for (i=0; i<nlsa; i++) { + h = &(g->sum[i]); + p = seprint(p, e, "lsa%d(", i); + p = seprintospflsaheader(p, e, &(h->hdr)); + p = seprint(p, e, " mask %V met %d)", + h->netmask, Net3(h->lsa.metric)); + } + } + break; + case LSAASext: + { + struct OspfASextLSA *h; + + for (i=0; i<nlsa; i++) { + h = &(g->as[i]); + p = seprint(p, e, " lsa%d(", i); + p = seprintospflsaheader(p, e, &(h->hdr)); + p = seprint(p, e, " mask %V extflg %1.1ux met %d fwdaddr %V extrtflg %ux)", + h->netmask, h->lsa.flag, Net3(h->lsa.metric), + h->lsa.fwdaddr, NetL(h->lsa.exrttag)); + } + } + break; + default: + p = seprint(p, e, "Not an LS update, lstype %d ", h->lstype); + p = seprint(p, e, " %.*H", len>64?64:len, a); + break; + } + return seprint(p, e, ")"); +} + +char* +seprintospflsack(char *p, char *e, void *a, int len) +{ + int nlsa, i; + struct OspfLSAhdr *h; + + h = (struct OspfLSAhdr *)a; + nlsa = len/sizeof(struct OspfLSAhdr); + p = seprint(p, e, "%d-%s(", nlsa, ospfpkttype(OSPFlsack)); + for (i=0; i<nlsa; i++) { + p = seprint(p, e, " lsa%d(", i); + p = seprintospflsaheader(p, e, &(h[i])); + p = seprint(p, e, ")"); + } + return seprint(p, e, ")"); +} + +static void +p_compile(Filter *f) +{ + sysfatal("unknown ospf field: %s", f->s); +} + +static int +p_filter(Filter *f, Msg *m) +{ + USED(f); + USED(m); + return 0; +} + +int +p_seprint(Msg *m) +{ + Ospfpkt *ospf; + int len, x; + char *p, *e; + + len = m->pe - m->ps; + if(len < OSPF_HDRSIZE) + return -1; + p = m->p; + e = m->e; + + /* adjust packet size */ + ospf = (Ospfpkt*)m->ps; + x = NetS(ospf->length); + if(x < len) + return -1; + x -= OSPF_HDRSIZE; + + p = seprint(p, e, "ver=%d type=%d len=%d r=%V a=%V c=%4.4ux %s ", + ospf->version, ospf->type, x, + ospf->router, ospf->area, NetS(ospf->sum), + ospfauth(ospf)); + + switch (ospf->type) { + case OSPFhello: + p = seprintospfhello(p, e, ospf->data, x); + break; + case OSPFdd: + p = seprintospfdatadesc(p, e, ospf->data, x); + break; + case OSPFlsrequest: + p = seprint(p, e, " %s->", ospfpkttype(ospf->type)); + goto Default; + case OSPFlsupdate: + p = seprintospflsupdate(p, e, ospf->data, x); + break; + case OSPFlsack: + p = seprintospflsack(p, e, ospf->data, x); + break; + default: +Default: + p = seprint(p, e, " data=%.*H", x>64?64:x, ospf->data); + } + m->p = p; + m->pr = nil; + return 0; +} + +Proto ospf = +{ + "ospf", + p_compile, + p_filter, + p_seprint, + nil, + nil, + defaultframer, +}; diff --git a/src/cmd/ip/snoopy/ppp.c b/src/cmd/ip/snoopy/ppp.c new file mode 100755 index 00000000..5c497a8c --- /dev/null +++ b/src/cmd/ip/snoopy/ppp.c @@ -0,0 +1,629 @@ +#include <u.h> +#include <libc.h> +#include <ip.h> +#include <libsec.h> +#include "dat.h" +#include "protos.h" + +/* PPP stuff */ +enum { + PPP_addr= 0xff, + PPP_ctl= 0x3, + PPP_period= 3*1000, /* period of retransmit process (in ms) */ +}; + +/* PPP protocols */ +enum { + PPP_ip= 0x21, /* internet */ + PPP_vjctcp= 0x2d, /* compressing van jacobson tcp */ + PPP_vjutcp= 0x2f, /* uncompressing van jacobson tcp */ + PPP_ml= 0x3d, /* multi link */ + PPP_comp= 0xfd, /* compressed packets */ + PPP_ipcp= 0x8021, /* ip control */ + PPP_ccp= 0x80fd, /* compression control */ + PPP_passwd= 0xc023, /* passwd authentication */ + PPP_lcp= 0xc021, /* link control */ + PPP_lqm= 0xc025, /* link quality monitoring */ + PPP_chap= 0xc223, /* challenge/response */ +}; + +/* LCP protocol (and IPCP) */ + + +typedef struct Lcppkt Lcppkt; +struct Lcppkt +{ + uchar code; + uchar id; + uchar len[2]; + uchar data[1]; +}; + +typedef struct Lcpopt Lcpopt; +struct Lcpopt +{ + uchar type; + uchar len; + uchar data[1]; +}; + +enum +{ + /* LCP codes */ + Lconfreq= 1, + Lconfack= 2, + Lconfnak= 3, + Lconfrej= 4, + Ltermreq= 5, + Ltermack= 6, + Lcoderej= 7, + Lprotorej= 8, + Lechoreq= 9, + Lechoack= 10, + Ldiscard= 11, + Lresetreq= 14, /* for ccp only */ + Lresetack= 15, /* for ccp only */ + + /* Lcp configure options */ + Omtu= 1, + Octlmap= 2, + Oauth= 3, + Oquality= 4, + Omagic= 5, + Opc= 7, + Oac= 8, + + /* authentication protocols */ + APmd5= 5, + APmschap= 128, + + /* Chap codes */ + Cchallenge= 1, + Cresponse= 2, + Csuccess= 3, + Cfailure= 4, + + /* ipcp configure options */ + Oipaddrs= 1, + Oipcompress= 2, + Oipaddr= 3, + Oipdns= 129, + Oipwins= 130, + Oipdns2= 131, + Oipwins2= 132, +}; + +char * +lcpcode[] = { + 0, + "confreq", + "confack", + "confnak", + "confrej", + "termreq", + "termack", + "coderej", + "protorej", + "echoreq", + "echoack", + "discard", + "id", + "timeremain", + "resetreq", + "resetack", +}; + +static Mux p_mux[] = +{ + {"ip", PPP_ip, }, + {"ppp_vjctcp", PPP_vjctcp, }, + {"ppp_vjutcp", PPP_vjutcp, }, + {"ppp_ml", PPP_ml, }, + {"ppp_comp", PPP_comp, }, + {"ppp_ipcp", PPP_ipcp, }, + {"ppp_ccp", PPP_ccp, }, + {"ppp_passwd", PPP_passwd, }, + {"ppp_lcp", PPP_lcp, }, + {"ppp_lqm", PPP_lqm, }, + {"ppp_chap", PPP_chap, }, + {0}, +}; + +enum +{ + OOproto, +}; + +static void +p_compile(Filter *f) +{ + Mux *m; + + for(m = p_mux; m->name != nil; m++) + if(strcmp(f->s, m->name) == 0){ + f->pr = m->pr; + f->ulv = m->val; + f->subop = OOproto; + return; + } + + sysfatal("unknown ppp field or protocol: %s", f->s); +} + +static int +p_filter(Filter *f, Msg *m) +{ + int proto; + int len; + + if(f->subop != OOproto) + return 0; + + len = m->pe - m->ps; + if(len < 3) + return -1; + + if(m->ps[0] == PPP_addr && m->ps[1] == PPP_ctl) + m->ps += 2; + + proto = *m->ps++; + if((proto&1) == 0) + proto = (proto<<8) | *m->ps++; + + if(proto == f->ulv) + return 1; + + return 0; +} + +static int +p_seprint(Msg *m) +{ + int proto; + int len; + + len = m->pe - m->ps; + if(len < 3) + return -1; + + if(m->ps[0] == PPP_addr && m->ps[1] == PPP_ctl) + m->ps += 2; + + proto = *m->ps++; + if((proto&1) == 0) + proto = (proto<<8) | *m->ps++; + + m->p = seprint(m->p, m->e, "pr=%ud len=%d", proto, len); + demux(p_mux, proto, proto, m, &dump); + + return 0; +} + +static int +p_seprintchap(Msg *m) +{ + Lcppkt *lcp; + char *p, *e; + int len; + + if(m->pe-m->ps < 4) + return -1; + + p = m->p; + e = m->e; + m->pr = nil; + + /* resize packet */ + lcp = (Lcppkt*)m->ps; + len = NetS(lcp->len); + if(m->ps+len < m->pe) + m->pe = m->ps+len; + else if(m->ps+len > m->pe) + return -1; + + p = seprint(p, e, "id=%d code=%d", lcp->id, lcp->code); + switch(lcp->code) { + default: + p = seprint(p, e, " data=%.*H", len>64?64:len, lcp->data); + break; + case 1: + case 2: + if(lcp->data[0] > len-4){ + p = seprint(p, e, "%.*H", len-4, lcp->data); + } else { + p = seprint(p, e, " %s=", lcp->code==1?"challenge ":"response "); + p = seprint(p, e, "%.*H", lcp->data[0], lcp->data+1); + p = seprint(p, e, " name="); + p = seprint(p, e, "%.*H", len-4-lcp->data[0]-1, lcp->data+lcp->data[0]+1); + } + break; + case 3: + case 4: + if(len > 64) + len = 64; + p = seprint(p, e, " %s=%.*H", lcp->code==3?"success ":"failure", + len>64?64:len, lcp->data); + break; + } + m->p = seprint(p, e, " len=%d", len); + return 0; +} + +static char* +seprintlcpopt(char *p, char *e, void *a, int len) +{ + Lcpopt *o; + int proto, x, period; + uchar *cp, *ecp; + + cp = a; + ecp = cp+len; + + for(; cp < ecp; cp += o->len){ + o = (Lcpopt*)cp; + if(cp + o->len > ecp || o->len == 0){ + p = seprint(p, e, " bad-opt-len=%d", o->len); + return p; + } + + switch(o->type){ + default: + p = seprint(p, e, " (type=%d len=%d)", o->type, o->len); + break; + case Omtu: + p = seprint(p, e, " mtu=%d", NetS(o->data)); + break; + case Octlmap: + p = seprint(p, e, " ctlmap=%ux", NetL(o->data)); + break; + case Oauth: + proto = NetS(o->data); + switch(proto) { + default: + p = seprint(p, e, " auth=%d", proto); + break; + case PPP_passwd: + p = seprint(p, e, " auth=passwd"); + break; + case PPP_chap: + p = seprint(p, e, " (auth=chap data=%2.2ux)", o->data[2]); + break; + } + break; + case Oquality: + proto = NetS(o->data); + switch(proto) { + default: + p = seprint(p, e, " qproto=%d", proto); + break; + case PPP_lqm: + x = NetL(o->data+2)*10; + period = (x+(PPP_period-1))/PPP_period; + p = seprint(p, e, " (qproto=lqm period=%d)", period); + break; + } + case Omagic: + p = seprint(p, e, " magic=%ux", NetL(o->data)); + break; + case Opc: + p = seprint(p, e, " protocol-compress"); + break; + case Oac: + p = seprint(p, e, " addr-compress"); + break; + } + } + return p; +} + + +static int +p_seprintlcp(Msg *m) +{ + Lcppkt *lcp; + char *p, *e; + int len; + + if(m->pe-m->ps < 4) + return -1; + + p = m->p; + e = m->e; + m->pr = nil; + + lcp = (Lcppkt*)m->ps; + len = NetS(lcp->len); + if(m->ps+len < m->pe) + m->pe = m->ps+len; + else if(m->ps+len > m->pe) + return -1; + + p = seprint(p, e, "id=%d code=%d", lcp->id, lcp->code); + switch(lcp->code) { + default: + p = seprint(p, e, " data=%.*H", len>64?64:len, lcp->data); + break; + case Lconfreq: + case Lconfack: + case Lconfnak: + case Lconfrej: + p = seprint(p, e, "=%s", lcpcode[lcp->code]); + p = seprintlcpopt(p, e, lcp->data, len-4); + break; + case Ltermreq: + case Ltermack: + case Lcoderej: + case Lprotorej: + case Lechoreq: + case Lechoack: + case Ldiscard: + p = seprint(p, e, "=%s", lcpcode[lcp->code]); + break; + } + m->p = seprint(p, e, " len=%d", len); + return 0; +} + +static char* +seprintipcpopt(char *p, char *e, void *a, int len) +{ + Lcpopt *o; + uchar *cp, *ecp; + + cp = a; + ecp = cp+len; + + for(; cp < ecp; cp += o->len){ + o = (Lcpopt*)cp; + if(cp + o->len > ecp){ + p = seprint(p, e, " bad opt len %ux", o->type); + return p; + } + + switch(o->type){ + default: + p = seprint(p, e, " (type=%d len=%d)", o->type, o->len); + break; + case Oipaddrs: + p = seprint(p, e, " ipaddrs(deprecated)"); + break; + case Oipcompress: + p = seprint(p, e, " ipcompress"); + break; + case Oipaddr: + p = seprint(p, e, " ipaddr=%V", o->data); + break; + case Oipdns: + p = seprint(p, e, " dnsaddr=%V", o->data); + break; + case Oipwins: + p = seprint(p, e, " winsaddr=%V", o->data); + break; + case Oipdns2: + p = seprint(p, e, " dns2addr=%V", o->data); + break; + case Oipwins2: + p = seprint(p, e, " wins2addr=%V", o->data); + break; + } + } + return p; +} + +static int +p_seprintipcp(Msg *m) +{ + Lcppkt *lcp; + char *p, *e; + int len; + + if(m->pe-m->ps < 4) + return -1; + + p = m->p; + e = m->e; + m->pr = nil; + + lcp = (Lcppkt*)m->ps; + len = NetS(lcp->len); + if(m->ps+len < m->pe) + m->pe = m->ps+len; + else if(m->ps+len > m->pe) + return -1; + + p = seprint(p, e, "id=%d code=%d", lcp->id, lcp->code); + switch(lcp->code) { + default: + p = seprint(p, e, " data=%.*H", len>64?64:len, lcp->data); + break; + case Lconfreq: + case Lconfack: + case Lconfnak: + case Lconfrej: + p = seprint(p, e, "=%s", lcpcode[lcp->code]); + p = seprintipcpopt(p, e, lcp->data, len-4); + break; + case Ltermreq: + case Ltermack: + p = seprint(p, e, "=%s", lcpcode[lcp->code]); + break; + } + m->p = seprint(p, e, " len=%d", len); + return 0; +} + +static char* +seprintccpopt(char *p, char *e, void *a, int len) +{ + Lcpopt *o; + uchar *cp, *ecp; + + cp = a; + ecp = cp+len; + + for(; cp < ecp; cp += o->len){ + o = (Lcpopt*)cp; + if(cp + o->len > ecp){ + p = seprint(p, e, " bad opt len %ux", o->type); + return p; + } + + switch(o->type){ + default: + p = seprint(p, e, " type=%d ", o->type); + break; + case 0: + p = seprint(p, e, " OUI=(%d %.2ux%.2ux%.2ux) ", o->type, + o->data[0], o->data[1], o->data[2]); + break; + case 17: + p = seprint(p, e, " Stac-LZS"); + break; + case 18: + p = seprint(p, e, " Microsoft-PPC=%ux", NetL(o->data)); + break; + } + } + return p; +} + +static int +p_seprintccp(Msg *m) +{ + Lcppkt *lcp; + char *p, *e; + int len; + + if(m->pe-m->ps < 4) + return -1; + + p = m->p; + e = m->e; + m->pr = nil; + + lcp = (Lcppkt*)m->ps; + len = NetS(lcp->len); + if(m->ps+len < m->pe) + m->pe = m->ps+len; + else if(m->ps+len > m->pe) + return -1; + + p = seprint(p, e, "id=%d code=%d", lcp->id, lcp->code); + switch(lcp->code) { + default: + p = seprint(p, e, " data=%.*H", len>64?64:len, lcp->data); + break; + case Lconfreq: + case Lconfack: + case Lconfnak: + case Lconfrej: + p = seprint(p, e, "=%s", lcpcode[lcp->code]); + p = seprintccpopt(p, e, lcp->data, len-4); + break; + case Ltermreq: + case Ltermack: + case Lresetreq: + case Lresetack: + p = seprint(p, e, "=%s", lcpcode[lcp->code]); + break; + } + m->p = seprint(p, e, " len=%d", len); + + return 0; +} + +static int +p_seprintcomp(Msg *m) +{ + char compflag[5]; + ushort x; + int i; + int len; + + len = m->pe-m->ps; + if(len < 2) + return -1; + + x = NetS(m->ps); + m->ps += 2; + i = 0; + if(x & (1<<15)) + compflag[i++] = 'r'; + if(x & (1<<14)) + compflag[i++] = 'f'; + if(x & (1<<13)) + compflag[i++] = 'c'; + if(x & (1<<12)) + compflag[i++] = 'e'; + compflag[i] = 0; + m->p = seprint(m->p, m->e, "flag=%s count=%.3ux", compflag, x&0xfff); + m->p = seprint(m->p, m->e, " data=%.*H", len>64?64:len, m->ps); + m->pr = nil; + return 0; +} + +Proto ppp = +{ + "ppp", + p_compile, + p_filter, + p_seprint, + p_mux, + nil, + defaultframer, +}; + +Proto ppp_ipcp = +{ + "ppp_ipcp", + p_compile, + p_filter, + p_seprintipcp, + nil, + nil, + defaultframer, +}; + +Proto ppp_lcp = +{ + "ppp_lcp", + p_compile, + p_filter, + p_seprintlcp, + nil, + nil, + defaultframer, +}; + +Proto ppp_ccp = +{ + "ppp_ccp", + p_compile, + p_filter, + p_seprintccp, + nil, + nil, + defaultframer, +}; + +Proto ppp_chap = +{ + "ppp_chap", + p_compile, + p_filter, + p_seprintchap, + nil, + nil, + defaultframer, +}; + +Proto ppp_comp = +{ + "ppp_comp", + p_compile, + p_filter, + p_seprintcomp, + nil, + nil, + defaultframer, +}; diff --git a/src/cmd/ip/snoopy/ppp_ccp.c b/src/cmd/ip/snoopy/ppp_ccp.c new file mode 100755 index 00000000..dab65f8f --- /dev/null +++ b/src/cmd/ip/snoopy/ppp_ccp.c @@ -0,0 +1 @@ +/* place holder, this stuff is really in ppp.c */ diff --git a/src/cmd/ip/snoopy/ppp_chap.c b/src/cmd/ip/snoopy/ppp_chap.c new file mode 100755 index 00000000..dab65f8f --- /dev/null +++ b/src/cmd/ip/snoopy/ppp_chap.c @@ -0,0 +1 @@ +/* place holder, this stuff is really in ppp.c */ diff --git a/src/cmd/ip/snoopy/ppp_comp.c b/src/cmd/ip/snoopy/ppp_comp.c new file mode 100755 index 00000000..dab65f8f --- /dev/null +++ b/src/cmd/ip/snoopy/ppp_comp.c @@ -0,0 +1 @@ +/* place holder, this stuff is really in ppp.c */ diff --git a/src/cmd/ip/snoopy/ppp_ipcp.c b/src/cmd/ip/snoopy/ppp_ipcp.c new file mode 100755 index 00000000..dab65f8f --- /dev/null +++ b/src/cmd/ip/snoopy/ppp_ipcp.c @@ -0,0 +1 @@ +/* place holder, this stuff is really in ppp.c */ diff --git a/src/cmd/ip/snoopy/ppp_lcp.c b/src/cmd/ip/snoopy/ppp_lcp.c new file mode 100755 index 00000000..dab65f8f --- /dev/null +++ b/src/cmd/ip/snoopy/ppp_lcp.c @@ -0,0 +1 @@ +/* place holder, this stuff is really in ppp.c */ diff --git a/src/cmd/ip/snoopy/pppoe_disc.c b/src/cmd/ip/snoopy/pppoe_disc.c new file mode 100755 index 00000000..4f17ebd7 --- /dev/null +++ b/src/cmd/ip/snoopy/pppoe_disc.c @@ -0,0 +1,172 @@ +#include <u.h> +#include <libc.h> +#include <ip.h> +#include "dat.h" +#include "protos.h" + +typedef struct Hdr Hdr; +struct Hdr { + uchar verstype; + uchar code; + uchar sessid[2]; + uchar length[2]; /* of payload */ +}; +enum +{ + HDRSIZE = 1+1+2+2 +}; + +static Mux p_mux[] = +{ + {"ppp", 0, } , + {0} +}; + +enum +{ + Overs, + Otype, + Ocode, + Osess, +}; + +static Field p_fields[] = +{ + {"v", Fnum, Overs, "version", } , + {"t", Fnum, Otype, "type", } , + {"c", Fnum, Ocode, "code" } , + {"s", Fnum, Osess, "sessid" } , + {0} +}; + +static void +p_compilesess(Filter *f) +{ +// Mux *m; + + if(f->op == '='){ + compile_cmp(pppoe_sess.name, f, p_fields); + return; + } +/* + for(m = p_mux; m->name != nil; m++) + if(strcmp(f->s, m->name) == 0){ + f->pr = m->pr; + f->ulv = m->val; + f->subop = Ot; + return; + } +*/ + sysfatal("unknown pppoe field or protocol: %s", f->s); +} +static void +p_compiledisc(Filter *f) +{ +// Mux *m; + + if(f->op == '='){ + compile_cmp(pppoe_disc.name, f, p_fields); + return; + } +/* + for(m = p_mux; m->name != nil; m++) + if(strcmp(f->s, m->name) == 0){ + f->pr = m->pr; + f->ulv = m->val; + f->subop = Ot; + return; + } +*/ + sysfatal("unknown pppoe field or protocol: %s", f->s); +} + +static int +p_filter(Filter *f, Msg *m) +{ + Hdr *h; + + if(m->pe - m->ps < HDRSIZE) + return 0; + + h = (Hdr*)m->ps; + m->ps += HDRSIZE; + + switch(f->subop){ + case Overs: + return (h->verstype>>4) == f->ulv; + case Otype: + return (h->verstype&0xF) == f->ulv; + case Ocode: + return h->code == f->ulv; + case Osess: + return NetS(h->sessid) == f->ulv; + } + return 0; +} + +/* BUG: print all the discovery types */ +static int +p_seprintdisc(Msg *m) +{ + Hdr *h; + int len; + + len = m->pe - m->ps; + if(len < HDRSIZE) + return -1; + + h = (Hdr*)m->ps; + m->ps += HDRSIZE; + + m->pr = nil; + + m->p = seprint(m->p, m->e, "v=%d t=%d c=0x%x s=0x%ux, len=%d", + h->verstype>>4, h->verstype&0xF, h->code, NetS(h->sessid), NetS(h->length)); + + return 0; +} + +static int +p_seprintsess(Msg *m) +{ + Hdr *h; + int len; + + len = m->pe - m->ps; + if(len < HDRSIZE) + return -1; + + h = (Hdr*)m->ps; + m->ps += HDRSIZE; + + /* this will call ppp for me */ + demux(p_mux, 0, 0, m, &dump); + + m->p = seprint(m->p, m->e, "v=%d t=%d c=0x%x s=0x%ux, len=%d", + h->verstype>>4, h->verstype&0xF, h->code, NetS(h->sessid), NetS(h->length)); + + return 0; +} + +Proto pppoe_disc = +{ + "pppoe_disc", + p_compiledisc, + p_filter, + p_seprintdisc, + p_mux, + p_fields, + defaultframer +}; + +Proto pppoe_sess = +{ + "pppoe_sess", + p_compilesess, + p_filter, + p_seprintsess, + p_mux, + p_fields, + defaultframer +}; + diff --git a/src/cmd/ip/snoopy/pppoe_sess.c b/src/cmd/ip/snoopy/pppoe_sess.c new file mode 100755 index 00000000..0472a907 --- /dev/null +++ b/src/cmd/ip/snoopy/pppoe_sess.c @@ -0,0 +1 @@ +/* placeholder; see pppoe_disc.c */ diff --git a/src/cmd/ip/snoopy/protos.c b/src/cmd/ip/snoopy/protos.c new file mode 100644 index 00000000..4cf78e27 --- /dev/null +++ b/src/cmd/ip/snoopy/protos.c @@ -0,0 +1,33 @@ +#include <u.h> +#include <libc.h> +#include "dat.h" +#include "protos.h" +Proto *protos[] = +{ + ðer, + &ip, + &ip6, + &dump, + &arp, + &rarp, + &udp, + &bootp, + &dhcp, + &hdlc, + &rtp, + &rtcp, + &tcp, + &il, + &icmp, + &icmp6, + &ninep, + &ospf, + &ppp, + &ppp_ccp, + &ppp_lcp, + &ppp_chap, + &ppp_ipcp, + &pppoe_sess, + &pppoe_disc, + 0, +}; diff --git a/src/cmd/ip/snoopy/protos.h b/src/cmd/ip/snoopy/protos.h new file mode 100644 index 00000000..501b59ec --- /dev/null +++ b/src/cmd/ip/snoopy/protos.h @@ -0,0 +1,25 @@ +extern Proto ether; +extern Proto ip; +extern Proto ip6; +extern Proto dump; +extern Proto arp; +extern Proto rarp; +extern Proto udp; +extern Proto bootp; +extern Proto dhcp; +extern Proto hdlc; +extern Proto rtp; +extern Proto rtcp; +extern Proto tcp; +extern Proto il; +extern Proto icmp; +extern Proto icmp6; +extern Proto ninep; +extern Proto ospf; +extern Proto ppp; +extern Proto ppp_ccp; +extern Proto ppp_lcp; +extern Proto ppp_chap; +extern Proto ppp_ipcp; +extern Proto pppoe_sess; +extern Proto pppoe_disc; diff --git a/src/cmd/ip/snoopy/rarp.c b/src/cmd/ip/snoopy/rarp.c new file mode 100755 index 00000000..149c8207 --- /dev/null +++ b/src/cmd/ip/snoopy/rarp.c @@ -0,0 +1 @@ +/* place holder, this stuff is really in arp.c */ diff --git a/src/cmd/ip/snoopy/rtcp.c b/src/cmd/ip/snoopy/rtcp.c new file mode 100755 index 00000000..1036caaf --- /dev/null +++ b/src/cmd/ip/snoopy/rtcp.c @@ -0,0 +1,97 @@ +#include <u.h> +#include <libc.h> +#include <ip.h> +#include "dat.h" +#include "protos.h" + +typedef struct Hdr Hdr; +struct Hdr { + uchar hdr; // RTCP header + uchar pt; // Packet type + uchar len[2]; // Report length + uchar ssrc[4]; // Synchronization source identifier + uchar ntp[8]; // NTP time stamp + uchar rtp[4]; // RTP time stamp + uchar pktc[4]; // Sender's packet count + uchar octc[4]; // Sender's octect count +}; + +typedef struct Report Report; +struct Report { + uchar ssrc[4]; // SSRC identifier + uchar lost[4]; // Fraction + cumu lost + uchar seqhi[4]; // Highest seq number received + uchar jitter[4]; // Interarrival jitter + uchar lsr[4]; // Last SR + uchar dlsr[4]; // Delay since last SR +}; + +enum{ + RTCPLEN = 28, // Minimum size of an RTCP header + REPORTLEN = 24, +}; + + +static void +p_compile(Filter *f) +{ + sysfatal("unknown rtcp field: %s", f->s); +} + +static int +p_filter(Filter *f, Msg *m) +{ + USED(f); + USED(m); + return 0; +} + +static int +p_seprint(Msg *m) +{ + Hdr*h; + Report*r; + int rc, i, frac; + float dlsr; + + if(m->pe - m->ps < RTCPLEN) + return -1; + + h = (Hdr*)m->ps; + if(m->pe - m->ps < (NetS(h->len) + 1) * 4) + return -1; + + rc = h->hdr & 0x1f; + m->ps += RTCPLEN; + m->p = seprint(m->p, m->e, "version=%d rc=%d tp=%d ssrc=%8ux ntp=%d.%.10ud rtp=%d pktc=%d octc=%d hlen=%d", + (h->hdr >> 6) & 3, rc, h->pt, NetL(h->ssrc), + NetL(h->ntp), (uint)NetL(&h->ntp[4]), NetL(h->rtp), + NetL(h->pktc), NetL(h->octc), + (NetS(h->len) + 1) * 4); + + for(i = 0; i < rc; i++){ + r = (Report*)m->ps; + m->ps += REPORTLEN; + + frac = (int)(((float)r->lost[0] * 100.) / 256.); + r->lost[0] = 0; + dlsr = (float)NetL(r->dlsr) / 65536.; + + m->p = seprint(m->p, m->e, "\n\trr(csrc=%8ux frac=%3d%% cumu=%10d seqhi=%10ud jitter=%10d lsr=%8ux dlsr=%f)", + NetL(r->ssrc), frac, NetL(r->lost), NetL(r->seqhi), + NetL(r->jitter), NetL(r->lsr), + dlsr); + } + m->pr = nil; + return 0; +} + +Proto rtcp = { + "rtcp", + p_compile, + p_filter, + p_seprint, + nil, + nil, + defaultframer, +}; diff --git a/src/cmd/ip/snoopy/rtp.c b/src/cmd/ip/snoopy/rtp.c new file mode 100755 index 00000000..1e8e5ab8 --- /dev/null +++ b/src/cmd/ip/snoopy/rtp.c @@ -0,0 +1,76 @@ +#include <u.h> +#include <libc.h> +#include <ip.h> +#include "dat.h" +#include "protos.h" + +typedef struct Hdr Hdr; +struct Hdr { + uchar hdr; // RTP header + uchar marker; // Payload and marker + uchar seq[2]; // Sequence number + uchar ts[4]; // Time stamp + uchar ssrc[4]; // Synchronization source identifier +}; + +enum{ + RTPLEN = 12, // Minimum size of an RTP header +}; + + +static void +p_compile(Filter *f) +{ + sysfatal("unknown rtp field: %s", f->s); +} + +static int +p_filter(Filter *f, Msg *m) +{ + USED(f); + USED(m); + return 0; +} + +static int +p_seprint(Msg *m) +{ + Hdr*h; + ushort seq; + ulong ssrc, ts; + int cc, i; + + if(m->pe - m->ps < RTPLEN) + return -1; + + h = (Hdr*)m->ps; + cc = h->hdr & 0xf; + if(m->pe - m->ps < RTPLEN + cc * 4) + return -1; + + m->ps += RTPLEN; + + seq = NetS(h->seq); + ts = NetL(h->ts); + ssrc = NetL(h->ssrc); + + m->p = seprint(m->p, m->e, "version=%d x=%d cc=%d seq=%d ts=%ld ssrc=%ulx", + (h->hdr >> 6) & 3, (h->hdr >> 4) & 1, cc, seq, ts, ssrc); + for(i = 0; i < cc; i++){ + m->p = seprint(m->p, m->e, " csrc[%d]=%d", + i, NetL(m->ps)); + m->ps += 4; + } + m->pr = nil; + return 0; +} + +Proto rtp = { + "rtp", + p_compile, + p_filter, + p_seprint, + nil, + nil, + defaultframer, +}; diff --git a/src/cmd/ip/snoopy/tcp.c b/src/cmd/ip/snoopy/tcp.c new file mode 100755 index 00000000..31ad5663 --- /dev/null +++ b/src/cmd/ip/snoopy/tcp.c @@ -0,0 +1,221 @@ +#include <u.h> +#include <libc.h> +#include <ip.h> +#include "dat.h" +#include "protos.h" + +typedef struct Hdr Hdr; +struct Hdr +{ + uchar sport[2]; + uchar dport[2]; + uchar seq[4]; + uchar ack[4]; + uchar flag[2]; + uchar win[2]; + uchar cksum[2]; + uchar urg[2]; + uchar opt[1]; +}; + +typedef struct PseudoHdr{ + uchar src[4]; + uchar dst[4]; + uchar zero; + uchar proto; + uchar length[2]; + uchar hdrdata[1580]; +} PseudoHdr; + +enum +{ + TCPLEN= 20, +}; + +enum +{ + Os, + Od, + Osd, +}; + +static Field p_fields[] = +{ + {"s", Fnum, Os, "source port", } , + {"d", Fnum, Od, "dest port", } , + {"a", Fnum, Osd, "source/dest port", } , + {"sd", Fnum, Osd, "source/dest port", } , + {0} +}; + +static Mux p_mux[] = +{ + {"ninep", 17007, }, /* exportfs */ + {"ninep", 564, }, /* 9fs */ + {"ninep", 17005, }, /* ocpu */ + {"ninep", 17010, }, /* ncpu */ + {"ninep", 17013, }, /* cpu */ + {0}, +}; + +enum +{ + EOLOPT = 0, + NOOPOPT = 1, + MSSOPT = 2, + MSS_LENGTH = 4, /* Mean segment size */ + WSOPT = 3, + WS_LENGTH = 3, /* Bits to scale window size by */ +}; + +static void +p_compile(Filter *f) +{ + Mux *m; + + if(f->op == '='){ + compile_cmp(udp.name, f, p_fields); + return; + } + for(m = p_mux; m->name != nil; m++) + if(strcmp(f->s, m->name) == 0){ + f->pr = m->pr; + f->ulv = m->val; + f->subop = Osd; + return; + } + sysfatal("unknown tcp field or protocol: %s", f->s); +} + +static int +p_filter(Filter *f, Msg *m) +{ + Hdr *h; + + if(m->pe - m->ps < TCPLEN) + return 0; + + h = (Hdr*)m->ps; + m->ps += ((NetS(h->flag)>>10)&0x3f); + + switch(f->subop){ + case Os: + return NetS(h->sport) == f->ulv; + case Od: + return NetS(h->dport) == f->ulv; + case Osd: + return NetS(h->sport) == f->ulv || NetS(h->dport) == f->ulv; + } + return 0; +} + +enum +{ + URG = 0x20, /* Data marked urgent */ + ACK = 0x10, /* Aknowledge is valid */ + PSH = 0x08, /* Whole data pipe is pushed */ + RST = 0x04, /* Reset connection */ + SYN = 0x02, /* Pkt. is synchronise */ + FIN = 0x01, /* Start close down */ +}; + +static char* +flags(int f) +{ + static char fl[20]; + char *p; + + p = fl; + if(f & URG) + *p++ = 'U'; + if(f & ACK) + *p++ = 'A'; + if(f & PSH) + *p++ = 'P'; + if(f & RST) + *p++ = 'R'; + if(f & SYN) + *p++ = 'S'; + if(f & FIN) + *p++ = 'F'; + *p = 0; + return fl; +} + + +static int +p_seprint(Msg *m) +{ + Hdr *h; + int dport, sport; + int len, flag, optlen; + uchar *optr; + + if(m->pe - m->ps < TCPLEN) + return -1; + h = (Hdr*)m->ps; + + /* get tcp header length */ + flag = NetS(h->flag); + len = (flag>>10)&~3; + flag &= 0x3ff; + m->ps += len; + + /* next protocol */ + dport = NetS(h->dport); + sport = NetS(h->sport); + demux(p_mux, sport, dport, m, &dump); + + m->p = seprint(m->p, m->e, "s=%d d=%d seq=%lud ack=%lud fl=%s win=%d ck=%4.4ux", + NetS(h->sport), dport, + (ulong)NetL(h->seq), (ulong)NetL(h->ack), + flags(flag), NetS(h->win), + NetS(h->cksum)); + + /* tcp options */ + len -= TCPLEN; + optr = h->opt; + while(len > 0) { + if(*optr == EOLOPT){ + m->p = seprint(m->p, m->e, " opt=EOL"); + break; + } + if(*optr == NOOPOPT) { + m->p = seprint(m->p, m->e, " opt=NOOP"); + len--; + optr++; + continue; + } + optlen = optr[1]; + if(optlen < 2 || optlen > len) + break; + switch(*optr) { + case MSSOPT: + m->p = seprint(m->p, m->e, " opt%d=(mss %ud)", optlen, nhgets(optr+2)); + break; + case WSOPT: + m->p = seprint(m->p, m->e, " opt%d=(wscale %ud)", optlen, *(optr+2)); + break; + default: + m->p = seprint(m->p, m->e, " opt%d=(%ud %.*H)", optlen, *optr, optlen-2,optr+2); + } + len -= optlen; + optr += optlen; + } + + if(Cflag){ + // editing in progress by ehg + } + return 0; +} + +Proto tcp = +{ + "tcp", + p_compile, + p_filter, + p_seprint, + p_mux, + p_fields, + defaultframer, +}; diff --git a/src/cmd/ip/snoopy/udp.c b/src/cmd/ip/snoopy/udp.c new file mode 100755 index 00000000..43d2b6f6 --- /dev/null +++ b/src/cmd/ip/snoopy/udp.c @@ -0,0 +1,131 @@ +#include <u.h> +#include <libc.h> +#include <ip.h> +#include "dat.h" +#include "protos.h" + +typedef struct Hdr Hdr; +struct Hdr +{ + uchar sport[2]; /* Source port */ + uchar dport[2]; /* Destination port */ + uchar len[2]; /* data length */ + uchar cksum[2]; /* Checksum */ +}; + +enum +{ + UDPLEN= 8, +}; + +enum +{ + Os, + Od, + Osd, + Osetport, +}; + +static Field p_fields[] = +{ + {"s", Fnum, Os, "source port", } , + {"d", Fnum, Od, "dest port", } , + {"a", Fnum, Osd, "source/dest port", } , + {"sd", Fnum, Osd, "source/dest port", } , + {0} +}; + +#define ANYPORT ~0UL + +static Mux p_mux[] = +{ + {"bootp", 67, }, + {"ninep", 6346, }, /* tvs */ + {"rtp", ANYPORT, }, + {"rtcp", ANYPORT, }, + {0}, +}; + +/* default next protocol, can be changed by p_filter, reset by p_compile */ +static Proto *defproto = &dump; + +static void +p_compile(Filter *f) +{ + Mux *m; + + if(f->op == '='){ + compile_cmp(udp.name, f, p_fields); + return; + } + for(m = p_mux; m->name != nil; m++) + if(strcmp(f->s, m->name) == 0){ + f->pr = m->pr; + f->ulv = m->val; + f->subop = Osd; + return; + } + + sysfatal("unknown udp field or protocol: %s", f->s); +} + +static int +p_filter(Filter *f, Msg *m) +{ + Hdr *h; + + if(m->pe - m->ps < UDPLEN) + return 0; + + h = (Hdr*)m->ps; + m->ps += UDPLEN; + + switch(f->subop){ + case Os: + return NetS(h->sport) == f->ulv; + case Od: + return NetS(h->dport) == f->ulv; + case Osd: + if(f->ulv == ANYPORT){ + defproto = f->pr; + return 1; + } + return NetS(h->sport) == f->ulv || NetS(h->dport) == f->ulv; + } + return 0; +} + +static int +p_seprint(Msg *m) +{ + Hdr *h; + int dport, sport; + + + if(m->pe - m->ps < UDPLEN) + return -1; + h = (Hdr*)m->ps; + m->ps += UDPLEN; + + /* next protocol */ + sport = NetS(h->sport); + dport = NetS(h->dport); + demux(p_mux, sport, dport, m, defproto); + defproto = &dump; + + m->p = seprint(m->p, m->e, "s=%d d=%d ck=%4.4ux ln=%4d", + NetS(h->sport), dport, + NetS(h->cksum), NetS(h->len)); + return 0; +} + +Proto udp = +{ + "udp", + p_compile, + p_filter, + p_seprint, + p_mux, + p_fields, + defaultframer, +}; |