9srv.c 4.3 KB

123456789101112131415161718192021222324252627282930313233343536373839404142434445464748495051525354555657585960616263646566676869707172737475767778798081828384858687888990919293949596979899100101102103104105106107108109110111112113114115116117118119120121122123124125126127128129130131132133134135136137138139140141142143144145146147148149150151152153154155156157158159160161162163164165166167168169170171172173174175176177178179180181182183184185186187188189190191192193194195196197198199200201202203204205206207208209210211212213214215216217218219220221222223224225226227228229230231232233234235236237238239240241242243244245246247248249250251
  1. /*
  2. * This file is part of the UCB release of Plan 9. It is subject to the license
  3. * terms in the LICENSE file found in the top-level directory of this
  4. * distribution and at http://akaros.cs.berkeley.edu/files/Plan9License. No
  5. * part of the UCB release of Plan 9, including this file, may be copied,
  6. * modified, propagated, or distributed except according to the terms contained
  7. * in the LICENSE file.
  8. */
  9. #include "stdinc.h"
  10. #include "9.h"
  11. typedef struct Srv Srv;
  12. struct Srv {
  13. int fd;
  14. int srvfd;
  15. char* service;
  16. char* mntpnt;
  17. Srv* next;
  18. Srv* prev;
  19. };
  20. static struct {
  21. VtLock* lock;
  22. Srv* head;
  23. Srv* tail;
  24. } sbox;
  25. static int
  26. srvFd(char* name, int mode, int fd, char** mntpnt)
  27. {
  28. int n, srvfd;
  29. char *p, buf[10];
  30. /*
  31. * Drop a file descriptor with given name and mode into /srv.
  32. * Create with ORCLOSE and don't close srvfd so it will be removed
  33. * automatically on process exit.
  34. */
  35. p = smprint("/srv/%s", name);
  36. if((srvfd = create(p, ORCLOSE|OWRITE, mode)) < 0){
  37. vtMemFree(p);
  38. p = smprint("#s/%s", name);
  39. if((srvfd = create(p, ORCLOSE|OWRITE, mode)) < 0){
  40. vtSetError("create %s: %r", p);
  41. vtMemFree(p);
  42. return -1;
  43. }
  44. }
  45. n = snprint(buf, sizeof(buf), "%d", fd);
  46. if(write(srvfd, buf, n) < 0){
  47. close(srvfd);
  48. vtSetError("write %s: %r", p);
  49. vtMemFree(p);
  50. return -1;
  51. }
  52. *mntpnt = p;
  53. return srvfd;
  54. }
  55. static void
  56. srvFree(Srv* srv)
  57. {
  58. if(srv->prev != nil)
  59. srv->prev->next = srv->next;
  60. else
  61. sbox.head = srv->next;
  62. if(srv->next != nil)
  63. srv->next->prev = srv->prev;
  64. else
  65. sbox.tail = srv->prev;
  66. if(srv->srvfd != -1)
  67. close(srv->srvfd);
  68. vtMemFree(srv->service);
  69. vtMemFree(srv->mntpnt);
  70. vtMemFree(srv);
  71. }
  72. static Srv*
  73. srvAlloc(char* service, int mode, int fd)
  74. {
  75. Dir *dir;
  76. Srv *srv;
  77. int srvfd;
  78. char *mntpnt;
  79. vtLock(sbox.lock);
  80. for(srv = sbox.head; srv != nil; srv = srv->next){
  81. if(strcmp(srv->service, service) != 0)
  82. continue;
  83. /*
  84. * If the service exists, but is stale,
  85. * free it up and let the name be reused.
  86. */
  87. if((dir = dirfstat(srv->srvfd)) != nil){
  88. free(dir);
  89. vtSetError("srv: already serving '%s'", service);
  90. vtUnlock(sbox.lock);
  91. return nil;
  92. }
  93. srvFree(srv);
  94. break;
  95. }
  96. if((srvfd = srvFd(service, mode, fd, &mntpnt)) < 0){
  97. vtUnlock(sbox.lock);
  98. return nil;
  99. }
  100. close(fd);
  101. srv = vtMemAllocZ(sizeof(Srv));
  102. srv->srvfd = srvfd;
  103. srv->service = vtStrDup(service);
  104. srv->mntpnt = mntpnt;
  105. if(sbox.tail != nil){
  106. srv->prev = sbox.tail;
  107. sbox.tail->next = srv;
  108. }
  109. else{
  110. sbox.head = srv;
  111. srv->prev = nil;
  112. }
  113. sbox.tail = srv;
  114. vtUnlock(sbox.lock);
  115. return srv;
  116. }
  117. static int
  118. cmdSrv(int argc, char* argv[])
  119. {
  120. Con *con;
  121. Srv *srv;
  122. char *usage = "usage: srv [-APWdp] [service]";
  123. int conflags, dflag, fd[2], mode, pflag, r;
  124. dflag = 0;
  125. pflag = 0;
  126. conflags = 0;
  127. mode = 0666;
  128. ARGBEGIN{
  129. default:
  130. return cliError(usage);
  131. case 'A':
  132. conflags |= ConNoAuthCheck;
  133. break;
  134. case 'I':
  135. conflags |= ConIPCheck;
  136. break;
  137. case 'N':
  138. conflags |= ConNoneAllow;
  139. break;
  140. case 'P':
  141. conflags |= ConNoPermCheck;
  142. mode = 0600;
  143. break;
  144. case 'W':
  145. conflags |= ConWstatAllow;
  146. mode = 0600;
  147. break;
  148. case 'd':
  149. dflag = 1;
  150. break;
  151. case 'p':
  152. pflag = 1;
  153. mode = 0600;
  154. break;
  155. }ARGEND
  156. if(pflag && (conflags&ConNoPermCheck)){
  157. vtSetError("srv: cannot use -P with -p");
  158. return 0;
  159. }
  160. switch(argc){
  161. default:
  162. return cliError(usage);
  163. case 0:
  164. vtRLock(sbox.lock);
  165. for(srv = sbox.head; srv != nil; srv = srv->next)
  166. consPrint("\t%s\t%d\n", srv->service, srv->srvfd);
  167. vtRUnlock(sbox.lock);
  168. return 1;
  169. case 1:
  170. if(!dflag)
  171. break;
  172. vtLock(sbox.lock);
  173. for(srv = sbox.head; srv != nil; srv = srv->next){
  174. if(strcmp(srv->service, argv[0]) != 0)
  175. continue;
  176. srvFree(srv);
  177. break;
  178. }
  179. vtUnlock(sbox.lock);
  180. if(srv == nil){
  181. vtSetError("srv: '%s' not found", argv[0]);
  182. return 0;
  183. }
  184. return 1;
  185. }
  186. if(pipe(fd) < 0){
  187. vtSetError("srv pipe: %r");
  188. return 0;
  189. }
  190. if((srv = srvAlloc(argv[0], mode, fd[0])) == nil){
  191. close(fd[0]); close(fd[1]);
  192. return 0;
  193. }
  194. if(pflag)
  195. r = consOpen(fd[1], srv->srvfd, -1);
  196. else{
  197. con = conAlloc(fd[1], srv->mntpnt, conflags);
  198. if(con == nil)
  199. r = 0;
  200. else
  201. r = 1;
  202. }
  203. if(r == 0){
  204. close(fd[1]);
  205. vtLock(sbox.lock);
  206. srvFree(srv);
  207. vtUnlock(sbox.lock);
  208. }
  209. return r;
  210. }
  211. int
  212. srvInit(void)
  213. {
  214. sbox.lock = vtLockAlloc();
  215. cliAddCmd("srv", cmdSrv);
  216. return 1;
  217. }