pump.c 3.3 KB

123456789101112131415161718192021222324252627282930313233343536373839404142434445464748495051525354555657585960616263646566676869707172737475767778798081828384858687888990919293949596979899100101102103104105106107108109110111112113114115116117118119120121122123124125126127128129130131132133134135136137138139140141142143144145146147148149150151152153154155156157158159160161162163164165166167168169170171172173174175176177178179180181182183184185186187188189190191192193194195196197198199200201202203204205206207208209210211212213214215216217218219220221222
  1. /* pump - copy through circular buffer */
  2. #include <u.h>
  3. #include <libc.h>
  4. uchar* buf;
  5. Lock arithlock; /* protect 64-bit accesses: unlikely to be atomic */
  6. uvlong nin;
  7. uvlong nout;
  8. ulong kilo;
  9. ulong max;
  10. long ssize;
  11. vlong tsize;
  12. int dsize;
  13. int done;
  14. int ibsize;
  15. int obsize;
  16. int verb;
  17. void doinput(int);
  18. void dooutput(int);
  19. static void
  20. usage(void)
  21. {
  22. fprint(2, "usage: pump [-f ofile] [-k KB-buffer] [-i ireadsize]\n"
  23. "\t[-o owritesize] [-b iando] [-s start-KB] [-d sleeptime] "
  24. "[files]\n");
  25. exits("usage");
  26. }
  27. void
  28. main(int argc, char *argv[])
  29. {
  30. int i, f, fo;
  31. char *file;
  32. kilo = 5000;
  33. obsize = ibsize = 8*1024;
  34. dsize = 0;
  35. fo = 1;
  36. ARGBEGIN {
  37. default:
  38. usage();
  39. case 'b':
  40. obsize = ibsize = atoi(EARGF(usage()));
  41. break;
  42. case 'd':
  43. dsize = atoi(EARGF(usage()));
  44. break;
  45. case 'f':
  46. file = EARGF(usage());
  47. fo = create(file, 1, 0666);
  48. if(fo < 0)
  49. sysfatal("can't create %s: %r", file);
  50. break;
  51. case 'i':
  52. ibsize = atoi(EARGF(usage()));
  53. break;
  54. case 'k':
  55. kilo = atoi(EARGF(usage()));
  56. break;
  57. case 'o':
  58. obsize = atoi(EARGF(usage()));
  59. break;
  60. case 's':
  61. ssize = atoi(EARGF(usage()));
  62. if(ssize <= 0)
  63. ssize = 800;
  64. ssize <<= 10;
  65. break;
  66. case 't':
  67. tsize = atoll(EARGF(usage()));
  68. tsize *= 10584000; /* minutes */
  69. break;
  70. } ARGEND
  71. kilo <<= 10;
  72. buf = malloc(kilo);
  73. if(buf == nil)
  74. sysfatal("no memory: %r");
  75. nin = 0;
  76. nout = 0;
  77. done = 0;
  78. max = 0;
  79. switch(rfork(RFPROC|RFNOWAIT|RFNAMEG|RFMEM)) {
  80. default:
  81. dooutput(fo);
  82. break;
  83. case 0:
  84. for(i=0; i<argc; i++) {
  85. f = open(argv[i], OREAD);
  86. if(f < 0) {
  87. fprint(2, "%s: can't open %s: %r\n",
  88. argv0, argv[i]);
  89. break;
  90. }
  91. doinput(f);
  92. close(f);
  93. }
  94. if(argc == 0)
  95. doinput(0);
  96. break;
  97. case -1:
  98. fprint(2, "%s: fork failed: %r\n", argv0);
  99. break;
  100. }
  101. done = 1;
  102. exits(0);
  103. }
  104. /* call with arithlock held */
  105. static int
  106. sleepunlocked(long ms)
  107. {
  108. int r;
  109. unlock(&arithlock);
  110. r = sleep(ms);
  111. lock(&arithlock);
  112. return r;
  113. }
  114. void
  115. dooutput(int f)
  116. {
  117. long n, l, c;
  118. lock(&arithlock);
  119. for (;;) {
  120. n = nin - nout;
  121. if(n == 0) {
  122. if(done)
  123. break;
  124. sleepunlocked(dsize);
  125. continue;
  126. }
  127. if(verb && n > max) {
  128. fprint(2, "n = %ld\n", n);
  129. max = n;
  130. }
  131. l = nout % kilo;
  132. unlock(&arithlock);
  133. if(kilo-l < n)
  134. n = kilo-l;
  135. if(n > obsize)
  136. n = obsize;
  137. c = write(f, buf+l, n);
  138. lock(&arithlock);
  139. if(c != n) {
  140. fprint(2, "%s: write error at offset %,lld: %r\n",
  141. argv0, seek(f, 0, 1));
  142. break;
  143. }
  144. nout += c;
  145. if(tsize && nout > tsize) {
  146. fprint(2, "%s: time limit exceeded\n", argv0);
  147. break;
  148. }
  149. }
  150. unlock(&arithlock);
  151. }
  152. void
  153. doinput(int f)
  154. {
  155. long n, l, c, xnin;
  156. lock(&arithlock);
  157. if(ssize > 0) {
  158. for (xnin = 0; xnin < ssize && !done; xnin += c) {
  159. n = kilo - (xnin - nout);
  160. if(n == 0)
  161. break;
  162. unlock(&arithlock);
  163. l = xnin % kilo;
  164. if(kilo-l < n)
  165. n = kilo-l;
  166. if(n > ibsize)
  167. n = ibsize;
  168. c = read(f, buf+l, n);
  169. lock(&arithlock);
  170. if(c <= 0) {
  171. if(c < 0)
  172. fprint(2, "%s: read error: %r\n", argv0);
  173. break;
  174. }
  175. }
  176. nin = xnin;
  177. }
  178. while(!done) {
  179. n = kilo - (nin - nout);
  180. if(n == 0) {
  181. sleepunlocked(0);
  182. continue;
  183. }
  184. l = nin % kilo;
  185. unlock(&arithlock);
  186. if(kilo-l < n)
  187. n = kilo-l;
  188. if(n > ibsize)
  189. n = ibsize;
  190. c = read(f, buf+l, n);
  191. lock(&arithlock);
  192. if(c <= 0) {
  193. if(c < 0)
  194. fprint(2, "%s: read error: %r\n", argv0);
  195. break;
  196. }
  197. nin += c;
  198. }
  199. unlock(&arithlock);
  200. }