consolefs.c 21 KB

123456789101112131415161718192021222324252627282930313233343536373839404142434445464748495051525354555657585960616263646566676869707172737475767778798081828384858687888990919293949596979899100101102103104105106107108109110111112113114115116117118119120121122123124125126127128129130131132133134135136137138139140141142143144145146147148149150151152153154155156157158159160161162163164165166167168169170171172173174175176177178179180181182183184185186187188189190191192193194195196197198199200201202203204205206207208209210211212213214215216217218219220221222223224225226227228229230231232233234235236237238239240241242243244245246247248249250251252253254255256257258259260261262263264265266267268269270271272273274275276277278279280281282283284285286287288289290291292293294295296297298299300301302303304305306307308309310311312313314315316317318319320321322323324325326327328329330331332333334335336337338339340341342343344345346347348349350351352353354355356357358359360361362363364365366367368369370371372373374375376377378379380381382383384385386387388389390391392393394395396397398399400401402403404405406407408409410411412413414415416417418419420421422423424425426427428429430431432433434435436437438439440441442443444445446447448449450451452453454455456457458459460461462463464465466467468469470471472473474475476477478479480481482483484485486487488489490491492493494495496497498499500501502503504505506507508509510511512513514515516517518519520521522523524525526527528529530531532533534535536537538539540541542543544545546547548549550551552553554555556557558559560561562563564565566567568569570571572573574575576577578579580581582583584585586587588589590591592593594595596597598599600601602603604605606607608609610611612613614615616617618619620621622623624625626627628629630631632633634635636637638639640641642643644645646647648649650651652653654655656657658659660661662663664665666667668669670671672673674675676677678679680681682683684685686687688689690691692693694695696697698699700701702703704705706707708709710711712713714715716717718719720721722723724725726727728729730731732733734735736737738739740741742743744745746747748749750751752753754755756757758759760761762763764765766767768769770771772773774775776777778779780781782783784785786787788789790791792793794795796797798799800801802803804805806807808809810811812813814815816817818819820821822823824825826827828829830831832833834835836837838839840841842843844845846847848849850851852853854855856857858859860861862863864865866867868869870871872873874875876877878879880881882883884885886887888889890891892893894895896897898899900901902903904905906907908909910911912913914915916917918919920921922923924925926927928929930931932933934935936937938939940941942943944945946947948949950951952953954955956957958959960961962963964965966967968969970971972973974975976977978979980981982983984985986987988989990991992993994995996997998999100010011002100310041005100610071008100910101011101210131014101510161017101810191020102110221023102410251026102710281029103010311032103310341035103610371038103910401041104210431044104510461047104810491050105110521053105410551056105710581059106010611062106310641065106610671068106910701071107210731074107510761077107810791080108110821083108410851086108710881089109010911092109310941095109610971098109911001101110211031104110511061107110811091110111111121113111411151116111711181119112011211122112311241125112611271128112911301131113211331134113511361137113811391140114111421143114411451146114711481149115011511152115311541155115611571158115911601161116211631164116511661167116811691170117111721173117411751176117711781179118011811182118311841185118611871188118911901191119211931194119511961197119811991200120112021203120412051206120712081209121012111212121312141215121612171218121912201221122212231224122512261227122812291230123112321233123412351236123712381239124012411242124312441245124612471248124912501251125212531254125512561257125812591260126112621263126412651266126712681269127012711272127312741275127612771278127912801281128212831284128512861287
  1. #include <u.h>
  2. #include <libc.h>
  3. #include <auth.h>
  4. #include <fcall.h>
  5. #include <bio.h>
  6. #include <ndb.h>
  7. #include <thread.h>
  8. /*
  9. * This fs presents a 1 level file system. It contains
  10. * up to three files per console (xxx and xxxctl and xxxstat)
  11. */
  12. typedef struct Console Console;
  13. typedef struct Fid Fid;
  14. typedef struct Request Request;
  15. typedef struct Reqlist Reqlist;
  16. typedef struct Fs Fs;
  17. enum
  18. {
  19. /* last 5 bits of qid.path */
  20. Textern= 0, /* fake parent of top level */
  21. Ttopdir, /* top level directory */
  22. Qctl,
  23. Qstat,
  24. Qdata,
  25. Bufsize= 32*1024, /* chars buffered per reader */
  26. Maxcons= 64, /* maximum consoles */
  27. Nhash= 64, /* Fid hash buckets */
  28. };
  29. #define TYPE(x) (((ulong)x.path) & 0xf)
  30. #define CONS(x) ((((ulong)x.path) >> 4)&0xfff)
  31. #define QID(c, x) (((c)<<4) | (x))
  32. struct Request
  33. {
  34. Request *next;
  35. Fid *fid;
  36. Fs *fs;
  37. Fcall f;
  38. uchar buf[1];
  39. };
  40. struct Reqlist
  41. {
  42. Lock;
  43. Request *first;
  44. Request *last;
  45. };
  46. struct Fid
  47. {
  48. Lock;
  49. Fid *next; /* hash list */
  50. Fid *cnext; /* list of Fid's on a console */
  51. int fid;
  52. int ref;
  53. int attached;
  54. int open;
  55. char *user;
  56. char mbuf[Bufsize]; /* message */
  57. int bufn;
  58. int used;
  59. Qid qid;
  60. Console *c;
  61. char buf[Bufsize];
  62. char *rp;
  63. char *wp;
  64. Reqlist r; /* active read requests */
  65. };
  66. struct Console
  67. {
  68. Lock;
  69. char *name;
  70. char *dev;
  71. int speed;
  72. int cronly;
  73. int ondemand; /* open only on demand */
  74. int chat; /* chat consoles are special */
  75. int pid; /* pid of reader */
  76. int fd;
  77. int cfd;
  78. int sfd;
  79. Fid *flist; /* open fids to broadcast to */
  80. };
  81. struct Fs
  82. {
  83. Lock;
  84. int fd; /* to kernel mount point */
  85. int messagesize;
  86. Fid *hash[Nhash];
  87. Console *cons[Maxcons];
  88. int ncons;
  89. };
  90. extern void console(Fs*, char*, char*, int, int, int);
  91. extern Fs* fsmount(char*);
  92. extern void fsreader(void*);
  93. extern void fsrun(void*);
  94. extern Fid* fsgetfid(Fs*, int);
  95. extern void fsputfid(Fs*, Fid*);
  96. extern int fsdirgen(Fs*, Qid, int, Dir*, uchar*, int);
  97. extern void fsreply(Fs*, Request*, char*);
  98. extern void fskick(Fs*, Fid*);
  99. extern int fsreopen(Fs*, Console*);
  100. extern void fsversion(Fs*, Request*, Fid*);
  101. extern void fsflush(Fs*, Request*, Fid*);
  102. extern void fsauth(Fs*, Request*, Fid*);
  103. extern void fsattach(Fs*, Request*, Fid*);
  104. extern void fswalk(Fs*, Request*, Fid*);
  105. extern void fsclwalk(Fs*, Request*, Fid*);
  106. extern void fsopen(Fs*, Request*, Fid*);
  107. extern void fscreate(Fs*, Request*, Fid*);
  108. extern void fsread(Fs*, Request*, Fid*);
  109. extern void fswrite(Fs*, Request*, Fid*);
  110. extern void fsclunk(Fs*, Request*, Fid*);
  111. extern void fsremove(Fs*, Request*, Fid*);
  112. extern void fsstat(Fs*, Request*, Fid*);
  113. extern void fswstat(Fs*, Request*, Fid*);
  114. void (*fcall[])(Fs*, Request*, Fid*) =
  115. {
  116. [Tflush] fsflush,
  117. [Tversion] fsversion,
  118. [Tauth] fsauth,
  119. [Tattach] fsattach,
  120. [Twalk] fswalk,
  121. [Topen] fsopen,
  122. [Tcreate] fscreate,
  123. [Tread] fsread,
  124. [Twrite] fswrite,
  125. [Tclunk] fsclunk,
  126. [Tremove] fsremove,
  127. [Tstat] fsstat,
  128. [Twstat] fswstat
  129. };
  130. char Eperm[] = "permission denied";
  131. char Eexist[] = "file does not exist";
  132. char Enotdir[] = "not a directory";
  133. char Eisopen[] = "file already open";
  134. char Ebadcount[] = "bad read/write count";
  135. char Enofid[] = "no such fid";
  136. char *consoledb = "/lib/ndb/consoledb";
  137. char *mntpt = "/mnt/consoles";
  138. int messagesize = 8192+IOHDRSZ;
  139. void
  140. fatal(char *fmt, ...)
  141. {
  142. va_list arg;
  143. char buf[1024];
  144. write(2, "consolefs: ", 10);
  145. va_start(arg, fmt);
  146. vseprint(buf, buf+1024, fmt, arg);
  147. va_end(arg);
  148. write(2, buf, strlen(buf));
  149. write(2, "\n", 1);
  150. threadexitsall(fmt);
  151. }
  152. void*
  153. emalloc(uint n)
  154. {
  155. void *p;
  156. p = malloc(n);
  157. if(p == nil)
  158. fatal("malloc failed: %r");
  159. memset(p, 0, n);
  160. return p;
  161. }
  162. int debug;
  163. Ndb *db;
  164. /*
  165. * any request that can get queued for a delayed reply
  166. */
  167. Request*
  168. allocreq(Fs *fs, int bufsize)
  169. {
  170. Request *r;
  171. r = emalloc(sizeof(Request)+bufsize);
  172. r->fs = fs;
  173. r->next = nil;
  174. return r;
  175. }
  176. /*
  177. * for maintaining lists of requests
  178. */
  179. void
  180. addreq(Reqlist *l, Request *r)
  181. {
  182. lock(l);
  183. if(l->first == nil)
  184. l->first = r;
  185. else
  186. l->last->next = r;
  187. l->last = r;
  188. r->next = nil;
  189. unlock(l);
  190. }
  191. /*
  192. * remove the first request from a list of requests
  193. */
  194. Request*
  195. remreq(Reqlist *l)
  196. {
  197. Request *r;
  198. lock(l);
  199. r = l->first;
  200. if(r != nil)
  201. l->first = r->next;
  202. unlock(l);
  203. return r;
  204. }
  205. /*
  206. * remove a request with the given tag from a list of requests
  207. */
  208. Request*
  209. remtag(Reqlist *l, int tag)
  210. {
  211. Request *or, **ll;
  212. lock(l);
  213. ll = &l->first;
  214. for(or = *ll; or; or = or->next){
  215. if(or->f.tag == tag){
  216. *ll = or->next;
  217. unlock(l);
  218. return or;
  219. }
  220. ll = &or->next;
  221. }
  222. unlock(l);
  223. return nil;
  224. }
  225. Qid
  226. parentqid(Qid q)
  227. {
  228. if(q.type & QTDIR)
  229. return (Qid){QID(0, Textern), 0, QTDIR};
  230. else
  231. return (Qid){QID(0, Ttopdir), 0, QTDIR};
  232. }
  233. int
  234. fsdirgen(Fs *fs, Qid parent, int i, Dir *d, uchar *buf, int nbuf)
  235. {
  236. static char name[64];
  237. char *p;
  238. int xcons;
  239. d->uid = d->gid = d->muid = "network";
  240. d->length = 0;
  241. d->atime = time(nil);
  242. d->mtime = d->atime;
  243. d->type = 'C';
  244. d->dev = '0';
  245. switch(TYPE(parent)){
  246. case Textern:
  247. if(i != 0)
  248. return -1;
  249. p = "consoles";
  250. d->mode = DMDIR|0555;
  251. d->qid.type = QTDIR;
  252. d->qid.path = QID(0, Ttopdir);
  253. d->qid.vers = 0;
  254. break;
  255. case Ttopdir:
  256. xcons = i/3;
  257. if(xcons >= fs->ncons)
  258. return -1;
  259. p = fs->cons[xcons]->name;
  260. switch(i%3){
  261. case 0:
  262. if(fs->cons[xcons]->cfd < 0)
  263. return 0;
  264. snprint(name, sizeof name, "%sctl", p);
  265. p = name;
  266. d->qid.type = QTFILE;
  267. d->qid.path = QID(xcons, Qctl);
  268. d->qid.vers = 0;
  269. break;
  270. case 1:
  271. if(fs->cons[xcons]->sfd < 0)
  272. return 0;
  273. snprint(name, sizeof name, "%sstat", p);
  274. p = name;
  275. d->qid.type = QTFILE;
  276. d->qid.path = QID(xcons, Qstat);
  277. d->qid.vers = 0;
  278. break;
  279. case 2:
  280. d->qid.type = QTFILE;
  281. d->qid.path = QID(xcons, Qdata);
  282. d->qid.vers = 0;
  283. break;
  284. }
  285. d->mode = 0666;
  286. break;
  287. default:
  288. return -1;
  289. }
  290. d->name = p;
  291. if(buf != nil)
  292. return convD2M(d, buf, nbuf);
  293. return 1;
  294. }
  295. /*
  296. * mount the user interface and start a request processor
  297. */
  298. Fs*
  299. fsmount(char *mntpt)
  300. {
  301. Fs *fs;
  302. int pfd[2], srv;
  303. char buf[32];
  304. int n;
  305. static void *v[2];
  306. fs = emalloc(sizeof(Fs));
  307. if(pipe(pfd) < 0)
  308. fatal("opening pipe: %r");
  309. /* start up the file system process */
  310. v[0] = fs;
  311. v[1] = pfd;
  312. proccreate(fsrun, v, 16*1024);
  313. /* Typically mounted before /srv exists */
  314. if(access("#s/consoles", AEXIST) < 0){
  315. srv = create("#s/consoles", OWRITE, 0666);
  316. if(srv < 0)
  317. fatal("post: %r");
  318. n = sprint(buf, "%d", pfd[1]);
  319. if(write(srv, buf, n) < 0)
  320. fatal("write srv: %r");
  321. close(srv);
  322. }
  323. mount(pfd[1], -1, mntpt, MBEFORE, "");
  324. close(pfd[1]);
  325. return fs;
  326. }
  327. /*
  328. * reopen a console
  329. */
  330. int
  331. fsreopen(Fs* fs, Console *c)
  332. {
  333. char buf[128];
  334. static void *v[2];
  335. if(c->pid){
  336. if(postnote(PNPROC, c->pid, "reopen") != 0)
  337. fprint(2, "postnote failed: %r\n");
  338. c->pid = 0;
  339. }
  340. if(c->fd >= 0){
  341. close(c->fd);
  342. close(c->cfd);
  343. close(c->sfd);
  344. c->cfd = -1;
  345. c->fd = -1;
  346. c->sfd = -1;
  347. }
  348. if(c->flist == nil && c->ondemand)
  349. return 0;
  350. c->fd = open(c->dev, ORDWR);
  351. if(c->fd < 0)
  352. return -1;
  353. snprint(buf, sizeof(buf), "%sctl", c->dev);
  354. c->cfd = open(buf, ORDWR);
  355. fprint(c->cfd, "b%d", c->speed);
  356. snprint(buf, sizeof(buf), "%sstat", c->dev);
  357. c->sfd = open(buf, OREAD);
  358. v[0] = fs;
  359. v[1] = c;
  360. proccreate(fsreader, v, 16*1024);
  361. return 0;
  362. }
  363. void
  364. change(Fs *fs, Console *c, int doreopen, int speed, int cronly, int ondemand)
  365. {
  366. lock(c);
  367. if(speed != c->speed){
  368. c->speed = speed;
  369. doreopen = 1;
  370. }
  371. if(ondemand != c->ondemand){
  372. c->ondemand = ondemand;
  373. doreopen = 1;
  374. }
  375. c->cronly = cronly;
  376. if(doreopen)
  377. fsreopen(fs, c);
  378. unlock(c);
  379. }
  380. /*
  381. * create a console interface
  382. */
  383. void
  384. console(Fs* fs, char *name, char *dev, int speed, int cronly, int ondemand)
  385. {
  386. Console *c;
  387. char *x;
  388. int i, doreopen;
  389. if(fs->ncons >= Maxcons)
  390. fatal("too many consoles, too little time");
  391. doreopen = 0;
  392. for(i = 0; i < fs->ncons; i++){
  393. c = fs->cons[i];
  394. if(strcmp(name, c->name) == 0){
  395. if(strcmp(dev, c->dev) != 0){
  396. /* new device */
  397. x = c->dev;
  398. c->dev = strdup(dev);
  399. free(x);
  400. doreopen = 1;
  401. }
  402. change(fs, c, doreopen, speed, cronly, ondemand);
  403. return;
  404. }
  405. }
  406. #ifdef sapedoesntlikethis
  407. /*
  408. * The code below prevents this from working. I can't
  409. * think of scenarios where the code below actually helps
  410. * Sape
  411. *
  412. * console=borneo dev=/dev/eia1
  413. * speed=9600
  414. * openondemand=1
  415. * console=tottie dev=/dev/eia1
  416. * speed=115200
  417. * openondemand=1
  418. */
  419. for(i = 0; i < fs->ncons; i++){
  420. c = fs->cons[i];
  421. if(strcmp(dev, c->dev) == 0){
  422. /* at least a rename */
  423. x = c->name;
  424. c->name = strdup(name);
  425. free(x);
  426. change(fs, c, doreopen, speed, cronly, ondemand);
  427. return;
  428. }
  429. }
  430. #endif
  431. c = emalloc(sizeof(Console));
  432. fs->cons[fs->ncons] = c;
  433. fs->ncons++;
  434. c->name = strdup(name);
  435. c->dev = strdup(dev);
  436. if(strcmp(c->dev, "/dev/null") == 0)
  437. c->chat = 1;
  438. else
  439. c->chat = 0;
  440. c->fd = -1;
  441. c->cfd = -1;
  442. c->sfd = -1;
  443. change(fs, c, 1, speed, cronly, ondemand);
  444. }
  445. /*
  446. * buffer data from console to a client.
  447. * circular q with writer able to catch up to reader.
  448. * the reader may miss data but always sees an in order sequence.
  449. */
  450. void
  451. fromconsole(Fid *f, char *p, int n)
  452. {
  453. char *rp, *wp, *ep;
  454. int pass;
  455. lock(f);
  456. rp = f->rp;
  457. wp = f->wp;
  458. ep = f->buf + sizeof(f->buf);
  459. pass = 0;
  460. while(n--){
  461. *wp++ = *p++;
  462. if(wp >= ep)
  463. wp = f->buf;
  464. if(rp == wp)
  465. pass = 1;
  466. }
  467. f->wp = wp;
  468. /* we overtook the read pointer, push it up so readers always
  469. * see the tail of what was written
  470. */
  471. if(pass){
  472. wp++;
  473. if(wp >= ep)
  474. f->rp = f->buf;
  475. else
  476. f->rp = wp;
  477. }
  478. unlock(f);
  479. }
  480. /*
  481. * broadcast a list of members to all listeners
  482. */
  483. void
  484. bcastmembers(Fs *fs, Console *c, char *msg, Fid *f)
  485. {
  486. int n;
  487. Fid *fl;
  488. char buf[512];
  489. sprint(buf, "[%s%s", msg, f->user);
  490. for(fl = c->flist; fl != nil && strlen(buf) + 64 < sizeof(buf); fl = fl->cnext){
  491. if(f == fl)
  492. continue;
  493. strcat(buf, ", ");
  494. strcat(buf, fl->user);
  495. }
  496. strcat(buf, "]\n");
  497. n = strlen(buf);
  498. for(fl = c->flist; fl; fl = fl->cnext){
  499. fromconsole(fl, buf, n);
  500. fskick(fs, fl);
  501. }
  502. }
  503. void
  504. handler(void*, char *msg)
  505. {
  506. if(strstr(msg, "reopen"))
  507. noted(NCONT);
  508. noted(NDFLT);
  509. }
  510. /*
  511. * a process to read console output and broadcast it (one per console)
  512. */
  513. void
  514. fsreader(void *v)
  515. {
  516. int n;
  517. Fid *fl;
  518. char buf[1024];
  519. Fs *fs;
  520. Console *c;
  521. void **a;
  522. a = v;
  523. fs = a[0];
  524. c = a[1];
  525. c->pid = getpid();
  526. notify(handler);
  527. if(c->chat)
  528. threadexits(nil);
  529. for(;;){
  530. n = read(c->fd, buf, sizeof(buf));
  531. if(n < 0)
  532. break;
  533. lock(c);
  534. for(fl = c->flist; fl; fl = fl->cnext){
  535. fromconsole(fl, buf, n);
  536. fskick(fs, fl);
  537. }
  538. unlock(c);
  539. }
  540. }
  541. void
  542. readdb(Fs *fs)
  543. {
  544. Ndbtuple *t, *nt;
  545. char *dev, *cons;
  546. int cronly, speed, ondemand;
  547. ndbreopen(db);
  548. /* start a listener for each console */
  549. for(;;){
  550. t = ndbparse(db);
  551. if(t == nil)
  552. break;
  553. dev = nil;
  554. cons = nil;
  555. speed = 9600;
  556. cronly = 0;
  557. ondemand = 0;
  558. for(nt = t; nt; nt = nt->entry){
  559. if(strcmp(nt->attr, "console") == 0)
  560. cons = nt->val;
  561. else if(strcmp(nt->attr, "dev") == 0)
  562. dev = nt->val;
  563. else if(strcmp(nt->attr, "speed") == 0)
  564. speed = atoi(nt->val);
  565. else if(strcmp(nt->attr, "cronly") == 0)
  566. cronly = 1;
  567. else if(strcmp(nt->attr, "openondemand") == 0)
  568. ondemand = 1;
  569. }
  570. if(dev != nil && cons != nil)
  571. console(fs, cons, dev, speed, cronly, ondemand);
  572. ndbfree(t);
  573. }
  574. }
  575. int dbmtime;
  576. /*
  577. * a request processor (one per Fs)
  578. */
  579. void
  580. fsrun(void *v)
  581. {
  582. int n, t;
  583. Request *r;
  584. Fid *f;
  585. Dir *d;
  586. void **a = v;
  587. Fs* fs;
  588. int *pfd;
  589. fs = a[0];
  590. pfd = a[1];
  591. fs->fd = pfd[0];
  592. for(;;){
  593. d = dirstat(consoledb);
  594. if(d != nil && d->mtime != dbmtime){
  595. dbmtime = d->mtime;
  596. readdb(fs);
  597. }
  598. free(d);
  599. r = allocreq(fs, messagesize);
  600. n = read9pmsg(fs->fd, r->buf, messagesize);
  601. if(n <= 0)
  602. fatal("unmounted");
  603. if(convM2S(r->buf, n, &r->f) == 0){
  604. fprint(2, "can't convert %ux %ux %ux\n", r->buf[0],
  605. r->buf[1], r->buf[2]);
  606. free(r);
  607. continue;
  608. }
  609. f = fsgetfid(fs, r->f.fid);
  610. r->fid = f;
  611. if(debug)
  612. fprint(2, "%F path %llux\n", &r->f, f->qid.path);
  613. t = r->f.type;
  614. r->f.type++;
  615. (*fcall[t])(fs, r, f);
  616. }
  617. }
  618. Fid*
  619. fsgetfid(Fs *fs, int fid)
  620. {
  621. Fid *f, *nf;
  622. lock(fs);
  623. for(f = fs->hash[fid%Nhash]; f; f = f->next){
  624. if(f->fid == fid){
  625. f->ref++;
  626. unlock(fs);
  627. return f;
  628. }
  629. }
  630. nf = emalloc(sizeof(Fid));
  631. nf->next = fs->hash[fid%Nhash];
  632. fs->hash[fid%Nhash] = nf;
  633. nf->fid = fid;
  634. nf->ref = 1;
  635. nf->wp = nf->buf;
  636. nf->rp = nf->wp;
  637. unlock(fs);
  638. return nf;
  639. }
  640. void
  641. fsputfid(Fs *fs, Fid *f)
  642. {
  643. Fid **l, *nf;
  644. lock(fs);
  645. if(--f->ref > 0){
  646. unlock(fs);
  647. return;
  648. }
  649. for(l = &fs->hash[f->fid%Nhash]; nf = *l; l = &nf->next)
  650. if(nf == f){
  651. *l = f->next;
  652. break;
  653. }
  654. unlock(fs);
  655. free(f->user);
  656. free(f);
  657. }
  658. void
  659. fsauth(Fs *fs, Request *r, Fid*)
  660. {
  661. fsreply(fs, r, "consolefs: authentication not required");
  662. }
  663. void
  664. fsversion(Fs *fs, Request *r, Fid*)
  665. {
  666. if(r->f.msize < 256){
  667. fsreply(fs, r, "message size too small");
  668. return;
  669. }
  670. messagesize = r->f.msize;
  671. if(messagesize > 8192+IOHDRSZ)
  672. messagesize = 8192+IOHDRSZ;
  673. r->f.msize = messagesize;
  674. if(strncmp(r->f.version, "9P2000", 6) != 0){
  675. fsreply(fs, r, "unrecognized 9P version");
  676. return;
  677. }
  678. r->f.version = "9P2000";
  679. fsreply(fs, r, nil);
  680. }
  681. void
  682. fsflush(Fs *fs, Request *r, Fid *f)
  683. {
  684. Request *or;
  685. or = remtag(&f->r, r->f.oldtag);
  686. if(or != nil){
  687. fsputfid(fs, or->fid);
  688. free(or);
  689. }
  690. fsreply(fs, r, nil);
  691. }
  692. void
  693. fsattach(Fs *fs, Request *r, Fid *f)
  694. {
  695. f->qid.type = QTDIR;
  696. f->qid.path = QID(0, Ttopdir);
  697. f->qid.vers = 0;
  698. if(r->f.uname[0])
  699. f->user = strdup(r->f.uname);
  700. else
  701. f->user = strdup("none");
  702. /* hold down the fid till the clunk */
  703. f->attached = 1;
  704. lock(fs);
  705. f->ref++;
  706. unlock(fs);
  707. r->f.qid = f->qid;
  708. fsreply(fs, r, nil);
  709. }
  710. void
  711. fswalk(Fs *fs, Request *r, Fid *f)
  712. {
  713. char *name;
  714. Dir d;
  715. int i, n, nqid, nwname;
  716. Qid qid, wqid[MAXWELEM];
  717. Fid *nf;
  718. char *err;
  719. if(f->attached == 0){
  720. fsreply(fs, r, Enofid);
  721. return;
  722. }
  723. nf = nil;
  724. if(r->f.fid != r->f.newfid){
  725. nf = fsgetfid(fs, r->f.newfid);
  726. nf->attached = f->attached;
  727. nf->open = f->open;
  728. nf->qid = f->qid;
  729. nf->user = strdup(f->user);
  730. nf->c = f->c;
  731. nf->wp = nf->buf;
  732. nf->rp = nf->wp;
  733. f = nf;
  734. }
  735. qid = f->qid;
  736. err = nil;
  737. nwname = r->f.nwname;
  738. nqid = 0;
  739. if(nwname > 0){
  740. for(; err == nil && nqid < nwname; nqid++){
  741. if(nqid >= MAXWELEM){
  742. err = "too many name elements";
  743. break;
  744. }
  745. name = r->f.wname[nqid];
  746. if(strcmp(name, "..") == 0)
  747. qid = parentqid(qid);
  748. else if(strcmp(name, ".") != 0){
  749. for(i = 0; ; i++){
  750. n = fsdirgen(fs, qid, i, &d, nil, 0);
  751. if(n < 0){
  752. err = Eexist;
  753. break;
  754. }
  755. if(n > 0 && strcmp(name, d.name) == 0){
  756. qid = d.qid;
  757. break;
  758. }
  759. }
  760. }
  761. wqid[nqid] = qid;
  762. }
  763. if(nf != nil && nqid < nwname)
  764. fsputfid(fs, nf);
  765. if(nqid == nwname)
  766. f->qid = qid;
  767. }
  768. memmove(r->f.wqid, wqid, nqid*sizeof(Qid));
  769. r->f.nwqid = nqid;
  770. fsreply(fs, r, err);
  771. }
  772. int
  773. ingroup(char *user, char *group)
  774. {
  775. Ndbtuple *t, *nt;
  776. Ndbs s;
  777. t = ndbsearch(db, &s, "group", group);
  778. if(t == nil)
  779. return 0;
  780. for(nt = t; nt; nt = nt->entry){
  781. if(strcmp(nt->attr, "uid") == 0)
  782. if(strcmp(nt->val, user) == 0)
  783. break;
  784. }
  785. ndbfree(t);
  786. return nt != nil;
  787. }
  788. int
  789. userok(char *u, char *cname)
  790. {
  791. Ndbtuple *t, *nt;
  792. Ndbs s;
  793. t = ndbsearch(db, &s, "console", cname);
  794. if(t == nil)
  795. return 0;
  796. for(nt = t; nt; nt = nt->entry){
  797. if(strcmp(nt->attr, "uid") == 0)
  798. if(strcmp(nt->val, u) == 0)
  799. break;
  800. if(strcmp(nt->attr, "gid") == 0)
  801. if(ingroup(u, nt->val))
  802. break;
  803. }
  804. ndbfree(t);
  805. return nt != nil;
  806. }
  807. int m2p[] ={
  808. [OREAD] 4,
  809. [OWRITE] 2,
  810. [ORDWR] 6
  811. };
  812. /*
  813. * broadcast a message to all listeners
  814. */
  815. void
  816. bcastmsg(Fs *fs, Console *c, char *msg, int n)
  817. {
  818. Fid *fl;
  819. for(fl = c->flist; fl; fl = fl->cnext){
  820. fromconsole(fl, msg, n);
  821. fskick(fs, fl);
  822. }
  823. }
  824. void
  825. fsopen(Fs *fs, Request *r, Fid *f)
  826. {
  827. int mode;
  828. Console *c;
  829. if(f->attached == 0){
  830. fsreply(fs, r, Enofid);
  831. return;
  832. }
  833. if(f->open){
  834. fsreply(fs, r, Eisopen);
  835. return;
  836. }
  837. mode = r->f.mode & 3;
  838. if((QTDIR & f->qid.type) && mode != OREAD){
  839. fsreply(fs, r, Eperm);
  840. return;
  841. }
  842. switch(TYPE(f->qid)){
  843. case Qdata:
  844. c = fs->cons[CONS(f->qid)];
  845. if(!userok(f->user, c->name)){
  846. fsreply(fs, r, Eperm);
  847. return;
  848. }
  849. f->rp = f->buf;
  850. f->wp = f->buf;
  851. f->c = c;
  852. lock(c);
  853. sprint(f->mbuf, "[%s] ", f->user);
  854. f->bufn = strlen(f->mbuf);
  855. f->used = 0;
  856. f->cnext = c->flist;
  857. c->flist = f;
  858. bcastmembers(fs, c, "+", f);
  859. if(c->pid == 0)
  860. fsreopen(fs, c);
  861. unlock(c);
  862. break;
  863. case Qctl:
  864. c = fs->cons[CONS(f->qid)];
  865. if(!userok(f->user, c->name)){
  866. fsreply(fs, r, Eperm);
  867. return;
  868. }
  869. f->c = c;
  870. break;
  871. case Qstat:
  872. c = fs->cons[CONS(f->qid)];
  873. if(!userok(f->user, c->name)){
  874. fsreply(fs, r, Eperm);
  875. return;
  876. }
  877. f->c = c;
  878. break;
  879. }
  880. f->open = 1;
  881. r->f.iounit = messagesize-IOHDRSZ;
  882. r->f.qid = f->qid;
  883. fsreply(fs, r, nil);
  884. }
  885. void
  886. fscreate(Fs *fs, Request *r, Fid*)
  887. {
  888. fsreply(fs, r, Eperm);
  889. }
  890. void
  891. fsread(Fs *fs, Request *r, Fid *f)
  892. {
  893. uchar *p, *e;
  894. int i, m, off;
  895. vlong offset;
  896. Dir d;
  897. char sbuf[ERRMAX];
  898. if(f->attached == 0){
  899. fsreply(fs, r, Enofid);
  900. return;
  901. }
  902. if((int)r->f.count < 0){
  903. fsreply(fs, r, Ebadcount);
  904. return;
  905. }
  906. if(QTDIR & f->qid.type){
  907. p = r->buf + IOHDRSZ;
  908. e = p + r->f.count;
  909. offset = r->f.offset;
  910. off = 0;
  911. for(i=0; p<e; i++, off+=m){
  912. m = fsdirgen(fs, f->qid, i, &d, p, e-p);
  913. if(m < 0)
  914. break;
  915. if(m > BIT16SZ && off >= offset)
  916. p += m;
  917. }
  918. r->f.data = (char*)r->buf + IOHDRSZ;
  919. r->f.count = (char*)p - r->f.data;
  920. } else {
  921. switch(TYPE(f->qid)){
  922. case Qdata:
  923. addreq(&f->r, r);
  924. fskick(fs, f);
  925. return;
  926. case Qctl:
  927. r->f.data = (char*)r->buf+IOHDRSZ;
  928. r->f.count = 0;
  929. break;
  930. case Qstat:
  931. if(r->f.count > sizeof(sbuf))
  932. r->f.count = sizeof(sbuf);
  933. i = pread(f->c->sfd, sbuf, r->f.count, r->f.offset);
  934. if(i < 0){
  935. errstr(sbuf, sizeof sbuf);
  936. fsreply(fs, r, sbuf);
  937. return;
  938. }
  939. r->f.data = sbuf;
  940. r->f.count = i;
  941. break;
  942. default:
  943. fsreply(fs, r, Eexist);
  944. return;
  945. }
  946. }
  947. fsreply(fs, r, nil);
  948. }
  949. void
  950. fswrite(Fs *fs, Request *r, Fid *f)
  951. {
  952. int i, eol = 0;
  953. if(f->attached == 0){
  954. fsreply(fs, r, Enofid);
  955. return;
  956. }
  957. if((int)r->f.count < 0){
  958. fsreply(fs, r, Ebadcount);
  959. return;
  960. }
  961. if(QTDIR & f->qid.type){
  962. fsreply(fs, r, Eperm);
  963. return;
  964. }
  965. switch(TYPE(f->qid)){
  966. default:
  967. fsreply(fs, r, Eperm);
  968. return;
  969. case Qctl:
  970. write(f->c->cfd, r->f.data, r->f.count);
  971. break;
  972. case Qdata:
  973. for(i = 0; i < r->f.count; i++){
  974. if(r->f.data[i] == '\n'){
  975. if(f->c->chat && f->used)
  976. eol = 1;
  977. if(f->c->cronly)
  978. r->f.data[i] = '\r';
  979. }
  980. else
  981. f->used = 1;
  982. }
  983. if(f->c->chat){
  984. fskick(fs, f);
  985. if(!f->used)
  986. break;
  987. if(f->bufn + r->f.count > Bufsize){
  988. r->f.count -= (f->bufn + r->f.count) % Bufsize;
  989. eol = 1;
  990. }
  991. strncat(f->mbuf, r->f.data, r->f.count);
  992. f->bufn += r->f.count;
  993. if(eol){
  994. bcastmsg(fs, f->c, f->mbuf, f->bufn);
  995. sprint(f->mbuf, "[%s] ", f->user);
  996. f->bufn = strlen(f->mbuf);
  997. f->used = 0;
  998. }
  999. }
  1000. else
  1001. write(f->c->fd, r->f.data, r->f.count);
  1002. break;
  1003. }
  1004. fsreply(fs, r, nil);
  1005. }
  1006. void
  1007. fsclunk(Fs *fs, Request *r, Fid *f)
  1008. {
  1009. Fid **l, *fl;
  1010. Request *nr;
  1011. if(f->open && TYPE(f->qid) == Qdata){
  1012. while((nr = remreq(&f->r)) != nil){
  1013. fsputfid(fs, f);
  1014. free(nr);
  1015. }
  1016. lock(f->c);
  1017. for(l = &f->c->flist; *l; l = &fl->cnext){
  1018. fl = *l;
  1019. if(fl == f){
  1020. *l = fl->cnext;
  1021. break;
  1022. }
  1023. }
  1024. bcastmembers(fs, f->c, "-", f);
  1025. if(f->c->ondemand && f->c->flist == nil)
  1026. fsreopen(fs, f->c);
  1027. unlock(f->c);
  1028. }
  1029. fsreply(fs, r, nil);
  1030. fsputfid(fs, f);
  1031. }
  1032. void
  1033. fsremove(Fs *fs, Request *r, Fid*)
  1034. {
  1035. fsreply(fs, r, Eperm);
  1036. }
  1037. void
  1038. fsstat(Fs *fs, Request *r, Fid *f)
  1039. {
  1040. int i, n;
  1041. Qid q;
  1042. Dir d;
  1043. q = parentqid(f->qid);
  1044. for(i = 0; ; i++){
  1045. r->f.stat = r->buf+IOHDRSZ;
  1046. n = fsdirgen(fs, q, i, &d, r->f.stat, messagesize-IOHDRSZ);
  1047. if(n < 0){
  1048. fsreply(fs, r, Eexist);
  1049. return;
  1050. }
  1051. r->f.nstat = n;
  1052. if(r->f.nstat > BIT16SZ && d.qid.path == f->qid.path)
  1053. break;
  1054. }
  1055. fsreply(fs, r, nil);
  1056. }
  1057. void
  1058. fswstat(Fs *fs, Request *r, Fid*)
  1059. {
  1060. fsreply(fs, r, Eperm);
  1061. }
  1062. void
  1063. fsreply(Fs *fs, Request *r, char *err)
  1064. {
  1065. int n;
  1066. uchar buf[8192+IOHDRSZ];
  1067. if(err){
  1068. r->f.type = Rerror;
  1069. r->f.ename = err;
  1070. }
  1071. n = convS2M(&r->f, buf, messagesize);
  1072. if(debug)
  1073. fprint(2, "%F path %llux n=%d\n", &r->f, r->fid->qid.path, n);
  1074. fsputfid(fs, r->fid);
  1075. if(write(fs->fd, buf, n) != n)
  1076. fatal("unmounted");
  1077. free(r);
  1078. }
  1079. /*
  1080. * called whenever input or a read request has been received
  1081. */
  1082. void
  1083. fskick(Fs *fs, Fid *f)
  1084. {
  1085. Request *r;
  1086. char *p, *rp, *wp, *ep;
  1087. int i;
  1088. lock(f);
  1089. while(f->rp != f->wp){
  1090. r = remreq(&f->r);
  1091. if(r == nil)
  1092. break;
  1093. p = (char*)r->buf;
  1094. rp = f->rp;
  1095. wp = f->wp;
  1096. ep = &f->buf[Bufsize];
  1097. for(i = 0; i < r->f.count && rp != wp; i++){
  1098. *p++ = *rp++;
  1099. if(rp >= ep)
  1100. rp = f->buf;
  1101. }
  1102. f->rp = rp;
  1103. r->f.data = (char*)r->buf;
  1104. r->f.count = p - (char*)r->buf;
  1105. fsreply(fs, r, nil);
  1106. }
  1107. unlock(f);
  1108. }
  1109. void
  1110. usage(void)
  1111. {
  1112. fprint(2, "usage: consolefs [-d] [-m mount-point] [-c console-db]\n");
  1113. threadexitsall("usage");
  1114. }
  1115. void
  1116. threadmain(int argc, char **argv)
  1117. {
  1118. fmtinstall('F', fcallfmt);
  1119. ARGBEGIN{
  1120. case 'd':
  1121. debug++;
  1122. break;
  1123. case 'c':
  1124. consoledb = ARGF();
  1125. if(consoledb == nil)
  1126. usage();
  1127. break;
  1128. case 'm':
  1129. mntpt = ARGF();
  1130. if(mntpt == nil)
  1131. usage();
  1132. break;
  1133. }ARGEND;
  1134. db = ndbopen(consoledb);
  1135. if(db == nil)
  1136. fatal("can't open %s: %r", consoledb);
  1137. fsmount(mntpt);
  1138. }