| #include <u.h> |
| #include <signal.h> |
| #include <libc.h> |
| #include <bio.h> |
| #include <fcall.h> |
| #include <9pclient.h> |
| #include <auth.h> |
| #include <thread.h> |
| |
| char *addr; |
| |
| void |
| usage(void) |
| { |
| fprint(2, "usage: 9p [-n] [-a address] [-A aname] cmd args...\n"); |
| fprint(2, "possible cmds:\n"); |
| fprint(2, " read name\n"); |
| fprint(2, " readfd name\n"); |
| fprint(2, " write [-l] name\n"); |
| fprint(2, " writefd name\n"); |
| fprint(2, " stat name\n"); |
| fprint(2, " rdwr name\n"); |
| fprint(2, " ls [-ldn] name\n"); |
| fprint(2, "without -a, name elem/path means /path on server unix!$ns/elem\n"); |
| threadexitsall("usage"); |
| } |
| |
| CFsys *(*nsmnt)(char*, char*) = nsamount; |
| CFsys *(*fsmnt)(int, char*) = fsamount; |
| |
| char *aname; |
| void xread(int, char**); |
| void xwrite(int, char**); |
| void xreadfd(int, char**); |
| void xwritefd(int, char**); |
| void xstat(int, char**); |
| void xls(int, char**); |
| void xrdwr(int, char**); |
| void xrm(int, char**); |
| void xcreate(int, char**); |
| void xcon(int, char**); |
| |
| struct { |
| char *s; |
| void (*f)(int, char**); |
| } cmds[] = { |
| "con", xcon, |
| "read", xread, |
| "write", xwrite, |
| "readfd", xreadfd, |
| "writefd", xwritefd, |
| "stat", xstat, |
| "rdwr", xrdwr, |
| "ls", xls, |
| "rm", xrm, |
| "create", xcreate, |
| }; |
| |
| void |
| threadmain(int argc, char **argv) |
| { |
| char *cmd; |
| int i; |
| |
| ARGBEGIN{ |
| case 'A': |
| aname = EARGF(usage()); |
| break; |
| case 'a': |
| addr = EARGF(usage()); |
| if(strchr(addr, '!') == nil) |
| addr = netmkaddr(addr, "tcp", "9fs"); |
| break; |
| case 'n': |
| nsmnt = nsmount; |
| fsmnt = fsmount; |
| break; |
| case 'D': |
| chatty9pclient = 1; |
| break; |
| default: |
| usage(); |
| }ARGEND |
| |
| signal(SIGINT, SIG_DFL); |
| |
| if(argc < 1) |
| usage(); |
| |
| cmd = argv[0]; |
| for(i=0; i<nelem(cmds); i++){ |
| if(strcmp(cmds[i].s, cmd) == 0){ |
| cmds[i].f(argc, argv); |
| threadexitsall(0); |
| } |
| } |
| usage(); |
| } |
| |
| CFsys* |
| xparse(char *name, char **path) |
| { |
| int fd; |
| char *p; |
| CFsys *fs; |
| |
| if(addr == nil){ |
| p = strchr(name, '/'); |
| if(p == nil) |
| p = name+strlen(name); |
| else |
| *p++ = 0; |
| *path = p; |
| fs = nsmnt(name, aname); |
| if(fs == nil) |
| sysfatal("mount: %r"); |
| }else{ |
| *path = name; |
| if((fd = dial(addr, nil, nil, nil)) < 0) |
| sysfatal("dial: %r"); |
| if((fs = fsmnt(fd, aname)) == nil) |
| sysfatal("mount: %r"); |
| } |
| return fs; |
| } |
| |
| CFid* |
| xopen(char *name, int mode) |
| { |
| CFid *fid; |
| CFsys *fs; |
| |
| fs = xparse(name, &name); |
| fid = fsopen(fs, name, mode); |
| if(fid == nil) |
| sysfatal("fsopen %s: %r", name); |
| return fid; |
| } |
| |
| int |
| xopenfd(char *name, int mode) |
| { |
| CFsys *fs; |
| |
| fs = xparse(name, &name); |
| return fsopenfd(fs, name, mode); |
| } |
| |
| void |
| xread(int argc, char **argv) |
| { |
| char buf[4096]; |
| int n; |
| CFid *fid; |
| |
| ARGBEGIN{ |
| default: |
| usage(); |
| }ARGEND |
| |
| if(argc != 1) |
| usage(); |
| |
| fid = xopen(argv[0], OREAD); |
| while((n = fsread(fid, buf, sizeof buf)) > 0) |
| if(write(1, buf, n) < 0) |
| sysfatal("write error: %r"); |
| fsclose(fid); |
| if(n < 0) |
| sysfatal("read error: %r"); |
| threadexitsall(0); |
| } |
| |
| void |
| xreadfd(int argc, char **argv) |
| { |
| char buf[4096]; |
| int n; |
| int fd; |
| |
| ARGBEGIN{ |
| default: |
| usage(); |
| }ARGEND |
| |
| if(argc != 1) |
| usage(); |
| |
| fd = xopenfd(argv[0], OREAD); |
| while((n = read(fd, buf, sizeof buf)) > 0) |
| if(write(1, buf, n) < 0) |
| sysfatal("write error: %r"); |
| if(n < 0) |
| sysfatal("read error: %r"); |
| threadexitsall(0); |
| } |
| |
| void |
| xwrite(int argc, char **argv) |
| { |
| char buf[4096]; |
| int n, did; |
| CFid *fid; |
| Biobuf *b; |
| char *p; |
| int byline; |
| |
| byline = 0; |
| ARGBEGIN{ |
| case 'l': |
| byline = 1; |
| break; |
| default: |
| usage(); |
| }ARGEND |
| |
| if(argc != 1) |
| usage(); |
| |
| did = 0; |
| fid = xopen(argv[0], OWRITE|OTRUNC); |
| if(byline){ |
| n = 0; |
| b = malloc(sizeof *b); |
| if(b == nil) |
| sysfatal("out of memory"); |
| Binit(b, 0, OREAD); |
| while((p = Brdstr(b, '\n', 0)) != nil){ |
| n = strlen(p); |
| did = 1; |
| if(fswrite(fid, p, n) != n) |
| fprint(2, "write: %r\n"); |
| } |
| free(b); |
| }else{ |
| while((n = read(0, buf, sizeof buf)) > 0){ |
| did = 1; |
| if(fswrite(fid, buf, n) != n) |
| sysfatal("write error: %r"); |
| } |
| } |
| if(n == 0 && !did){ |
| if(fswrite(fid, buf, 0) != 0) |
| sysfatal("write error: %r"); |
| } |
| if(n < 0) |
| sysfatal("read error: %r"); |
| fsclose(fid); |
| threadexitsall(0); |
| } |
| |
| void |
| xwritefd(int argc, char **argv) |
| { |
| char buf[4096]; |
| int n; |
| int fd; |
| |
| ARGBEGIN{ |
| default: |
| usage(); |
| }ARGEND |
| |
| if(argc != 1) |
| usage(); |
| |
| fd = xopenfd(argv[0], OWRITE|OTRUNC); |
| while((n = read(0, buf, sizeof buf)) > 0) |
| if(write(fd, buf, n) != n) |
| sysfatal("write error: %r"); |
| if(n < 0) |
| sysfatal("read error: %r"); |
| threadexitsall(0); |
| } |
| |
| void |
| xstat(int argc, char **argv) |
| { |
| Dir *d; |
| CFsys *fs; |
| char *name; |
| |
| ARGBEGIN{ |
| default: |
| usage(); |
| }ARGEND |
| |
| if(argc != 1) |
| usage(); |
| |
| name = argv[0]; |
| fs = xparse(name, &name); |
| if((d = fsdirstat(fs, name)) == 0) |
| sysfatal("dirstat: %r"); |
| fmtinstall('D', dirfmt); |
| fmtinstall('M', dirmodefmt); |
| print("%D\n", d); |
| threadexitsall(0); |
| } |
| |
| void |
| xrdwr(int argc, char **argv) |
| { |
| char buf[4096]; |
| int n; |
| CFid *fid; |
| |
| ARGBEGIN{ |
| default: |
| usage(); |
| }ARGEND |
| |
| if(argc != 1) |
| usage(); |
| |
| fid = xopen(argv[0], ORDWR); |
| for(;;){ |
| fsseek(fid, 0, 0); |
| if((n = fsread(fid, buf, sizeof buf)) < 0) |
| fprint(2, "read: %r\n"); |
| else{ |
| if(write(1, buf, n) < 0 || write(1, "\n", 1) < 0) |
| sysfatal("write error: %r"); |
| } |
| n = read(0, buf, sizeof buf); |
| if(n <= 0) |
| break; |
| if(buf[n-1] == '\n') |
| n--; |
| if(fswrite(fid, buf, n) != n) |
| fprint(2, "write: %r\n"); |
| } |
| fsclose(fid); |
| threadexitsall(0); |
| } |
| |
| void |
| xcreate(int argc, char **argv) |
| { |
| int i; |
| CFsys *fs; |
| CFid *fid; |
| char *p; |
| |
| ARGBEGIN{ |
| default: |
| usage(); |
| }ARGEND |
| |
| if(argc == 0) |
| usage(); |
| |
| for(i=0; i<argc; i++){ |
| fs = xparse(argv[i], &p); |
| if((fid=fscreate(fs, p, OREAD, 0666)) == nil) |
| fprint(2, "create %s: %r\n", argv[i]); |
| else |
| fsclose(fid); |
| fsunmount(fs); |
| } |
| } |
| |
| void |
| xrm(int argc, char **argv) |
| { |
| int i; |
| CFsys *fs; |
| char *p; |
| |
| ARGBEGIN{ |
| default: |
| usage(); |
| }ARGEND |
| |
| if(argc == 0) |
| usage(); |
| |
| for(i=0; i<argc; i++){ |
| fs = xparse(argv[i], &p); |
| if(fsremove(fs, p) < 0) |
| fprint(2, "remove %s: %r\n", argv[i]); |
| fsunmount(fs); |
| } |
| } |
| |
| void |
| rdcon(void *v) |
| { |
| int n; |
| char buf[4096]; |
| CFid *fid; |
| |
| fid = v; |
| for(;;){ |
| n = read(0, buf, sizeof buf); |
| if(n <= 0) |
| threadexitsall(0); |
| if(buf[0] == 'R'-'A'+1) |
| threadexitsall(0); |
| if(fswrite(fid, buf, n) != n) |
| fprint(2, "write: %r\n"); |
| } |
| } |
| |
| void |
| xcon(int argc, char **argv) |
| { |
| char buf[4096], *r, *w, *e; |
| int n, nocr; |
| CFid *fid; |
| |
| nocr = 1; |
| |
| ARGBEGIN{ |
| case 'r': |
| nocr = 0; |
| break; |
| default: |
| usage(); |
| }ARGEND |
| |
| if(argc != 1) |
| usage(); |
| |
| fid = xopen(argv[0], ORDWR); |
| proccreate(rdcon, fid, 32768); |
| for(;;){ |
| n = fsread(fid, buf, sizeof buf); |
| if(n <= 0) |
| threadexitsall(0); |
| if(nocr){ |
| for(r=w=buf, e=buf+n; r<e; r++) |
| if(*r != '\r') |
| *w++ = *r; |
| n = w-buf; |
| } |
| if(write(1, buf, n) != n) |
| threadexitsall(0); |
| } |
| fsclose(fid); |
| threadexitsall(0); |
| } |
| |
| static char *mon[] = |
| { |
| "Jan", "Feb", "Mar", "Apr", "May", "Jun", |
| "Jul", "Aug", "Sep", "Oct", "Nov", "Dec" |
| }; |
| |
| |
| int |
| timefmt(Fmt *fmt) |
| { |
| ulong u; |
| static ulong time0; |
| Tm *tm; |
| |
| if(time0 == 0) |
| time0 = time(0); |
| u = va_arg(fmt->args, ulong); |
| tm = localtime(u); |
| if((long)(time0-u) < 6*30*86400) |
| return fmtprint(fmt, "%s %2d %02d:%02d", |
| mon[tm->mon], tm->mday, tm->hour, tm->min); |
| return fmtprint(fmt, "%s %2d %5d", |
| mon[tm->mon], tm->mday, tm->year+1900); |
| } |
| |
| static int |
| dircmp(const void *va, const void *vb) |
| { |
| Dir *a, *b; |
| |
| a = (Dir*)va; |
| b = (Dir*)vb; |
| return strcmp(a->name, b->name); |
| } |
| |
| static int |
| timecmp(const void *va, const void *vb) |
| { |
| Dir *a, *b; |
| |
| a = (Dir*)va; |
| b = (Dir*)vb; |
| if(a->mtime < b->mtime) |
| return -1; |
| else if(a->mtime > b->mtime) |
| return 1; |
| else |
| return 0; |
| } |
| |
| char *dot[] = { "." }; |
| |
| void |
| xls(int argc, char **argv) |
| { |
| char *err, *name, *xname, *f[4], buf[4096]; |
| int i, j, l, sort; |
| int lflag, dflag, tflag, n, len[4]; |
| Dir *d; |
| CFid *fid; |
| CFsys *fs; |
| |
| err = nil; |
| sort = 1; |
| lflag = dflag = tflag = 0; |
| ARGBEGIN{ |
| case 'n': |
| sort = 0; |
| break; |
| case 'l': |
| lflag = 1; |
| break; |
| case 'd': |
| dflag = 1; |
| break; |
| case 't': |
| tflag = 1; |
| break; |
| }ARGEND |
| |
| fmtinstall('D', dirfmt); |
| fmtinstall('M', dirmodefmt); |
| quotefmtinstall(); |
| fmtinstall('T', timefmt); |
| |
| if(argc == 0){ |
| argv = dot; |
| argc = 1; |
| } |
| for(i=0; i<argc; i++){ |
| name = argv[i]; |
| fs = xparse(name, &xname); |
| if((d = fsdirstat(fs, xname)) == nil){ |
| fprint(2, "dirstat %s: %r\n", name); |
| fsunmount(fs); |
| err = "errors"; |
| continue; |
| } |
| if((d->mode&DMDIR) && !dflag){ |
| if((fid = fsopen(fs, xname, OREAD)) == nil){ |
| fprint(2, "open %s: %r\n", name); |
| fsunmount(fs); |
| free(d); |
| err = "errors"; |
| continue; |
| } |
| free(d); |
| n = fsdirreadall(fid, &d); |
| fsclose(fid); |
| if(n < 0){ |
| fprint(2, "dirreadall %s: %r\n", name); |
| fsunmount(fs); |
| err = "errors"; |
| continue; |
| } |
| if(sort){ |
| if(tflag) |
| qsort(d, n, sizeof d[0], timecmp); |
| else |
| qsort(d, n, sizeof d[0], dircmp); |
| } |
| for(j=0; j<4; j++) |
| len[j] = 0; |
| for(i=0; i<n; i++){ |
| d[i].type = 'M'; |
| d[i].dev = 0; |
| snprint(buf, sizeof buf, "%d %s %s %lld", |
| d[i].dev, d[i].uid, d[i].gid, d[i].length); |
| getfields(buf, f, 4, 0, " "); |
| for(j=0; j<4; j++){ |
| l = strlen(f[j]); |
| if(l > len[j]) |
| len[j] = l; |
| } |
| } |
| for(i=0; i<n; i++){ |
| if(!lflag){ |
| print("%q\n", d[i].name); |
| continue; |
| } |
| print("%M %C %*d %*s %*s %*lld %T %q\n", |
| d[i].mode, d[i].type, len[0], d[i].dev, |
| -len[1], d[i].uid, -len[2], d[i].gid, |
| len[3], d[i].length, d[i].mtime, d[i].name); |
| } |
| }else{ |
| d->type = 'M'; |
| d->dev = 0; |
| if(lflag) |
| print("%M %C %d %s %s %lld %T %q\n", |
| d->mode, d->type, d->dev, |
| d->uid, d->gid, d->length, d->mtime, d->name); |
| else |
| print("%q\n", d->name); |
| } |
| free(d); |
| } |
| threadexitsall(err); |
| } |
| |