blob: 4d19e5e682e0c241caa2d8a51cca73c140fc4017 [file] [log] [blame]
rscd3df3082003-12-06 18:08:52 +00001#include <u.h>
rsc3e610792004-10-22 17:05:24 +00002#include <signal.h>
rscd3df3082003-12-06 18:08:52 +00003#include <libc.h>
rsc276bf4e2005-02-11 19:44:04 +00004#include <bio.h>
rscd3df3082003-12-06 18:08:52 +00005#include <fcall.h>
rsceb423b52005-01-04 22:09:51 +00006#include <9pclient.h>
rsc0d4243f2005-02-11 16:53:27 +00007#include <auth.h>
rscf063dad2004-02-29 22:42:33 +00008#include <thread.h>
rscd3df3082003-12-06 18:08:52 +00009
10char *addr;
11
12void
13usage(void)
14{
rsc011090f2005-03-21 17:27:25 +000015 fprint(2, "usage: 9p [-a address] [-A aname] cmd args...\n");
rscd3df3082003-12-06 18:08:52 +000016 fprint(2, "possible cmds:\n");
17 fprint(2, " read name\n");
rsc32f69c32003-12-11 17:48:38 +000018 fprint(2, " readfd name\n");
rsc276bf4e2005-02-11 19:44:04 +000019 fprint(2, " write [-l] name\n");
rsc32f69c32003-12-11 17:48:38 +000020 fprint(2, " writefd name\n");
rscd3df3082003-12-06 18:08:52 +000021 fprint(2, " stat name\n");
rsc05abefb2005-02-13 18:32:38 +000022 fprint(2, " rdwr name\n");
rsc1b404fe2006-02-05 15:34:46 +000023 fprint(2, " ls [-ldn] name\n");
rscd3df3082003-12-06 18:08:52 +000024 fprint(2, "without -a, name elem/path means /path on server unix!$ns/elem\n");
rsc38c10d12005-01-17 21:29:00 +000025 threadexitsall("usage");
rscd3df3082003-12-06 18:08:52 +000026}
27
rsc011090f2005-03-21 17:27:25 +000028char *aname;
rscd3df3082003-12-06 18:08:52 +000029void xread(int, char**);
30void xwrite(int, char**);
rsc32f69c32003-12-11 17:48:38 +000031void xreadfd(int, char**);
32void xwritefd(int, char**);
rscd3df3082003-12-06 18:08:52 +000033void xstat(int, char**);
34void xls(int, char**);
rsc05abefb2005-02-13 18:32:38 +000035void xrdwr(int, char**);
rsc1b404fe2006-02-05 15:34:46 +000036void xrm(int, char**);
37void xcreate(int, char**);
rsce29d0c82006-01-06 18:00:42 +000038void xcon(int, char**);
rscd3df3082003-12-06 18:08:52 +000039
40struct {
41 char *s;
42 void (*f)(int, char**);
43} cmds[] = {
rsce29d0c82006-01-06 18:00:42 +000044 "con", xcon,
rscd3df3082003-12-06 18:08:52 +000045 "read", xread,
46 "write", xwrite,
rsc32f69c32003-12-11 17:48:38 +000047 "readfd", xreadfd,
48 "writefd", xwritefd,
rscd3df3082003-12-06 18:08:52 +000049 "stat", xstat,
rsc05abefb2005-02-13 18:32:38 +000050 "rdwr", xrdwr,
rsc1d2f5612005-03-18 19:19:33 +000051 "ls", xls,
rsc1b404fe2006-02-05 15:34:46 +000052 "rm", xrm,
53 "create", xcreate,
rscd3df3082003-12-06 18:08:52 +000054};
55
56void
rscf063dad2004-02-29 22:42:33 +000057threadmain(int argc, char **argv)
rscd3df3082003-12-06 18:08:52 +000058{
59 char *cmd;
60 int i;
61
62 ARGBEGIN{
rsc011090f2005-03-21 17:27:25 +000063 case 'A':
64 aname = EARGF(usage());
65 break;
rscd3df3082003-12-06 18:08:52 +000066 case 'a':
67 addr = EARGF(usage());
rsc1d2f5612005-03-18 19:19:33 +000068 if(strchr(addr, '!') == nil)
69 addr = netmkaddr(addr, "tcp", "9fs");
rscd3df3082003-12-06 18:08:52 +000070 break;
rsc17cdbb92004-12-27 01:22:48 +000071 case 'D':
rsc0d4243f2005-02-11 16:53:27 +000072 chatty9pclient = 1;
73 break;
rscd3df3082003-12-06 18:08:52 +000074 default:
75 usage();
76 }ARGEND
77
rsc3e610792004-10-22 17:05:24 +000078 signal(SIGINT, SIG_DFL);
79
rscd3df3082003-12-06 18:08:52 +000080 if(argc < 1)
81 usage();
82
83 cmd = argv[0];
84 for(i=0; i<nelem(cmds); i++){
85 if(strcmp(cmds[i].s, cmd) == 0){
86 cmds[i].f(argc, argv);
rscf063dad2004-02-29 22:42:33 +000087 threadexitsall(0);
rscd3df3082003-12-06 18:08:52 +000088 }
89 }
90 usage();
91}
92
rsceb423b52005-01-04 22:09:51 +000093CFsys*
rscd3df3082003-12-06 18:08:52 +000094xparse(char *name, char **path)
95{
96 int fd;
rscd3df3082003-12-06 18:08:52 +000097 char *p;
rsceb423b52005-01-04 22:09:51 +000098 CFsys *fs;
rscd3df3082003-12-06 18:08:52 +000099
100 if(addr == nil){
101 p = strchr(name, '/');
102 if(p == nil)
103 p = name+strlen(name);
104 else
105 *p++ = 0;
106 *path = p;
rsc011090f2005-03-21 17:27:25 +0000107 fs = nsamount(name, aname);
rsc32f69c32003-12-11 17:48:38 +0000108 if(fs == nil)
109 sysfatal("mount: %r");
110 }else{
rscd3df3082003-12-06 18:08:52 +0000111 *path = name;
rsc32f69c32003-12-11 17:48:38 +0000112 if((fd = dial(addr, nil, nil, nil)) < 0)
113 sysfatal("dial: %r");
rsc011090f2005-03-21 17:27:25 +0000114 if((fs = fsamount(fd, aname)) == nil)
115 sysfatal("fsamount: %r");
rsc32f69c32003-12-11 17:48:38 +0000116 }
rscd3df3082003-12-06 18:08:52 +0000117 return fs;
118}
119
rsceb423b52005-01-04 22:09:51 +0000120CFid*
rscd3df3082003-12-06 18:08:52 +0000121xopen(char *name, int mode)
122{
rsceb423b52005-01-04 22:09:51 +0000123 CFid *fid;
124 CFsys *fs;
rscd3df3082003-12-06 18:08:52 +0000125
126 fs = xparse(name, &name);
127 fid = fsopen(fs, name, mode);
128 if(fid == nil)
129 sysfatal("fsopen %s: %r", name);
130 return fid;
131}
132
rsc32f69c32003-12-11 17:48:38 +0000133int
134xopenfd(char *name, int mode)
135{
rsceb423b52005-01-04 22:09:51 +0000136 CFsys *fs;
rsc32f69c32003-12-11 17:48:38 +0000137
138 fs = xparse(name, &name);
139 return fsopenfd(fs, name, mode);
140}
141
rscd3df3082003-12-06 18:08:52 +0000142void
143xread(int argc, char **argv)
144{
rsc05abefb2005-02-13 18:32:38 +0000145 char buf[4096];
rscd3df3082003-12-06 18:08:52 +0000146 int n;
rsceb423b52005-01-04 22:09:51 +0000147 CFid *fid;
rscd3df3082003-12-06 18:08:52 +0000148
149 ARGBEGIN{
150 default:
151 usage();
152 }ARGEND
153
154 if(argc != 1)
155 usage();
156
157 fid = xopen(argv[0], OREAD);
158 while((n = fsread(fid, buf, sizeof buf)) > 0)
159 write(1, buf, n);
rsc286237e2005-10-29 17:37:10 +0000160 fsclose(fid);
rscd3df3082003-12-06 18:08:52 +0000161 if(n < 0)
162 sysfatal("read error: %r");
rsc38c10d12005-01-17 21:29:00 +0000163 threadexitsall(0);
rscd3df3082003-12-06 18:08:52 +0000164}
165
166void
rsc32f69c32003-12-11 17:48:38 +0000167xreadfd(int argc, char **argv)
168{
rsc05abefb2005-02-13 18:32:38 +0000169 char buf[4096];
rsc32f69c32003-12-11 17:48:38 +0000170 int n;
171 int fd;
172
173 ARGBEGIN{
174 default:
175 usage();
176 }ARGEND
177
178 if(argc != 1)
179 usage();
180
181 fd = xopenfd(argv[0], OREAD);
182 while((n = read(fd, buf, sizeof buf)) > 0)
183 write(1, buf, n);
184 if(n < 0)
185 sysfatal("read error: %r");
rsc38c10d12005-01-17 21:29:00 +0000186 threadexitsall(0);
rsc32f69c32003-12-11 17:48:38 +0000187}
188
189void
rscd3df3082003-12-06 18:08:52 +0000190xwrite(int argc, char **argv)
191{
rsc05abefb2005-02-13 18:32:38 +0000192 char buf[4096];
rsc3e610792004-10-22 17:05:24 +0000193 int n, did;
rsceb423b52005-01-04 22:09:51 +0000194 CFid *fid;
rsc276bf4e2005-02-11 19:44:04 +0000195 Biobuf *b;
196 char *p;
197 int byline;
rscd3df3082003-12-06 18:08:52 +0000198
rsc276bf4e2005-02-11 19:44:04 +0000199 byline = 0;
rscd3df3082003-12-06 18:08:52 +0000200 ARGBEGIN{
rsc276bf4e2005-02-11 19:44:04 +0000201 case 'l':
202 byline = 1;
rsc05abefb2005-02-13 18:32:38 +0000203 break;
rscd3df3082003-12-06 18:08:52 +0000204 default:
205 usage();
206 }ARGEND
207
208 if(argc != 1)
209 usage();
210
rsc3e610792004-10-22 17:05:24 +0000211 did = 0;
rscd3df3082003-12-06 18:08:52 +0000212 fid = xopen(argv[0], OWRITE|OTRUNC);
rsc276bf4e2005-02-11 19:44:04 +0000213 if(byline){
214 n = 0;
215 b = malloc(sizeof *b);
216 if(b == nil)
217 sysfatal("out of memory");
218 Binit(b, 0, OREAD);
219 while((p = Brdstr(b, '\n', 0)) != nil){
220 n = strlen(p);
221 did = 1;
222 if(fswrite(fid, p, n) != n)
rsc05abefb2005-02-13 18:32:38 +0000223 fprint(2, "write: %r\n");
rsc276bf4e2005-02-11 19:44:04 +0000224 }
225 free(b);
226 }else{
227 while((n = read(0, buf, sizeof buf)) > 0){
228 did = 1;
229 if(fswrite(fid, buf, n) != n)
230 sysfatal("write error: %r");
231 }
rsc3e610792004-10-22 17:05:24 +0000232 }
233 if(n == 0 && !did){
234 if(fswrite(fid, buf, 0) != 0)
235 sysfatal("write error: %r");
236 }
rscd3df3082003-12-06 18:08:52 +0000237 if(n < 0)
238 sysfatal("read error: %r");
rsc286237e2005-10-29 17:37:10 +0000239 fsclose(fid);
rsc38c10d12005-01-17 21:29:00 +0000240 threadexitsall(0);
rscd3df3082003-12-06 18:08:52 +0000241}
242
243void
rsc32f69c32003-12-11 17:48:38 +0000244xwritefd(int argc, char **argv)
245{
rsc05abefb2005-02-13 18:32:38 +0000246 char buf[4096];
rsc32f69c32003-12-11 17:48:38 +0000247 int n;
248 int fd;
249
250 ARGBEGIN{
251 default:
252 usage();
253 }ARGEND
254
255 if(argc != 1)
256 usage();
257
258 fd = xopenfd(argv[0], OWRITE|OTRUNC);
259 while((n = read(0, buf, sizeof buf)) > 0)
260 if(write(fd, buf, n) != n)
261 sysfatal("write error: %r");
262 if(n < 0)
263 sysfatal("read error: %r");
rsc38c10d12005-01-17 21:29:00 +0000264 threadexitsall(0);
rsc32f69c32003-12-11 17:48:38 +0000265}
266
267void
rscd3df3082003-12-06 18:08:52 +0000268xstat(int argc, char **argv)
269{
270 Dir *d;
rsceb423b52005-01-04 22:09:51 +0000271 CFsys *fs;
272 char *name;
rscd3df3082003-12-06 18:08:52 +0000273
274 ARGBEGIN{
275 default:
276 usage();
277 }ARGEND
278
279 if(argc != 1)
280 usage();
281
rsceb423b52005-01-04 22:09:51 +0000282 name = argv[0];
283 fs = xparse(name, &name);
284 if((d = fsdirstat(fs, name)) == 0)
285 sysfatal("dirstat: %r");
rscd3df3082003-12-06 18:08:52 +0000286 fmtinstall('D', dirfmt);
287 fmtinstall('M', dirmodefmt);
288 print("%D\n", d);
rsc38c10d12005-01-17 21:29:00 +0000289 threadexitsall(0);
rscd3df3082003-12-06 18:08:52 +0000290}
rsc05abefb2005-02-13 18:32:38 +0000291
292void
293xrdwr(int argc, char **argv)
294{
295 char buf[4096];
296 int n;
297 CFid *fid;
298
299 ARGBEGIN{
300 default:
301 usage();
302 }ARGEND
303
304 if(argc != 1)
305 usage();
306
307 fid = xopen(argv[0], ORDWR);
308 for(;;){
rsc1bb8ccf2006-02-14 19:41:02 +0000309 fsseek(fid, 0, 0);
rsc05abefb2005-02-13 18:32:38 +0000310 if((n = fsread(fid, buf, sizeof buf)) < 0)
311 fprint(2, "read: %r\n");
312 else{
313 write(1, buf, n);
314 write(1, "\n", 1);
315 }
316 n = read(0, buf, sizeof buf);
317 if(n <= 0)
318 break;
319 if(buf[n-1] == '\n')
320 n--;
321 if(fswrite(fid, buf, n) != n)
322 fprint(2, "write: %r\n");
323 }
rsc286237e2005-10-29 17:37:10 +0000324 fsclose(fid);
rsc05abefb2005-02-13 18:32:38 +0000325 threadexitsall(0);
326}
rsc1d2f5612005-03-18 19:19:33 +0000327
rsce29d0c82006-01-06 18:00:42 +0000328void
rsc1b404fe2006-02-05 15:34:46 +0000329xcreate(int argc, char **argv)
330{
331 int i;
332 CFsys *fs;
333 CFid *fid;
334 char *p;
335
336 ARGBEGIN{
337 default:
338 usage();
339 }ARGEND
340
341 if(argc == 0)
342 usage();
343
344 for(i=0; i<argc; i++){
345 fs = xparse(argv[i], &p);
346 if((fid=fscreate(fs, p, OREAD, 0666)) == nil)
347 fprint(2, "create %s: %r\n", argv[i]);
348 else
349 fsclose(fid);
350 fsunmount(fs);
351 }
352}
353
354void
355xrm(int argc, char **argv)
356{
357 int i;
358 CFsys *fs;
359 char *p;
360
361 ARGBEGIN{
362 default:
363 usage();
364 }ARGEND
365
366 if(argc == 0)
367 usage();
368
369 for(i=0; i<argc; i++){
370 fs = xparse(argv[i], &p);
371 if(fsremove(fs, p) < 0)
372 fprint(2, "remove %s: %r\n", argv[i]);
373 fsunmount(fs);
374 }
375}
376
377void
rsce29d0c82006-01-06 18:00:42 +0000378rdcon(void *v)
379{
rsced0601e2006-01-06 18:03:09 +0000380 int n;
rsce29d0c82006-01-06 18:00:42 +0000381 char buf[4096];
382 CFid *fid;
383
384 fid = v;
385 for(;;){
386 n = read(0, buf, sizeof buf);
387 if(n <= 0)
388 threadexitsall(0);
rsced0601e2006-01-06 18:03:09 +0000389 if(buf[0] == 'R'-'A'+1)
390 threadexitsall(0);
rsce29d0c82006-01-06 18:00:42 +0000391 if(fswrite(fid, buf, n) != n)
392 fprint(2, "write: %r\n");
393 }
394}
395
396void
397xcon(int argc, char **argv)
398{
399 char buf[4096], *r, *w, *e;
400 int n, nocr;
401 CFid *fid;
402
403 nocr = 1;
404
405 ARGBEGIN{
406 case 'r':
407 nocr = 0;
408 break;
409 default:
410 usage();
411 }ARGEND
412
413 if(argc != 1)
414 usage();
415
416 fid = xopen(argv[0], ORDWR);
rsced0601e2006-01-06 18:03:09 +0000417 proccreate(rdcon, fid, 32768);
rsce29d0c82006-01-06 18:00:42 +0000418 for(;;){
rsced0601e2006-01-06 18:03:09 +0000419 n = fsread(fid, buf, sizeof buf);
rsce29d0c82006-01-06 18:00:42 +0000420 if(n <= 0)
421 threadexitsall(0);
422 if(nocr){
423 for(r=w=buf, e=buf+n; r<e; r++)
424 if(*r != '\r')
425 *w++ = *r;
426 n = w-buf;
427 }
428 if(write(1, buf, n) != n)
429 threadexitsall(0);
430 }
431 fsclose(fid);
432 threadexitsall(0);
433}
434
rsc1d2f5612005-03-18 19:19:33 +0000435static char *mon[] =
436{
437 "Jan", "Feb", "Mar", "Apr", "May", "Jun",
438 "Jul", "Aug", "Sep", "Oct", "Nov", "Dec"
439};
440
441
442int
443timefmt(Fmt *fmt)
444{
445 ulong u;
446 static ulong time0;
447 Tm *tm;
448
449 if(time0 == 0)
450 time0 = time(0);
451 u = va_arg(fmt->args, ulong);
452 tm = localtime(u);
453 if((long)(time0-u) < 6*30*86400)
454 return fmtprint(fmt, "%s %2d %02d:%02d",
455 mon[tm->mon], tm->mday, tm->hour, tm->min);
rscbcac59d2005-08-11 17:33:43 +0000456 return fmtprint(fmt, "%s %2d %5d",
457 mon[tm->mon], tm->mday, tm->year+1900);
rsc1d2f5612005-03-18 19:19:33 +0000458}
459
460static int
461dircmp(const void *va, const void *vb)
462{
463 Dir *a, *b;
464
465 a = (Dir*)va;
466 b = (Dir*)vb;
467 return strcmp(a->name, b->name);
468}
469
470void
471xls(int argc, char **argv)
472{
rsca919ad82005-08-10 18:54:14 +0000473 char *err, *name, *xname, *f[4], buf[4096];
rsc1b404fe2006-02-05 15:34:46 +0000474 int nf, i, j, l, sort;
rsc1d2f5612005-03-18 19:19:33 +0000475 int lflag, dflag, n, len[4];
476 Dir *d;
477 CFid *fid;
478 CFsys *fs;
rsca919ad82005-08-10 18:54:14 +0000479
480 err = nil;
rsc1b404fe2006-02-05 15:34:46 +0000481 sort = 0;
rsc1d2f5612005-03-18 19:19:33 +0000482 lflag = dflag = 0;
483 ARGBEGIN{
rsc1b404fe2006-02-05 15:34:46 +0000484 case 'n':
485 sort = 0;
486 break;
rsc1d2f5612005-03-18 19:19:33 +0000487 case 'l':
488 lflag = 1;
489 break;
490 case 'd':
491 dflag = 1;
492 break;
493 }ARGEND
494
495 fmtinstall('D', dirfmt);
496 fmtinstall('M', dirmodefmt);
497 quotefmtinstall();
498 fmtinstall('T', timefmt);
499
500 for(i=0; i<argc; i++){
501 name = argv[i];
502 fs = xparse(name, &xname);
503 if((d = fsdirstat(fs, xname)) == nil){
504 fprint(2, "dirstat %s: %r\n", name);
505 fsunmount(fs);
rsca919ad82005-08-10 18:54:14 +0000506 err = "errors";
rsc1d2f5612005-03-18 19:19:33 +0000507 continue;
508 }
509 if((d->mode&DMDIR) && !dflag){
510 if((fid = fsopen(fs, xname, OREAD)) == nil){
511 fprint(2, "open %s: %r\n", name);
512 fsunmount(fs);
513 free(d);
rsca919ad82005-08-10 18:54:14 +0000514 err = "errors";
rsc1d2f5612005-03-18 19:19:33 +0000515 continue;
516 }
517 free(d);
518 n = fsdirreadall(fid, &d);
519 fsclose(fid);
520 if(n < 0){
521 fprint(2, "dirreadall %s: %r\n", name);
522 fsunmount(fs);
rsca919ad82005-08-10 18:54:14 +0000523 err = "errors";
rsc1d2f5612005-03-18 19:19:33 +0000524 continue;
525 }
rsc1b404fe2006-02-05 15:34:46 +0000526 if(sort)
527 qsort(d, n, sizeof d[0], dircmp);
rsc1d2f5612005-03-18 19:19:33 +0000528 for(j=0; j<5; j++)
529 len[j] = 0;
530 for(i=0; i<n; i++){
531 d[i].type = 'M';
532 d[i].dev = 0;
533 snprint(buf, sizeof buf, "%d %s %s %lld",
534 d[i].dev, d[i].uid, d[i].gid, d[i].length);
535 nf = getfields(buf, f, 4, 0, " ");
536 for(j=0; j<4; j++){
537 l = strlen(f[j]);
538 if(l > len[j])
539 len[j] = l;
540 }
541 }
542 for(i=0; i<n; i++)
543 print("%M %C %*d %*s %*s %*lld %T %q\n",
544 d[i].mode, d[i].type, len[0], d[i].dev,
545 -len[1], d[i].uid, -len[2], d[i].gid,
546 len[3], d[i].length, d[i].mtime, d[i].name);
547
548 }else{
549 d->type = 'M';
550 d->dev = 0;
551 print("%M %C %d %s %s %lld %T %q\n",
552 d->mode, d->type, d->dev,
553 d->uid, d->gid, d->length, d->mtime, d->name);
554 }
555 free(d);
556 }
rsca919ad82005-08-10 18:54:14 +0000557 threadexitsall(err);
rsc1d2f5612005-03-18 19:19:33 +0000558}
559