ref: bc1ff6985c03402497a052d2b4e2a299b3cdf218
dir: /sys/src/cmd/usb/disk/disk.c/
/* * usb/disk - usb mass storage file server * * supports only the scsi command interface, not ata. */ #include <u.h> #include <libc.h> #include <ctype.h> #include <fcall.h> #include <thread.h> #include "scsireq.h" #include "usb.h" #include "usbfs.h" #include "ums.h" enum { Qdir = 0, Qctl, Qraw, Qdata, Qpart, Qmax = Maxparts, }; typedef struct Dirtab Dirtab; struct Dirtab { char *name; int mode; }; ulong ctlmode = 0664; /* * Partition management (adapted from disk/partfs) */ Part * lookpart(Umsc *lun, char *name) { Part *part, *p; part = lun->part; for(p=part; p < &part[Qmax]; p++){ if(p->inuse && strcmp(p->name, name) == 0) return p; } return nil; } Part * freepart(Umsc *lun) { Part *part, *p; part = lun->part; for(p=part; p < &part[Qmax]; p++){ if(!p->inuse) return p; } return nil; } int addpart(Umsc *lun, char *name, vlong start, vlong end, ulong mode) { Part *p; if(start < 0 || start > end || end > lun->blocks){ werrstr("bad partition boundaries"); return -1; } if(lookpart(lun, name) != nil) { werrstr("partition name already in use"); return -1; } p = freepart(lun); if(p == nil){ werrstr("no free partition slots"); return -1; } p->inuse = 1; free(p->name); p->id = p - lun->part; p->name = estrdup(name); p->offset = start; p->length = end - start; p->mode = mode; return 0; } int delpart(Umsc *lun, char *s) { Part *p; p = lookpart(lun, s); if(p == nil || p->id <= Qdata){ werrstr("partition not found"); return -1; } p->inuse = 0; free(p->name); p->name = nil; p->vers++; return 0; } void fixlength(Umsc *lun, vlong blocks) { Part *part, *p; part = lun->part; part[Qdata].length = blocks; for(p=&part[Qdata+1]; p < &part[Qmax]; p++){ if(p->inuse && p->offset + p->length > blocks){ if(p->offset > blocks){ p->offset =blocks; p->length = 0; }else p->length = blocks - p->offset; } } } void makeparts(Umsc *lun) { addpart(lun, "/", 0, 0, DMDIR | 0555); addpart(lun, "ctl", 0, 0, 0664); addpart(lun, "raw", 0, 0, 0640); addpart(lun, "data", 0, lun->blocks, 0640); } /* * ctl parsing & formatting (adapted from partfs) */ static char* ctlstring(Usbfs *fs) { Part *p, *part; Fmt fmt; Umsc *lun; Ums *ums; ums = fs->dev->aux; lun = fs->aux; part = &lun->part[0]; fmtstrinit(&fmt); fmtprint(&fmt, "dev %s\n", fs->dev->dir); fmtprint(&fmt, "lun %ld\n", lun - &ums->lun[0]); if(lun->flags & Finqok) fmtprint(&fmt, "inquiry %s\n", lun->inq); if(lun->blocks > 0) fmtprint(&fmt, "geometry %llud %ld\n", lun->blocks, lun->lbsize); for (p = &part[Qdata+1]; p < &part[Qmax]; p++) if (p->inuse) fmtprint(&fmt, "part %s %lld %lld\n", p->name, p->offset, p->offset + p->length); return fmtstrflush(&fmt); } static int ctlparse(Usbfs *fs, char *msg) { vlong start, end; char *argv[16]; int argc; Umsc *lun; lun = fs->aux; argc = tokenize(msg, argv, nelem(argv)); if(argc < 1){ werrstr("empty control message"); return -1; } if(strcmp(argv[0], "part") == 0){ if(argc != 4){ werrstr("part takes 3 args"); return -1; } start = strtoll(argv[2], 0, 0); end = strtoll(argv[3], 0, 0); return addpart(lun, argv[1], start, end, 0640); }else if(strcmp(argv[0], "delpart") == 0){ if(argc != 2){ werrstr("delpart takes 1 arg"); return -1; } return delpart(lun, argv[1]); } werrstr("unknown ctl"); return -1; } /* * These are used by scuzz scsireq */ int exabyte, force6bytecmds; int diskdebug; static void ding(void *, char *msg) { if(strstr(msg, "alarm") != nil) noted(NCONT); noted(NDFLT); } static int getmaxlun(Dev *dev) { uchar max; int r; max = 0; r = Rd2h|Rclass|Riface; if(usbcmd(dev, r, Getmaxlun, 0, 0, &max, 1) < 0){ dprint(2, "disk: %s: getmaxlun failed: %r\n", dev->dir); }else{ max &= 017; /* 15 is the max. allowed */ dprint(2, "disk: %s: maxlun %d\n", dev->dir, max); } return max; } static int umsreset(Ums *ums) { int r; r = Rh2d|Rclass|Riface; if(usbcmd(ums->dev, r, Umsreset, 0, 0, nil, 0) < 0){ fprint(2, "disk: reset: %r\n"); return -1; } return 0; } static int umsrecover(Ums *ums) { if(umsreset(ums) < 0) return -1; if(unstall(ums->dev, ums->epin, Ein) < 0) dprint(2, "disk: unstall epin: %r\n"); /* do we need this when epin == epout? */ if(unstall(ums->dev, ums->epout, Eout) < 0) dprint(2, "disk: unstall epout: %r\n"); return 0; } static void umsfatal(Ums *ums) { int i; devctl(ums->dev, "detach"); for(i = 0; i < ums->maxlun; i++) usbfsdel(&ums->lun[i].fs); } static int ispow2(uvlong ul) { return (ul & (ul - 1)) == 0; } /* * return smallest power of 2 >= n */ static int log2(int n) { int i; for(i = 0; (1 << i) < n; i++) ; return i; } static int umscapacity(Umsc *lun) { uchar data[32]; lun->blocks = 0; lun->capacity = 0; lun->lbsize = 0; memset(data, 0, sizeof data); if(SRrcapacity(lun, data) < 0 && SRrcapacity(lun, data) < 0) return -1; lun->blocks = GETBELONG(data); lun->lbsize = GETBELONG(data+4); if(lun->blocks == 0xFFFFFFFF){ if(SRrcapacity16(lun, data) < 0){ lun->lbsize = 0; lun->blocks = 0; return -1; }else{ lun->lbsize = GETBELONG(data + 8); lun->blocks = (uvlong)GETBELONG(data)<<32 | GETBELONG(data + 4); } } lun->blocks++; /* SRcapacity returns LBA of last block */ lun->capacity = (vlong)lun->blocks * lun->lbsize; fixlength(lun, lun->blocks); if(diskdebug) fprint(2, "disk: logical block size %lud, # blocks %llud\n", lun->lbsize, lun->blocks); return 0; } static int umsinit(Ums *ums) { uchar i; Umsc *lun; int some; umsreset(ums); ums->maxlun = getmaxlun(ums->dev); ums->lun = mallocz((ums->maxlun+1) * sizeof(*ums->lun), 1); some = 0; for(i = 0; i <= ums->maxlun; i++){ lun = &ums->lun[i]; lun->ums = ums; lun->umsc = lun; lun->lun = i; lun->flags = Fopen | Fusb | Frw10; if(SRinquiry(lun) < 0 && SRinquiry(lun) < 0){ dprint(2, "disk: lun %d inquiry failed\n", i); continue; } switch(lun->inquiry[0]){ case Devdir: case Devworm: /* a little different than the others */ case Devcd: case Devmo: break; default: fprint(2, "disk: lun %d is not a disk (type %#02x)\n", i, lun->inquiry[0]); continue; } SRstart(lun, 1); /* * we ignore the device type reported by inquiry. * Some devices return a wrong value but would still work. */ some++; lun->inq = smprint("%.48s", (char *)lun->inquiry+8); umscapacity(lun); } if(some == 0){ dprint(2, "disk: all luns failed\n"); devctl(ums->dev, "detach"); return -1; } return 0; } /* * called by SR*() commands provided by scuzz's scsireq */ long umsrequest(Umsc *umsc, ScsiPtr *cmd, ScsiPtr *data, int *status) { Cbw cbw; Csw csw; int n, nio, left; Ums *ums; ums = umsc->ums; memcpy(cbw.signature, "USBC", 4); cbw.tag = ++ums->seq; cbw.datalen = data->count; cbw.flags = data->write? CbwDataOut: CbwDataIn; cbw.lun = umsc->lun; if(cmd->count < 1 || cmd->count > 16) print("disk: umsrequest: bad cmd count: %ld\n", cmd->count); cbw.len = cmd->count; assert(cmd->count <= sizeof(cbw.command)); memcpy(cbw.command, cmd->p, cmd->count); memset(cbw.command + cmd->count, 0, sizeof(cbw.command) - cmd->count); werrstr(""); /* we use %r later even for n == 0 */ if(diskdebug){ fprint(2, "disk: cmd: tag %#lx: ", cbw.tag); for(n = 0; n < cbw.len; n++) fprint(2, " %2.2x", cbw.command[n]&0xFF); fprint(2, " datalen: %ld\n", cbw.datalen); } /* issue tunnelled scsi command */ if(write(ums->epout->dfd, &cbw, CbwLen) != CbwLen){ fprint(2, "disk: cmd: %r\n"); goto Fail; } /* transfer the data */ nio = data->count; if(nio != 0){ if(data->write) n = write(ums->epout->dfd, data->p, nio); else{ n = read(ums->epin->dfd, data->p, nio); left = nio - n; if (n >= 0 && left > 0) /* didn't fill data->p? */ memset(data->p + n, 0, left); } nio = n; if(diskdebug) if(n < 0) fprint(2, "disk: data: %r\n"); else fprint(2, "disk: data: %d bytes\n", n); if(n <= 0) if(data->write == 0) unstall(ums->dev, ums->epin, Ein); } /* read the transfer's status */ n = read(ums->epin->dfd, &csw, CswLen); if(n <= 0){ /* n == 0 means "stalled" */ unstall(ums->dev, ums->epin, Ein); n = read(ums->epin->dfd, &csw, CswLen); } if(n != CswLen || strncmp(csw.signature, "USBS", 4) != 0){ dprint(2, "disk: read n=%d: status: %r\n", n); goto Fail; } if(csw.tag != cbw.tag){ dprint(2, "disk: status tag mismatch\n"); goto Fail; } if(csw.status >= CswPhaseErr){ dprint(2, "disk: phase error\n"); goto Fail; } if(csw.dataresidue == 0 || ums->wrongresidues) csw.dataresidue = data->count - nio; if(diskdebug){ fprint(2, "disk: status: %2.2ux residue: %ld\n", csw.status, csw.dataresidue); if(cbw.command[0] == ScmdRsense){ fprint(2, "sense data:"); for(n = 0; n < data->count - csw.dataresidue; n++) fprint(2, " %2.2x", data->p[n]); fprint(2, "\n"); } } switch(csw.status){ case CswOk: *status = STok; break; case CswFailed: *status = STcheck; break; default: dprint(2, "disk: phase error\n"); goto Fail; } ums->nerrs = 0; return data->count - csw.dataresidue; Fail: *status = STharderr; if(ums->nerrs++ > 15){ fprint(2, "disk: %s: too many errors: device detached\n", ums->dev->dir); umsfatal(ums); }else umsrecover(ums); return -1; } static int dwalk(Usbfs *fs, Fid *fid, char *name) { Umsc *lun; Part *p; lun = fs->aux; if((fid->qid.type & QTDIR) == 0){ werrstr("walk in non-directory"); return -1; } if(strcmp(name, "..") == 0) return 0; p = lookpart(lun, name); if(p == nil){ werrstr(Enotfound); return -1; } fid->qid.path = p->id | fs->qid; fid->qid.vers = p->vers; fid->qid.type = p->mode >> 24; return 0; } static int dstat(Usbfs *fs, Qid qid, Dir *d); static void dostat(Usbfs *fs, int path, Dir *d) { Umsc *lun; Part *p; lun = fs->aux; p = &lun->part[path]; d->qid.path = path; d->qid.vers = p->vers; d->qid.type =p->mode >> 24; d->mode = p->mode; d->length = (vlong) p->length * lun->lbsize; strecpy(d->name, d->name + Namesz - 1, p->name); } static int dirgen(Usbfs *fs, Qid, int n, Dir *d, void*) { Umsc *lun; int i; lun = fs->aux; for(i = Qctl; i < Qmax; i++){ if(lun->part[i].inuse == 0) continue; if(n-- == 0) break; } if(i == Qmax) return -1; dostat(fs, i, d); d->qid.path |= fs->qid; return 0; } static int dstat(Usbfs *fs, Qid qid, Dir *d) { int path; path = qid.path & ~fs->qid; dostat(fs, path, d); d->qid.path |= fs->qid; return 0; } static int dopen(Usbfs *fs, Fid *fid, int) { ulong path; Umsc *lun; path = fid->qid.path & ~fs->qid; lun = fs->aux; switch(path){ case Qraw: lun->phase = Pcmd; break; } return 0; } /* * check i/o parameters and compute values needed later. * we shift & mask manually to avoid run-time calls to _divv and _modv, * since we don't need general division nor its cost. */ static int setup(Umsc *lun, Part *p, char *data, int count, vlong offset) { long nb, lbsize, lbshift, lbmask; uvlong bno; if(count < 0 || lun->lbsize <= 0 && umscapacity(lun) < 0 || lun->lbsize == 0) return -1; lbsize = lun->lbsize; assert(ispow2(lbsize)); lbshift = log2(lbsize); lbmask = lbsize - 1; bno = offset >> lbshift; /* offset / lbsize */ nb = ((offset + count + lbsize - 1) >> lbshift) - bno; if(bno + nb > p->length) /* past end of partition? */ nb = p->length - bno; if(nb * lbsize > Maxiosize) nb = Maxiosize / lbsize; lun->nb = nb; if(bno >= p->length || nb == 0) return 0; bno += p->offset; /* start of partition */ lun->offset = bno; lun->off = offset & lbmask; /* offset % lbsize */ if(lun->off == 0 && (count & lbmask) == 0) lun->bufp = data; else /* not transferring full, aligned blocks; need intermediary */ lun->bufp = lun->buf; return count; } /* * Upon SRread/SRwrite errors we assume the medium may have changed, * and ask again for the capacity of the media. * BUG: How to proceed to avoid confussing dossrv?? */ static long dread(Usbfs *fs, Fid *fid, void *data, long count, vlong offset) { long n; ulong path; char buf[64]; char *s; Part *p; Umsc *lun; Ums *ums; Qid q; q = fid->qid; path = fid->qid.path & ~fs->qid; ums = fs->dev->aux; lun = fs->aux; qlock(ums); switch(path){ case Qdir: count = usbdirread(fs, q, data, count, offset, dirgen, nil); break; case Qctl: s = ctlstring(fs); count = usbreadbuf(data, count, offset, s, strlen(s)); free(s); break; case Qraw: if(lun->lbsize <= 0 && umscapacity(lun) < 0){ count = -1; break; } switch(lun->phase){ case Pcmd: qunlock(ums); werrstr("phase error"); return -1; case Pdata: lun->data.p = data; lun->data.count = count; lun->data.write = 0; count = umsrequest(lun,&lun->cmd,&lun->data,&lun->status); lun->phase = Pstatus; if(count < 0) lun->lbsize = 0; /* medium may have changed */ break; case Pstatus: n = snprint(buf, sizeof buf, "%11.0ud ", lun->status); count = usbreadbuf(data, count, 0LL, buf, n); lun->phase = Pcmd; break; } break; case Qdata: default: p = &lun->part[path]; if(!p->inuse){ count = -1; werrstr(Eperm); break; } count = setup(lun, p, data, count, offset); if (count <= 0) break; n = SRread(lun, lun->bufp, lun->nb * lun->lbsize); if(n < 0){ lun->lbsize = 0; /* medium may have changed */ count = -1; } else if (lun->bufp == data) count = n; else{ /* * if n == lun->nb*lun->lbsize (as expected), * just copy count bytes. */ if(lun->off + count > n) count = n - lun->off; /* short read */ if(count > 0) memmove(data, lun->bufp + lun->off, count); } break; } qunlock(ums); return count; } static long dwrite(Usbfs *fs, Fid *fid, void *data, long count, vlong offset) { long len, ocount; ulong path; uvlong bno; Ums *ums; Part *p; Umsc *lun; char *s; ums = fs->dev->aux; lun = fs->aux; path = fid->qid.path & ~fs->qid; qlock(ums); switch(path){ case Qdir: count = -1; werrstr(Eperm); break; case Qctl: s = emallocz(count+1, 1); memmove(s, data, count); if(s[count-1] == '\n') s[count-1] = 0; if(ctlparse(fs, s) == -1) count = -1; free(s); break; case Qraw: if(lun->lbsize <= 0 && umscapacity(lun) < 0){ count = -1; break; } switch(lun->phase){ case Pcmd: if(count != 6 && count != 10){ qunlock(ums); werrstr("bad command length"); return -1; } memmove(lun->rawcmd, data, count); lun->cmd.p = lun->rawcmd; lun->cmd.count = count; lun->cmd.write = 1; lun->phase = Pdata; break; case Pdata: lun->data.p = data; lun->data.count = count; lun->data.write = 1; count = umsrequest(lun,&lun->cmd,&lun->data,&lun->status); lun->phase = Pstatus; if(count < 0) lun->lbsize = 0; /* medium may have changed */ break; case Pstatus: lun->phase = Pcmd; werrstr("phase error"); count = -1; break; } break; case Qdata: default: p = &lun->part[path]; if(!p->inuse){ count = -1; werrstr(Eperm); break; } len = ocount = count; count = setup(lun, p, data, count, offset); if (count <= 0) break; bno = lun->offset; if (lun->bufp == lun->buf) { count = SRread(lun, lun->bufp, lun->nb * lun->lbsize); if(count < 0) { lun->lbsize = 0; /* medium may have changed */ break; } /* * if count == lun->nb*lun->lbsize, as expected, just * copy len (the original count) bytes of user data. */ if(lun->off + len > count) len = count - lun->off; /* short read */ if(len > 0) memmove(lun->bufp + lun->off, data, len); } lun->offset = bno; count = SRwrite(lun, lun->bufp, lun->nb * lun->lbsize); if(count < 0) lun->lbsize = 0; /* medium may have changed */ else{ if(lun->off + len > count) count -= lun->off; /* short write */ /* never report more bytes written than requested */ if(count < 0) count = 0; else if(count > ocount) count = ocount; } break; } qunlock(ums); return count; } int findendpoints(Ums *ums) { Ep *ep; Usbdev *ud; ulong csp, sc; int i, epin, epout; epin = epout = -1; ud = ums->dev->usb; for(i = 0; i < nelem(ud->ep); i++){ if((ep = ud->ep[i]) == nil) continue; csp = ep->iface->csp; sc = Subclass(csp); if(!(Class(csp) == Clstorage && (Proto(csp) == Protobulk))) continue; if(sc != Subatapi && sc != Sub8070 && sc != Subscsi) fprint(2, "disk: subclass %#ulx not supported. trying anyway\n", sc); if(ep->type == Ebulk){ if(ep->dir == Eboth || ep->dir == Ein) if(epin == -1) epin = ep->id; if(ep->dir == Eboth || ep->dir == Eout) if(epout == -1) epout = ep->id; } } dprint(2, "disk: ep ids: in %d out %d\n", epin, epout); if(epin == -1 || epout == -1) return -1; ums->epin = openep(ums->dev, epin); if(ums->epin == nil){ fprint(2, "disk: openep %d: %r\n", epin); return -1; } if(epout == epin){ incref(ums->epin); ums->epout = ums->epin; }else ums->epout = openep(ums->dev, epout); if(ums->epout == nil){ fprint(2, "disk: openep %d: %r\n", epout); closedev(ums->epin); return -1; } if(ums->epin == ums->epout) opendevdata(ums->epin, ORDWR); else{ opendevdata(ums->epin, OREAD); opendevdata(ums->epout, OWRITE); } if(ums->epin->dfd < 0 || ums->epout->dfd < 0){ fprint(2, "disk: open i/o ep data: %r\n"); closedev(ums->epin); closedev(ums->epout); return -1; } dprint(2, "disk: ep in %s out %s\n", ums->epin->dir, ums->epout->dir); devctl(ums->epin, "timeout 2000"); devctl(ums->epout, "timeout 2000"); if(usbdebug > 1 || diskdebug > 2){ devctl(ums->epin, "debug 1"); devctl(ums->epout, "debug 1"); devctl(ums->dev, "debug 1"); } return 0; } static int usage(void) { werrstr("usage: usb/disk [-d] [-N nb]"); return -1; } static void umsdevfree(void *a) { Ums *ums = a; if(ums == nil) return; closedev(ums->epin); closedev(ums->epout); ums->epin = ums->epout = nil; free(ums->lun); free(ums); } static Usbfs diskfs = { .walk = dwalk, .open = dopen, .read = dread, .write = dwrite, .stat = dstat, }; int diskmain(Dev *dev, int argc, char **argv) { Ums *ums; Umsc *lun; int i, devid; devid = dev->id; ARGBEGIN{ case 'd': scsidebug(diskdebug); diskdebug++; break; case 'N': devid = atoi(EARGF(usage())); break; default: return usage(); }ARGEND if(argc != 0) { return usage(); } // notify(ding); ums = dev->aux = emallocz(sizeof(Ums), 1); ums->maxlun = -1; ums->dev = dev; dev->free = umsdevfree; if(findendpoints(ums) < 0){ werrstr("disk: endpoints not found"); return -1; } /* * SanDISK 512M gets residues wrong. */ if(dev->usb->vid == 0x0781 && dev->usb->did == 0x5150) ums->wrongresidues = 1; if(umsinit(ums) < 0){ dprint(2, "disk: umsinit: %r\n"); return -1; } for(i = 0; i <= ums->maxlun; i++){ lun = &ums->lun[i]; lun->fs = diskfs; snprint(lun->fs.name, sizeof(lun->fs.name), "sdU%d.%d", devid, i); lun->fs.dev = dev; incref(dev); lun->fs.aux = lun; makeparts(lun); usbfsadd(&lun->fs); } return 0; }