wifi: add packet timestamping support
This commit is contained in:
parent
cf5316a402
commit
a87b6909bc
2 changed files with 46 additions and 4 deletions
|
@ -85,6 +85,19 @@ wifihdrlen(Wifipkt *w)
|
||||||
return n;
|
return n;
|
||||||
}
|
}
|
||||||
|
|
||||||
|
static uvlong
|
||||||
|
getts(uchar *d)
|
||||||
|
{
|
||||||
|
return (uvlong)d[0] |
|
||||||
|
(uvlong)d[1]<<8 |
|
||||||
|
(uvlong)d[2]<<16 |
|
||||||
|
(uvlong)d[3]<<24 |
|
||||||
|
(uvlong)d[4]<<32 |
|
||||||
|
(uvlong)d[5]<<40 |
|
||||||
|
(uvlong)d[6]<<48 |
|
||||||
|
(uvlong)d[7]<<56;
|
||||||
|
}
|
||||||
|
|
||||||
void
|
void
|
||||||
wifiiq(Wifi *wifi, Block *b)
|
wifiiq(Wifi *wifi, Block *b)
|
||||||
{
|
{
|
||||||
|
@ -93,6 +106,8 @@ wifiiq(Wifi *wifi, Block *b)
|
||||||
Etherpkt *e;
|
Etherpkt *e;
|
||||||
int hdrlen;
|
int hdrlen;
|
||||||
|
|
||||||
|
if(b->flag & Btimestamp)
|
||||||
|
assert(b->rp - b->base >= 8);
|
||||||
if(BLEN(b) < WIFIHDRSIZE)
|
if(BLEN(b) < WIFIHDRSIZE)
|
||||||
goto drop;
|
goto drop;
|
||||||
w = (Wifipkt*)b->rp;
|
w = (Wifipkt*)b->rp;
|
||||||
|
@ -115,6 +130,7 @@ wifiiq(Wifi *wifi, Block *b)
|
||||||
case 0x04: /* control */
|
case 0x04: /* control */
|
||||||
break;
|
break;
|
||||||
case 0x08: /* data */
|
case 0x08: /* data */
|
||||||
|
b->flag &= ~Btimestamp;
|
||||||
b->rp += hdrlen;
|
b->rp += hdrlen;
|
||||||
switch(w->fc[0] & 0xf0){
|
switch(w->fc[0] & 0xf0){
|
||||||
default:
|
default:
|
||||||
|
@ -136,9 +152,7 @@ wifiiq(Wifi *wifi, Block *b)
|
||||||
memmove(e->d, dstaddr(&h), Eaddrlen);
|
memmove(e->d, dstaddr(&h), Eaddrlen);
|
||||||
memmove(e->s, srcaddr(&h), Eaddrlen);
|
memmove(e->s, srcaddr(&h), Eaddrlen);
|
||||||
memmove(e->type, s.type, 2);
|
memmove(e->type, s.type, 2);
|
||||||
|
|
||||||
dmatproxy(b, 0, wifi->ether->ea, &wifi->dmat);
|
dmatproxy(b, 0, wifi->ether->ea, &wifi->dmat);
|
||||||
|
|
||||||
etheriq(wifi->ether, b);
|
etheriq(wifi->ether, b);
|
||||||
return;
|
return;
|
||||||
}
|
}
|
||||||
|
@ -310,6 +324,7 @@ wifiprobe(Wifi *wifi, Wnode *wn)
|
||||||
|
|
||||||
b = allocb(WIFIHDRSIZE + 512);
|
b = allocb(WIFIHDRSIZE + 512);
|
||||||
w = (Wifipkt*)b->wp;
|
w = (Wifipkt*)b->wp;
|
||||||
|
|
||||||
w->fc[0] = 0x40; /* probe request */
|
w->fc[0] = 0x40; /* probe request */
|
||||||
w->fc[1] = 0x00; /* STA->STA */
|
w->fc[1] = 0x00; /* STA->STA */
|
||||||
memmove(w->a1, wifi->ether->bcast, Eaddrlen); /* ??? */
|
memmove(w->a1, wifi->ether->bcast, Eaddrlen); /* ??? */
|
||||||
|
@ -377,6 +392,7 @@ sendassoc(Wifi *wifi, Wnode *bss)
|
||||||
memmove(w->a1, bss->bssid, Eaddrlen); /* ??? */
|
memmove(w->a1, bss->bssid, Eaddrlen); /* ??? */
|
||||||
memmove(w->a2, wifi->ether->ea, Eaddrlen);
|
memmove(w->a2, wifi->ether->ea, Eaddrlen);
|
||||||
memmove(w->a3, bss->bssid, Eaddrlen);
|
memmove(w->a3, bss->bssid, Eaddrlen);
|
||||||
|
|
||||||
b->wp += WIFIHDRSIZE;
|
b->wp += WIFIHDRSIZE;
|
||||||
p = b->wp;
|
p = b->wp;
|
||||||
|
|
||||||
|
@ -461,12 +477,17 @@ recvbeacon(Wifi *wifi, Wnode *wn, uchar *d, int len)
|
||||||
if(len < 0)
|
if(len < 0)
|
||||||
return;
|
return;
|
||||||
|
|
||||||
d += 8; /* timestamp */
|
/* timestamp */
|
||||||
|
wn->ts = getts(d);
|
||||||
|
d += 8;
|
||||||
wn->ival = d[0] | d[1]<<8;
|
wn->ival = d[0] | d[1]<<8;
|
||||||
d += 2;
|
d += 2;
|
||||||
wn->cap = d[0] | d[1]<<8;
|
wn->cap = d[0] | d[1]<<8;
|
||||||
d += 2;
|
d += 2;
|
||||||
|
|
||||||
|
wn->dtimcount = 0;
|
||||||
|
wn->dtimperiod = 1;
|
||||||
|
|
||||||
rsnset = 0;
|
rsnset = 0;
|
||||||
for(e = d + len; d+2 <= e; d = x){
|
for(e = d + len; d+2 <= e; d = x){
|
||||||
d += 2;
|
d += 2;
|
||||||
|
@ -513,6 +534,13 @@ recvbeacon(Wifi *wifi, Wnode *wn, uchar *d, int len)
|
||||||
if(d != x)
|
if(d != x)
|
||||||
wn->channel = d[0];
|
wn->channel = d[0];
|
||||||
break;
|
break;
|
||||||
|
case 5:
|
||||||
|
if(x - d < 2)
|
||||||
|
break;
|
||||||
|
wn->dtimcount = d[0];
|
||||||
|
if(d[1] > 0)
|
||||||
|
wn->dtimperiod = d[1];
|
||||||
|
break;
|
||||||
case 221: /* vendor specific */
|
case 221: /* vendor specific */
|
||||||
len = x - d;
|
len = x - d;
|
||||||
if(rsnset || len < sizeof(wpa1oui) || memcmp(d, wpa1oui, sizeof(wpa1oui)) != 0)
|
if(rsnset || len < sizeof(wpa1oui) || memcmp(d, wpa1oui, sizeof(wpa1oui)) != 0)
|
||||||
|
@ -619,6 +647,7 @@ wifiproc(void *arg)
|
||||||
w = (Wifipkt*)b->rp;
|
w = (Wifipkt*)b->rp;
|
||||||
if(w->fc[1] & 0x40)
|
if(w->fc[1] & 0x40)
|
||||||
continue;
|
continue;
|
||||||
|
b->flag &= ~Btimestamp;
|
||||||
wifiiq(wifi, b);
|
wifiiq(wifi, b);
|
||||||
b = nil;
|
b = nil;
|
||||||
}
|
}
|
||||||
|
@ -637,6 +666,8 @@ wifiproc(void *arg)
|
||||||
if((wn = nodelookup(wifi, w->a3, 1)) == nil)
|
if((wn = nodelookup(wifi, w->a3, 1)) == nil)
|
||||||
continue;
|
continue;
|
||||||
wn->lastseen = MACHP(0)->ticks;
|
wn->lastseen = MACHP(0)->ticks;
|
||||||
|
if(b->flag & Btimestamp)
|
||||||
|
wn->rs = getts(b->rp - 8);
|
||||||
b->rp += wifihdrlen(w);
|
b->rp += wifihdrlen(w);
|
||||||
recvbeacon(wifi, wn, b->rp, BLEN(b));
|
recvbeacon(wifi, wn, b->rp, BLEN(b));
|
||||||
|
|
||||||
|
@ -655,6 +686,8 @@ wifiproc(void *arg)
|
||||||
if((wn = nodelookup(wifi, w->a3, 0)) == nil)
|
if((wn = nodelookup(wifi, w->a3, 0)) == nil)
|
||||||
continue;
|
continue;
|
||||||
wn->lastseen = MACHP(0)->ticks;
|
wn->lastseen = MACHP(0)->ticks;
|
||||||
|
if(b->flag & Btimestamp)
|
||||||
|
wn->rs = getts(b->rp - 8);
|
||||||
switch(w->fc[0] & 0xf0){
|
switch(w->fc[0] & 0xf0){
|
||||||
case 0x10: /* assoc response */
|
case 0x10: /* assoc response */
|
||||||
case 0x30: /* reassoc response */
|
case 0x30: /* reassoc response */
|
||||||
|
@ -804,8 +837,9 @@ Scan:
|
||||||
wifideauth(wifi, wn); /* stuck in auth, start over */
|
wifideauth(wifi, wn); /* stuck in auth, start over */
|
||||||
if(wn->status == Sconn || wn->status == Sunauth)
|
if(wn->status == Sconn || wn->status == Sunauth)
|
||||||
sendauth(wifi, wn);
|
sendauth(wifi, wn);
|
||||||
if(wn->status == Sauth)
|
if(wn->status == Sauth){
|
||||||
sendassoc(wifi, wn);
|
sendassoc(wifi, wn);
|
||||||
|
}
|
||||||
}
|
}
|
||||||
tsleep(&up->sleep, return0, 0, 500);
|
tsleep(&up->sleep, return0, 0, 500);
|
||||||
}
|
}
|
||||||
|
|
|
@ -50,6 +50,10 @@ struct Wnode
|
||||||
ulong txerror;
|
ulong txerror;
|
||||||
|
|
||||||
/* stuff from beacon */
|
/* stuff from beacon */
|
||||||
|
uvlong rs;
|
||||||
|
uvlong ts;
|
||||||
|
uchar dtimcount;
|
||||||
|
uchar dtimperiod;
|
||||||
int ival;
|
int ival;
|
||||||
int cap;
|
int cap;
|
||||||
int channel;
|
int channel;
|
||||||
|
@ -109,6 +113,10 @@ struct Wifipkt
|
||||||
uchar a4[Eaddrlen];
|
uchar a4[Eaddrlen];
|
||||||
};
|
};
|
||||||
|
|
||||||
|
enum {
|
||||||
|
Btimestamp = 1<<15,
|
||||||
|
};
|
||||||
|
|
||||||
Wifi *wifiattach(Ether *ether, void (*transmit)(Wifi*, Wnode*, Block*));
|
Wifi *wifiattach(Ether *ether, void (*transmit)(Wifi*, Wnode*, Block*));
|
||||||
void wifiiq(Wifi*, Block*);
|
void wifiiq(Wifi*, Block*);
|
||||||
int wifihdrlen(Wifipkt*);
|
int wifihdrlen(Wifipkt*);
|
||||||
|
|
Loading…
Reference in a new issue