snoopy: add vlan protocol (802.1q)

This commit is contained in:
cinap_lenrek 2022-02-27 03:33:01 +00:00
parent 9944e16b16
commit fd1cfc824a
3 changed files with 110 additions and 4 deletions

View file

@ -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

View file

@ -43,6 +43,7 @@ PROTOS=\
tcp\ tcp\
ttls\ ttls\
udp\ udp\
vlan\
POBJS=${PROTOS:%=%.$O} POBJS=${PROTOS:%=%.$O}

View 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
};