9srv.c 3.5 KB

123456789101112131415161718192021222324252627282930313233343536373839404142434445464748495051525354555657585960616263646566676869707172737475767778798081828384858687888990919293949596979899100101102103104105106107108109110111112113114115116117118119120121122123124125126127128129130131132133134135136137138139140141142143144145146147148149150151152153154155156157158159160161162163164165166167168169170171172173174175176177178179180181182183184185186187188189190191192193194195196197198199200201202203204205206207208209210211212213214215216217218219220221222223
  1. #include "stdinc.h"
  2. #include "9.h"
  3. typedef struct Srv Srv;
  4. typedef struct Srv {
  5. int fd;
  6. int srvfd;
  7. char* service;
  8. char* mntpnt;
  9. Srv* next;
  10. Srv* prev;
  11. } Srv;
  12. static struct {
  13. VtLock* lock;
  14. Srv* head;
  15. Srv* tail;
  16. } sbox;
  17. static int
  18. srvFd(char* name, int mode, int fd, char** mntpnt)
  19. {
  20. int n, srvfd;
  21. char *p, buf[10];
  22. /*
  23. * Drop a file descriptor with given name and mode into /srv.
  24. * Create with ORCLOSE and don't close srvfd so it will be removed
  25. * automatically on process exit.
  26. */
  27. p = smprint("/srv/%s", name);
  28. if((srvfd = create(p, ORCLOSE|OWRITE, mode)) < 0){
  29. vtMemFree(p);
  30. p = smprint("#s/%s", name);
  31. if((srvfd = create(p, ORCLOSE|OWRITE, mode)) < 0){
  32. vtSetError("create %s: %r", p);
  33. vtMemFree(p);
  34. return -1;
  35. }
  36. }
  37. n = snprint(buf, sizeof(buf), "%d", fd);
  38. if(write(srvfd, buf, n) < 0){
  39. close(srvfd);
  40. vtSetError("write %s: %r", p);
  41. vtMemFree(p);
  42. return -1;
  43. }
  44. *mntpnt = p;
  45. return srvfd;
  46. }
  47. static void
  48. srvFree(Srv* srv)
  49. {
  50. if(srv->prev != nil)
  51. srv->prev->next = srv->next;
  52. else
  53. sbox.head = srv->next;
  54. if(srv->next != nil)
  55. srv->next->prev = srv->prev;
  56. else
  57. sbox.tail = srv->prev;
  58. if(srv->srvfd != -1)
  59. close(srv->srvfd);
  60. vtMemFree(srv->service);
  61. vtMemFree(srv->mntpnt);
  62. vtMemFree(srv);
  63. }
  64. static Srv*
  65. srvAlloc(char* service, int mode, int fd)
  66. {
  67. Srv *srv;
  68. int srvfd;
  69. char *mntpnt;
  70. vtLock(sbox.lock);
  71. for(srv = sbox.head; srv != nil; srv = srv->next){
  72. if(strcmp(srv->service, service) != 0)
  73. continue;
  74. vtSetError("srv: already serving '%s'", service);
  75. vtUnlock(sbox.lock);
  76. return nil;
  77. }
  78. if((srvfd = srvFd(service, mode, fd, &mntpnt)) < 0){
  79. vtUnlock(sbox.lock);
  80. return nil;
  81. }
  82. close(fd);
  83. srv = vtMemAllocZ(sizeof(Srv));
  84. srv->srvfd = srvfd;
  85. srv->service = vtStrDup(service);
  86. srv->mntpnt = mntpnt;
  87. if(sbox.tail != nil){
  88. srv->prev = sbox.tail;
  89. sbox.tail->next = srv;
  90. }
  91. else{
  92. sbox.head = srv;
  93. srv->prev = nil;
  94. }
  95. sbox.tail = srv;
  96. vtUnlock(sbox.lock);
  97. return srv;
  98. }
  99. static int
  100. cmdSrv(int argc, char* argv[])
  101. {
  102. Srv *srv;
  103. int Aflag, dflag, Pflag, fd[2], mode, pflag, r;
  104. char *usage = "usage: srv [-Adp] [service]";
  105. Con *con;
  106. Aflag = dflag = pflag = Pflag = 0;
  107. mode = 0666;
  108. ARGBEGIN{
  109. default:
  110. return cliError(usage);
  111. case 'A':
  112. Aflag = 1;
  113. break;
  114. case 'd':
  115. dflag = 1;
  116. break;
  117. case 'p':
  118. pflag = 1;
  119. mode = 0600;
  120. break;
  121. case 'P':
  122. Pflag = 1;
  123. mode = 0600;
  124. break;
  125. }ARGEND
  126. if(pflag && Pflag){
  127. vtSetError("srv: cannot use -P with -p");
  128. return 0;
  129. }
  130. switch(argc){
  131. default:
  132. return cliError(usage);
  133. case 0:
  134. vtRLock(sbox.lock);
  135. for(srv = sbox.head; srv != nil; srv = srv->next)
  136. consPrint("\t%s\t%d\n", srv->service, srv->srvfd);
  137. vtRUnlock(sbox.lock);
  138. return 1;
  139. case 1:
  140. if(!dflag)
  141. break;
  142. vtLock(sbox.lock);
  143. for(srv = sbox.head; srv != nil; srv = srv->next){
  144. if(strcmp(srv->service, argv[0]) != 0)
  145. continue;
  146. srvFree(srv);
  147. break;
  148. }
  149. vtUnlock(sbox.lock);
  150. if(srv == nil){
  151. vtSetError("srv: '%s' not found", argv[0]);
  152. return 0;
  153. }
  154. return 1;
  155. }
  156. if(pipe(fd) < 0){
  157. vtSetError("srv pipe: %r");
  158. return 0;
  159. }
  160. if((srv = srvAlloc(argv[0], mode, fd[0])) == nil){
  161. close(fd[0]); close(fd[1]);
  162. return 0;
  163. }
  164. if(pflag)
  165. r = consOpen(fd[1], srv->srvfd, -1);
  166. else{
  167. con = conAlloc(fd[1], srv->mntpnt);
  168. if(con == nil)
  169. r = 0;
  170. else{
  171. r = 1;
  172. con->noauth = Aflag;
  173. con->noperm = Pflag;
  174. }
  175. }
  176. if(r == 0){
  177. close(fd[1]);
  178. vtLock(sbox.lock);
  179. srvFree(srv);
  180. vtUnlock(sbox.lock);
  181. }
  182. return r;
  183. }
  184. int
  185. srvInit(void)
  186. {
  187. sbox.lock = vtLockAlloc();
  188. cliAddCmd("srv", cmdSrv);
  189. return 1;
  190. }