tty.c 3.8 KB

123456789101112131415161718192021222324252627282930313233343536373839404142434445464748495051525354555657585960616263646566676869707172737475767778798081828384858687888990919293949596979899100101102103104105106107108109110111112113114115116117118119120121122123124125126127128129130131132133134135136137138139140141142143144145146147148149150151152153154155156157158159160161162163164165166167168169170171172173174175176177178179180181182183184185186187188189190191192193194195196197198199200201202203204205
  1. #include <u.h>
  2. #include <libc.h>
  3. int echo = 1;
  4. int raw;
  5. int ctrlp;
  6. void
  7. usage(void)
  8. {
  9. fprint(2, "usage: aux/tty [-p] [-f comfile] cmd args...\n");
  10. exits("usage");
  11. }
  12. void
  13. main(int argc, char *argv[])
  14. {
  15. int frchld[2];
  16. int tochld[2];
  17. int pid;
  18. int i, j;
  19. char *file;
  20. file = nil;
  21. ARGBEGIN{
  22. case 'p':
  23. ctrlp++;
  24. break;
  25. case 'f':
  26. file = EARGF(usage());
  27. break;
  28. }ARGEND
  29. if(file != nil){
  30. int fd;
  31. if((fd = open(file, ORDWR)) == -1)
  32. sysfatal("open %s", file);
  33. dup(fd, 0);
  34. dup(fd, 1);
  35. dup(fd, 2);
  36. close(fd);
  37. }
  38. pipe(frchld);
  39. pipe(tochld);
  40. switch(pid = rfork(RFPROC|RFFDG|RFNOTEG)){
  41. case -1:
  42. sysfatal("fork");
  43. case 0:
  44. close(tochld[0]);
  45. close(frchld[1]);
  46. dup(tochld[1], 0);
  47. dup(frchld[0], 1);
  48. dup(frchld[0], 2);
  49. close(tochld[1]);
  50. close(frchld[0]);
  51. exec(argv[0], argv);
  52. sysfatal("exec");
  53. default:
  54. close(tochld[1]);
  55. close(frchld[0]);
  56. break;
  57. }
  58. static char buf[512];
  59. static char obuf[512];
  60. int nfr, nto;
  61. int wpid;
  62. switch(wpid = rfork(RFPROC|RFFDG)){
  63. case -1:
  64. sysfatal("rfork");
  65. case 0:
  66. close(0);
  67. while((nfr = read(frchld[1], buf, sizeof buf)) > 0){
  68. int i, j;
  69. j = 0;
  70. for(i = 0; i < nfr; i++){
  71. if(buf[i] == '\n'){
  72. if(j > 0){
  73. write(1, buf, j);
  74. j = 0;
  75. }
  76. write(1, "\r\n", 2);
  77. } else {
  78. buf[j++] = buf[i];
  79. }
  80. }
  81. if(write(1, buf, j) != j)
  82. exits("write");
  83. }
  84. fprint(1, "aux/tty: got eof\n");
  85. postnote(PNPROC, getppid(), "interrupt");
  86. exits(nil);
  87. default:
  88. close(frchld[1]);
  89. j = 0;
  90. while((nto = read(0, buf, sizeof buf)) > 0){
  91. int oldj;
  92. oldj = j;
  93. for(i = 0; i < nto; i++){
  94. if(buf[i] == '\r' || buf[i] == '\n'){
  95. obuf[j++] = '\n';
  96. write(tochld[0], obuf, j);
  97. if(echo){
  98. obuf[j-1] = '\r';
  99. obuf[j++] = '\n';
  100. write(1, obuf+oldj, j-oldj);
  101. }
  102. j = 0;
  103. } else if(ctrlp && buf[i] == '\x10'){ // ctrl-p
  104. if(j > 0){
  105. if(echo)write(1, obuf+oldj, j-oldj);
  106. write(tochld[0], obuf, j);
  107. j = 0;
  108. }
  109. int fd;
  110. fd = open("/dev/reboot", OWRITE);
  111. if(fd != -1){
  112. fprint(1, "aux/tty: rebooting the system\n");
  113. sleep(2000);
  114. write(fd, "reboot", 6);
  115. close(fd);
  116. } else {
  117. fprint(1, "aux/tty: open /dev/reboot: %r\n");
  118. }
  119. } else if(buf[i] == '\003'){ // ctrl-c
  120. if(j > 0){
  121. if(echo)write(1, obuf+oldj, j-oldj);
  122. write(tochld[0], obuf, j);
  123. j = 0;
  124. }
  125. fprint(1, "aux/tty: sent interrupt to %d\n", pid);
  126. postnote(PNGROUP, pid, "interrupt");
  127. continue;
  128. } else if(buf[i] == '\x1d' ){ // ctrl-]
  129. if(j > 0){
  130. if(echo)write(1, obuf+oldj, j-oldj);
  131. write(tochld[0], obuf, j);
  132. j = 0;
  133. }
  134. fprint(1, "aux/tty: sent interrupt to %d\n", pid);
  135. postnote(PNGROUP, pid, "term");
  136. continue;
  137. } else if(buf[i] == '\004'){ // ctrl-d
  138. if(j > 0){
  139. if(echo)write(1, obuf+oldj, j-oldj);
  140. write(tochld[0], obuf, j);
  141. j = 0;
  142. }
  143. fprint(1, "aux/tty: sent eof to %d\n", pid);
  144. write(tochld[0], obuf, 0); //eof
  145. continue;
  146. } else if(buf[i] == 0x15){ // ctrl-u
  147. if(!raw){
  148. while(j > 0){
  149. j--;
  150. if(echo)write(1, "\b \b", 3); // bs
  151. else write(tochld[0], "\x15", 1); // bs
  152. }
  153. } else {
  154. obuf[j++] = buf[i];
  155. }
  156. continue;
  157. } else if(buf[i] == 0x7f || buf[i] == '\b'){ // backspace
  158. if(!raw){
  159. if(j > 0){
  160. j--;
  161. if(echo)write(1, "\b \b", 3); // bs
  162. else write(tochld[0], "\b", 1); // bs
  163. }
  164. } else {
  165. obuf[j++] = '\b';
  166. }
  167. continue;
  168. } else {
  169. obuf[j++] = buf[i];
  170. }
  171. }
  172. if(j > 0){
  173. if(raw){
  174. if(echo)write(1, obuf, j);
  175. write(tochld[0], obuf, j);
  176. j = 0;
  177. } else if(echo && j > oldj){
  178. write(1, obuf+oldj, j-oldj);
  179. }
  180. }
  181. }
  182. close(0);
  183. close(1);
  184. close(2);
  185. close(tochld[0]);
  186. postnote(PNGROUP, pid, "hangup");
  187. waitpid();
  188. waitpid();
  189. }
  190. exits(nil);
  191. }