CoreNEURON
progressbar.cpp
Go to the documentation of this file.
1 /**
2  * \file
3  * \author Trevor Fountain
4  * \author Johannes Buchner
5  * \author Erik Garrison
6  * \date 2010-2014
7  * \copyright BSD 3-Clause
8  *
9  * progressbar -- a C class (by convention) for displaying progress
10  * on the command line (to stdout).
11  */
13 
14 #include <cassert>
15 #include <cstddef>
16 #include <climits>
17 #include <unistd.h>
18 
19 /// How wide we assume the screen is if termcap fails.
20 enum { DEFAULT_SCREEN_WIDTH = 80 };
21 
22 /// The smallest that the bar can ever be (not including borders)
23 enum { MINIMUM_BAR_WIDTH = 10 };
24 
25 /// The format in which the estimated remaining time will be reported
26 static const char* const ETA_FORMAT = "t: %-6.2f ETA:%2dh%02dm%02ds";
27 
28 /// The maximum number of characters that the ETA_FORMAT can ever yield
29 enum { ETA_FORMAT_LENGTH = 13 };
30 
31 /// Amount of screen width taken up by whitespace (i.e. whitespace between label/bar/ETA components)
32 enum { WHITESPACE_LENGTH = 2 };
33 
34 /// The amount of width taken up by the border of the bar component.
35 enum { BAR_BORDER_WIDTH = 2 };
36 
37 /// The maximum number of bar redraws (to avoid frequent output in long runs)
38 enum { BAR_DRAW_COUNT_MAX = 500 };
39 
41 
42 /// Models a duration of time broken into hour/minute/second components. The number of seconds
43 /// should be less than the
44 /// number of seconds in one minute, and the number of minutes should be less than the number of
45 /// minutes in one hour.
47  int hours;
48  int minutes;
49  int seconds;
50 };
51 
52 static void progressbar_draw(const progressbar* bar);
53 static int progressbar_remaining_seconds(const progressbar* bar);
54 
55 /**
56  * Create a new progress bar with the specified label, max number of steps, and format string.
57  * Note that `format` must be exactly three characters long, e.g. "<->" to render a progress
58  * bar like "<---------->". Returns nullptr if there isn't enough memory to allocate a progressbar
59  */
60 progressbar* progressbar_new_with_format(const char* label, unsigned long max, const char* format) {
61  auto* new_bar = static_cast<progressbar*>(malloc(sizeof(progressbar)));
62  if (new_bar == nullptr) {
63  return nullptr;
64  }
65 
66  new_bar->max = max;
67  new_bar->value = 0;
68  new_bar->draw_time_interval = isatty(STDOUT_FILENO) ? BAR_DRAW_INTERVAL
70  new_bar->t = 0;
71  new_bar->start = time(nullptr);
72  assert(3 == strlen(format) && "format must be 3 characters in length");
73  new_bar->format.begin = format[0];
74  new_bar->format.fill = format[1];
75  new_bar->format.end = format[2];
76 
77  progressbar_update_label(new_bar, label);
78  progressbar_draw(new_bar);
79  new_bar->prev_t = difftime(time(nullptr), new_bar->start);
80  new_bar->drawn_count = 1;
81 
82  return new_bar;
83 }
84 
85 /**
86  * Create a new progress bar with the specified label and max number of steps.
87  */
88 progressbar* progressbar_new(const char* label, unsigned long max) {
89  return progressbar_new_with_format(label, max, "|=|");
90 }
91 
92 void progressbar_update_label(progressbar* bar, const char* label) {
93  bar->label = label;
94 }
95 
96 /**
97  * Delete an existing progress bar.
98  */
100  free(bar);
101 }
102 
103 /**
104  * Increment an existing progressbar by `value` steps.
105  * Additionally issues a redraw in case a certain time interval has elapsed (min: 1sec)
106  * Reasons for a larger interval are:
107  * - Stdout is not TTY
108  * - Respect BAR_DRAW_COUNT_MAX
109  */
110 void progressbar_update(progressbar* bar, unsigned long value, double t) {
111  bar->value = value;
112  bar->t = t;
113  int sim_time = difftime(time(nullptr), bar->start);
114 
115  // If there is not enough time passed to redraw the progress bar return
116  if ((sim_time - bar->prev_t) < bar->draw_time_interval) {
117  return;
118  }
119 
120  progressbar_draw(bar);
121 
122  bar->drawn_count++;
123  bar->prev_t = sim_time;
124 
125  if (bar->drawn_count >= BAR_DRAW_COUNT_MAX || sim_time < 15) {
126  // Dont change the interval after the limit. Simulation should be over any moment and
127  // avoid the calc of draw_time_interval which could raise DIV/0
128  // Also, dont do it the first 15sec to avoid really bad estimates which could potentially
129  // delay a better estimate too far away in the future.
130  return;
131  }
132 
133  // Sample ETA to calculate the next interval until the redraw of the progressbar
134  int eta_s = progressbar_remaining_seconds(bar);
135  bar->draw_time_interval = eta_s / (BAR_DRAW_COUNT_MAX - bar->drawn_count);
136 
138  bar->draw_time_interval = isatty(STDOUT_FILENO)
141  : bar->draw_time_interval)
143  }
144 }
145 
146 /**
147  * Increment an existing progressbar by a single step.
148  */
149 void progressbar_inc(progressbar* bar, double t) {
150  progressbar_update(bar, bar->value + 1, t);
151 }
152 
153 static void progressbar_write_char(FILE* file, const int ch, const size_t times) {
154  for (std::size_t i = 0; i < times; ++i) {
155  fputc(ch, file);
156  }
157 }
158 
159 static int progressbar_max(int x, int y) {
160  return x > y ? x : y;
161 }
162 
163 static unsigned int get_screen_width(void) {
164  return DEFAULT_SCREEN_WIDTH;
165 }
166 
167 static int progressbar_bar_width(int screen_width, int label_length) {
169  screen_width - label_length - ETA_FORMAT_LENGTH - WHITESPACE_LENGTH);
170 }
171 
172 static int progressbar_label_width(int screen_width, int label_length, int bar_width) {
173  int eta_width = ETA_FORMAT_LENGTH;
174 
175  // If the progressbar is too wide to fit on the screen, we must sacrifice the label.
176  if (label_length + 1 + bar_width + 1 + ETA_FORMAT_LENGTH > screen_width) {
177  return progressbar_max(0, screen_width - bar_width - eta_width - WHITESPACE_LENGTH);
178  } else {
179  return label_length;
180  }
181 }
182 
184  double offset = difftime(time(nullptr), bar->start);
185  if (bar->value > 0 && offset > 0) {
186  return (offset / (double) bar->value) * (bar->max - bar->value);
187  } else {
188  return 0;
189  }
190 }
191 
193  int hours = seconds / 3600;
194  seconds -= hours * 3600;
195  int minutes = seconds / 60;
196  seconds -= minutes * 60;
197 
198  progressbar_time_components components = {hours, minutes, seconds};
199  return components;
200 }
201 
202 static void progressbar_draw(const progressbar* bar) {
203  int screen_width = get_screen_width();
204  int label_length = strlen(bar->label);
205  int bar_width = progressbar_bar_width(screen_width, label_length);
206  int label_width = progressbar_label_width(screen_width, label_length, bar_width);
207 
208  int progressbar_completed = (bar->value >= bar->max);
209  int bar_piece_count = bar_width - BAR_BORDER_WIDTH;
210  int bar_piece_current = (progressbar_completed)
211  ? bar_piece_count
212  : bar_piece_count * ((double) bar->value / bar->max);
213 
215  (progressbar_completed)
216  ? progressbar_calc_time_components(difftime(time(nullptr), bar->start))
218 
219  if (label_width == 0) {
220  // The label would usually have a trailing space, but in the case that we don't print
221  // a label, the bar can use that space instead.
222  bar_width += 1;
223  } else {
224  // Draw the label
225  fwrite(bar->label, 1, label_width, stdout);
226  fputc(' ', stdout);
227  }
228 
229  // Draw the progressbar
230  fputc(bar->format.begin, stdout);
231  progressbar_write_char(stdout, bar->format.fill, bar_piece_current);
232  progressbar_write_char(stdout, ' ', bar_piece_count - bar_piece_current);
233  fputc(bar->format.end, stdout);
234 
235  // Draw the ETA
236  fputc(' ', stdout);
237  fprintf(stdout, ETA_FORMAT, bar->t, eta.hours, eta.minutes, eta.seconds);
238  fputc('\r', stdout);
239  fflush(stdout);
240 }
241 
242 /**
243  * Finish a progressbar, indicating 100% completion, and free it.
244  */
246  // Make sure we fill the progressbar so things look complete.
247  progressbar_draw(bar);
248 
249  // Print a newline, so that future outputs to stdout look prettier
250  fprintf(stdout, "\n");
251 
252  // We've finished with this progressbar, so go ahead and free it.
253  progressbar_free(bar);
254 }
progressbar::t
double t
current time (added for simulation)
Definition: progressbar.hpp:47
ETA_FORMAT
static const char *const ETA_FORMAT
The format in which the estimated remaining time will be reported.
Definition: progressbar.cpp:26
progressbar_draw
static void progressbar_draw(const progressbar *bar)
Definition: progressbar.cpp:202
progressbar_finish
void progressbar_finish(progressbar *bar)
Finish a progressbar, indicating 100% completion, and free it.
Definition: progressbar.cpp:245
progressbar_time_components::seconds
int seconds
Definition: progressbar.cpp:49
progressbar_max
static int progressbar_max(int x, int y)
Definition: progressbar.cpp:159
progressbar_inc
void progressbar_inc(progressbar *bar, double t)
Increment an existing progressbar by a single step.
Definition: progressbar.cpp:149
progressbar_update
void progressbar_update(progressbar *bar, unsigned long value, double t)
Increment an existing progressbar by value steps.
Definition: progressbar.cpp:110
progressbar_new_with_format
progressbar * progressbar_new_with_format(const char *label, unsigned long max, const char *format)
Create a new progress bar with the specified label, max number of steps, and format string.
Definition: progressbar.cpp:60
progressbar::value
unsigned long value
current value
Definition: progressbar.hpp:26
ETA_FORMAT_LENGTH
@ ETA_FORMAT_LENGTH
Definition: progressbar.cpp:29
progressbar_calc_time_components
static progressbar_time_components progressbar_calc_time_components(int seconds)
Definition: progressbar.cpp:192
progressbar
Progressbar data structure (do not modify or create directly)
Definition: progressbar.hpp:21
BAR_DRAW_INTERVAL
@ BAR_DRAW_INTERVAL
Definition: progressbar.cpp:40
BAR_DRAW_COUNT_MAX
@ BAR_DRAW_COUNT_MAX
Definition: progressbar.cpp:38
progressbar::max
unsigned long max
maximum value
Definition: progressbar.hpp:23
progressbar_remaining_seconds
static int progressbar_remaining_seconds(const progressbar *bar)
Definition: progressbar.cpp:183
progressbar_update_label
void progressbar_update_label(progressbar *bar, const char *label)
Set the label of the progressbar.
Definition: progressbar.cpp:92
progressbar::draw_time_interval
time_t draw_time_interval
time interval between consecutive bar redraws (seconds)
Definition: progressbar.hpp:32
progressbar::start
time_t start
time progressbar was started
Definition: progressbar.hpp:38
coreneuron::t
double t
Definition: register_mech.cpp:22
progressbar_time_components::hours
int hours
Definition: progressbar.cpp:47
i
#define i
Definition: md1redef.h:19
progressbar::format
struct progressbar::@8 format
characters for the beginning, filling and end of the progressbar.
progressbar_free
void progressbar_free(progressbar *bar)
Delete an existing progress bar.
Definition: progressbar.cpp:99
progressbar_new
progressbar * progressbar_new(const char *label, unsigned long max)
Create a new progress bar with the specified label and max number of steps.
Definition: progressbar.cpp:88
DEFAULT_SCREEN_WIDTH
@ DEFAULT_SCREEN_WIDTH
Definition: progressbar.cpp:20
progressbar_time_components::minutes
int minutes
Definition: progressbar.cpp:48
progressbar_write_char
static void progressbar_write_char(FILE *file, const int ch, const size_t times)
Definition: progressbar.cpp:153
progressbar::fill
char fill
Definition: progressbar.hpp:53
get_screen_width
static unsigned int get_screen_width(void)
Definition: progressbar.cpp:163
progressbar::drawn_count
unsigned long drawn_count
number of redrawn bars
Definition: progressbar.hpp:35
progressbar::end
char end
Definition: progressbar.hpp:54
progressbar::prev_t
time_t prev_t
time progressbar was drawn for last time
Definition: progressbar.hpp:41
BAR_DRAW_INTERVAL_NOTTY
@ BAR_DRAW_INTERVAL_NOTTY
Definition: progressbar.cpp:40
progressbar::label
const char * label
label
Definition: progressbar.hpp:44
MINIMUM_BAR_WIDTH
@ MINIMUM_BAR_WIDTH
Definition: progressbar.cpp:23
progressbar.hpp
progressbar_time_components
Models a duration of time broken into hour/minute/second components.
Definition: progressbar.cpp:46
progressbar_label_width
static int progressbar_label_width(int screen_width, int label_length, int bar_width)
Definition: progressbar.cpp:172
progressbar::begin
char begin
Definition: progressbar.hpp:52
BAR_BORDER_WIDTH
@ BAR_BORDER_WIDTH
Definition: progressbar.cpp:35
progressbar_bar_width
static int progressbar_bar_width(int screen_width, int label_length)
Definition: progressbar.cpp:167
WHITESPACE_LENGTH
@ WHITESPACE_LENGTH
Definition: progressbar.cpp:32