xfile.c 2.5 KB

123456789101112131415161718192021222324252627282930313233343536373839404142434445464748495051525354555657585960616263646566676869707172737475767778798081828384858687888990919293949596979899100101102103104105106107108109110111112113114115116117118119120121122123124125126127128129130131132133134135136137138139140141142143144145146147148149150151152153154155156157158159160161
  1. #include <u.h>
  2. #include <libc.h>
  3. #include <fcall.h>
  4. #include <thread.h>
  5. #include <9p.h>
  6. #include "dat.h"
  7. #include "fns.h"
  8. static Xfs *xhead;
  9. static Xfile *freelist;
  10. static Lock xlock, freelock;
  11. int client;
  12. Xfs *
  13. getxfs(char *name)
  14. {
  15. int fd;
  16. Dir *dir;
  17. Xfs *xf, *fxf;
  18. if(name==0 || name[0]==0)
  19. name = deffile;
  20. if(name == 0){
  21. errno = Enofilsys;
  22. return 0;
  23. }
  24. fd = open(name, rdonly ? OREAD : ORDWR);
  25. if(fd < 0){
  26. errno = Enonexist;
  27. return 0;
  28. }
  29. if((dir = dirfstat(fd)) == 0){
  30. errno = Eio;
  31. close(fd);
  32. return 0;
  33. }
  34. lock(&xlock);
  35. for(fxf=0, xf=xhead; xf; xf=xf->next){
  36. if(xf->ref == 0){
  37. if(fxf == 0)
  38. fxf = xf;
  39. continue;
  40. }
  41. if(xf->qid.path != dir->qid.path || xf->qid.vers != dir->qid.vers)
  42. continue;
  43. if(strcmp(xf->name, name) != 0 || xf->dev < 0)
  44. continue;
  45. chat("incref \"%s\", dev=%d...", xf->name, xf->dev);
  46. ++xf->ref;
  47. unlock(&xlock);
  48. close(fd);
  49. free(dir);
  50. return xf;
  51. }
  52. if(fxf==0){
  53. fxf = malloc(sizeof(Xfs));
  54. if(fxf==0){
  55. unlock(&xlock);
  56. close(fd);
  57. free(dir);
  58. errno = Enomem;
  59. return 0;
  60. }
  61. fxf->next = xhead;
  62. xhead = fxf;
  63. }
  64. chat("alloc \"%s\", dev=%d...", name, fd);
  65. fxf->name = strdup(name);
  66. fxf->ref = 1;
  67. fxf->qid = dir->qid;
  68. fxf->dev = fd;
  69. fxf->fmt = 0;
  70. fxf->ptr = 0;
  71. free(dir);
  72. if( ext2fs(fxf)<0 ){
  73. xhead = fxf->next;
  74. free(fxf);
  75. unlock(&xlock);
  76. return 0;
  77. }
  78. unlock(&xlock);
  79. return fxf;
  80. }
  81. void
  82. refxfs(Xfs *xf, int delta)
  83. {
  84. lock(&xlock);
  85. xf->ref += delta;
  86. if(xf->ref == 0){
  87. /*mchat("free \"%s\", dev=%d...", xf->name, xf->dev);
  88. dumpbuf();*/
  89. CleanSuper(xf);
  90. syncbuf();
  91. free(xf->name);
  92. purgebuf(xf);
  93. if(xf->dev >= 0){
  94. close(xf->dev);
  95. xf->dev = -1;
  96. }
  97. }
  98. unlock(&xlock);
  99. }
  100. Xfile *
  101. xfile(Fid *fid, int flag)
  102. {
  103. Xfile *f;
  104. f = (Xfile*)fid->aux;
  105. switch(flag){
  106. default:
  107. panic("xfile");
  108. case Asis:
  109. return (f && f->xf && f->xf->dev < 0) ? 0 : f;
  110. case Clean:
  111. if (f) chat("Clean and fid->aux already exists\n");
  112. break;
  113. case Clunk:
  114. if(f){
  115. clean(f);
  116. lock(&freelock);
  117. f->next = freelist;
  118. freelist = f;
  119. unlock(&freelock);
  120. fid->aux = 0;
  121. }
  122. return 0;
  123. }
  124. if(f)
  125. return clean(f);
  126. lock(&freelock);
  127. if(f = freelist){ /* assign = */
  128. freelist = f->next;
  129. unlock(&freelock);
  130. } else {
  131. unlock(&freelock);
  132. f = malloc(sizeof(Xfile));
  133. }
  134. fid->aux = f;
  135. f->fid = fid->fid;
  136. f->client = client;
  137. f->xf = 0;
  138. f->ptr = 0;
  139. f->root = 0;
  140. return f;
  141. }
  142. Xfile *
  143. clean(Xfile *f)
  144. {
  145. if(f->xf && f->root){
  146. refxfs(f->xf, -1);
  147. f->xf = 0;
  148. }
  149. f->xf = 0;
  150. f->root = 0;
  151. f->dirindex = 0;
  152. return f;
  153. }