devip: various bugfixes and cleanups for arp code

- fix missing runlock(ifc) when ifcid != a->ifcid in rxmitsols() (thanks erik quanstro)
- don't leak packets when transfering blocks from arp entry hold list to droplist
- free rest of droplist when bwrite() errors in arpenter(), remove useless checks (ifc != nil)
- free arp entry hold list from cleanarpent()
- consistent use of nil for pointers
This commit is contained in:
cinap_lenrek 2015-09-27 22:17:02 +02:00
parent 920783505c
commit 4449a34756

View file

@ -61,6 +61,18 @@ arpinit(Fs *f)
kproc("rxmitproc", rxmitproc, f->arp); kproc("rxmitproc", rxmitproc, f->arp);
} }
static void
freeblistchain(Block *bp)
{
Block *next;
while(bp != nil){
next = bp->list;
freeblist(bp);
bp = next;
}
}
/* /*
* create a new arp entry for an ip address. * create a new arp entry for an ip address.
*/ */
@ -68,7 +80,7 @@ static Arpent*
newarp6(Arp *arp, uchar *ip, Ipifc *ifc, int addrxt) newarp6(Arp *arp, uchar *ip, Ipifc *ifc, int addrxt)
{ {
uint t; uint t;
Block *next, *xp; Block *xp;
Arpent *a, *e, *f, **l; Arpent *a, *e, *f, **l;
Medium *m = ifc->m; Medium *m = ifc->m;
int empty; int empty;
@ -87,31 +99,23 @@ newarp6(Arp *arp, uchar *ip, Ipifc *ifc, int addrxt)
/* dump waiting packets */ /* dump waiting packets */
xp = a->hold; xp = a->hold;
a->hold = nil; a->hold = nil;
if(isv4(a->ip))
if(isv4(a->ip)){ freeblistchain(xp);
while(xp){
next = xp->list;
freeblist(xp);
xp = next;
}
}
else { /* queue icmp unreachable for rxmitproc later on, w/o arp lock */ else { /* queue icmp unreachable for rxmitproc later on, w/o arp lock */
if(xp){ if(xp != nil){
if(arp->dropl == nil) if(arp->dropf == nil)
arp->dropf = xp; arp->dropf = xp;
else else
arp->dropl->list = xp; arp->dropl->list = xp;
arp->dropl = a->last;
for(next = xp->list; next; next = next->list)
xp = next;
arp->dropl = xp;
wakeup(&arp->rxmtq); wakeup(&arp->rxmtq);
} }
} }
a->last = nil;
/* take out of current chain */ /* take out of current chain */
l = &arp->hash[haship(a->ip)]; l = &arp->hash[haship(a->ip)];
for(f = *l; f; f = f->hash){ for(f = *l; f != nil; f = f->hash){
if(f == a){ if(f == a){
*l = a->hash; *l = a->hash;
break; break;
@ -137,18 +141,18 @@ newarp6(Arp *arp, uchar *ip, Ipifc *ifc, int addrxt)
/* put to the end of re-transmit chain; addrxt is 0 when isv4(a->ip) */ /* put to the end of re-transmit chain; addrxt is 0 when isv4(a->ip) */
if(!ipismulticast(a->ip) && addrxt){ if(!ipismulticast(a->ip) && addrxt){
l = &arp->rxmt; l = &arp->rxmt;
empty = (*l==nil); empty = (*l == nil);
for(f = *l; f; f = f->nextrxt){ for(f = *l; f != nil; f = f->nextrxt){
if(f == a){ if(f == a){
*l = a->nextrxt; *l = a->nextrxt;
break; break;
} }
l = &f->nextrxt; l = &f->nextrxt;
} }
for(f = *l; f; f = f->nextrxt){ for(f = *l; f != nil; f = f->nextrxt)
l = &f->nextrxt; l = &f->nextrxt;
}
*l = a; *l = a;
if(empty) if(empty)
wakeup(&arp->rxmtq); wakeup(&arp->rxmtq);
@ -173,7 +177,7 @@ cleanarpent(Arp *arp, Arpent *a)
/* take out of current chain */ /* take out of current chain */
l = &arp->hash[haship(a->ip)]; l = &arp->hash[haship(a->ip)];
for(f = *l; f; f = f->hash){ for(f = *l; f != nil; f = f->hash){
if(f == a){ if(f == a){
*l = a->hash; *l = a->hash;
break; break;
@ -183,7 +187,7 @@ cleanarpent(Arp *arp, Arpent *a)
/* take out of re-transmit chain */ /* take out of re-transmit chain */
l = &arp->rxmt; l = &arp->rxmt;
for(f = *l; f; f = f->nextrxt){ for(f = *l; f != nil; f = f->nextrxt){
if(f == a){ if(f == a){
*l = a->nextrxt; *l = a->nextrxt;
break; break;
@ -192,8 +196,8 @@ cleanarpent(Arp *arp, Arpent *a)
} }
a->nextrxt = nil; a->nextrxt = nil;
a->hash = nil; a->hash = nil;
a->hold = nil; freeblistchain(a->hold);
a->last = nil; a->hold = a->last = nil;
a->ifc = nil; a->ifc = nil;
} }
@ -218,7 +222,7 @@ arpget(Arp *arp, Block *bp, int version, Ipifc *ifc, uchar *ip, uchar *mac)
qlock(arp); qlock(arp);
hash = haship(ip); hash = haship(ip);
for(a = arp->hash[hash]; a; a = a->hash){ for(a = arp->hash[hash]; a != nil; a = a->hash){
if(memcmp(ip, a->ip, sizeof(a->ip)) == 0) if(memcmp(ip, a->ip, sizeof(a->ip)) == 0)
if(type == a->type) if(type == a->type)
break; break;
@ -231,12 +235,12 @@ arpget(Arp *arp, Block *bp, int version, Ipifc *ifc, uchar *ip, uchar *mac)
a->utime = NOW; a->utime = NOW;
if(a->state == AWAIT){ if(a->state == AWAIT){
if(bp != nil){ if(bp != nil){
if(a->hold)
a->last->list = bp;
else
a->hold = bp;
a->last = bp;
bp->list = nil; bp->list = nil;
if(a->hold == nil)
a->hold = bp;
else
a->last->list = bp;
a->last = bp;
} }
return a; /* return with arp qlocked */ return a; /* return with arp qlocked */
} }
@ -274,7 +278,7 @@ arpresolve(Arp *arp, Arpent *a, Medium *type, uchar *mac)
if(!isv4(a->ip)){ if(!isv4(a->ip)){
l = &arp->rxmt; l = &arp->rxmt;
for(f = *l; f; f = f->nextrxt){ for(f = *l; f != nil; f = f->nextrxt){
if(f == a){ if(f == a){
*l = a->nextrxt; *l = a->nextrxt;
break; break;
@ -288,7 +292,8 @@ arpresolve(Arp *arp, Arpent *a, Medium *type, uchar *mac)
a->state = AOK; a->state = AOK;
a->utime = NOW; a->utime = NOW;
bp = a->hold; bp = a->hold;
a->hold = nil; assert(bp->list == nil);
a->hold = a->last = nil;
qunlock(arp); qunlock(arp);
return bp; return bp;
@ -307,10 +312,8 @@ arpenter(Fs *fs, int version, uchar *ip, uchar *mac, int n, int refresh)
arp = fs->arp; arp = fs->arp;
if(n != 6){ if(n != 6)
// print("arp: len = %d\n", n);
return; return;
}
switch(version){ switch(version){
case V4: case V4:
@ -326,16 +329,14 @@ arpenter(Fs *fs, int version, uchar *ip, uchar *mac, int n, int refresh)
return; /* to supress warnings */ return; /* to supress warnings */
} }
if(r == nil){ if(r == nil)
// print("arp: no route for entry\n");
return; return;
}
ifc = r->ifc; ifc = r->ifc;
type = ifc->m; type = ifc->m;
qlock(arp); qlock(arp);
for(a = arp->hash[haship(ip)]; a; a = a->hash){ for(a = arp->hash[haship(ip)]; a != nil; a = a->hash){
if(a->type != type || (a->state != AWAIT && a->state != AOK)) if(a->type != type || (a->state != AWAIT && a->state != AOK))
continue; continue;
@ -346,7 +347,7 @@ arpenter(Fs *fs, int version, uchar *ip, uchar *mac, int n, int refresh)
if(version == V6){ if(version == V6){
/* take out of re-transmit chain */ /* take out of re-transmit chain */
l = &arp->rxmt; l = &arp->rxmt;
for(f = *l; f; f = f->nextrxt){ for(f = *l; f != nil; f = f->nextrxt){
if(f == a){ if(f == a){
*l = a->nextrxt; *l = a->nextrxt;
break; break;
@ -358,29 +359,27 @@ arpenter(Fs *fs, int version, uchar *ip, uchar *mac, int n, int refresh)
a->ifc = ifc; a->ifc = ifc;
a->ifcid = ifc->ifcid; a->ifcid = ifc->ifcid;
bp = a->hold; bp = a->hold;
a->hold = nil; a->hold = a->last = nil;
if(version == V4) if(version == V4)
ip += IPv4off; ip += IPv4off;
a->utime = NOW; a->utime = NOW;
a->ctime = a->utime; a->ctime = a->utime;
qunlock(arp); qunlock(arp);
while(bp){ while(bp != nil){
next = bp->list; next = bp->list;
if(ifc != nil){ if(waserror()){
if(waserror()){ freeblistchain(next);
runlock(ifc);
nexterror();
}
rlock(ifc);
if(ifc->m != nil)
ifc->m->bwrite(ifc, bp, version, ip);
else
freeb(bp);
runlock(ifc); runlock(ifc);
poperror(); nexterror();
} else }
freeb(bp); rlock(ifc);
if(ifc->m != nil)
ifc->m->bwrite(ifc, bp, version, ip);
else
freeblist(bp);
runlock(ifc);
poperror();
bp = next; bp = next;
} }
return; return;
@ -404,7 +403,6 @@ arpwrite(Fs *fs, char *s, int len)
int n; int n;
Route *r; Route *r;
Arp *arp; Arp *arp;
Block *bp;
Arpent *a; Arpent *a;
Medium *m; Medium *m;
char *f[4], buf[256]; char *f[4], buf[256];
@ -430,21 +428,14 @@ arpwrite(Fs *fs, char *s, int len)
a->hash = nil; a->hash = nil;
a->state = 0; a->state = 0;
a->utime = 0; a->utime = 0;
while(a->hold != nil){ freeblistchain(a->hold);
bp = a->hold->list; a->hold = a->last = nil;
freeblist(a->hold);
a->hold = bp;
}
} }
memset(arp->hash, 0, sizeof(arp->hash)); memset(arp->hash, 0, sizeof(arp->hash));
/* clear all pkts on these lists (rxmt, dropf/l) */ /* clear all pkts on these lists (rxmt, dropf/l) */
arp->rxmt = nil; arp->rxmt = nil;
arp->dropl = nil; freeblistchain(arp->dropf);
while(arp->dropf != nil){ arp->dropf = arp->dropl = nil;
bp = arp->dropf->list;
freeblist(arp->dropf);
arp->dropf = bp;
}
qunlock(arp); qunlock(arp);
} else if(strcmp(f[0], "add") == 0){ } else if(strcmp(f[0], "add") == 0){
switch(n){ switch(n){
@ -482,18 +473,13 @@ arpwrite(Fs *fs, char *s, int len)
if (parseip(ip, f[1]) == -1) if (parseip(ip, f[1]) == -1)
error(Ebadip); error(Ebadip);
qlock(arp);
for(a = arp->hash[haship(ip)]; a; a = a->hash) qlock(arp);
for(a = arp->hash[haship(ip)]; a != nil; a = a->hash)
if(memcmp(ip, a->ip, sizeof(a->ip)) == 0) if(memcmp(ip, a->ip, sizeof(a->ip)) == 0)
break; break;
if(a){ if(a != nil){
while(a->hold != nil){
bp = a->hold->list;
freeblist(a->hold);
a->hold = bp;
}
cleanarpent(arp, a); cleanarpent(arp, a);
memset(a->ip, 0, sizeof(a->ip)); memset(a->ip, 0, sizeof(a->ip));
memset(a->mac, 0, sizeof(a->mac)); memset(a->mac, 0, sizeof(a->mac));
@ -565,7 +551,7 @@ rxmitsols(Arp *arp)
f = arp->f; f = arp->f;
a = arp->rxmt; a = arp->rxmt;
if(a==nil){ if(a == nil){
nrxt = 0; nrxt = 0;
goto dodrops; /* return nrxt; */ goto dodrops; /* return nrxt; */
} }
@ -573,24 +559,23 @@ rxmitsols(Arp *arp)
if(nrxt > 3*ReTransTimer/4) if(nrxt > 3*ReTransTimer/4)
goto dodrops; /* return nrxt; */ goto dodrops; /* return nrxt; */
for(; a; a = a->nextrxt){ for(; a != nil; a = a->nextrxt){
ifc = a->ifc; ifc = a->ifc;
assert(ifc != nil); if(a->rxtsrem > 0 && ifc != nil && canrlock(ifc)){
if((a->rxtsrem <= 0) || !(canrlock(ifc)) || (a->ifcid != ifc->ifcid)){ if(a->ifcid == ifc->ifcid)
xp = a->hold; break;
a->hold = nil; runlock(ifc);
if(xp){
if(arp->dropl == nil)
arp->dropf = xp;
else
arp->dropl->list = xp;
}
cleanarpent(arp, a);
} }
else xp = a->hold;
break; a->hold = nil;
if(xp != nil){
if(arp->dropf == nil)
arp->dropf = xp;
else
arp->dropl->list = xp;
arp->dropl = a->last;
}
cleanarpent(arp, a);
} }
if(a == nil) if(a == nil)
goto dodrops; goto dodrops;
@ -605,34 +590,33 @@ rxmitsols(Arp *arp)
/* put to the end of re-transmit chain */ /* put to the end of re-transmit chain */
l = &arp->rxmt; l = &arp->rxmt;
for(b = *l; b; b = b->nextrxt){ for(b = *l; b != nil; b = b->nextrxt){
if(b == a){ if(b == a){
*l = a->nextrxt; *l = a->nextrxt;
break; break;
} }
l = &b->nextrxt; l = &b->nextrxt;
} }
for(b = *l; b; b = b->nextrxt){ for(b = *l; b != nil; b = b->nextrxt)
l = &b->nextrxt; l = &b->nextrxt;
}
*l = a; *l = a;
a->rxtsrem--; a->rxtsrem--;
a->nextrxt = nil; a->nextrxt = nil;
a->rtime = NOW + ReTransTimer; a->rtime = NOW + ReTransTimer;
a = arp->rxmt; a = arp->rxmt;
if(a==nil) if(a == nil)
nrxt = 0; nrxt = 0;
else else
nrxt = a->rtime - NOW; nrxt = a->rtime - NOW;
dodrops: dodrops:
xp = arp->dropf; xp = arp->dropf;
arp->dropf = nil; arp->dropf = arp->dropl = nil;
arp->dropl = nil;
qunlock(arp); qunlock(arp);
for(; xp; xp = next){ for(; xp != nil; xp = next){
next = xp->list; next = xp->list;
icmphostunr(f, ifc, xp, Icmp6_adr_unreach, 1); icmphostunr(f, ifc, xp, Icmp6_adr_unreach, 1);
} }
@ -645,11 +629,8 @@ static int
rxready(void *v) rxready(void *v)
{ {
Arp *arp = (Arp *) v; Arp *arp = (Arp *) v;
int x;
x = ((arp->rxmt != nil) || (arp->dropf != nil)); return arp->rxmt != nil || arp->dropf != nil;
return x;
} }
static void static void
@ -659,9 +640,8 @@ rxmitproc(void *v)
long wakeupat; long wakeupat;
arp->rxmitp = up; arp->rxmitp = up;
//print("arp rxmitproc started\n");
if(waserror()){ if(waserror()){
arp->rxmitp = 0; arp->rxmitp = nil;
pexit("hangup", 1); pexit("hangup", 1);
} }
for(;;){ for(;;){