HILA
Loading...
Searching...
No Matches
input.h
Go to the documentation of this file.
1/** @file input.h */
2#ifndef PARAM_INPUT_H
3#define PARAM_INPUT_H
4
5#include <string>
6#include "defs.h"
7#include "lattice.h"
8#include "plumbing/com_mpi.h"
9
10namespace hila {
11
12/**
13 * @brief hila::input - Class for parsing runtime parameter files.
14 *
15 * @details Input files consist normally of "key <value>" -pairs.
16 *
17 * It is important to note that the data structure employed by the input class is not a dictionary,
18 * but technically a stack. This means that the values are read in sequentially, thus they must be
19 * in the file in the order they are read in.
20 *
21 * In examples of other methods we will refer to the input object as f. Creating the input object is
22 * done with:
23 *
24 * \code{.cpp}
25 * hila::input f("filename"); // initialization with filename opens the file for input
26 * \endcode
27 *
28 * One can also initialize the input object without specifying the filename:
29 *
30 * \code{.cpp}
31 * hila::input f; // initialization with filename opens the file for input
32 * \endcode
33 *
34 * in which case the `hila::input::open` method needs to be called separately.
35 *
36 * __Comment character '#'__: everything after # in input file is a comment to the end of the line.
37 *
38 * __NOTE__: methods which broadcast to all nodes (default) must be called from all nodes
39 * synchronously. These include open(), get(), get_value() with bcast=true, get_item with
40 * bcast=true.
41 *
42 * Thus:
43 * \code{.cpp}
44 * if (hila::myrank() == 0) {
45 * double v = f.get("a value");
46 * ...
47 * }
48 * \endcode
49 * deadlocks (if there are more than 1 rank). Method `f.get_value(v,"a value",false)` can be used in
50 * this context.
51 */
52class input {
53
54 private:
55 std::ifstream inputfile;
56 bool is_initialized = false;
57 std::string filename;
58 bool use_cin;
59 int file_number; // running file index
60
61 std::string linebuffer;
62 size_t lb_start = 0; // linebuffer start index
63 bool is_line_printed;
64 bool speaking = true; // false does not print information
65
66 std::vector<std::string> cmdline_p;
67
68 public:
69 input() {}
70 ~input() {
71 close();
72 }
73 input(const std::string &fname) {
74 open(fname);
75 }
76
77 /**
78 * @brief Open file that parameters are read from
79 * @details If input class is initialized with path to file that needs to be opened,
80 * then the file will be opened automatically, and this method does not need to be called.
81 *
82 * If no argument is given then filename will be interperated as default input file name
83 *
84 * In the case that use_cmdline is True and if no argument is given and filename is given with
85 * -i {alternative_file} cmdline argument then {alternative_file} will be the specified file to
86 * be opened.
87 *
88 * @param fname Path to file that needs to be read
89 * @param use_cmdline Default True
90 * @param exit_on_error If true exit on error. Return value is passed to all MPI nodes. Default
91 * True
92 * @return true
93 * @return false
94 */
95 bool open(const std::string &fname, bool use_cmdline = true, bool exit_on_error = true);
96
97 /** @brief Closes input parameter file
98 * @details After `f.close()` the file can be reopened with `f.open("file")` and the stack is
99 * reset. File is also closed when variable "f" goes out of scope.
100 */
101 void close();
102
103 /**
104 * @brief Silence print output during file reading
105 * @details
106 * \code{.cpp}
107 * f.quiet(); // don't print read items to hila::out
108 * f.quiet(false); // re-enable printing
109 * \endcode
110 * By default hila::input methods print everything read to hila::out0 for logging. `f.quiet()`
111 * disables this.
112 *
113 * @param really
114 */
115 void quiet(bool really = true) {
116 speaking = !really;
117 }
118
119 /**
120 * @brief returntype is a special class for resolving get("label") return type
121 */
123 public:
124 std::string label;
125 input *parent;
126
127 returntype(const std::string &str, input *a) : label(str), parent(a) {}
128
129 /// cast operator does the conversion - disable nullptr_t cast used
130 /// in hila types
131 /// disable type "char" because then c++ does not know how to initialize strings
132 template <typename T, std::enable_if_t<(hila::is_complex_or_arithmetic<T>::value &&
133 !std::is_same<T, char>::value) ||
134 std::is_same<T, std::string>::value,
135 int> = 0>
136 operator T() {
137 T val;
138 if (!parent->get_value(val, label, true))
140 return val;
141 }
142
143 template <typename T, int n>
144 operator Vector<n, T>() {
145 Vector<n, T> val;
146 if (!parent->get_value(val, label, true))
148 return val;
149 }
150
151 operator CoordinateVector() {
153 if (!parent->get_value(val, label, true))
155 return val;
156 }
157
158 template <typename T, std::enable_if_t<hila::is_complex_or_arithmetic<T>::value ||
159 std::is_same<T, std::string>::value,
160 int> = 0>
161 operator std::vector<T>() {
162 std::vector<T> val;
163 if (!parent->get_value(val, label, true))
165 return val;
166 }
167 };
168
169 /**
170 * @brief Get next value in stack of read in input string from parameters file.
171 * @details Use as
172 * \code{.cpp}
173 * var = f.get("key");
174 * \endcode
175 * reads in a key-value pair
176 * \code{.txt}
177 * key <value(s)>
178 * \endcode
179 * from the input file f, and returns the value of type of variable var.
180 * The value is broadcast to all MPI nodes. The method infers
181 * the type of the returned variable from the type of the assignment.
182 *
183 * Key is an alphanumeric string, which may contain words separated by whitespace.
184 *
185 * The get method simply traverses the parsed in stack of input values in the given
186 * parameters file. A functionality of the method is that the given argument key is skipped and
187 * the value that this key is assigned to is then fetched and returned.
188 *
189 * Technically the key's can also be read in if no argument is given to get. The main
190 * functionality of the key argument is to book keep by the user that all parameters are read
191 * in. If the key that is wanting to be read doesn't exist, then get throws an error and the
192 * program is stopped.
193 *
194 * With this logic one could construct a parameters file of only values:
195 *
196 * \code {.txt}
197 * value_1
198 * value_2
199 * .
200 * .
201 * .
202 * value_n
203 * \endcode
204 *
205 * and read in the values in a loop by simply calling hila::input::get() consecutively, but this
206 * is not advised.
207 *
208 * The type of the value to be read in is inferred by the variable the value is fetched into. If
209 * the value cannot be casted into the variable, then the parser will throw an error and crash.
210 *
211 * ## Types which the parser supports with examples
212 *
213 * Multiple items are separated by commas, whitespace is not significant.
214 *
215 * ### Any arithmetic type (ints/floats)
216 * @code
217 * int i = f.get("number of cats");
218 * @endcode
219 * __matches__ " number of cats 5 "
220 *
221 * ### Complex<float/double>
222 * @code{.cpp}
223 * Complex<double> phase = f.get("complex phase");
224 * @endcode
225 * __matches__ "complex phase (0.4, 0.5)"
226 *
227 * complex values are given in pairs within ( , )
228 *
229 * ### std::string
230 * @code{.cpp}
231 * std::string s = f.get("key");
232 * @endcode
233 * __matches__ "key <string value>" where string value is either
234 *
235 * - sequence of non-whitespace chars, delimited by whitespace, eol, ','
236 * or '#'.
237 *
238 * - characters enclosed by quotes "..". These have to pair
239 * within the same line. Quote marks are removed.
240 *
241 * ### CoordinateVector
242 * @code{.cpp}
243 * CoordinateVector v;
244 * v = f.get("lattice size");
245 * @endcode
246 * __matches__ "lattice size 32, 32, 32, 32" (if NDIM == 4).
247 *
248 * ### Vector<T,int>
249 * @code{.cpp}
250 * Vector<5,int> vec = f.get("initial vector")
251 * @endcode
252 * __matches__ "initial vector 1, 2, 3, 4, 5"
253 *
254 * ### std::vector<T>
255 * @code{.cpp}
256 * std::vector<double> dvec = f.get("vec");
257 * @endcode
258 * __matches__ "vec 3,4, 5.5, 7.8, 4" and returns a vector of double values.
259 * The numbers are read until they are not followed by a comma. If comma
260 * is the last non-whitespace character on the line, reading continues to
261 * the next line.
262 *
263 * T is one of the other supported types.
264 *
265 * __NOTE__: The parser will crash if during reading in a multi valued line (Vector,
266 * std::vector) the line ends with a ",". The parser will interpret that there is data on the
267 * next line to read in and will fail during the next get call.
268 *
269 * @param key Parameter to be read in.
270 * @return returntype Value corresponding to input key
271 */
272 inline returntype get(const std::string &key) {
273 return returntype(key, this);
274 }
275
276 inline returntype get() {
277 return returntype("", this);
278 }
279
280 /**
281 * @brief Read input (alternative to get())
282 * @details
283 * Val can be any value used in get()-method above. If broadcast==false,
284 * the value is not broadcast to other nodes. The return value is false if
285 * the value could not be read successfully, true otherwise.
286 * This method does not exit on error (but an error message may be printed)
287 * Example:
288 * \code{.cpp}
289 * int i,j;
290 * bool success;
291 * success = get_value(i, "key", false); // only node 0 gets i
292 * success = get_value(j,"key2"); // all nodes get j
293 *\endcode
294 * __NOTE__: if broadcast == false the return value is correct only on node 0.
295 *
296 * Supported types same as for hila::input::get
297 * @tparam T
298 * @param val variable to store gotten value in. Infers the type for the fetched value
299 * @param label key to fetch value for
300 * @param bcast default true. If true the value will be broadcasted to all nodes
301 * @return true Returns true on success
302 * @return false Returns false if return value is correct only on node 0.
303 */
304 template <typename T>
305 bool get_value(T &val, const std::string &label, bool bcast = true) {
306 val = {};
307 bool no_error = handle_key(label); // removes whitespace
308
309 if (hila::myrank() == 0 && no_error) {
310 std::string tok;
311 if (!(get_token(tok) && is_value(tok, val))) {
312
313 if (speaking)
314 hila::out << "Error: expecting a value of type '" << type_id<T>() << "' after '"
315 << label << "'\n";
316
317 no_error = false;
318 }
319 }
320
321 if (bcast) {
322 // string has to be treated separately
323 if constexpr (std::is_same<T, std::string>::value) {
324 hila::broadcast(val);
325 hila::broadcast(no_error);
326 } else {
327 hila::broadcast2(val, no_error);
328 }
329 }
330
331 return no_error;
332 }
333
334 // get_value for Complex<T>
335 template <typename T>
336 bool get_value(Complex<T> &val, const std::string &label, bool bcast = true) {
337 val = 0;
338 bool no_error = handle_key(label); // removes whitespace
339
340 if (hila::myrank() == 0 && no_error) {
341 std::string tok;
342 T re, im;
343
344 no_error =
345 (match_token("(") && get_token(tok) && is_value(tok, re) && match_token(",") &&
346 get_token(tok) && is_value(tok, im) && match_token(")"));
347 if (!no_error && speaking) {
348 hila::out << "Error: expecting complex value '(re,im)' after '" << label << "'\n";
349 }
350
351 val = Complex<T>(re, im);
352 }
353
354 if (bcast) {
355 hila::broadcast2(val, no_error);
356 }
357 return no_error;
358 }
359 // get_value for #Vector<n,T>
360 template <int n, typename T>
361 bool get_value(Vector<n, T> &val, const std::string &label, bool bcast = true) {
362 val = 0;
363 bool no_error = true;
364
365 if (hila::myrank() == 0) {
366 no_error = get_value(val[0], label, false);
367 for (int i = 1; i < n && no_error; i++) {
368 no_error = get_value(val[i], ",", false);
369 }
370
371 if (!no_error && speaking) {
372 hila::out << "Error: expecting " << n << " comma-separated " << type_id<T>()
373 << "s after '" << label << "'\n";
374 }
375 }
376
377 if (bcast) {
378 hila::broadcast2(val, no_error);
379 }
380 return no_error;
381 }
382
383 // get_value for #CoordinateVector
384 template <int n = NDIM>
385 bool get_value(CoordinateVector &val, const std::string &label, bool bcast = true) {
387 bool b = get_value(iv, label, bcast);
388 val = iv;
389 return b;
390 }
391
392 // get_value for std::vector<T>
393 template <typename T>
394 bool get_value(std::vector<T> &val, const std::string &label, bool bcast = true) {
395 val = {};
396 bool no_error = true;
397
398 if (hila::myrank() == 0) {
399 T v;
400 no_error = get_value(v, label, false);
401 val.push_back(v);
402 while (no_error && match_token(",")) {
403 no_error = get_value(v, "", false);
404 val.push_back(v);
405 }
406
407 if (!no_error && speaking) {
408 hila::out << "Error: expecting a comma-separated list of " << type_id<T>()
409 << "s after '" << label << "'\n";
410 }
411 }
412
413 if (bcast) {
414 hila::broadcast(val);
415 hila::broadcast(no_error);
416 }
417
418 return no_error;
419 }
420 /** @} */
421
422 /**
423 * @brief Identify item from a list
424 * @details"items" contains the allowed entries. Return value is the
425 * index of the item found in input file.
426 *
427 * If the value of the optional bool parameter broadcast is:
428 * - true (default): the result is broadcast to all nodes
429 * and the program exits if no matches found.
430 *
431 * - false: result is not broadcast, and if no match found returns -1.
432 *
433 * Special item values:
434 *
435 * - "%f" matches a float or double value
436 * - "%i" matches an int or long
437 * - "%s" matches any string value
438 *
439 * If one of these is matched, it has to be read again with corresponding
440 * get() or get_value() -method. get_item doesn't progress the stack, so get() will fetch the
441 * next value which the item result corresponds to
442 *
443 * Examples:
444 * \code{.cpp}
445 * i = f.get_item("colour", {"red","green","blue"});
446 * \endcode
447 * will return value 1 if f contains "colour green", and quits the
448 * program if none of the 3 alternatives are found.
449 * \code{.cpp}
450 * double clover;
451 * int i = f.get_item("c_sw", {"tree","perturbative","%f"} );
452 * if (i == 2)
453 * clover = f.get();
454 * else { ...
455 * \endcode
456 * If file contains:
457 * - c_sw perturbative get_item() returns 1
458 * - c_sw 1.343 get_item() returns 2 and subsequent get() sets c_sw = 1.343
459 * - c_sw abcd error message and quit
460 *
461 * __NOTE__: "%s" matches anything. It should be the last item in the list.
462 * (The items are tested in order and first to match is returned.)
463 *
464 * @param label key to match in parameters file
465 * @param items list of items to identify with
466 * @param bcast Default true, if true broadcast to all MPI ranks
467 * @return int index of identified item in users defined list
468 */
469 int get_item(const std::string &label, const std::vector<std::string> &items,
470 bool bcast = true);
471
472 private:
473 /// a helper method to give type name
474 template <typename T>
475 inline const char *type_id() {
476 return nullptr;
477 }
478
479 bool peek_token(std::string &tok);
480 bool get_token(std::string &tok);
481 bool match_token(const std::string &tok);
482
483 void scan_cmdline(const std::string &key, int &end_of_key);
484
485 bool scan_string(std::string &val);
486
487 template <typename T, std::enable_if_t<std::is_arithmetic<T>::value, int> = 0>
488 bool is_value(const std::string &s, T &val) {
489 std::istringstream ss(s);
490 ss >> val;
491 if (ss.fail() || ss.bad())
492 return false;
493 else
494 return true;
495 }
496
497 bool is_value(const std::string &s, std::string &val);
498
499 bool contains_word_list(const std::string &list, int &end_of_key);
500
501 std::string remove_quotes(const std::string &val);
502
503 void print_linebuf(int eok);
504
505 bool get_line();
506 bool handle_key(const std::string &c);
507
508 bool remove_whitespace();
509};
510
511/// give here specializations of the type_id helper
512
513template <>
514inline const char *input::type_id<int>() {
515 return "int";
516}
517template <>
518inline const char *input::type_id<long>() {
519 return "long";
520}
521template <>
522inline const char *input::type_id<long long>() {
523 return "long long";
524}
525template <>
526inline const char *input::type_id<unsigned int>() {
527 return "unsigned int";
528}
529template <>
530inline const char *input::type_id<unsigned long>() {
531 return "unsigned long";
532}
533template <>
534inline const char *input::type_id<unsigned long long>() {
535 return "unsigned long long";
536}
537
538template <>
539inline const char *input::type_id<float>() {
540 return "float";
541}
542template <>
543inline const char *input::type_id<double>() {
544 return "double";
545}
546template <>
547inline const char *input::type_id<std::string>() {
548 return "string";
549}
550template <>
551inline const char *input::type_id<Complex<float>>() {
552 return "complex value";
553}
554template <>
555inline const char *input::type_id<Complex<double>>() {
556 return "complex value";
557}
558
559
560} // namespace hila
561
562#endif
Complex definition.
Definition cmplx.h:56
Matrix class which defines matrix operations.
Definition matrix.h:1620
returntype is a special class for resolving get("label") return type
Definition input.h:122
hila::input - Class for parsing runtime parameter files.
Definition input.h:52
void close()
Closes input parameter file.
Definition input.cpp:94
returntype get(const std::string &key)
Get next value in stack of read in input string from parameters file.
Definition input.h:272
bool get_value(T &val, const std::string &label, bool bcast=true)
Read input (alternative to get())
Definition input.h:305
bool open(const std::string &fname, bool use_cmdline=true, bool exit_on_error=true)
Open file that parameters are read from.
Definition input.cpp:27
void quiet(bool really=true)
Silence print output during file reading.
Definition input.h:115
int get_item(const std::string &label, const std::vector< std::string > &items, bool bcast=true)
Identify item from a list.
Definition input.cpp:361
CoordinateVector_t< int > CoordinateVector
CoordinateVector alias for CoordinateVector_t.
This file defines all includes for HILA.
Implement hila::swap for gauge fields.
Definition array.h:981
int myrank()
rank of this node
Definition com_mpi.cpp:234
std::ostream out
this is our default output file stream
T broadcast(T &var, int rank=0)
Broadcast the value of var to all MPI ranks from rank (default=0).
Definition com_mpi.h:168
void finishrun()
Normal, controlled exit - all nodes must call this. Prints timing information and information about c...
void broadcast2(T &t, U &u, int rank=0)
and broadcast with two values
Definition com_mpi.h:261