fsys.c 12 KB

123456789101112131415161718192021222324252627282930313233343536373839404142434445464748495051525354555657585960616263646566676869707172737475767778798081828384858687888990919293949596979899100101102103104105106107108109110111112113114115116117118119120121122123124125126127128129130131132133134135136137138139140141142143144145146147148149150151152153154155156157158159160161162163164165166167168169170171172173174175176177178179180181182183184185186187188189190191192193194195196197198199200201202203204205206207208209210211212213214215216217218219220221222223224225226227228229230231232233234235236237238239240241242243244245246247248249250251252253254255256257258259260261262263264265266267268269270271272273274275276277278279280281282283284285286287288289290291292293294295296297298299300301302303304305306307308309310311312313314315316317318319320321322323324325326327328329330331332333334335336337338339340341342343344345346347348349350351352353354355356357358359360361362363364365366367368369370371372373374375376377378379380381382383384385386387388389390391392393394395396397398399400401402403404405406407408409410411412413414415416417418419420421422423424425426427428429430431432433434435436437438439440441442443444445446447448449450451452453454455456457458459460461462463464465466467468469470471472473474475476477478479480481482483484485486487488489490491492493494495496497498499500501502503504505506507508509510511512513514515516517518519520521522523524525526527528529530531532533534535536537538539540541542543544545546547548549550551552553554555556557558559560561562563564565566567568569570571572573574575576577578579580581582583584585586587588589590591592593594595596597598599600601602603604605606607608609610611612613614615616617618619620621622623624625626627628629630631632633634635636637638639640641642643644645646647648649650651652653654655656657658659660661662663664665666667668669670671672673674675676677678679680681682683684685686687688689690691692693694695696697698699700701702703704705706707708709710711712713714715716717718719720721722723724725726727728729730731732733
  1. #include <u.h>
  2. #include <libc.h>
  3. #include <draw.h>
  4. #include <thread.h>
  5. #include <cursor.h>
  6. #include <mouse.h>
  7. #include <keyboard.h>
  8. #include <frame.h>
  9. #include <fcall.h>
  10. #include <plumb.h>
  11. #include "dat.h"
  12. #include "fns.h"
  13. static int cfd;
  14. static int sfd;
  15. enum
  16. {
  17. Nhash = 16,
  18. DEBUG = 0
  19. };
  20. static Fid *fids[Nhash];
  21. Fid *newfid(int);
  22. static Xfid* fsysflush(Xfid*, Fid*);
  23. static Xfid* fsysauth(Xfid*, Fid*);
  24. static Xfid* fsysversion(Xfid*, Fid*);
  25. static Xfid* fsysattach(Xfid*, Fid*);
  26. static Xfid* fsyswalk(Xfid*, Fid*);
  27. static Xfid* fsysopen(Xfid*, Fid*);
  28. static Xfid* fsyscreate(Xfid*, Fid*);
  29. static Xfid* fsysread(Xfid*, Fid*);
  30. static Xfid* fsyswrite(Xfid*, Fid*);
  31. static Xfid* fsysclunk(Xfid*, Fid*);
  32. static Xfid* fsysremove(Xfid*, Fid*);
  33. static Xfid* fsysstat(Xfid*, Fid*);
  34. static Xfid* fsyswstat(Xfid*, Fid*);
  35. Xfid* (*fcall[Tmax])(Xfid*, Fid*) =
  36. {
  37. [Tflush] = fsysflush,
  38. [Tversion] = fsysversion,
  39. [Tauth] = fsysauth,
  40. [Tattach] = fsysattach,
  41. [Twalk] = fsyswalk,
  42. [Topen] = fsysopen,
  43. [Tcreate] = fsyscreate,
  44. [Tread] = fsysread,
  45. [Twrite] = fsyswrite,
  46. [Tclunk] = fsysclunk,
  47. [Tremove]= fsysremove,
  48. [Tstat] = fsysstat,
  49. [Twstat] = fsyswstat,
  50. };
  51. char Eperm[] = "permission denied";
  52. char Eexist[] = "file does not exist";
  53. char Enotdir[] = "not a directory";
  54. Dirtab dirtab[]=
  55. {
  56. { ".", QTDIR, Qdir, 0500|DMDIR },
  57. { "acme", QTDIR, Qacme, 0500|DMDIR },
  58. { "cons", QTFILE, Qcons, 0600 },
  59. { "consctl", QTFILE, Qconsctl, 0000 },
  60. { "draw", QTDIR, Qdraw, 0000|DMDIR }, /* to suppress graphics progs started in acme */
  61. { "editout", QTFILE, Qeditout, 0200 },
  62. { "index", QTFILE, Qindex, 0400 },
  63. { "label", QTFILE, Qlabel, 0600 },
  64. { "new", QTDIR, Qnew, 0500|DMDIR },
  65. { nil, }
  66. };
  67. Dirtab dirtabw[]=
  68. {
  69. { ".", QTDIR, Qdir, 0500|DMDIR },
  70. { "addr", QTFILE, QWaddr, 0600 },
  71. { "body", QTAPPEND, QWbody, 0600|DMAPPEND },
  72. { "ctl", QTFILE, QWctl, 0600 },
  73. { "data", QTFILE, QWdata, 0600 },
  74. { "editout", QTFILE, QWeditout, 0200 },
  75. { "event", QTFILE, QWevent, 0600 },
  76. { "rdsel", QTFILE, QWrdsel, 0400 },
  77. { "wrsel", QTFILE, QWwrsel, 0200 },
  78. { "tag", QTAPPEND, QWtag, 0600|DMAPPEND },
  79. { nil, }
  80. };
  81. typedef struct Mnt Mnt;
  82. struct Mnt
  83. {
  84. QLock;
  85. int id;
  86. Mntdir *md;
  87. };
  88. Mnt mnt;
  89. Xfid* respond(Xfid*, Fcall*, char*);
  90. int dostat(int, Dirtab*, uchar*, int, uint);
  91. uint getclock(void);
  92. char *user = "Wile E. Coyote";
  93. int clockfd;
  94. static int closing = 0;
  95. int messagesize = Maxblock+IOHDRSZ; /* good start */
  96. void fsysproc(void *);
  97. void
  98. fsysinit(void)
  99. {
  100. int p[2];
  101. int n, fd;
  102. char buf[256];
  103. if(pipe(p) < 0)
  104. error("can't create pipe");
  105. cfd = p[0];
  106. sfd = p[1];
  107. fmtinstall('F', fcallfmt);
  108. clockfd = open("/dev/time", OREAD|OCEXEC);
  109. fd = open("/dev/user", OREAD);
  110. if(fd >= 0){
  111. n = read(fd, buf, sizeof buf-1);
  112. if(n > 0){
  113. buf[n] = 0;
  114. user = estrdup(buf);
  115. }
  116. close(fd);
  117. }
  118. proccreate(fsysproc, nil, STACK);
  119. }
  120. void
  121. fsysproc(void *)
  122. {
  123. int n;
  124. Xfid *x;
  125. Fid *f;
  126. Fcall t;
  127. uchar *buf;
  128. x = nil;
  129. for(;;){
  130. buf = emalloc(messagesize+UTFmax); /* overflow for appending partial rune in xfidwrite */
  131. n = read9pmsg(sfd, buf, messagesize);
  132. if(n <= 0){
  133. if(closing)
  134. break;
  135. error("i/o error on server channel");
  136. }
  137. if(x == nil){
  138. sendp(cxfidalloc, nil);
  139. x = recvp(cxfidalloc);
  140. }
  141. x->buf = buf;
  142. if(convM2S(buf, n, x) != n)
  143. error("convert error in convM2S");
  144. if(DEBUG)
  145. fprint(2, "%F\n", &x->Fcall);
  146. if(fcall[x->type] == nil)
  147. x = respond(x, &t, "bad fcall type");
  148. else{
  149. if(x->type==Tversion || x->type==Tauth)
  150. f = nil;
  151. else
  152. f = newfid(x->fid);
  153. x->f = f;
  154. x = (*fcall[x->type])(x, f);
  155. }
  156. }
  157. }
  158. Mntdir*
  159. fsysaddid(Rune *dir, int ndir, Rune **incl, int nincl)
  160. {
  161. Mntdir *m;
  162. int id;
  163. qlock(&mnt);
  164. id = ++mnt.id;
  165. m = emalloc(sizeof *m);
  166. m->id = id;
  167. m->dir = dir;
  168. m->ref = 1; /* one for Command, one will be incremented in attach */
  169. m->ndir = ndir;
  170. m->next = mnt.md;
  171. m->incl = incl;
  172. m->nincl = nincl;
  173. mnt.md = m;
  174. qunlock(&mnt);
  175. return m;
  176. }
  177. void
  178. fsysdelid(Mntdir *idm)
  179. {
  180. Mntdir *m, *prev;
  181. int i;
  182. char buf[64];
  183. if(idm == nil)
  184. return;
  185. qlock(&mnt);
  186. if(--idm->ref > 0){
  187. qunlock(&mnt);
  188. return;
  189. }
  190. prev = nil;
  191. for(m=mnt.md; m; m=m->next){
  192. if(m == idm){
  193. if(prev)
  194. prev->next = m->next;
  195. else
  196. mnt.md = m->next;
  197. for(i=0; i<m->nincl; i++)
  198. free(m->incl[i]);
  199. free(m->incl);
  200. free(m->dir);
  201. free(m);
  202. qunlock(&mnt);
  203. return;
  204. }
  205. prev = m;
  206. }
  207. qunlock(&mnt);
  208. sprint(buf, "fsysdelid: can't find id %d\n", idm->id);
  209. sendp(cerr, estrdup(buf));
  210. }
  211. /*
  212. * Called only in exec.c:/^run(), from a different FD group
  213. */
  214. Mntdir*
  215. fsysmount(Rune *dir, int ndir, Rune **incl, int nincl)
  216. {
  217. char buf[256];
  218. Mntdir *m;
  219. /* close server side so don't hang if acme is half-exited */
  220. close(sfd);
  221. m = fsysaddid(dir, ndir, incl, nincl);
  222. sprint(buf, "%d", m->id);
  223. if(mount(cfd, -1, "/mnt/acme", MREPL, buf) < 0){
  224. fsysdelid(m);
  225. return nil;
  226. }
  227. close(cfd);
  228. bind("/mnt/acme", "/mnt/wsys", MREPL);
  229. if(bind("/mnt/acme", "/dev", MBEFORE) < 0){
  230. fsysdelid(m);
  231. return nil;
  232. }
  233. return m;
  234. }
  235. void
  236. fsysclose(void)
  237. {
  238. closing = 1;
  239. close(cfd);
  240. close(sfd);
  241. }
  242. Xfid*
  243. respond(Xfid *x, Fcall *t, char *err)
  244. {
  245. int n;
  246. if(err){
  247. t->type = Rerror;
  248. t->ename = err;
  249. }else
  250. t->type = x->type+1;
  251. t->fid = x->fid;
  252. t->tag = x->tag;
  253. if(x->buf == nil)
  254. x->buf = emalloc(messagesize);
  255. n = convS2M(t, x->buf, messagesize);
  256. if(n <= 0)
  257. error("convert error in convS2M");
  258. if(write(sfd, x->buf, n) != n)
  259. error("write error in respond");
  260. free(x->buf);
  261. x->buf = nil;
  262. if(DEBUG)
  263. fprint(2, "r: %F\n", t);
  264. return x;
  265. }
  266. static
  267. Xfid*
  268. fsysversion(Xfid *x, Fid*)
  269. {
  270. Fcall t;
  271. if(x->msize < 256)
  272. return respond(x, &t, "version: message size too small");
  273. messagesize = x->msize;
  274. t.msize = messagesize;
  275. if(strncmp(x->version, "9P2000", 6) != 0)
  276. return respond(x, &t, "unrecognized 9P version");
  277. t.version = "9P2000";
  278. return respond(x, &t, nil);
  279. }
  280. static
  281. Xfid*
  282. fsysauth(Xfid *x, Fid*)
  283. {
  284. return respond(x, nil, "acme: authentication not required");
  285. }
  286. static
  287. Xfid*
  288. fsysflush(Xfid *x, Fid*)
  289. {
  290. sendp(x->c, xfidflush);
  291. return nil;
  292. }
  293. static
  294. Xfid*
  295. fsysattach(Xfid *x, Fid *f)
  296. {
  297. Fcall t;
  298. int id;
  299. Mntdir *m;
  300. if(strcmp(x->uname, user) != 0)
  301. return respond(x, &t, Eperm);
  302. f->busy = TRUE;
  303. f->open = FALSE;
  304. f->qid.path = Qdir;
  305. f->qid.type = QTDIR;
  306. f->qid.vers = 0;
  307. f->dir = dirtab;
  308. f->nrpart = 0;
  309. f->w = nil;
  310. t.qid = f->qid;
  311. f->mntdir = nil;
  312. id = atoi(x->aname);
  313. qlock(&mnt);
  314. for(m=mnt.md; m; m=m->next)
  315. if(m->id == id){
  316. f->mntdir = m;
  317. m->ref++;
  318. break;
  319. }
  320. if(m == nil)
  321. sendp(cerr, estrdup("unknown id in attach"));
  322. qunlock(&mnt);
  323. return respond(x, &t, nil);
  324. }
  325. static
  326. Xfid*
  327. fsyswalk(Xfid *x, Fid *f)
  328. {
  329. Fcall t;
  330. int c, i, j, id;
  331. Qid q;
  332. uchar type;
  333. ulong path;
  334. Fid *nf;
  335. Dirtab *d, *dir;
  336. Window *w;
  337. char *err;
  338. nf = nil;
  339. w = nil;
  340. if(f->open)
  341. return respond(x, &t, "walk of open file");
  342. if(x->fid != x->newfid){
  343. nf = newfid(x->newfid);
  344. if(nf->busy)
  345. return respond(x, &t, "newfid already in use");
  346. nf->busy = TRUE;
  347. nf->open = FALSE;
  348. nf->mntdir = f->mntdir;
  349. if(f->mntdir)
  350. f->mntdir->ref++;
  351. nf->dir = f->dir;
  352. nf->qid = f->qid;
  353. nf->w = f->w;
  354. nf->nrpart = 0; /* not open, so must be zero */
  355. if(nf->w)
  356. incref(nf->w);
  357. f = nf; /* walk f */
  358. }
  359. t.nwqid = 0;
  360. err = nil;
  361. dir = nil;
  362. id = WIN(f->qid);
  363. q = f->qid;
  364. if(x->nwname > 0){
  365. for(i=0; i<x->nwname; i++){
  366. if((q.type & QTDIR) == 0){
  367. err = Enotdir;
  368. break;
  369. }
  370. if(strcmp(x->wname[i], "..") == 0){
  371. type = QTDIR;
  372. path = Qdir;
  373. id = 0;
  374. if(w){
  375. winclose(w);
  376. w = nil;
  377. }
  378. Accept:
  379. if(i == MAXWELEM){
  380. err = "name too long";
  381. break;
  382. }
  383. q.type = type;
  384. q.vers = 0;
  385. q.path = QID(id, path);
  386. t.wqid[t.nwqid++] = q;
  387. continue;
  388. }
  389. /* is it a numeric name? */
  390. for(j=0; (c=x->wname[i][j]); j++)
  391. if(c<'0' || '9'<c)
  392. goto Regular;
  393. /* yes: it's a directory */
  394. if(w) /* name has form 27/23; get out before losing w */
  395. break;
  396. id = atoi(x->wname[i]);
  397. qlock(&row);
  398. w = lookid(id, FALSE);
  399. if(w == nil){
  400. qunlock(&row);
  401. break;
  402. }
  403. incref(w); /* we'll drop reference at end if there's an error */
  404. path = Qdir;
  405. type = QTDIR;
  406. qunlock(&row);
  407. dir = dirtabw;
  408. goto Accept;
  409. Regular:
  410. // if(FILE(f->qid) == Qacme) /* empty directory */
  411. // break;
  412. if(strcmp(x->wname[i], "new") == 0){
  413. if(w)
  414. error("w set in walk to new");
  415. sendp(cnewwindow, nil); /* signal newwindowthread */
  416. w = recvp(cnewwindow); /* receive new window */
  417. incref(w);
  418. type = QTDIR;
  419. path = QID(w->id, Qdir);
  420. id = w->id;
  421. dir = dirtabw;
  422. goto Accept;
  423. }
  424. if(id == 0)
  425. d = dirtab;
  426. else
  427. d = dirtabw;
  428. d++; /* skip '.' */
  429. for(; d->name; d++)
  430. if(strcmp(x->wname[i], d->name) == 0){
  431. path = d->qid;
  432. type = d->type;
  433. dir = d;
  434. goto Accept;
  435. }
  436. break; /* file not found */
  437. }
  438. if(i==0 && err == nil)
  439. err = Eexist;
  440. }
  441. if(err!=nil || t.nwqid<x->nwname){
  442. if(nf){
  443. nf->busy = FALSE;
  444. fsysdelid(nf->mntdir);
  445. }
  446. }else if(t.nwqid == x->nwname){
  447. if(w){
  448. f->w = w;
  449. w = nil; /* don't drop the reference */
  450. }
  451. if(dir)
  452. f->dir = dir;
  453. f->qid = q;
  454. }
  455. if(w != nil)
  456. winclose(w);
  457. return respond(x, &t, err);
  458. }
  459. static
  460. Xfid*
  461. fsysopen(Xfid *x, Fid *f)
  462. {
  463. Fcall t;
  464. int m;
  465. /* can't truncate anything, so just disregard */
  466. x->mode &= ~(OTRUNC|OCEXEC);
  467. /* can't execute or remove anything */
  468. if(x->mode==OEXEC || (x->mode&ORCLOSE))
  469. goto Deny;
  470. switch(x->mode){
  471. default:
  472. goto Deny;
  473. case OREAD:
  474. m = 0400;
  475. break;
  476. case OWRITE:
  477. m = 0200;
  478. break;
  479. case ORDWR:
  480. m = 0600;
  481. break;
  482. }
  483. if(((f->dir->perm&~(DMDIR|DMAPPEND))&m) != m)
  484. goto Deny;
  485. sendp(x->c, xfidopen);
  486. return nil;
  487. Deny:
  488. return respond(x, &t, Eperm);
  489. }
  490. static
  491. Xfid*
  492. fsyscreate(Xfid *x, Fid*)
  493. {
  494. Fcall t;
  495. return respond(x, &t, Eperm);
  496. }
  497. static
  498. int
  499. idcmp(void *a, void *b)
  500. {
  501. return *(int*)a - *(int*)b;
  502. }
  503. static
  504. Xfid*
  505. fsysread(Xfid *x, Fid *f)
  506. {
  507. Fcall t;
  508. uchar *b;
  509. int i, id, n, o, e, j, k, *ids, nids;
  510. Dirtab *d, dt;
  511. Column *c;
  512. uint clock, len;
  513. char buf[16];
  514. if(f->qid.type & QTDIR){
  515. if(FILE(f->qid) == Qacme){ /* empty dir */
  516. t.data = nil;
  517. t.count = 0;
  518. respond(x, &t, nil);
  519. return x;
  520. }
  521. o = x->offset;
  522. e = x->offset+x->count;
  523. clock = getclock();
  524. b = emalloc(messagesize);
  525. id = WIN(f->qid);
  526. n = 0;
  527. if(id > 0)
  528. d = dirtabw;
  529. else
  530. d = dirtab;
  531. d++; /* first entry is '.' */
  532. for(i=0; d->name!=nil && i<e; i+=len){
  533. len = dostat(WIN(x->f->qid), d, b+n, x->count-n, clock);
  534. if(len <= BIT16SZ)
  535. break;
  536. if(i >= o)
  537. n += len;
  538. d++;
  539. }
  540. if(id == 0){
  541. qlock(&row);
  542. nids = 0;
  543. ids = nil;
  544. for(j=0; j<row.ncol; j++){
  545. c = row.col[j];
  546. for(k=0; k<c->nw; k++){
  547. ids = realloc(ids, (nids+1)*sizeof(int));
  548. ids[nids++] = c->w[k]->id;
  549. }
  550. }
  551. qunlock(&row);
  552. qsort(ids, nids, sizeof ids[0], idcmp);
  553. j = 0;
  554. dt.name = buf;
  555. for(; j<nids && i<e; i+=len){
  556. k = ids[j];
  557. sprint(dt.name, "%d", k);
  558. dt.qid = QID(k, Qdir);
  559. dt.type = QTDIR;
  560. dt.perm = DMDIR|0700;
  561. len = dostat(k, &dt, b+n, x->count-n, clock);
  562. if(len == 0)
  563. break;
  564. if(i >= o)
  565. n += len;
  566. j++;
  567. }
  568. free(ids);
  569. }
  570. t.data = (char*)b;
  571. t.count = n;
  572. respond(x, &t, nil);
  573. free(b);
  574. return x;
  575. }
  576. sendp(x->c, xfidread);
  577. return nil;
  578. }
  579. static
  580. Xfid*
  581. fsyswrite(Xfid *x, Fid*)
  582. {
  583. sendp(x->c, xfidwrite);
  584. return nil;
  585. }
  586. static
  587. Xfid*
  588. fsysclunk(Xfid *x, Fid *f)
  589. {
  590. fsysdelid(f->mntdir);
  591. sendp(x->c, xfidclose);
  592. return nil;
  593. }
  594. static
  595. Xfid*
  596. fsysremove(Xfid *x, Fid*)
  597. {
  598. Fcall t;
  599. return respond(x, &t, Eperm);
  600. }
  601. static
  602. Xfid*
  603. fsysstat(Xfid *x, Fid *f)
  604. {
  605. Fcall t;
  606. t.stat = emalloc(messagesize-IOHDRSZ);
  607. t.nstat = dostat(WIN(x->f->qid), f->dir, t.stat, messagesize-IOHDRSZ, getclock());
  608. x = respond(x, &t, nil);
  609. free(t.stat);
  610. return x;
  611. }
  612. static
  613. Xfid*
  614. fsyswstat(Xfid *x, Fid*)
  615. {
  616. Fcall t;
  617. return respond(x, &t, Eperm);
  618. }
  619. Fid*
  620. newfid(int fid)
  621. {
  622. Fid *f, *ff, **fh;
  623. ff = nil;
  624. fh = &fids[fid&(Nhash-1)];
  625. for(f=*fh; f; f=f->next)
  626. if(f->fid == fid)
  627. return f;
  628. else if(ff==nil && f->busy==FALSE)
  629. ff = f;
  630. if(ff){
  631. ff->fid = fid;
  632. return ff;
  633. }
  634. f = emalloc(sizeof *f);
  635. f->fid = fid;
  636. f->next = *fh;
  637. *fh = f;
  638. return f;
  639. }
  640. uint
  641. getclock()
  642. {
  643. char buf[32];
  644. buf[0] = '\0';
  645. pread(clockfd, buf, sizeof buf, 0);
  646. return atoi(buf);
  647. }
  648. int
  649. dostat(int id, Dirtab *dir, uchar *buf, int nbuf, uint clock)
  650. {
  651. Dir d;
  652. d.qid.path = QID(id, dir->qid);
  653. d.qid.vers = 0;
  654. d.qid.type = dir->type;
  655. d.mode = dir->perm;
  656. d.length = 0; /* would be nice to do better */
  657. d.name = dir->name;
  658. d.uid = user;
  659. d.gid = user;
  660. d.muid = user;
  661. d.atime = clock;
  662. d.mtime = clock;
  663. return convD2M(&d, buf, nbuf);
  664. }