fsys.c 13 KB

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