progress.c 4.8 KB

123456789101112131415161718192021222324252627282930313233343536373839404142434445464748495051525354555657585960616263646566676869707172737475767778798081828384858687888990919293949596979899100101102103104105106107108109110111112113114115116117118119120121122123124125126127128129130131132133134135136137138139140141142143144145146147148149150151152153
  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. p->inited = 1;
  58. }
  59. void FAST_FUNC bb_progress_update(bb_progress_t *p,
  60. const char *curfile,
  61. off_t beg_range,
  62. off_t transferred,
  63. off_t totalsize)
  64. {
  65. off_t abbrevsize;
  66. unsigned since_last_update, elapsed;
  67. unsigned ratio;
  68. int barlength, i;
  69. ratio = 100;
  70. if (totalsize) {
  71. /* long long helps to have it working even if !LFS */
  72. ratio = (unsigned) (100ULL * (transferred+beg_range) / totalsize);
  73. if (ratio > 100) ratio = 100;
  74. }
  75. #if ENABLE_UNICODE_SUPPORT
  76. init_unicode();
  77. /* libbb candidate? */
  78. {
  79. wchar_t wbuf21[21];
  80. char *buf = xstrdup(curfile);
  81. unsigned len;
  82. /* trim to 20 wide chars max (sets wbuf21[20] to 0)
  83. * also, in case mbstowcs fails, we at least
  84. * dont get garbage */
  85. memset(wbuf21, 0, sizeof(wbuf21));
  86. /* convert to wide chars, no more than 20 */
  87. len = mbstowcs(wbuf21, curfile, 20); /* NB: may return -1 */
  88. /* back to multibyte; cant overflow */
  89. wcstombs(buf, wbuf21, INT_MAX);
  90. len = (len > 20) ? 0 : 20 - len;
  91. fprintf(stderr, "\r%s%*s%4d%% ", buf, len, "", ratio);
  92. free(buf);
  93. }
  94. #else
  95. fprintf(stderr, "\r%-20.20s%4d%% ", curfile, ratio);
  96. #endif
  97. barlength = get_tty2_width() - 49;
  98. if (barlength > 0) {
  99. /* god bless gcc for variable arrays :) */
  100. i = barlength * ratio / 100;
  101. {
  102. char buf[i+1];
  103. memset(buf, '*', i);
  104. buf[i] = '\0';
  105. fprintf(stderr, "|%s%*s|", buf, barlength - i, "");
  106. }
  107. }
  108. i = 0;
  109. abbrevsize = transferred + beg_range;
  110. while (abbrevsize >= 100000) {
  111. i++;
  112. abbrevsize >>= 10;
  113. }
  114. /* see http://en.wikipedia.org/wiki/Tera */
  115. fprintf(stderr, "%6d%c ", (int)abbrevsize, " kMGTPEZY"[i]);
  116. elapsed = monotonic_sec();
  117. since_last_update = elapsed - p->lastupdate_sec;
  118. if (transferred > p->lastsize) {
  119. p->lastupdate_sec = elapsed;
  120. p->lastsize = transferred;
  121. if (since_last_update >= STALLTIME) {
  122. /* We "cut off" these seconds from elapsed time
  123. * by adjusting start time */
  124. p->start_sec += since_last_update;
  125. }
  126. since_last_update = 0; /* we are un-stalled now */
  127. }
  128. elapsed -= p->start_sec; /* now it's "elapsed since start" */
  129. if (since_last_update >= STALLTIME) {
  130. fprintf(stderr, " - stalled -");
  131. } else {
  132. off_t to_download = totalsize - beg_range;
  133. if (!totalsize || transferred <= 0 || (int)elapsed <= 0 || transferred > to_download) {
  134. fprintf(stderr, "--:--:-- ETA");
  135. } else {
  136. /* to_download / (transferred/elapsed) - elapsed: */
  137. int eta = (int) ((unsigned long long)to_download*elapsed/transferred - elapsed);
  138. /* (long long helps to have working ETA even if !LFS) */
  139. i = eta % 3600;
  140. fprintf(stderr, "%02d:%02d:%02d ETA", eta / 3600, i / 60, i % 60);
  141. }
  142. }
  143. }