pump.c 3.3 KB

123456789101112131415161718192021222324252627282930313233343536373839404142434445464748495051525354555657585960616263646566676869707172737475767778798081828384858687888990919293949596979899100101102103104105106107108109110111112113114115116117118119120121122123124125126127128129130131132133134135136137138139140141142143144145146147148149150151152153154155156157158159160161162163164165166167168169170171172173174175176177178179180181182183184185186187188189190191192193194195196197198199200201202203204205206207208209210211212213214215216217218219220221
  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: %r\n", argv0);
  141. break;
  142. }
  143. nout += c;
  144. if(tsize && nout > tsize) {
  145. fprint(2, "%s: time limit exceeded\n", argv0);
  146. break;
  147. }
  148. }
  149. unlock(&arithlock);
  150. }
  151. void
  152. doinput(int f)
  153. {
  154. long n, l, c, xnin;
  155. lock(&arithlock);
  156. if(ssize > 0) {
  157. for (xnin = 0; xnin < ssize && !done; xnin += c) {
  158. n = kilo - (xnin - nout);
  159. if(n == 0)
  160. break;
  161. unlock(&arithlock);
  162. l = xnin % kilo;
  163. if(kilo-l < n)
  164. n = kilo-l;
  165. if(n > ibsize)
  166. n = ibsize;
  167. c = read(f, buf+l, n);
  168. lock(&arithlock);
  169. if(c <= 0) {
  170. if(c < 0)
  171. fprint(2, "%s: read error: %r\n", argv0);
  172. break;
  173. }
  174. }
  175. nin = xnin;
  176. }
  177. while(!done) {
  178. n = kilo - (nin - nout);
  179. if(n == 0) {
  180. sleepunlocked(0);
  181. continue;
  182. }
  183. l = nin % kilo;
  184. unlock(&arithlock);
  185. if(kilo-l < n)
  186. n = kilo-l;
  187. if(n > ibsize)
  188. n = ibsize;
  189. c = read(f, buf+l, n);
  190. lock(&arithlock);
  191. if(c <= 0) {
  192. if(c < 0)
  193. fprint(2, "%s: read error: %r\n", argv0);
  194. break;
  195. }
  196. nin += c;
  197. }
  198. unlock(&arithlock);
  199. }