progress.c 4.8 KB

123456789101112131415161718192021222324252627282930313233343536373839404142434445464748495051525354555657585960616263646566676869707172737475767778798081828384858687888990919293949596979899100101102103104105106107108109110111112113114115116117118119120121122123124125126127128129130131132133134135136137138139140141142143144145146147148149150151152
  1. /* vi: set sw=4 ts=4: */
  2. /*
  3. * Progress bar code.
  4. */
  5. /* Original copyright notice which applies to the CONFIG_FEATURE_WGET_STATUSBAR stuff,
  6. * much of which was blatantly stolen from openssh.
  7. */
  8. /*-
  9. * Copyright (c) 1992, 1993
  10. * The Regents of the University of California. All rights reserved.
  11. *
  12. * Redistribution and use in source and binary forms, with or without
  13. * modification, are permitted provided that the following conditions
  14. * are met:
  15. * 1. Redistributions of source code must retain the above copyright
  16. * notice, this list of conditions and the following disclaimer.
  17. * 2. Redistributions in binary form must reproduce the above copyright
  18. * notice, this list of conditions and the following disclaimer in the
  19. * documentation and/or other materials provided with the distribution.
  20. *
  21. * 3. <BSD Advertising Clause omitted per the July 22, 1999 licensing change
  22. * ftp://ftp.cs.berkeley.edu/pub/4bsd/README.Impt.License.Change>
  23. *
  24. * 4. Neither the name of the University nor the names of its contributors
  25. * may be used to endorse or promote products derived from this software
  26. * without specific prior written permission.
  27. *
  28. * THIS SOFTWARE IS PROVIDED BY THE REGENTS AND CONTRIBUTORS ``AS IS'' AND
  29. * ANY EXPRESS OR IMPLIED WARRANTIES, INCLUDING, BUT NOT LIMITED TO, THE
  30. * IMPLIED WARRANTIES OF MERCHANTABILITY AND FITNESS FOR A PARTICULAR PURPOSE
  31. * ARE DISCLAIMED. IN NO EVENT SHALL THE REGENTS OR CONTRIBUTORS BE LIABLE
  32. * FOR ANY DIRECT, INDIRECT, INCIDENTAL, SPECIAL, EXEMPLARY, OR CONSEQUENTIAL
  33. * DAMAGES (INCLUDING, BUT NOT LIMITED TO, PROCUREMENT OF SUBSTITUTE GOODS
  34. * OR SERVICES; LOSS OF USE, DATA, OR PROFITS; OR BUSINESS INTERRUPTION)
  35. * HOWEVER CAUSED AND ON ANY THEORY OF LIABILITY, WHETHER IN CONTRACT, STRICT
  36. * LIABILITY, OR TORT (INCLUDING NEGLIGENCE OR OTHERWISE) ARISING IN ANY WAY
  37. * OUT OF THE USE OF THIS SOFTWARE, EVEN IF ADVISED OF THE POSSIBILITY OF
  38. * SUCH DAMAGE.
  39. */
  40. #include "libbb.h"
  41. #include "unicode.h"
  42. enum {
  43. /* Seconds when xfer considered "stalled" */
  44. STALLTIME = 5
  45. };
  46. static unsigned int get_tty2_width(void)
  47. {
  48. unsigned width;
  49. get_terminal_width_height(2, &width, NULL);
  50. return width;
  51. }
  52. void FAST_FUNC bb_progress_init(bb_progress_t *p)
  53. {
  54. p->start_sec = monotonic_sec();
  55. p->lastupdate_sec = p->start_sec;
  56. p->lastsize = 0;
  57. }
  58. void FAST_FUNC bb_progress_update(bb_progress_t *p,
  59. const char *curfile,
  60. off_t beg_range,
  61. off_t transferred,
  62. off_t totalsize)
  63. {
  64. off_t abbrevsize;
  65. unsigned since_last_update, elapsed;
  66. unsigned ratio;
  67. int barlength, i;
  68. ratio = 100;
  69. if (totalsize) {
  70. /* long long helps to have it working even if !LFS */
  71. ratio = (unsigned) (100ULL * (transferred+beg_range) / totalsize);
  72. if (ratio > 100) ratio = 100;
  73. }
  74. #if ENABLE_FEATURE_ASSUME_UNICODE
  75. init_unicode();
  76. /* libbb candidate? */
  77. {
  78. wchar_t wbuf21[21];
  79. char *buf = xstrdup(curfile);
  80. unsigned len;
  81. /* trim to 20 wide chars max (sets wbuf21[20] to 0)
  82. * also, in case mbstowcs fails, we at least
  83. * dont get garbage */
  84. memset(wbuf21, 0, sizeof(wbuf21));
  85. /* convert to wide chars, no more than 20 */
  86. len = mbstowcs(wbuf21, curfile, 20); /* NB: may return -1 */
  87. /* back to multibyte; cant overflow */
  88. wcstombs(buf, wbuf21, INT_MAX);
  89. len = (len > 20) ? 0 : 20 - len;
  90. fprintf(stderr, "\r%s%*s%4d%% ", buf, len, "", ratio);
  91. free(buf);
  92. }
  93. #else
  94. fprintf(stderr, "\r%-20.20s%4d%% ", curfile, ratio);
  95. #endif
  96. barlength = get_tty2_width() - 49;
  97. if (barlength > 0) {
  98. /* god bless gcc for variable arrays :) */
  99. i = barlength * ratio / 100;
  100. {
  101. char buf[i+1];
  102. memset(buf, '*', i);
  103. buf[i] = '\0';
  104. fprintf(stderr, "|%s%*s|", buf, barlength - i, "");
  105. }
  106. }
  107. i = 0;
  108. abbrevsize = transferred + beg_range;
  109. while (abbrevsize >= 100000) {
  110. i++;
  111. abbrevsize >>= 10;
  112. }
  113. /* see http://en.wikipedia.org/wiki/Tera */
  114. fprintf(stderr, "%6d%c ", (int)abbrevsize, " kMGTPEZY"[i]);
  115. elapsed = monotonic_sec();
  116. since_last_update = elapsed - p->lastupdate_sec;
  117. if (transferred > p->lastsize) {
  118. p->lastupdate_sec = elapsed;
  119. p->lastsize = transferred;
  120. if (since_last_update >= STALLTIME) {
  121. /* We "cut off" these seconds from elapsed time
  122. * by adjusting start time */
  123. p->start_sec += since_last_update;
  124. }
  125. since_last_update = 0; /* we are un-stalled now */
  126. }
  127. elapsed -= p->start_sec; /* now it's "elapsed since start" */
  128. if (since_last_update >= STALLTIME) {
  129. fprintf(stderr, " - stalled -");
  130. } else {
  131. off_t to_download = totalsize - beg_range;
  132. if (!totalsize || transferred <= 0 || (int)elapsed <= 0 || transferred > to_download) {
  133. fprintf(stderr, "--:--:-- ETA");
  134. } else {
  135. /* to_download / (transferred/elapsed) - elapsed: */
  136. int eta = (int) ((unsigned long long)to_download*elapsed/transferred - elapsed);
  137. /* (long long helps to have working ETA even if !LFS) */
  138. i = eta % 3600;
  139. fprintf(stderr, "%02d:%02d:%02d ETA", eta / 3600, i / 60, i % 60);
  140. }
  141. }
  142. }