#include #include #include #include #define Extern extern #include "statfs.h" char *e[] = { [Ebadfid] "Bad fid", [Enotdir] "Not a directory", [Edupfid] "Fid already in use", [Eopen] "Fid already opened", [Exmnt] "Cannot .. past mount point", [Enoauth] "Authentication failed", }; void Xnop(Fsrpc *r) { Fcall thdr; reply(&r->work, &thdr, 0); r->busy = 0; } void Xsession(Fsrpc *r) { Fcall thdr; memset(thdr.authid, 0, sizeof(thdr.authid)); memset(thdr.authdom, 0, sizeof(thdr.authdom)); memset(thdr.chal, 0, sizeof(thdr.chal)); reply(&r->work, &thdr, 0); r->busy = 0; } void Xflush(Fsrpc *r) { Fsrpc *t, *e; Fcall thdr; e = &Workq[Nr_workbufs]; for(t = Workq; t < e; t++) { if(t->work.tag == r->work.oldtag) { DEBUG(2, "\tQ busy %d pid %d can %d\n", t->busy, t->pid, t->canint); if(t->busy && t->pid) { t->flushtag = r->work.tag; DEBUG(2, "\tset flushtag %d\n", r->work.tag); if(t->canint) postnote(PNPROC, t->pid, "flush"); r->busy = 0; return; } } } reply(&r->work, &thdr, 0); DEBUG(2, "\tflush reply\n"); r->busy = 0; } void Xattach(Fsrpc *r) { Fcall thdr; Fid *f; Rpc *rpc; ulong t; t = msec(); f = newfid(r->work.fid); if(f == 0) { reply(&r->work, &thdr, e[Ebadfid]); r->busy = 0; return; } f->f = root; thdr.qid = f->f->qid; reply(&r->work, &thdr, 0); r->busy = 0; t = msec()-t; rpc = &stats->rpc[Tattach]; rpc->time += t; if(t < rpc->loms) rpc->loms = t; if(t > rpc->hims) rpc->hims = t; } void Xclone(Fsrpc *r) { Fcall thdr; Fid *f, *n; ulong t; Rpc *rpc; t = msec(); f = getfid(r->work.fid); if(f == 0) { reply(&r->work, &thdr, e[Ebadfid]); r->busy = 0; return; } n = newfid(r->work.newfid); if(n == 0) { reply(&r->work, &thdr, e[Edupfid]); r->busy = 0; return; } n->f = f->f; reply(&r->work, &thdr, 0); r->busy = 0; t = msec()-t; rpc = &stats->rpc[Tclone]; rpc->time += t; if(t < rpc->loms) rpc->loms = t; if(t > rpc->hims) rpc->hims = t; } void Xwalk(Fsrpc *r) { char err[ERRLEN]; Fcall thdr; Fid *f; File *nf; Rpc *rpc; ulong t; t = msec(); f = getfid(r->work.fid); if(f == 0) { reply(&r->work, &thdr, e[Ebadfid]); r->busy = 0; return; } if(strcmp(r->work.name, "..") == 0) { if(f->f->parent == 0) { reply(&r->work, &thdr, e[Exmnt]); r->busy = 0; return; } f->f = f->f->parent; thdr.qid = f->f->qid; reply(&r->work, &thdr, 0); r->busy = 0; return; } nf = file(f->f, r->work.name); if(nf == 0) { errstr(err); reply(&r->work, &thdr, err); r->busy = 0; return; } f->f = nf; thdr.qid = nf->qid; reply(&r->work, &thdr, 0); r->busy = 0; t = msec()-t; rpc = &stats->rpc[Twalk]; rpc->time += t; if(t < rpc->loms) rpc->loms = t; if(t > rpc->hims) rpc->hims = t; } void Xclunk(Fsrpc *r) { Fcall thdr; Fid *f; Rpc *rpc; ulong t; int fid; t = msec(); f = getfid(r->work.fid); if(f == 0) { reply(&r->work, &thdr, e[Ebadfid]); r->busy = 0; return; } if(f->fid >= 0) close(f->fid); fid = r->work.fid; reply(&r->work, &thdr, 0); r->busy = 0; t = msec()-t; rpc = &stats->rpc[Tclunk]; rpc->time += t; if(t < rpc->loms) rpc->loms = t; if(t > rpc->hims) rpc->hims = t; if(f->nread || f->nwrite) fidreport(f); freefid(fid); } void Xstat(Fsrpc *r) { char err[ERRLEN], path[128]; Fcall thdr; Fid *f; int s; Rpc *rpc; ulong t; t = msec(); f = getfid(r->work.fid); if(f == 0) { reply(&r->work, &thdr, e[Ebadfid]); r->busy = 0; return; } if(f->fid >= 0) s = fstat(f->fid, thdr.stat); else { makepath(path, f->f, ""); s = stat(path, thdr.stat); } if(s < 0) { errstr(err); reply(&r->work, &thdr, err); r->busy = 0; return; } reply(&r->work, &thdr, 0); r->busy = 0; t = msec()-t; rpc = &stats->rpc[Tstat]; rpc->time += t; if(t < rpc->loms) rpc->loms = t; if(t > rpc->hims) rpc->hims = t; } void Xcreate(Fsrpc *r) { char err[ERRLEN], path[128]; Fcall thdr; Fid *f; File *nf; Rpc *rpc; ulong t; t = msec(); f = getfid(r->work.fid); if(f == 0) { reply(&r->work, &thdr, e[Ebadfid]); r->busy = 0; return; } makepath(path, f->f, r->work.name); f->fid = create(path, r->work.mode, r->work.perm); if(f->fid < 0) { errstr(err); reply(&r->work, &thdr, err); r->busy = 0; return; } nf = file(f->f, r->work.name); if(nf == 0) { errstr(err); reply(&r->work, &thdr, err); r->busy = 0; return; } f->mode = r->work.mode; f->f = nf; f->offset = 0; thdr.qid = f->f->qid; reply(&r->work, &thdr, 0); r->busy = 0; t = msec()-t; rpc = &stats->rpc[Tcreate]; rpc->time += t; if(t < rpc->loms) rpc->loms = t; if(t > rpc->hims) rpc->hims = t; } void Xremove(Fsrpc *r) { char err[ERRLEN], path[128]; Fcall thdr; Fid *f; ulong t; Rpc *rpc; t = msec(); f = getfid(r->work.fid); if(f == 0) { reply(&r->work, &thdr, e[Ebadfid]); r->busy = 0; return; } makepath(path, f->f, ""); DEBUG(2, "\tremove: %s\n", path); if(remove(path) < 0) { errstr(err); reply(&r->work, &thdr, err); freefid(r->work.fid); r->busy = 0; return; } f->f->inval = 1; if(f->fid >= 0) close(f->fid); freefid(r->work.fid); reply(&r->work, &thdr, 0); r->busy = 0; t = msec()-t; rpc = &stats->rpc[Tremove]; rpc->time += t; if(t < rpc->loms) rpc->loms = t; if(t > rpc->hims) rpc->hims = t; } void Xwstat(Fsrpc *r) { char err[ERRLEN], path[128]; Fcall thdr; Fid *f; int s; ulong t; Rpc *rpc; t = msec(); f = getfid(r->work.fid); if(f == 0) { reply(&r->work, &thdr, e[Ebadfid]); r->busy = 0; return; } if(f->fid >= 0) s = fwstat(f->fid, r->work.stat); else { makepath(path, f->f, ""); s = wstat(path, r->work.stat); } if(s < 0) { errstr(err); reply(&r->work, &thdr, err); } else reply(&r->work, &thdr, 0); r->busy = 0; t = msec()-t; rpc = &stats->rpc[Twstat]; rpc->time += t; if(t < rpc->loms) rpc->loms = t; if(t > rpc->hims) rpc->hims = t; } void Xclwalk(Fsrpc *r) { Fcall thdr; reply(&r->work, &thdr, "exportfs: no Tclwalk"); r->busy = 0; } void slave(Fsrpc *f) { Proc *p; int pid; static int nproc; for(;;) { for(p = Proclist; p; p = p->next) { if(p->busy == 0) { f->pid = p->pid; p->busy = 1; pid = rendezvous(p->pid, (ulong)f); if(pid != p->pid) fatal("rendezvous sync fail"); return; } } if(++nproc > MAXPROC) fatal("too many procs"); pid = rfork(RFPROC|RFMEM); if(pid < 0) fatal("rfork"); if(pid == 0) blockingslave(); p = malloc(sizeof(Proc)); if(p == 0) fatal("out of memory"); p->busy = 0; p->pid = pid; p->next = Proclist; Proclist = p; rendezvous(pid, (ulong)p); } } void blockingslave(void) { Proc *m; int pid; Fsrpc *p; Fcall thdr; notify(flushaction); pid = getpid(); m = (Proc*)rendezvous(pid, 0); for(;;) { p = (Fsrpc*)rendezvous(pid, pid); if((int)p == ~0) /* Interrupted */ continue; DEBUG(2, "\tslave: %d %F b %d p %d\n", pid, &p->work, p->busy, p->pid); if(p->flushtag != NOTAG) return; switch(p->work.type) { case Tread: slaveread(p); break; case Twrite: slavewrite(p); break; case Topen: slaveopen(p); break; default: reply(&p->work, &thdr, "exportfs: slave type error"); } if(p->flushtag != NOTAG) { p->work.type = Tflush; p->work.tag = p->flushtag; reply(&p->work, &thdr, 0); } p->busy = 0; m->busy = 0; } } void slaveopen(Fsrpc *p) { char err[ERRLEN], path[128]; Fcall *work, thdr; Fid *f; ulong t; Rpc *rpc; work = &p->work; t = msec(); f = getfid(work->fid); if(f == 0) { reply(work, &thdr, e[Ebadfid]); return; } if(f->fid >= 0) { close(f->fid); f->fid = -1; } makepath(path, f->f, ""); DEBUG(2, "\topen: %s %d\n", path, work->mode); p->canint = 1; if(p->flushtag != NOTAG) return; /* There is a race here I ignore because there are no locks */ f->fid = open(path, work->mode); p->canint = 0; if(f->fid < 0) { errstr(err); reply(work, &thdr, err); return; } DEBUG(2, "\topen: fd %d\n", f->fid); f->mode = work->mode; f->offset = 0; thdr.qid = f->f->qid; reply(work, &thdr, 0); t = msec()-t; rpc = &stats->rpc[Topen]; rpc->time += t; if(t < rpc->loms) rpc->loms = t; if(t > rpc->hims) rpc->hims = t; } void slaveread(Fsrpc *p) { char data[MAXFDATA], err[ERRLEN]; Fcall *work, thdr; Fid *f; int n, r; ulong t; Rpc *rpc; work = &p->work; t = msec(); f = getfid(work->fid); if(f == 0) { reply(work, &thdr, e[Ebadfid]); return; } if(work->offset != f->offset) fileseek(f, work->offset); n = (work->count > MAXFDATA) ? MAXFDATA : work->count; p->canint = 1; if(p->flushtag != NOTAG) return; r = read(f->fid, data, n); p->canint = 0; if(r < 0) { errstr(err); reply(work, &thdr, err); return; } DEBUG(2, "\tread: fd=%d %d bytes\n", f->fid, r); f->offset += r; thdr.data = data; thdr.count = r; stats->totread += r; f->nread++; f->bread += r; reply(work, &thdr, 0); t = msec()-t; rpc = &stats->rpc[Tread]; rpc->time += t; if(t < rpc->loms) rpc->loms = t; if(t > rpc->hims) rpc->hims = t; } void slavewrite(Fsrpc *p) { char err[ERRLEN]; Fcall *work, thdr; Fid *f; int n; ulong t; Rpc *rpc; work = &p->work; t = msec(); f = getfid(work->fid); if(f == 0) { reply(work, &thdr, e[Ebadfid]); return; } if(work->offset != f->offset) fileseek(f, work->offset); n = (work->count > MAXFDATA) ? MAXFDATA : work->count; p->canint = 1; if(p->flushtag != NOTAG) return; n = write(f->fid, work->data, n); p->canint = 0; if(n < 0) { errstr(err); reply(work, &thdr, err); return; } DEBUG(2, "\twrite: %d bytes fd=%d\n", n, f->fid); f->offset += n; thdr.count = n; f->nwrite++; f->bwrite += n; stats->totwrite += n; reply(work, &thdr, 0); t = msec()-t; rpc = &stats->rpc[Twrite]; rpc->time += t; if(t < rpc->loms) rpc->loms = t; if(t > rpc->hims) rpc->hims = t; } void fileseek(Fid *f, vlong offset) { char chunk[DIRCHUNK]; int n, nbytes, r; if(f->f->qid.path&CHDIR) { if(offset < f->offset) reopen(f); for(nbytes = offset - f->offset; nbytes; nbytes -= r) { n = (nbytes > DIRCHUNK) ? DIRCHUNK : nbytes; r = read(f->fid, chunk, n); if(r <= 0) { DEBUG(2,"\tdir seek error\n"); return; } f->offset += r; } } else f->offset = seek(f->fid, offset, 0); } void reopen(Fid *f) { USED(f); fatal("reopen"); } void flushaction(void *a, char *cause) { USED(a); if(strncmp(cause, "kill", 4) == 0) noted(NDFLT); noted(NCONT); }