snoopy: add vlan protocol (802.1q)
This commit is contained in:
parent
9944e16b16
commit
fd1cfc824a
3 changed files with 110 additions and 4 deletions
|
@ -16,7 +16,7 @@ struct Hdr {
|
||||||
#define ETHERMAXTU 1514 /* maximum transmit size */
|
#define ETHERMAXTU 1514 /* maximum transmit size */
|
||||||
#define ETHERHDRSIZE 14 /* size of an ethernet header */
|
#define ETHERHDRSIZE 14 /* size of an ethernet header */
|
||||||
|
|
||||||
static Mux p_mux[] =
|
Mux ethertypes[] =
|
||||||
{
|
{
|
||||||
{"ip", 0x0800, } ,
|
{"ip", 0x0800, } ,
|
||||||
{"arp", 0x0806, } ,
|
{"arp", 0x0806, } ,
|
||||||
|
@ -27,6 +27,7 @@ static Mux p_mux[] =
|
||||||
{"eapol", 0x888e, },
|
{"eapol", 0x888e, },
|
||||||
{"aoe", 0x88a2, } ,
|
{"aoe", 0x88a2, } ,
|
||||||
{"cec", 0xbcbc, } ,
|
{"cec", 0xbcbc, } ,
|
||||||
|
{"vlan", 0x8100, } ,
|
||||||
{0}
|
{0}
|
||||||
};
|
};
|
||||||
|
|
||||||
|
@ -57,7 +58,7 @@ p_compile(Filter *f)
|
||||||
compile_cmp(ether.name, f, p_fields);
|
compile_cmp(ether.name, f, p_fields);
|
||||||
return;
|
return;
|
||||||
}
|
}
|
||||||
for(m = p_mux; m->name != nil; m++)
|
for(m = ethertypes; m->name != nil; m++)
|
||||||
if(strcmp(f->s, m->name) == 0){
|
if(strcmp(f->s, m->name) == 0){
|
||||||
f->pr = m->pr;
|
f->pr = m->pr;
|
||||||
f->ulv = m->val;
|
f->ulv = m->val;
|
||||||
|
@ -106,7 +107,7 @@ p_seprint(Msg *m)
|
||||||
m->ps += ETHERHDRSIZE;
|
m->ps += ETHERHDRSIZE;
|
||||||
|
|
||||||
t = NetS(h->type);
|
t = NetS(h->type);
|
||||||
demux(p_mux, t, t, m, &dump);
|
demux(ethertypes, t, t, m, &dump);
|
||||||
|
|
||||||
m->p = seprint(m->p, m->e, "s=%E d=%E pr=%4.4ux ln=%d", h->s, h->d,
|
m->p = seprint(m->p, m->e, "s=%E d=%E pr=%4.4ux ln=%d", h->s, h->d,
|
||||||
t, len);
|
t, len);
|
||||||
|
@ -119,7 +120,7 @@ Proto ether =
|
||||||
p_compile,
|
p_compile,
|
||||||
p_filter,
|
p_filter,
|
||||||
p_seprint,
|
p_seprint,
|
||||||
p_mux,
|
ethertypes,
|
||||||
"%#.4lux",
|
"%#.4lux",
|
||||||
p_fields,
|
p_fields,
|
||||||
defaultframer
|
defaultframer
|
||||||
|
|
|
@ -43,6 +43,7 @@ PROTOS=\
|
||||||
tcp\
|
tcp\
|
||||||
ttls\
|
ttls\
|
||||||
udp\
|
udp\
|
||||||
|
vlan\
|
||||||
|
|
||||||
POBJS=${PROTOS:%=%.$O}
|
POBJS=${PROTOS:%=%.$O}
|
||||||
|
|
||||||
|
|
104
sys/src/cmd/ip/snoopy/vlan.c
Normal file
104
sys/src/cmd/ip/snoopy/vlan.c
Normal file
|
@ -0,0 +1,104 @@
|
||||||
|
#include <u.h>
|
||||||
|
#include <libc.h>
|
||||||
|
#include <ip.h>
|
||||||
|
#include "dat.h"
|
||||||
|
#include "protos.h"
|
||||||
|
|
||||||
|
extern Mux ethertypes[];
|
||||||
|
|
||||||
|
typedef struct Hdr Hdr;
|
||||||
|
struct Hdr {
|
||||||
|
uchar tag[2];
|
||||||
|
uchar type[2];
|
||||||
|
};
|
||||||
|
|
||||||
|
enum
|
||||||
|
{
|
||||||
|
Ov, /* vlan */
|
||||||
|
Oq, /* qprio */
|
||||||
|
Ot, /* type */
|
||||||
|
};
|
||||||
|
|
||||||
|
static Field p_fields[] =
|
||||||
|
{
|
||||||
|
{"v", Fnum, Ov, "vlan id" } ,
|
||||||
|
{"q", Fnum, Oq, "queue prio" } ,
|
||||||
|
{"t", Fnum, Ot, "type" } ,
|
||||||
|
{0}
|
||||||
|
};
|
||||||
|
|
||||||
|
static void
|
||||||
|
p_compile(Filter *f)
|
||||||
|
{
|
||||||
|
Mux *m;
|
||||||
|
|
||||||
|
if(f->op == '='){
|
||||||
|
compile_cmp(vlan.name, f, p_fields);
|
||||||
|
return;
|
||||||
|
}
|
||||||
|
for(m = ethertypes; 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 vlan field or protocol: %s", f->s);
|
||||||
|
}
|
||||||
|
|
||||||
|
static int
|
||||||
|
p_filter(Filter *f, Msg *m)
|
||||||
|
{
|
||||||
|
Hdr *h;
|
||||||
|
|
||||||
|
if(m->pe - m->ps < 4)
|
||||||
|
return 0;
|
||||||
|
|
||||||
|
h = (Hdr*)m->ps;
|
||||||
|
m->ps += 4;
|
||||||
|
|
||||||
|
switch(f->subop){
|
||||||
|
case Ov:
|
||||||
|
return (NetS(h->tag) & 0xFFF) == f->ulv;
|
||||||
|
case Oq:
|
||||||
|
return (NetS(h->tag) >> 12) == f->ulv;
|
||||||
|
case Ot:
|
||||||
|
return NetS(h->type) == f->ulv;
|
||||||
|
}
|
||||||
|
return 0;
|
||||||
|
}
|
||||||
|
|
||||||
|
static int
|
||||||
|
p_seprint(Msg *m)
|
||||||
|
{
|
||||||
|
uint v, q, t;
|
||||||
|
int len;
|
||||||
|
Hdr *h;
|
||||||
|
|
||||||
|
len = m->pe - m->ps;
|
||||||
|
if(len < 4)
|
||||||
|
return -1;
|
||||||
|
|
||||||
|
h = (Hdr*)m->ps;
|
||||||
|
m->ps += 4;
|
||||||
|
|
||||||
|
q = NetS(h->tag) >> 12;
|
||||||
|
v = NetS(h->tag) & 0xFFF;
|
||||||
|
t = NetS(h->type);
|
||||||
|
demux(ethertypes, t, t, m, &dump);
|
||||||
|
|
||||||
|
m->p = seprint(m->p, m->e, "v=%ud q=%ux pr=%4.4ux ln=%d", v, q, t, len);
|
||||||
|
return 0;
|
||||||
|
}
|
||||||
|
|
||||||
|
Proto vlan =
|
||||||
|
{
|
||||||
|
"vlan",
|
||||||
|
p_compile,
|
||||||
|
p_filter,
|
||||||
|
p_seprint,
|
||||||
|
ethertypes,
|
||||||
|
"%#.4lux",
|
||||||
|
p_fields,
|
||||||
|
defaultframer
|
||||||
|
};
|
Loading…
Reference in a new issue