HILA
Loading...
Searching...
No Matches
input.cpp
1#include <sstream>
2#include <iostream>
3#include <fstream>
4#include <regex>
5#include <type_traits>
6#include "defs.h"
7#include "input.h"
8#include "cmdline.h"
9#include <errno.h>
10#include <iomanip>
11
12#define COMMENT_CHAR '#'
13
14#define CMDLINE_USED_FLAG "#ÄÄ#"
15
16static int input_file_count = 0;
17
18//////////////////////////////////////////////////////////////////////////////
19// Parameter file input system
20// Check input.h for user instructions
21//////////////////////////////////////////////////////////////////////////////
22
23namespace hila {
24
25static std::string empty_key("");
26
27bool input::open(const std::string &file_name, bool use_cmdline, bool exit_on_error) {
28
29 std::string fname;
30
31 file_number = ++input_file_count;
32 cmdline_p.clear();
33
34 if (use_cmdline && input_file_count == 1 && hila::cmdline.flag_present("-i")) {
35 // input_file_count guarantees this is done only 1st time
36
37 fname = hila::cmdline.get_string("-i");
38
39 } else {
40 fname = file_name;
41 }
42
43 bool got_error = false;
44 if (hila::myrank() == 0) {
45 if (is_initialized) {
46 if (speaking)
47 hila::out0 << "Error: file '" << fname << "' cannot be opened because '" << filename
48 << "' is open in this input variable\n";
49
50 got_error = true;
51 } else {
52 filename = fname;
53 if (fname == "-") {
54 use_cin = true;
55 if (speaking)
56 hila::print_dashed_line("Reading from standard input");
57 } else {
58 use_cin = false;
59 inputfile.open(fname);
60 if (!inputfile.is_open()) {
61 if(hila::partitions.number() > 1) {
62 // try to open inputfile from upper level
63 filename = "../" + fname;
64 inputfile.open(filename);
65 }
66 }
67
68 if (inputfile.is_open()) {
69 is_initialized = true;
70 if (speaking)
71 hila::print_dashed_line("Reading file " + filename);
72
73 } else {
74 if (speaking) {
75 if (exit_on_error)
76 hila::out0 << "ERROR: ";
77 hila::out0 << "Input file '" << fname << "' could not be opened\n";
78 }
79
80 got_error = true;
81 }
82 }
83 }
84 }
85
86 hila::broadcast(got_error);
87 if (got_error && exit_on_error) {
89 }
90
91 return !got_error;
92}
93
95 if (is_initialized && !use_cin) {
96 if (inputfile.is_open())
97 inputfile.close();
98 is_initialized = false;
99 }
100
101 // check if some cmdline -p args are unused
102 bool is_unused = false;
103 if (hila::myrank() == 0 && file_number == 1 && cmdline_p.size() > 0) {
104 for (int i = 0; i < cmdline_p.size(); i += 2) {
105 if (cmdline_p[i] != CMDLINE_USED_FLAG) {
106 hila::out0 << "Error: unused command line -p argument, flag " << cmdline_p[i]
107 << '\n';
108 is_unused = true;
109 }
110 }
111 }
112 hila::broadcast(is_unused);
113 if (is_unused)
115
116 if (speaking)
117 hila::print_dashed_line();
118
119 // automatic cleaning of other vars
120}
121
122// read one line skipping comments and initial whitespace
123bool input::get_line() {
124 if (hila::myrank() == 0) {
125 do {
126 std::istream *inptr;
127
128 if (use_cin)
129 inptr = &std::cin;
130 else
131 inptr = &inputfile;
132
133 *inptr >> std::ws; // remove initial whitespace
134 if (!std::getline(*inptr, linebuffer)) {
135 linebuffer.clear();
136 lb_start = 0;
137 return false;
138 }
139 } while (linebuffer.at(0) == COMMENT_CHAR);
140 size_t i = linebuffer.find(COMMENT_CHAR);
141 if (i != std::string::npos)
142 linebuffer.resize(i);
143 lb_start = 0;
144
145 is_line_printed = false; // not yet printed
146 }
147 return true; // sync this at another spot
148}
149
150// print the read-in line with a bit special formatting
151void input::print_linebuf(int end_of_key) {
152 if (hila::myrank() == 0 && speaking) {
153
154 if (is_line_printed)
155 return;
156 is_line_printed = true;
157
158 int i = 0;
159 while (i < linebuffer.size() && std::isspace(linebuffer[i]))
160 i++;
161 for (; i < linebuffer.size() && i < end_of_key; i++) {
162 hila::out0 << linebuffer[i];
163 }
164 hila::out0 << ' ';
165 if (end_of_key > 0) {
166 for (int j = i; j < 20; j++)
167 hila::out0 << ' ';
168 }
169
170 while (i < linebuffer.size() && std::isspace(linebuffer[i]))
171 i++;
172 if (i < linebuffer.size()) {
173 hila::out0 << linebuffer.substr(i);
174 }
175 hila::out0 << std::endl; // use endl to show output here
176 }
177}
178
179// remove leading whitespace, incl. lines
180bool input::remove_whitespace() {
181 if (hila::myrank() == 0) {
182 while (lb_start < linebuffer.size() && std::isspace(linebuffer[lb_start]))
183 lb_start++;
184 if (lb_start == linebuffer.size())
185 return get_line();
186 }
187 return true; // do not broadcast yet, should be used only in node 0
188}
189
190// returns true if line contains the word list at the beginning of line. list
191// contains the word separated by whitespace. If match found, advances the
192// lb_start to new position. end_of_key is the index where key match on line ends
193
194bool input::contains_word_list(const std::string &list, int &end_of_key) {
195 const char *p = linebuffer.c_str() + lb_start;
196 const char *q = list.c_str();
197 while (std::isspace(*p))
198 p++;
199 while (std::isspace(*q))
200 q++;
201
202 bool last_space = false;
203 char last_char = 0;
204 while (*p && *q) {
205 // compare non-space chars
206 while (*p && *q && *p == *q && !std::isspace(*q)) {
207 last_char = *p;
208 p++;
209 q++;
210 last_space = false;
211 }
212 if (std::isspace(*q) && std::isspace(*p)) {
213 // matching spaces, skip
214 while (std::isspace(*p))
215 p++;
216 while (std::isspace(*q))
217 q++;
218 last_space = true;
219 }
220
221 if (*p != *q)
222 break;
223 }
224 // if line contained the words in list, *q = 0.
225
226 while (std::isspace(*q))
227 q++;
228 // if *q != 0 then some chars do not match
229 // if *q == 0 then p has to fully match. This is true if
230 // the last match was space or *p == 0 or *p is space or comma (,)
231 if (*q != 0 || !(last_space || *p == 0 || std::isspace(*p) || last_char == ',' || *p == ','))
232 return false;
233
234 end_of_key = p - linebuffer.c_str();
235
236 while (std::isspace(*p))
237 p++;
238 lb_start = p - linebuffer.c_str();
239 return true;
240}
241
242// find tokens separated by whitespace or ,
243// inside of " .. " is one token too.
244// returns true if the next token (on the same line) is found
245
246bool input::peek_token(std::string &tok) {
247 if (!remove_whitespace())
248 return false;
249 size_t i;
250 bool in_quotes = false;
251 for (i = lb_start; i < linebuffer.size() &&
252 ((!std::isspace(linebuffer[i]) && linebuffer[i] != ',') || in_quotes);
253 i++) {
254 if (linebuffer[i] == '"') {
255 in_quotes = !in_quotes;
256 }
257 }
258 if (i == lb_start && linebuffer[i] == ',')
259 i++; // this happens for only ,
260
261 if (in_quotes) {
262 hila::out0 << "Error: unbalanced quotes\n";
263 exit(1); // unclean exit
264 }
265 tok = linebuffer.substr(lb_start, i - lb_start);
266 return true;
267}
268
269// consumes the token too
270
271bool input::get_token(std::string &tok) {
272 if (peek_token(tok)) {
273 lb_start += tok.size();
274 return true;
275 }
276 return false;
277}
278
279// match_token returns true and consumes the token if it matches the argument
280
281bool input::match_token(const std::string &tok) {
282 std::string s;
283 if (peek_token(s) && s == tok) {
284 lb_start += tok.size();
285 return true;
286 }
287 return false;
288}
289
290// require the (typically beginning of line) key for parameters
291
292bool input::handle_key(const std::string &key) {
293 if (hila::myrank() == 0) {
294 // check the linebuffer for stuff
295 remove_whitespace();
296
297 int end_of_key = 0;
298 if (key.size() > 0 && !contains_word_list(key, end_of_key)) {
299 if (speaking) {
300 hila::out0 << "Error: expecting key '" << key << "', found instead:\n";
301 hila::out0 << "\"" << linebuffer << "\"\n----\n";
302 }
303 return false;
304 }
305
306 // check the command line args, if this is the first input file
307 scan_cmdline(key, end_of_key);
308
309 print_linebuf(end_of_key);
310 }
311 return true;
312}
313
314
315void input::scan_cmdline(const std::string &key, int &end_of_key) {
316
317 if (file_number == 1) {
318 if (cmdline_p.size() == 0 && hila::cmdline.flag_present("-p")) {
319 cmdline_p = hila::cmdline.values("-p");
320 }
321
322 for (int i = 0; i < cmdline_p.size(); i += 2) {
323 if (cmdline_p[i] == key) {
324
325 cmdline_p[i] = CMDLINE_USED_FLAG;
326 linebuffer = key + " " + cmdline_p.at(i + 1);
327 lb_start = key.size();
328 end_of_key = lb_start;
329 if (speaking)
330 hila::out0 << "OVERRIDE from command line: " << key << ":\n";
331 break;
332 }
333 }
334 }
335}
336
337
338// is the input string int/double/string and return it
339
340
341// a trivial function, useful for template
342bool input::is_value(const std::string &str, std::string &val) {
343 val = remove_quotes(str);
344 return true;
345}
346
347std::string input::remove_quotes(const std::string &val) {
348 size_t i, j;
349 std::string res;
350 res = val;
351 for (j = i = 0; i < val.size(); i++)
352 if (val[i] != '"')
353 res[j++] = val[i];
354 res.resize(j);
355 return res;
356}
357
358// expects "label <item>" -line, where <item> matches one of the std::strings in
359// items. returns the index of the item. If not found, errors out
360
361int input::get_item(const std::string &label, const std::vector<std::string> &items, bool bcast) {
362
363 bool no_error = handle_key(label);
364 int item = -1;
365 double d;
366 std::string s;
367
368 if (hila::myrank() == 0) {
369 if (no_error && peek_token(s)) {
370
371 for (int i = 0; i < items.size() && item < 0; i++) {
372 double dval;
373 long lval;
374 int end_of_key;
375
376 // clang-format off
377 if ((items[i] == "%s") ||
378 (items[i] == "%f" && is_value(s,dval)) ||
379 (items[i] == "%i" && is_value(s,lval)) ||
380 contains_word_list(items[i],end_of_key)) {
381 item = i;
382 }
383 // clang-format on
384 }
385 }
386
387 if (item < 0) {
388
389 // error, nothing was found
390 no_error = false;
391 if (speaking) {
392 hila::out0 << "Input '" << label << "' must be one of: ";
393 for (int i = 0; i < items.size(); i++) {
394 if (items[i] == "%s")
395 hila::out0 << "<string> ";
396 else if (items[i] == "%f")
397 hila::out0 << "<float/double> ";
398 else if (items[i] == "%i")
399 hila::out0 << "<int/long> ";
400 else
401 hila::out0 << '\'' << items[i] << "' ";
402 }
403 hila::out0 << '\n';
404 }
405 }
406 }
407
408 if (bcast) {
409 hila::broadcast2(item, no_error);
410
411 // with broadcast exit on error
412 if (!no_error)
414 }
415
416 return item;
417}
418
419} // namespace hila
void close()
Closes input parameter file.
Definition input.cpp:94
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
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
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 out0
This writes output only from main process (node 0)
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