Ccons.c 6.5 KB

123456789101112131415161718192021222324252627282930313233343536373839404142434445464748495051525354555657585960616263646566676869707172737475767778798081828384858687888990919293949596979899100101102103104105106107108109110111112113114115116117118119120121122123124125126127128129130131132133134135136137138139140141142143144145146147148149150151152153154155156157158159160161162163164165166167168169170171172173174175176177178179180181182183184185186187188189190191192193194195196197198199200201202203204205206207208209210211212213214215216217218219220221222223224225226227228229230231232233234235236237238239240241242243244245246247248249250251252253254255256257258259260261262263264265266267268269270271272273274275276277278279280281282283284285286287288289290291292293294295296297298299300301302303304305306307308309310311312313314315316317318319320321322323324325326327328329330331332333334335336337338339340341342343344345346347348349350351352353354355356357358359360361362363364365366367368369370371372373374375376377378379380381382383384385386387388389390391392393394
  1. #include "stdinc.h"
  2. #include "9.h"
  3. enum {
  4. Nl = 256, /* max. command line length */
  5. Nq = 8*1024, /* amount of I/O buffered */
  6. };
  7. typedef struct Q {
  8. VtLock* lock;
  9. VtRendez* full;
  10. VtRendez* empty;
  11. char q[Nq];
  12. int n;
  13. int r;
  14. int w;
  15. } Q;
  16. typedef struct Cons {
  17. VtLock* lock;
  18. int ref;
  19. int closed;
  20. int fd;
  21. int srvfd;
  22. int ctlfd;
  23. Q* iq; /* points to console.iq */
  24. Q* oq; /* points to console.oq */
  25. } Cons;
  26. static struct {
  27. Q* iq; /* input */
  28. Q* oq; /* output */
  29. char l[Nl]; /* command line assembly */
  30. int nl; /* current line length */
  31. int nopens;
  32. char* prompt;
  33. int np;
  34. } console;
  35. static void
  36. consClose(Cons* cons)
  37. {
  38. vtLock(cons->lock);
  39. cons->closed = 1;
  40. cons->ref--;
  41. if(cons->ref > 0){
  42. vtLock(cons->iq->lock);
  43. vtWakeup(cons->iq->full);
  44. vtUnlock(cons->iq->lock);
  45. vtLock(cons->oq->lock);
  46. vtWakeup(cons->oq->empty);
  47. vtUnlock(cons->oq->lock);
  48. vtUnlock(cons->lock);
  49. return;
  50. }
  51. if(cons->ctlfd != -1){
  52. close(cons->ctlfd);
  53. cons->srvfd = -1;
  54. }
  55. if(cons->srvfd != -1){
  56. close(cons->srvfd);
  57. cons->srvfd = -1;
  58. }
  59. if(cons->fd != -1){
  60. close(cons->fd);
  61. cons->fd = -1;
  62. }
  63. vtUnlock(cons->lock);
  64. vtLockFree(cons->lock);
  65. vtMemFree(cons);
  66. console.nopens--;
  67. }
  68. static void
  69. consIProc(void* v)
  70. {
  71. Q *q;
  72. Cons *cons;
  73. int n, w;
  74. char buf[Nq/4];
  75. vtThreadSetName("consI");
  76. cons = v;
  77. q = cons->iq;
  78. for(;;){
  79. /*
  80. * Can't tell the difference between zero-length read
  81. * and eof, so keep calling read until we get an error.
  82. */
  83. if(cons->closed || (n = read(cons->fd, buf, Nq/4)) < 0)
  84. break;
  85. vtLock(q->lock);
  86. while(Nq - q->n < n && !cons->closed)
  87. vtSleep(q->full);
  88. w = Nq - q->w;
  89. if(w < n){
  90. memmove(&q->q[q->w], buf, w);
  91. memmove(&q->q[0], buf + w, n - w);
  92. }
  93. else
  94. memmove(&q->q[q->w], buf, n);
  95. q->w = (q->w + n) % Nq;
  96. q->n += n;
  97. vtWakeup(q->empty);
  98. vtUnlock(q->lock);
  99. }
  100. consClose(cons);
  101. }
  102. static void
  103. consOProc(void* v)
  104. {
  105. Q *q;
  106. Cons *cons;
  107. char buf[Nq];
  108. int lastn, n, r;
  109. vtThreadSetName("consO");
  110. cons = v;
  111. q = cons->oq;
  112. vtLock(q->lock);
  113. lastn = 0;
  114. for(;;){
  115. while(lastn == q->n && !cons->closed)
  116. vtSleep(q->empty);
  117. if((n = q->n - lastn) > Nq)
  118. n = Nq;
  119. if(n > q->w){
  120. r = n - q->w;
  121. memmove(buf, &q->q[Nq - r], r);
  122. memmove(buf+r, &q->q[0], n - r);
  123. }
  124. else
  125. memmove(buf, &q->q[q->w - n], n);
  126. lastn = q->n;
  127. vtUnlock(q->lock);
  128. if(cons->closed || write(cons->fd, buf, n) < 0)
  129. break;
  130. vtLock(q->lock);
  131. vtWakeup(q->empty);
  132. }
  133. consClose(cons);
  134. }
  135. int
  136. consOpen(int fd, int srvfd, int ctlfd)
  137. {
  138. Cons *cons;
  139. cons = vtMemAllocZ(sizeof(Cons));
  140. cons->lock = vtLockAlloc();
  141. cons->fd = fd;
  142. cons->srvfd = srvfd;
  143. cons->ctlfd = ctlfd;
  144. cons->iq = console.iq;
  145. cons->oq = console.oq;
  146. console.nopens++;
  147. vtLock(cons->lock);
  148. cons->ref = 2;
  149. cons->closed = 0;
  150. if(vtThread(consOProc, cons) < 0){
  151. cons->ref--;
  152. vtUnlock(cons->lock);
  153. consClose(cons);
  154. return 0;
  155. }
  156. vtUnlock(cons->lock);
  157. if(ctlfd >= 0)
  158. consIProc(cons);
  159. else if(vtThread(consIProc, cons) < 0){
  160. consClose(cons);
  161. return 0;
  162. }
  163. return 1;
  164. }
  165. static int
  166. qWrite(Q* q, char* p, int n)
  167. {
  168. int w;
  169. vtLock(q->lock);
  170. if(n > Nq - q->w){
  171. w = Nq - q->w;
  172. memmove(&q->q[q->w], p, w);
  173. memmove(&q->q[0], p + w, n - w);
  174. q->w = n - w;
  175. }
  176. else{
  177. memmove(&q->q[q->w], p, n);
  178. q->w += n;
  179. }
  180. q->n += n;
  181. vtWakeup(q->empty);
  182. vtUnlock(q->lock);
  183. return n;
  184. }
  185. static Q*
  186. qAlloc(void)
  187. {
  188. Q *q;
  189. q = vtMemAllocZ(sizeof(Q));
  190. q->lock = vtLockAlloc();
  191. q->full = vtRendezAlloc(q->lock);
  192. q->empty = vtRendezAlloc(q->lock);
  193. q->n = q->r = q->w = 0;
  194. return q;
  195. }
  196. static void
  197. consProc(void*)
  198. {
  199. Q *q;
  200. int argc, i, n, r;
  201. char *argv[20], buf[Nq], *lp, *wbuf;
  202. vtThreadSetName("cons");
  203. q = console.iq;
  204. qWrite(console.oq, console.prompt, console.np);
  205. vtLock(q->lock);
  206. for(;;){
  207. while((n = q->n) == 0)
  208. vtSleep(q->empty);
  209. r = Nq - q->r;
  210. if(r < n){
  211. memmove(buf, &q->q[q->r], r);
  212. memmove(buf + r, &q->q[0], n - r);
  213. }
  214. else
  215. memmove(buf, &q->q[q->r], n);
  216. q->r = (q->r + n) % Nq;
  217. q->n -= n;
  218. vtWakeup(q->full);
  219. vtUnlock(q->lock);
  220. for(i = 0; i < n; i++){
  221. switch(buf[i]){
  222. case '\004': /* ^D */
  223. if(console.nl == 0){
  224. qWrite(console.oq, "\n", 1);
  225. break;
  226. }
  227. /*FALLTHROUGH*/
  228. default:
  229. if(console.nl < Nl-1){
  230. qWrite(console.oq, &buf[i], 1);
  231. console.l[console.nl++] = buf[i];
  232. }
  233. continue;
  234. case '\b':
  235. if(console.nl != 0){
  236. qWrite(console.oq, &buf[i], 1);
  237. console.nl--;
  238. }
  239. continue;
  240. case '\n':
  241. qWrite(console.oq, &buf[i], 1);
  242. break;
  243. case '\025': /* ^U */
  244. qWrite(console.oq, "^U\n", 3);
  245. console.nl = 0;
  246. break;
  247. case '\027': /* ^W */
  248. console.l[console.nl] = '\0';
  249. wbuf = vtMemAlloc(console.nl+1);
  250. memmove(wbuf, console.l, console.nl+1);
  251. argc = tokenize(wbuf, argv, nelem(argv));
  252. if(argc > 0)
  253. argc--;
  254. console.nl = 0;
  255. lp = console.l;
  256. for(i = 0; i < argc; i++)
  257. lp += sprint(lp, "%q ", argv[i]);
  258. console.nl = lp - console.l;
  259. vtMemFree(wbuf);
  260. qWrite(console.oq, "^W\n", 3);
  261. if(console.nl == 0)
  262. break;
  263. qWrite(console.oq, console.l, console.nl);
  264. continue;
  265. case '\177':
  266. qWrite(console.oq, "\n", 1);
  267. console.nl = 0;
  268. break;
  269. }
  270. console.l[console.nl] = '\0';
  271. if(console.nl != 0)
  272. cliExec(console.l);
  273. console.nl = 0;
  274. qWrite(console.oq, console.prompt, console.np);
  275. }
  276. vtLock(q->lock);
  277. }
  278. }
  279. int
  280. consWrite(char* buf, int len)
  281. {
  282. if(console.oq == nil)
  283. return write(2, buf, len);
  284. if(console.nopens == 0)
  285. write(2, buf, len);
  286. return qWrite(console.oq, buf, len);
  287. }
  288. int
  289. consPrompt(char* prompt)
  290. {
  291. char buf[ERRMAX];
  292. if(prompt == nil)
  293. prompt = "prompt";
  294. vtMemFree(console.prompt);
  295. console.np = snprint(buf, sizeof(buf), "%s: ", prompt);
  296. console.prompt = vtStrDup(buf);
  297. return console.np;
  298. }
  299. int
  300. consTTY(void)
  301. {
  302. int ctl, fd;
  303. char *name, *p;
  304. name = "/dev/cons";
  305. if((fd = open(name, ORDWR)) < 0){
  306. name = "#c/cons";
  307. if((fd = open(name, ORDWR)) < 0){
  308. vtSetError("consTTY: open %s: %r", name);
  309. return 0;
  310. }
  311. }
  312. p = smprint("%sctl", name);
  313. if((ctl = open(p, OWRITE)) < 0){
  314. close(fd);
  315. vtSetError("consTTY: open %s: %r", p);
  316. free(p);
  317. return 0;
  318. }
  319. if(write(ctl, "rawon", 5) < 0){
  320. close(ctl);
  321. close(fd);
  322. vtSetError("consTTY: write %s: %r", p);
  323. free(p);
  324. return 0;
  325. }
  326. free(p);
  327. if(consOpen(fd, fd, ctl) == 0){
  328. close(ctl);
  329. close(fd);
  330. return 0;
  331. }
  332. return 1;
  333. }
  334. int
  335. consInit(void)
  336. {
  337. console.iq = qAlloc();
  338. console.oq = qAlloc();
  339. console.nl = 0;
  340. consPrompt(nil);
  341. if(vtThread(consProc, nil) < 0){
  342. vtFatal("can't start console proc");
  343. return 0;
  344. }
  345. return 1;
  346. }