9srv.c 3.8 KB

123456789101112131415161718192021222324252627282930313233343536373839404142434445464748495051525354555657585960616263646566676869707172737475767778798081828384858687888990919293949596979899100101102103104105106107108109110111112113114115116117118119120121122123124125126127128129130131132133134135136137138139140141142143144145146147148149150151152153154155156157158159160161162163164165166167168169170171172173174175176177178179180181182183184185186187188189190191192193194195196197198199200201202203204205206207208209210211212213214215216217218219220221222223224225226227228229230231232233234235236237238
  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. Dir *dir;
  68. Srv *srv;
  69. int srvfd;
  70. char *mntpnt;
  71. vtLock(sbox.lock);
  72. for(srv = sbox.head; srv != nil; srv = srv->next){
  73. if(strcmp(srv->service, service) != 0)
  74. continue;
  75. /*
  76. * If the service exists, but is stale,
  77. * free it up and let the name be reused.
  78. */
  79. if((dir = dirfstat(srv->srvfd)) != nil){
  80. free(dir);
  81. vtSetError("srv: already serving '%s'", service);
  82. vtUnlock(sbox.lock);
  83. return nil;
  84. }
  85. srvFree(srv);
  86. break;
  87. }
  88. if((srvfd = srvFd(service, mode, fd, &mntpnt)) < 0){
  89. vtUnlock(sbox.lock);
  90. return nil;
  91. }
  92. close(fd);
  93. srv = vtMemAllocZ(sizeof(Srv));
  94. srv->srvfd = srvfd;
  95. srv->service = vtStrDup(service);
  96. srv->mntpnt = mntpnt;
  97. if(sbox.tail != nil){
  98. srv->prev = sbox.tail;
  99. sbox.tail->next = srv;
  100. }
  101. else{
  102. sbox.head = srv;
  103. srv->prev = nil;
  104. }
  105. sbox.tail = srv;
  106. vtUnlock(sbox.lock);
  107. return srv;
  108. }
  109. static int
  110. cmdSrv(int argc, char* argv[])
  111. {
  112. Con *con;
  113. Srv *srv;
  114. char *usage = "usage: srv [-APWdp] [service]";
  115. int Aflag, Pflag, Wflag, dflag, fd[2], mode, pflag, r;
  116. Aflag = Pflag = Wflag = dflag = pflag = 0;
  117. mode = 0666;
  118. ARGBEGIN{
  119. default:
  120. return cliError(usage);
  121. case 'A':
  122. Aflag = 1;
  123. break;
  124. case 'P':
  125. Pflag = 1;
  126. mode = 0600;
  127. break;
  128. case 'W':
  129. Wflag = 1;
  130. mode = 0600;
  131. break;
  132. case 'd':
  133. dflag = 1;
  134. break;
  135. case 'p':
  136. pflag = 1;
  137. mode = 0600;
  138. break;
  139. }ARGEND
  140. if(pflag && Pflag){
  141. vtSetError("srv: cannot use -P with -p");
  142. return 0;
  143. }
  144. switch(argc){
  145. default:
  146. return cliError(usage);
  147. case 0:
  148. vtRLock(sbox.lock);
  149. for(srv = sbox.head; srv != nil; srv = srv->next)
  150. consPrint("\t%s\t%d\n", srv->service, srv->srvfd);
  151. vtRUnlock(sbox.lock);
  152. return 1;
  153. case 1:
  154. if(!dflag)
  155. break;
  156. vtLock(sbox.lock);
  157. for(srv = sbox.head; srv != nil; srv = srv->next){
  158. if(strcmp(srv->service, argv[0]) != 0)
  159. continue;
  160. srvFree(srv);
  161. break;
  162. }
  163. vtUnlock(sbox.lock);
  164. if(srv == nil){
  165. vtSetError("srv: '%s' not found", argv[0]);
  166. return 0;
  167. }
  168. return 1;
  169. }
  170. if(pipe(fd) < 0){
  171. vtSetError("srv pipe: %r");
  172. return 0;
  173. }
  174. if((srv = srvAlloc(argv[0], mode, fd[0])) == nil){
  175. close(fd[0]); close(fd[1]);
  176. return 0;
  177. }
  178. if(pflag)
  179. r = consOpen(fd[1], srv->srvfd, -1);
  180. else{
  181. con = conAlloc(fd[1], srv->mntpnt);
  182. if(con == nil)
  183. r = 0;
  184. else{
  185. r = 1;
  186. con->noauth = Aflag;
  187. con->noperm = Pflag;
  188. con->wstatallow = Wflag;
  189. }
  190. }
  191. if(r == 0){
  192. close(fd[1]);
  193. vtLock(sbox.lock);
  194. srvFree(srv);
  195. vtUnlock(sbox.lock);
  196. }
  197. return r;
  198. }
  199. int
  200. srvInit(void)
  201. {
  202. sbox.lock = vtLockAlloc();
  203. cliAddCmd("srv", cmdSrv);
  204. return 1;
  205. }