On Tue, 26 Jun 2007 14:50:10 -0400 "Ed L. Cashin" <[EMAIL PROTECTED]> wrote:
> Handle multiple network paths to AoE device. > > ... > > > struct buf *inprocess; /* the one we're currently working on */ > - ushort lostjumbo; > - ushort nframes; /* number of frames below */ > - struct frame *frames; > + struct aoetgt *targets[NTARGETS]; > + struct aoetgt **tgt; /* target in use when working */ > + struct aoetgt **htgt; /* target needing rexmit assistance */ > +//int ios[64]; > }; whats that? > static ssize_t aoedisk_show_netif(struct gendisk * disk, char *page) > { > struct aoedev *d = disk->private_data; > + struct net_device *nds[8], **nd, **nnd, **ne; > + struct aoetgt **t, **te; > + struct aoeif *ifp, *e; > + char *p; > + > + memset(nds, 0, ARRAY_SIZE(nds)); bug: this will only zero eight bytes. > + nd = nds; > + ne = nd + ARRAY_SIZE(nds); > + t = d->targets; > + te = t + NTARGETS; > + for (; t<te && *t; t++) { > + ifp = (*t)->ifs; > + e = ifp + NAOEIFS; > + for (; ifp<e && ifp->nd; ifp++) { > + for (nnd=nds; nnd<nd; nnd++) > + if (*nnd == ifp->nd) > + break; > + if (nnd == nd) > + if (nd != ne) > + *nd++ = ifp->nd; > + } > + } There are multiple little coding-style warts here. scripts/checkpatch.pl will point them out. > > - return snprintf(page, PAGE_SIZE, "%s\n", d->ifp->name); > + ne = nd; > + nd = nds; > + if (*nd == NULL) > + return snprintf(page, PAGE_SIZE, "none\n"); > + for (p=page; nd<ne; nd++) > + p += snprintf(p, PAGE_SIZE - (p-page), "%s%s", > + p == page ? "" : ",", (*nd)->name); > + p += snprintf(p, PAGE_SIZE - (p-page), "\n"); > + return p-page; > } > /* firmware version */ > > .. > > spin_lock_irqsave(&d->lock, flags); > - d->flags &= ~DEVFL_MAXBCNT; > - d->flags |= DEVFL_PAUSE; > + aoecmd_cleanslate(d); > +loop: > + skb = aoecmd_ata_id(d); > spin_unlock_irqrestore(&d->lock, flags); > + if (!skb && !msleep_interruptible(200)) { > + spin_lock_irqsave(&d->lock, flags); > + goto loop; > + } > + aoenet_xmit(skb); > aoecmd_cfg(major, minor); > - > return 0; > } interruptible sleep? Does this code work as intended when there's a signal pending? (Maybe that's what the interruptible sleep is for: don't know, and am not inclined to work it out given the (low) level of comments in here and the (lower) level of changelogging). > ... > > +static struct frame * > +freeframe(struct aoedev *d) > { > + struct frame *f, *e; > + struct aoetgt **t; > + ulong n; > + > + if (d->targets[0] == NULL) { /* shouldn't happen, but I'm paranoid */ > + printk(KERN_ERR "aoe: NULL TARGETS!\n"); > + return NULL; > + } > + t = d->targets; > + do { > + if (t != d->htgt) > + if ((*t)->ifp->nd) > + if ((*t)->nout < (*t)->maxout) { ugh. Do this: do { if (t == d->htgt) continue; if (!(*t)->ifp->nd) continue; if ((*t)->nout >= (*t)->maxout) continue; <stuff> } while (++t ...) > +#define ATASCNT(raw) (((struct aoe_atahdr *)(((struct aoe_hdr > *)raw)+1))->scnt) This could be implemented as a (possibly inlined) C function, therefore it shuld be implemented that way. > + > static void > rexmit_timer(ulong vp) > { > struct aoedev *d; > + struct aoetgt *t, **tt, **te; > + struct aoeif *ifp; > struct frame *f, *e; > struct sk_buff *sl; > register long timeout; > @@ -373,31 +451,75 @@ rexmit_timer(ulong vp) > spin_unlock_irqrestore(&d->lock, flags); > return; > } > - f = d->frames; > - e = f + d->nframes; > - for (; f<e; f++) { > - if (f->tag != FREETAG && tsince(f->tag) >= timeout) { > + tt = d->targets; > + te = tt + NTARGETS; > + for (; tt<te && *tt; tt++) { > + t = *tt; > + f = t->frames; > + e = f + t->nframes; > + for (; f<e; f++) { > + if (f->tag == FREETAG > + || tsince(f->tag) < timeout) > + continue; > n = f->waited += timeout; > n /= HZ; > - if (n > aoe_deadsecs) { /* waited too long for response > */ > + if (n > aoe_deadsecs) { /* waited too long. device > failure. */ > aoedev_downdev(d); > break; > } > - rexmit(d, f); > + > + if (n > HELPWAIT) /* see if another target can help */ > + if (tt != d->targets || d->targets[1]) poease find a way to avoid pulling this stunt. > + d->htgt = tt; > + > + if (t->nout == t->maxout) { > + if (t->maxout > 1) > + t->maxout--; > + t->lastwadj = jiffies; > + } > + > + ifp = getif(t, f->skb->dev); > + if (ifp && ++ifp->lost > (t->nframes << 1)) > + if (ifp != t->ifs || t->ifs[1].nd) { here too. > + ejectif(t, ifp); > + ifp = NULL; > + } > + > + if (ATASCNT(aoe_hdr(f->skb)) > DEFAULTBCNT / 512) > + if (ifp && ++ifp->lostjumbo > (t->nframes << 1)) > + if (ifp->maxbcnt != DEFAULTBCNT) { more. > + printk(KERN_INFO "aoe: e%ld.%d: too many lost > jumbo on %s:%012llx - " > + "falling back to %d frames.\n", > + d->aoemajor, d->aoeminor, > + ifp->nd->name, mac_addr(t->addr), > + DEFAULTBCNT); > + ifp->maxbcnt = 0; > + } > + resend(d, t, f); > + } > + > + /* window check */ > + if (t->nout == t->maxout) > + if (t->maxout < t->nframes) > + if ((jiffies - t->lastwadj)/HZ > 10) { more. Just use &&? > + t->maxout++; > + t->lastwadj = jiffies; > } > } > - if (d->flags & DEVFL_KICKME) { > + > + if (d->sendq_hd) { > + n = d->rttavg <<= 1; > + if (n > MAXTIMER) > + d->rttavg = MAXTIMER; > + } > + > + if (d->flags & DEVFL_KICKME || d->htgt) { > d->flags &= ~DEVFL_KICKME; > aoecmd_work(d); > } > > sl = d->sendq_hd; > d->sendq_hd = d->sendq_tl = NULL; > - if (sl) { > - n = d->rttavg <<= 1; > - if (n > MAXTIMER) > - d->rttavg = MAXTIMER; > - } > > d->timer.expires = jiffies + TIMERTICK; > add_timer(&d->timer); > +static struct aoetgt * > +addtgt(struct aoedev *d, char *addr, ulong nframes) > +{ > + struct aoetgt *t, **tt, **te; > + struct frame *f, *e; > + > + tt = d->targets; > + te = tt + NTARGETS; > + for (; tt<te; tt++) { > + if (*tt == NULL) > + break; > + else if (memcmp((*tt)->addr, addr, 6) > 0) { > + memmove(tt+1, tt, (void *)te-(void *)(tt+1)); > + break; > + } > + } Wow. Perhaps there are so few comments because nobody understands what it's all doing > + if (tt == te) > + return NULL; > + > + t = kcalloc(1, sizeof *t, GFP_ATOMIC); > + f = kcalloc(nframes, sizeof *f, GFP_ATOMIC); Is the GFP_ATOMIC unavoidable? It is vastly less reliable than GFP_KERNEL. > + switch (!t || !f) { Oh dear. Please, use an `if' statement rather than party tricks like this. > + case 0: > + t->nframes = nframes; > + t->frames = f; > + e = f + nframes; > + for (; f<e; f++) { > + f->tag = FREETAG; > + f->skb = new_skb(ETH_ZLEN); > + if (!f->skb) > + break; > + } > + if (f == e) > + break; > + while (f > t->frames) { > + f--; > + dev_kfree_skb(f->skb); > + } > + default: > + if (f) > + kfree(f); > + if (t) > + kfree(t); kfree(NULL) is legal. > + return NULL; > + } > + > + memcpy(t->addr, addr, sizeof t->addr); > + t->ifp = t->ifs; > + t->maxout = t->nframes; > + return *tt = t; > +} > + > void > aoecmd_cfg_rsp(struct sk_buff *skb) > { > struct aoedev *d; > struct aoe_hdr *h; > struct aoe_cfghdr *ch; > + struct aoetgt *t; > + struct aoeif *ifp; > ulong flags, sysminor, aoemajor; > struct sk_buff *sl; > enum { MAXFRAMES = 16 }; > @@ -754,7 +952,7 @@ aoecmd_cfg_rsp(struct sk_buff *skb) > if (n > MAXFRAMES) /* keep it reasonable */ > n = MAXFRAMES; > > - d = aoedev_by_sysminor_m(sysminor, n); > + d = aoedev_by_sysminor_m(sysminor); > if (d == NULL) { > printk(KERN_INFO "aoe: device sysminor_m failure\n"); > return; > @@ -762,38 +960,70 @@ aoecmd_cfg_rsp(struct sk_buff *skb) > > spin_lock_irqsave(&d->lock, flags); > > - /* permit device to migrate mac and network interface */ > - d->ifp = skb->dev; > - memcpy(d->addr, h->src, sizeof d->addr); > - if (!(d->flags & DEVFL_MAXBCNT)) { > - n = d->ifp->mtu; > + t = gettgt(d, h->src); > + if (!t) { > + t = addtgt(d, h->src, n); > + if (!t) { > + printk(KERN_INFO "aoe: device addtgt failure; too many > targets?\n"); No, more likely a GFP_ATOMIC allocation failed. Returning -ENOMEM is better than just guessing. > + spin_unlock_irqrestore(&d->lock, flags); > + return; > + } > + } > + ifp = getif(t, skb->dev); > + if (!ifp) { > + if (!(ifp = addif(t, skb->dev))) { Preferred style is ifp = addif(t, skb->dev); if (!ifp) { (checkpatch will report this) > + printk(KERN_INFO "aoe: device addif failure; too many > interfaces?\n"); > + spin_unlock_irqrestore(&d->lock, flags); > + return; > + } > + } > + if (ifp->maxbcnt) { > + n = ifp->nd->mtu; > n -= sizeof (struct aoe_hdr) + sizeof (struct aoe_atahdr); > n /= 512; > if (n > ch->scnt) > n = ch->scnt; > n = n ? n * 512 : DEFAULTBCNT; > - if (n != d->maxbcnt) { > + if (n != ifp->maxbcnt) { > printk(KERN_INFO > - "aoe: e%ld.%ld: setting %d byte data frames on > %s\n", > - d->aoemajor, d->aoeminor, n, d->ifp->name); > - d->maxbcnt = n; > + "aoe: e%ld.%d: setting %d byte data frames on > %s:%012llx\n", > + d->aoemajor, d->aoeminor, n, ifp->nd->name, > + (unsigned long long) mac_addr(t->addr)); > + ifp->maxbcnt = n; > } > } > > /* don't change users' perspective */ > - if (d->nopen && !(d->flags & DEVFL_PAUSE)) { > + if (d->nopen) { > spin_unlock_irqrestore(&d->lock, flags); > return; > } > - d->flags |= DEVFL_PAUSE; /* force pause */ > - d->mintimer = MINTIMER; > d->fw_ver = be16_to_cpu(ch->fwver); > > - /* check for already outstanding ataid */ > - sl = aoedev_isbusy(d) == 0 ? aoecmd_ata_id(d) : NULL; > + sl = aoecmd_ata_id(d); > > spin_unlock_irqrestore(&d->lock, flags); > > aoenet_xmit(sl); > } > - To unsubscribe from this list: send the line "unsubscribe linux-kernel" in the body of a message to [EMAIL PROTECTED] More majordomo info at http://vger.kernel.org/majordomo-info.html Please read the FAQ at http://www.tux.org/lkml/