ref: 85d0f0e1a2c05afee5a101c1e51a3ddd2ff2a5f6
dir: /sys/src/cmd/nusb/ether/cdc.c/
/* * generic CDC */ #include <u.h> #include <libc.h> #include <thread.h> #include "usb.h" #include "dat.h" #include <ip.h> static int cdcreceive(Dev *ep) { Block *b; int n; b = allocb(Maxpkt); if((n = read(ep->dfd, b->wp, b->lim - b->base)) < 0){ freeb(b); return -1; } b->wp += n; etheriq(b); return 0; } static void cdctransmit(Dev *ep, Block *b) { int n; n = BLEN(b); if(write(ep->dfd, b->rp, n) < 0){ freeb(b); return; } freeb(b); /* * this may not work with all CDC devices. the * linux driver sends one more random byte * instead of a zero byte transaction. maybe we * should do the same? */ if((n % ep->maxpkt) == 0) write(ep->dfd, "", 0); } int cdcinit(Dev *d) { int i; Usbdev *ud; uchar *b; Desc *dd; char *mac; ud = d->usb; for(i = 0; i < nelem(ud->ddesc); i++) if((dd = ud->ddesc[i]) != nil){ b = (uchar*)&dd->data; if(b[1] == Dfunction && b[2] == Fnether){ mac = loaddevstr(d, b[3]); if(mac != nil && strlen(mac) != 12){ free(mac); mac = nil; } if(mac != nil){ parseether(macaddr, mac); free(mac); epreceive = cdcreceive; eptransmit = cdctransmit; return 0; } } } return -1; }