tool_cb_prg.c 4.4 KB

123456789101112131415161718192021222324252627282930313233343536373839404142434445464748495051525354555657585960616263646566676869707172737475767778798081828384858687888990919293949596979899100101102103104105106107108109110111112113114115116117118119120121122123124125126127128129130131132133134135136137138139140141142143144145146147148149150
  1. /***************************************************************************
  2. * _ _ ____ _
  3. * Project ___| | | | _ \| |
  4. * / __| | | | |_) | |
  5. * | (__| |_| | _ <| |___
  6. * \___|\___/|_| \_\_____|
  7. *
  8. * Copyright (C) 1998 - 2014, Daniel Stenberg, <daniel@haxx.se>, et al.
  9. *
  10. * This software is licensed as described in the file COPYING, which
  11. * you should have received as part of this distribution. The terms
  12. * are also available at http://curl.haxx.se/docs/copyright.html.
  13. *
  14. * You may opt to use, copy, modify, merge, publish, distribute and/or sell
  15. * copies of the Software, and permit persons to whom the Software is
  16. * furnished to do so, under the terms of the COPYING file.
  17. *
  18. * This software is distributed on an "AS IS" basis, WITHOUT WARRANTY OF ANY
  19. * KIND, either express or implied.
  20. *
  21. ***************************************************************************/
  22. #include "tool_setup.h"
  23. #define ENABLE_CURLX_PRINTF
  24. /* use our own printf() functions */
  25. #include "curlx.h"
  26. #include "tool_cfgable.h"
  27. #include "tool_cb_prg.h"
  28. #include "tool_util.h"
  29. #include "memdebug.h" /* keep this as LAST include */
  30. /*
  31. ** callback for CURLOPT_XFERINFOFUNCTION
  32. */
  33. #define MAX_BARLENGTH 256
  34. int tool_progress_cb(void *clientp,
  35. curl_off_t dltotal, curl_off_t dlnow,
  36. curl_off_t ultotal, curl_off_t ulnow)
  37. {
  38. /* The original progress-bar source code was written for curl by Lars Aas,
  39. and this new edition inherits some of his concepts. */
  40. char line[MAX_BARLENGTH+1];
  41. char format[40];
  42. double frac;
  43. double percent;
  44. int barwidth;
  45. int num;
  46. struct timeval now = tvnow();
  47. struct ProgressData *bar = (struct ProgressData *)clientp;
  48. curl_off_t total;
  49. curl_off_t point;
  50. /* expected transfer size */
  51. total = dltotal + ultotal + bar->initial_size;
  52. /* we've come this far */
  53. point = dlnow + ulnow + bar->initial_size;
  54. if(bar->calls && (tvdiff(now, bar->prevtime) < 100L) && point < total)
  55. /* after first call, limit progress-bar updating to 10 Hz */
  56. /* update when we're at 100% even if last update is less than 200ms ago */
  57. return 0;
  58. if(point > total)
  59. /* we have got more than the expected total! */
  60. total = point;
  61. /* simply count invokes */
  62. bar->calls++;
  63. if(total < 1) {
  64. curl_off_t prevblock = bar->prev / 1024;
  65. curl_off_t thisblock = point / 1024;
  66. while(thisblock > prevblock) {
  67. fprintf(bar->out, "#");
  68. prevblock++;
  69. }
  70. }
  71. else if(point != bar->prev) {
  72. frac = (double)point / (double)total;
  73. percent = frac * 100.0f;
  74. barwidth = bar->width - 7;
  75. num = (int) (((double)barwidth) * frac);
  76. if(num > MAX_BARLENGTH)
  77. num = MAX_BARLENGTH;
  78. memset(line, '#', num);
  79. line[num] = '\0';
  80. snprintf(format, sizeof(format), "\r%%-%ds %%5.1f%%%%", barwidth);
  81. fprintf(bar->out, format, line, percent);
  82. }
  83. fflush(bar->out);
  84. bar->prev = point;
  85. bar->prevtime = now;
  86. return 0;
  87. }
  88. void progressbarinit(struct ProgressData *bar,
  89. struct OperationConfig *config)
  90. {
  91. #ifdef __EMX__
  92. /* 20000318 mgs */
  93. int scr_size[2];
  94. #endif
  95. char *colp;
  96. memset(bar, 0, sizeof(struct ProgressData));
  97. /* pass this through to progress function so
  98. * it can display progress towards total file
  99. * not just the part that's left. (21-may-03, dbyron) */
  100. if(config->use_resume)
  101. bar->initial_size = config->resume_from;
  102. /* TODO: get terminal width through ansi escapes or something similar.
  103. try to update width when xterm is resized... - 19990617 larsa */
  104. #ifndef __EMX__
  105. /* 20000318 mgs
  106. * OS/2 users most likely won't have this env var set, and besides that
  107. * we're using our own way to determine screen width */
  108. colp = curlx_getenv("COLUMNS");
  109. if(colp) {
  110. char *endptr;
  111. long num = strtol(colp, &endptr, 10);
  112. if((endptr != colp) && (endptr == colp + strlen(colp)) && (num > 0))
  113. bar->width = (int)num;
  114. else
  115. bar->width = 79;
  116. curl_free(colp);
  117. }
  118. else
  119. bar->width = 79;
  120. #else
  121. /* 20000318 mgs
  122. * We use this emx library call to get the screen width, and subtract
  123. * one from what we got in order to avoid a problem with the cursor
  124. * advancing to the next line if we print a string that is as long as
  125. * the screen is wide. */
  126. _scrsize(scr_size);
  127. bar->width = scr_size[0] - 1;
  128. #endif
  129. bar->out = config->global->errors;
  130. }