The Battle for Wesnoth  1.19.10+dev
wesnoth.cpp
Go to the documentation of this file.
1 /*
2  Copyright (C) 2003 - 2025
3  by David White <dave@whitevine.net>
4  Part of the Battle for Wesnoth Project https://www.wesnoth.org/
5 
6  This program is free software; you can redistribute it and/or modify
7  it under the terms of the GNU General Public License as published by
8  the Free Software Foundation; either version 2 of the License, or
9  (at your option) any later version.
10  This program is distributed in the hope that it will be useful,
11  but WITHOUT ANY WARRANTY.
12 
13  See the COPYING file for more details.
14 */
15 
16 #include "addon/manager.hpp"
17 #include "build_info.hpp"
18 #include "commandline_argv.hpp"
19 #include "commandline_options.hpp" // for commandline_options, etc
20 #include "config.hpp" // for config, config::error, etc
21 #include "cursor.hpp" // for set, CURSOR_TYPE::NORMAL, etc
22 #include "events.hpp"
23 #include "filesystem.hpp" // for filesystem::file_exists, filesystem::io_exception, etc
24 #include "floating_label.hpp"
25 #include "font/error.hpp" // for error
26 #include "font/font_config.hpp" // for load_font_config, etc
27 #include "formula/formula.hpp" // for formula_error
28 #include "game_config.hpp" // for path, debug, debug_lua, etc
29 #include "game_config_manager.hpp" // for game_config_manager, etc
30 #include "game_end_exceptions.hpp"
31 #include "game_launcher.hpp" // for game_launcher, etc
32 #include "gettext.hpp"
33 #include "gui/core/event/handler.hpp" // for tmanager
35 #include "gui/dialogs/message.hpp" // for show_error_message
37 #include "gui/dialogs/title_screen.hpp" // for title_screen, etc
38 #include "gui/gui.hpp" // for init
39 #include "log.hpp" // for LOG_STREAM, general, logger, etc
44 #include "sdl/exception.hpp" // for exception
45 #include "serialization/binary_or_text.hpp" // for config_writer
46 #include "serialization/chrono.hpp"
47 #include "serialization/parser.hpp" // for read
48 #include "serialization/preprocessor.hpp" // for preproc_define, etc
49 #include "serialization/schema_validator.hpp" // for strict_validation_enabled and schema_validator
50 #include "sound.hpp" // for commit_music_changes, etc
51 #include "utils/optimer.hpp"
52 #include "formula/string_utils.hpp" // VGETTEXT
53 #include <functional>
54 #include "game_version.hpp" // for version_info
55 #include "video.hpp" // for video::error and video::quit
56 #include "wesconfig.h" // for PACKAGE
57 #include "widgets/button.hpp" // for button
58 #include "wml_exception.hpp" // for wml_exception
59 
61 #ifdef _WIN32
62 #include "log_windows.hpp"
63 
64 #include <float.h>
65 #endif // _WIN32
66 
67 #ifndef _MSC_VER
68 #include <fenv.h>
69 #endif // _MSC_VER
70 
71 #include <SDL2/SDL.h> // for SDL_Init, SDL_INIT_TIMER
72 
73 #include <boost/program_options/errors.hpp> // for error
74 #include <boost/algorithm/string/predicate.hpp> // for checking cmdline options
75 #include "utils/optional_fwd.hpp"
76 
77 #include <cerrno> // for ENOMEM
78 #include <clocale> // for setlocale, LC_ALL, etc
79 #include <cstdio> // for remove, fprintf, stderr
80 #include <cstdlib> // for srand, exit
81 #include <exception> // for exception
82 #include <vector>
83 #include <iostream>
84 
85 //#define NO_CATCH_AT_GAME_END
86 
87 #ifdef _WIN32
88 
89 #ifdef INADDR_ANY
90 #undef INADDR_ANY
91 #endif
92 
93 #ifdef INADDR_BROADCAST
94 #undef INADDR_BROADCAST
95 #endif
96 
97 #ifdef INADDR_NONE
98 #undef INADDR_NONE
99 #endif
100 
101 #include <windows.h>
102 
103 #endif // _WIN32
104 
105 #ifdef DEBUG_WINDOW_LAYOUT_GRAPHS
106 #include "gui/widgets/debug.hpp"
107 #endif
108 
109 // this file acts as a launcher for various gui2 dialogs
110 // so this reduces some boilerplate.
111 using namespace gui2::dialogs;
112 
113 static lg::log_domain log_config("config");
114 #define LOG_CONFIG LOG_STREAM(info, log_config)
115 
116 #define LOG_GENERAL LOG_STREAM(info, lg::general())
117 
118 static lg::log_domain log_preprocessor("preprocessor");
119 #define LOG_PREPROC LOG_STREAM(info, log_preprocessor)
120 
121 // this is needed to allow identical functionality with clean refactoring
122 // play_game only returns on an error, all returns within play_game can
123 // be replaced with this
124 static void safe_exit(int res)
125 {
126  LOG_GENERAL << "exiting with code " << res;
127  exit(res);
128 }
129 
130 static void handle_preprocess_string(const commandline_options& cmdline_opts)
131 {
132  preproc_map defines_map;
133 
134  if(cmdline_opts.preprocess_input_macros) {
135  std::string file = *cmdline_opts.preprocess_input_macros;
136  if(!filesystem::file_exists(file)) {
137  PLAIN_LOG << "please specify an existing file. File " << file << " doesn't exist.";
138  return;
139  }
140 
141  PLAIN_LOG << "Reading cached defines from: " << file;
142 
143  config cfg;
144 
145  try {
146  cfg = io::read(*filesystem::istream_file(file));
147  } catch(const config::error& e) {
148  PLAIN_LOG << "Caught a config error while parsing file '" << file << "':\n" << e.message;
149  }
150 
151  int read = 0;
152 
153  // use static preproc_define::read_pair(config) to make a object
154  for(const auto [_, cfg] : cfg.all_children_view()) {
155  const preproc_map::value_type def = preproc_define::read_pair(cfg);
156  defines_map[def.first] = def.second;
157  ++read;
158  }
159 
160  PLAIN_LOG << "Read " << read << " defines.";
161  }
162 
163  if(cmdline_opts.preprocess_defines) {
164  // add the specified defines
165  for(const std::string& define : *cmdline_opts.preprocess_defines) {
166  if(define.empty()) {
167  PLAIN_LOG << "empty define supplied";
168  continue;
169  }
170 
171  LOG_PREPROC << "adding define: " << define;
172  defines_map.emplace(define, preproc_define(define));
173  }
174  }
175 
176  // add the WESNOTH_VERSION define
177  defines_map["WESNOTH_VERSION"] = preproc_define(game_config::wesnoth_version.str());
178 
179  // preprocess resource
180  PLAIN_LOG << "preprocessing specified string: " << *cmdline_opts.preprocess_source_string;
181  const utils::ms_optimer timer(
182  [](const auto& timer) { PLAIN_LOG << "preprocessing finished. Took " << timer << " ticks."; });
183  std::cout << preprocess_string(*cmdline_opts.preprocess_source_string, &defines_map) << std::endl;
184  PLAIN_LOG << "added " << defines_map.size() << " defines.";
185 }
186 
187 static void handle_preprocess_command(const commandline_options& cmdline_opts)
188 {
189  preproc_map input_macros;
190 
191  if(cmdline_opts.preprocess_input_macros) {
192  std::string file = *cmdline_opts.preprocess_input_macros;
193  if(filesystem::file_exists(file) == false) {
194  PLAIN_LOG << "please specify an existing file. File " << file << " doesn't exist.";
195  return;
196  }
197 
198  PLAIN_LOG << "Reading cached defines from: " << file;
199 
200  config cfg;
201 
202  try {
203  cfg = io::read(*filesystem::istream_file(file));
204  } catch(const config::error& e) {
205  PLAIN_LOG << "Caught a config error while parsing file '" << file << "':\n" << e.message;
206  }
207 
208  int read = 0;
209 
210  // use static preproc_define::read_pair(config) to make a object
211  for(const auto [_, cfg] : cfg.all_children_view()) {
212  const preproc_map::value_type def = preproc_define::read_pair(cfg);
213  input_macros[def.first] = def.second;
214  ++read;
215  }
216 
217  PLAIN_LOG << "Read " << read << " defines.";
218  }
219 
220  const std::string resourceToProcess(*cmdline_opts.preprocess_path);
221  const std::string targetDir(*cmdline_opts.preprocess_target);
222 
223  const utils::ms_optimer timer(
224  [](const auto& timer) { PLAIN_LOG << "preprocessing finished. Took " << timer << " ticks."; });
225 
226  // If the users add the SKIP_CORE define we won't preprocess data/core
227  bool skipCore = false;
228  bool skipTerrainGFX = false;
229 
230  // The 'core_defines_map' is the one got from data/core macros
231  preproc_map defines_map(input_macros);
232 
233  if(cmdline_opts.preprocess_defines) {
234  // add the specified defines
235  for(const std::string& define : *cmdline_opts.preprocess_defines) {
236  if(define.empty()) {
237  PLAIN_LOG << "empty define supplied";
238  continue;
239  }
240 
241  LOG_PREPROC << "adding define: " << define;
242  defines_map.emplace(define, preproc_define(define));
243 
244  if(define == "SKIP_CORE") {
245  PLAIN_LOG << "'SKIP_CORE' defined.";
246  skipCore = true;
247  } else if(define == "NO_TERRAIN_GFX") {
248  PLAIN_LOG << "'NO_TERRAIN_GFX' defined.";
249  skipTerrainGFX = true;
250  }
251  }
252  }
253 
254  // add the WESNOTH_VERSION define
255  defines_map["WESNOTH_VERSION"] = preproc_define(game_config::wesnoth_version.str());
256 
257  PLAIN_LOG << "added " << defines_map.size() << " defines.";
258 
259  // preprocess core macros first if we don't skip the core
260  if(skipCore == false) {
261  PLAIN_LOG << "preprocessing common macros from 'data/core' ...";
262 
263  // process each folder explicitly to gain speed
264  preprocess_resource(game_config::path + "/data/core/macros", &defines_map);
265 
266  if(skipTerrainGFX == false) {
267  preprocess_resource(game_config::path + "/data/core/terrain-graphics", &defines_map);
268  }
269 
270  PLAIN_LOG << "acquired " << (defines_map.size() - input_macros.size()) << " 'data/core' defines.";
271  } else {
272  PLAIN_LOG << "skipped 'data/core'";
273  }
274 
275  // preprocess resource
276  PLAIN_LOG << "preprocessing specified resource: " << resourceToProcess << " ...";
277 
278  preprocess_resource(resourceToProcess, &defines_map, true, true, targetDir);
279  PLAIN_LOG << "acquired " << (defines_map.size() - input_macros.size()) << " total defines.";
280 
281  if(cmdline_opts.preprocess_output_macros) {
282  std::string outputFileName = "_MACROS_.cfg";
283  if(!cmdline_opts.preprocess_output_macros->empty()) {
284  outputFileName = *cmdline_opts.preprocess_output_macros;
285  }
286 
287  std::string outputPath = targetDir + "/" + outputFileName;
288 
289  PLAIN_LOG << "writing '" << outputPath << "' with " << defines_map.size() << " defines.";
290 
292  if(!out->fail()) {
293  config_writer writer(*out, false);
294 
295  for(auto& define_pair : defines_map) {
296  define_pair.second.write(writer, define_pair.first);
297  }
298  } else {
299  PLAIN_LOG << "couldn't open the file.";
300  }
301  }
302 }
303 
304 static int handle_validate_command(const std::string& file, abstract_validator& validator, const std::vector<std::string>& defines) {
305  preproc_map defines_map;
306  // add the WESNOTH_VERSION define
307  defines_map["WESNOTH_VERSION"] = preproc_define(game_config::wesnoth_version.str());
308  defines_map["SCHEMA_VALIDATION"] = preproc_define();
309  for(const std::string& define : defines) {
310  if(define.empty()) {
311  PLAIN_LOG << "empty define supplied";
312  continue;
313  }
314 
315  LOG_PREPROC << "adding define: " << define;
316  defines_map.emplace(define, preproc_define(define));
317  }
318  PLAIN_LOG << "Validating " << file << " against schema " << validator.name_;
320  config result = io::read(*preprocess_file(file, &defines_map), &validator);
321  if(lg::broke_strict()) {
322  std::cout << "validation failed\n";
323  } else {
324  std::cout << "validation succeeded\n";
325  }
326  return lg::broke_strict();
327 }
328 
329 /** Process commandline-arguments */
330 static int process_command_args(commandline_options& cmdline_opts)
331 {
332  // Options that output info unaffected by other options and return.
333  if(cmdline_opts.help) {
334  std::cout << cmdline_opts;
335  return 0;
336  }
337 
338  if(cmdline_opts.logdomains) {
339  std::cout << lg::list_log_domains(*cmdline_opts.logdomains);
340  return 0;
341  }
342 
343  if(cmdline_opts.version) {
344  std::cout << "Battle for Wesnoth" << " " << game_config::wesnoth_version.str() << "\n\n";
345  std::cout << "Library versions:\n" << game_config::library_versions_report() << '\n';
346  std::cout << "Optional features:\n" << game_config::optional_features_report();
347  return 0;
348  }
349 
350  if(cmdline_opts.simple_version) {
351  std::cout << game_config::wesnoth_version.str() << "\n";
352  return 0;
353  }
354 
355  // Options that don't change behavior based on any others should be checked alphabetically below.
356 
357  if(cmdline_opts.no_log_sanitize) {
358  lg::set_log_sanitize(false);
359  }
360 
361  if(cmdline_opts.usercache_dir) {
363  }
364 
365  if(cmdline_opts.userdata_dir) {
367  }
368 
369  // earliest possible point to ensure the userdata directory is known
371  filesystem::set_user_data_dir(std::string());
372  }
373 
374  // userdata is initialized, so initialize logging to file if enabled
375  // If true, output will be redirected to file, else output be written to console.
376  // On Windows, if Wesnoth was not started from a console, one will be allocated.
377  if(cmdline_opts.log_to_file
378  || (!cmdline_opts.no_log_to_file
379  && !getenv("WESNOTH_NO_LOG_FILE")
380  // command line options that imply not redirecting output to a log file
381  && !cmdline_opts.data_path
382  && !cmdline_opts.userdata_path
383  && !cmdline_opts.usercache_path
384  && !cmdline_opts.report
385  && !cmdline_opts.do_diff
386  && !cmdline_opts.do_patch
387  && !cmdline_opts.preprocess
388  && !cmdline_opts.render_image
389  && !cmdline_opts.screenshot
390  && !cmdline_opts.nogui
391  && !cmdline_opts.headless_unit_test
392  && !cmdline_opts.validate_schema
393  && !cmdline_opts.validate_wml
394  )
395  )
396  {
398  }
399 #ifdef _WIN32
400  // This forces a Windows console to be attached to the process even
401  // if Wesnoth is an IMAGE_SUBSYSTEM_WINDOWS_GUI executable because it
402  // turns Wesnoth into a CLI application. (unless --wnoconsole is given)
403  else if(!cmdline_opts.no_console) {
405  }
406 #endif
407 
408  if(cmdline_opts.log) {
409  for(const auto& log_pair : *cmdline_opts.log) {
410  const std::string log_domain = log_pair.second;
411  const lg::severity severity = log_pair.first;
412  if(!lg::set_log_domain_severity(log_domain, severity)) {
413  PLAIN_LOG << "unknown log domain: " << log_domain;
414  return 2;
415  }
416  }
417  }
418 
419  if(!cmdline_opts.nobanner) {
420  const auto now = std::chrono::system_clock::now();
421  PLAIN_LOG << "Battle for Wesnoth v" << game_config::revision << " " << game_config::build_arch();
422  PLAIN_LOG << "Started on " << chrono::format_local_timestamp(now, "%a %b %d %T %Y") << '\n';
423  }
424 
425  if(cmdline_opts.usercache_path) {
426  std::cout << filesystem::get_cache_dir();
427  return 0;
428  }
429 
430  if(cmdline_opts.userdata_path) {
431  std::cout << filesystem::get_user_data_dir();
432  return 0;
433  }
434 
435  if(cmdline_opts.data_dir) {
436  game_config::path = filesystem::normalize_path(*cmdline_opts.data_dir, true, true);
437  if(!cmdline_opts.nobanner) {
438  PLAIN_LOG << "Overriding data directory with '" << game_config::path << "'";
439  }
440  } else {
441  // if a pre-defined path does not exist this will empty it
443  if(game_config::path.empty()) {
444  if(std::string exe_dir = filesystem::get_exe_dir(); !exe_dir.empty()) {
445  if(std::string auto_dir = filesystem::autodetect_game_data_dir(std::move(exe_dir)); !auto_dir.empty()) {
446  if(!cmdline_opts.nobanner) {
447  PLAIN_LOG << "Automatically found a possible data directory at: " << auto_dir;
448  }
449  game_config::path = filesystem::normalize_path(auto_dir, true, true);
450  }
451  } else {
452  PLAIN_LOG << "Cannot find game data directory. Specify one with --data-dir";
453  return 1;
454  }
455  }
456  }
457 
459  PLAIN_LOG << "Could not find game data directory '" << game_config::path << "'";
460  return 1;
461  }
462 
463  if(cmdline_opts.data_path) {
464  std::cout << game_config::path;
465  return 0;
466  }
467 
468  if(cmdline_opts.debug_lua) {
469  game_config::debug_lua = true;
470  }
471 
472  if(cmdline_opts.allow_insecure) {
474  }
475 
476  if(cmdline_opts.addon_server_info) {
478  }
479 
480  if(cmdline_opts.strict_lua) {
482  }
483 
484  if(cmdline_opts.log_precise_timestamps) {
486  }
487 
488  if(cmdline_opts.rng_seed) {
489  srand(*cmdline_opts.rng_seed);
490  }
491 
492  if(cmdline_opts.render_image) {
493  SDL_setenv("SDL_VIDEODRIVER", "dummy", 1);
494  }
495 
496  if(cmdline_opts.strict_validation) {
498  }
499 
500  if(cmdline_opts.report) {
501  std::cout << "\n========= BUILD INFORMATION =========\n\n" << game_config::full_build_report();
502  return 0;
503  }
504 
505  if(cmdline_opts.validate_schema) {
507  validator.set_create_exceptions(false); // Don't crash if there's an error, just go ahead anyway
508  return handle_validate_command(*cmdline_opts.validate_schema, validator, {});
509  }
510 
511  if(cmdline_opts.do_diff) {
512  std::ifstream in_left(cmdline_opts.diff_left);
513  std::ifstream in_right(cmdline_opts.diff_right);
514  config left = io::read(in_left);
515  config right = io::read(in_right);
516  std::ostream* os = &std::cout;
517  if(cmdline_opts.output_file) {
518  os = new std::ofstream(*cmdline_opts.output_file);
519  }
521  out.write(right.get_diff(left));
522  if(os != &std::cout) delete os;
523  return 0;
524  }
525 
526  if(cmdline_opts.do_patch) {
527  std::ifstream in_base(cmdline_opts.diff_left);
528  std::ifstream in_diff(cmdline_opts.diff_right);
529  config base = io::read(in_base);
530  config diff = io::read(in_diff);
531  base.apply_diff(diff);
532  std::ostream* os = &std::cout;
533  if(cmdline_opts.output_file) {
534  os = new std::ofstream(*cmdline_opts.output_file);
535  }
537  out.write(base);
538  if(os != &std::cout) delete os;
539  return 0;
540  }
541 
542  if(cmdline_opts.generate_spritesheet) {
543  PLAIN_LOG << "sheet path " << *cmdline_opts.generate_spritesheet;
545  return 0;
546  }
547 
548  // Options changing their behavior dependent on some others should be checked below.
549 
550  if(cmdline_opts.preprocess) {
551  handle_preprocess_command(cmdline_opts);
552  return 0;
553  }
554 
555  if(cmdline_opts.preprocess_source_string.has_value()) {
556  handle_preprocess_string(cmdline_opts);
557  return 0;
558  }
559 
560  if(cmdline_opts.validate_wml) {
561  std::string schema_path;
562  if(cmdline_opts.validate_with) {
563  schema_path = *cmdline_opts.validate_with;
564  if(!filesystem::file_exists(schema_path)) {
565  if(auto check = filesystem::get_wml_location(schema_path)) {
566  schema_path = check.value();
567  } else {
568  PLAIN_LOG << "Could not find schema file: " << schema_path;
569  }
570  } else {
571  schema_path = filesystem::normalize_path(schema_path);
572  }
573  } else {
574  schema_path = filesystem::get_wml_location("schema/game_config.cfg").value();
575  }
577  validator.set_create_exceptions(false); // Don't crash if there's an error, just go ahead anyway
578  return handle_validate_command(*cmdline_opts.validate_wml, validator,
579  cmdline_opts.preprocess_defines.value_or<decltype(cmdline_opts.preprocess_defines)::value_type>({}));
580  }
581 
582  if(cmdline_opts.preprocess_defines || cmdline_opts.preprocess_input_macros || cmdline_opts.preprocess_path) {
583  // It would be good if this was supported for running tests too, possibly for other uses.
584  // For the moment show an error message instead of leaving the user wondering why it doesn't work.
585  PLAIN_LOG << "That --preprocess-* option is only supported when using --preprocess or --validate.";
586  return 2;
587  }
588 
589  // Not the most intuitive solution, but I wanted to leave current semantics for now
590  return -1;
591 }
592 
593 /**
594  * I would prefer to setup locale first so that early error
595  * messages can get localized, but we need the game_launcher
596  * initialized to have filesystem::get_intl_dir() to work. Note: setlocale()
597  * does not take GUI language setting into account.
598  */
599 static void init_locale()
600 {
601 #if defined _WIN32 || defined __APPLE__
602  setlocale(LC_ALL, "English");
603 #else
604  std::setlocale(LC_ALL, "C");
605 #endif
606 
607  const std::string& intl_dir = filesystem::get_intl_dir();
608 
609  translation::bind_textdomain(PACKAGE, intl_dir.c_str(), "UTF-8");
610  translation::bind_textdomain(PACKAGE "-lib", intl_dir.c_str(), "UTF-8");
612 }
613 
614 /**
615  * Print an alert and instructions to stderr about early initialization errors.
616  *
617  * This is provided as an aid for users dealing with potential data dir
618  * configuration issues. The first code to read core WML *has* the
619  * responsibility to call this function in the event of a problem, to inform
620  * the user of the most likely possible cause and suggest a course of action
621  * to solve the issue.
622  */
624 {
625  // NOTE: wrap output to 80 columns.
626  PLAIN_LOG << '\n'
627  << "An error at this point during initialization usually indicates that the data\n"
628  << "directory above was not correctly set or detected. Try passing the correct path\n"
629  << "in the command line with the --data-dir switch or as the only argument.";
630 }
631 
632 /**
633  * Handles the lua script command line arguments if present.
634  * This function will only run once.
635  */
637 {
638  static bool first_time = true;
639 
640  if(!first_time) {
641  return;
642  }
643 
644  first_time = false;
645 
646  if(!game->init_lua_script()) {
647  // PLAIN_LOG << "error when loading lua scripts at startup";
648  // PLAIN_LOG << "could not load lua script: " << *cmdline_opts.script_file;
649  }
650 }
651 
652 #ifdef _MSC_VER
653 static void check_fpu()
654 {
655  uint32_t f_control;
656 
657  if(_controlfp_s(&f_control, 0, 0) == 0) {
658  uint32_t unused;
659  uint32_t rounding_mode = f_control & _MCW_RC;
660 
661  if(rounding_mode != _RC_NEAR) {
662  PLAIN_LOG << "Floating point rounding mode is currently '"
663  << ((rounding_mode == _RC_CHOP)
664  ? "chop"
665  : (rounding_mode == _RC_UP)
666  ? "up"
667  : (rounding_mode == _RC_DOWN)
668  ? "down"
669  : (rounding_mode == _RC_NEAR) ? "near" : "unknown")
670  << "' setting to 'near'";
671 
672  if(_controlfp_s(&unused, _RC_NEAR, _MCW_RC)) {
673  PLAIN_LOG << "failed to set floating point rounding type to 'near'";
674  }
675  }
676 
677 #ifndef _M_AMD64
678  uint32_t precision_mode = f_control & _MCW_PC;
679  if(precision_mode != _PC_53) {
680  PLAIN_LOG << "Floating point precision mode is currently '"
681  << ((precision_mode == _PC_53)
682  ? "double"
683  : (precision_mode == _PC_24)
684  ? "single"
685  : (precision_mode == _PC_64) ? "double extended" : "unknown")
686  << "' setting to 'double'";
687 
688  if(_controlfp_s(&unused, _PC_53, _MCW_PC)) {
689  PLAIN_LOG << "failed to set floating point precision type to 'double'";
690  }
691  }
692 #endif
693 
694  } else {
695  PLAIN_LOG << "_controlfp_s failed.";
696  }
697 }
698 #else
699 static void check_fpu()
700 {
701  switch(fegetround()) {
702  case FE_TONEAREST:
703  break;
704  case FE_DOWNWARD:
705  STREAMING_LOG << "Floating point precision mode is currently 'downward'";
706  goto reset_fpu;
707  case FE_TOWARDZERO:
708  STREAMING_LOG << "Floating point precision mode is currently 'toward-zero'";
709  goto reset_fpu;
710  case FE_UPWARD:
711  STREAMING_LOG << "Floating point precision mode is currently 'upward'";
712  goto reset_fpu;
713  default:
714  STREAMING_LOG << "Floating point precision mode is currently 'unknown'";
715  goto reset_fpu;
716  reset_fpu:
717  STREAMING_LOG << " - setting to 'nearest'\n";
718  fesetround(FE_TONEAREST);
719  break;
720  }
721 }
722 #endif
723 
724 /**
725  * Setups the game environment and enters
726  * the titlescreen or game loops.
727  */
728 static int do_gameloop(commandline_options& cmdline_opts)
729 {
730  srand(std::time(nullptr));
731 
732  const auto game = std::make_unique<game_launcher>(cmdline_opts);
733 
734  init_locale();
735 
736  bool res;
737 
738  // Do initialize fonts before reading the game config, to have game
739  // config error messages displayed. fonts will be re-initialized later
740  // when the language is read from the game config.
741  res = font::load_font_config();
742  if(res == false) {
743  PLAIN_LOG << "could not initialize fonts";
744  // The most common symptom of a bogus data dir path -- warn the user.
746  return 1;
747  }
748 
749  res = game->init_language();
750  if(res == false) {
751  PLAIN_LOG << "could not initialize the language";
752  return 1;
753  }
754 
755  res = game->init_video();
756  if(res == false) {
757  PLAIN_LOG << "could not initialize display";
758  return 1;
759  }
760 
761  check_fpu();
762  const cursor::manager cursor_manager;
764 
765 #if(defined(_X11) && !defined(__APPLE__)) || defined(_WIN32)
766  SDL_EventState(SDL_SYSWMEVENT, SDL_ENABLE);
767 #endif
768 
769  gui2::init();
770  gui2::switch_theme(prefs::get().gui2_theme());
771  const gui2::event::manager gui_event_manager;
772 
773  // if the log directory is not writable, then this is the error condition so show the error message.
774  // if the log directory is writable, then there's no issue.
775  // if the optional isn't set, then logging to file has been disabled, so there's no issue.
776  if(!lg::log_dir_writable().value_or(true)) {
777  utils::string_map symbols;
778  symbols["logdir"] = filesystem::get_logs_dir();
779  std::string msg = VGETTEXT("Unable to create log files in directory $logdir. This is often caused by incorrect folder permissions, anti-virus software restricting folder access, or using OneDrive to manage your My Documents folder.", symbols);
780  gui2::show_message(_("Logging Failure"), msg, message::ok_button);
781  }
782 
783  game_config_manager config_manager(cmdline_opts);
784 
788  }
789 
790  loading_screen::display([&res, &config_manager, &cmdline_opts]() {
793 
794  if(res == false) {
795  PLAIN_LOG << "could not initialize game config";
796  return;
797  }
798 
800 
801  res = font::load_font_config();
802  if(res == false) {
803  PLAIN_LOG << "could not re-initialize fonts for the current language";
804  return;
805  }
806 
807  if(!game_config::no_addons && !cmdline_opts.noaddons) {
809 
811  }
812  });
813 
814  if(res == false) {
815  return 1;
816  }
817 
818  plugins_manager plugins_man(new application_lua_kernel);
819 
820  const plugins_context::reg_vec callbacks {
821  {"play_multiplayer", std::bind(&game_launcher::play_multiplayer, game.get(), game_launcher::mp_mode::CONNECT)},
822  {"play_local", std::bind(&game_launcher::play_multiplayer, game.get(), game_launcher::mp_mode::LOCAL)},
823  {"play_campaign", std::bind(&game_launcher::play_campaign, game.get())},
824  };
825 
826  const plugins_context::areg_vec accessors {
827  {"command_line", std::bind(&commandline_options::to_config, &cmdline_opts)},
828  };
829 
830  plugins_context plugins("titlescreen", callbacks, accessors);
831 
832  plugins.set_callback("exit", [](const config& cfg) { safe_exit(cfg["code"].to_int(0)); }, false);
833 
834  while(true) {
835  if(!game->has_load_data()) {
836  auto cfg = config_manager.game_config().optional_child("titlescreen_music");
837  if(cfg) {
838  for(const config& i : cfg->child_range("music")) {
840  }
841 
842  config title_music_config;
843  title_music_config["name"] = game_config::title_music;
844  title_music_config["append"] = true;
845  title_music_config["immediate"] = true;
846  sound::play_music_config(title_music_config);
847  } else {
850  }
851  }
852 
853  handle_lua_script_args(&*game, cmdline_opts);
854 
855  plugins.play_slice();
856  plugins.play_slice();
857 
858  if(!cmdline_opts.unit_test.empty()) {
859  return static_cast<int>(game->unit_test());
860  }
861 
862  if(game->play_test() == false) {
863  return 0;
864  }
865 
866  if(game->play_screenshot_mode() == false) {
867  return 0;
868  }
869 
870  if(game->play_render_image_mode() == false) {
871  return 0;
872  }
873 
874  // Start directly a campaign
875  if(game->goto_campaign() == false) {
876  if(game->jump_to_campaign_id().empty())
877  continue; // Go to main menu
878  else
879  return 1; // we got an error starting the campaign from command line
880  }
881 
882  // Start directly a multiplayer
883  // Eventually with a specified server
884  if(game->goto_multiplayer() == false) {
885  continue; // Go to main menu
886  }
887 
888  // Start directly a commandline multiplayer game
889  if(game->play_multiplayer_commandline() == false) {
890  return 0;
891  }
892 
893  if(game->goto_editor() == false) {
894  return 0;
895  }
896 
897  const font::floating_label_context label_manager;
898 
900 
901  // If loading a game, skip the titlescreen entirely
902  if(game->has_load_data() && game->load_game()) {
904  continue;
905  }
906 
907  int retval;
908  { // scope to not keep the title screen alive all game
909  title_screen dlg(*game);
910 
911  // Allows re-layout on resize.
912  // Since RELOAD_UI is not checked here, it causes
913  // the dialog to be closed and reshown with changes.
915  dlg.show();
916  }
917  retval = dlg.get_retval();
918  }
919 
920  switch(retval) {
922  LOG_GENERAL << "quitting game...";
923  return 0;
925  game->play_multiplayer(game_launcher::mp_mode::CONNECT);
926  break;
928  game->play_multiplayer(game_launcher::mp_mode::HOST);
929  break;
931  game->play_multiplayer(game_launcher::mp_mode::LOCAL);
932  break;
934  loading_screen::display([&config_manager]() {
935  config_manager.reload_changed_game_config();
936  gui2::init();
937  gui2::switch_theme(prefs::get().gui2_theme());
938  });
939  break;
941  game->start_editor();
942  break;
945  break;
947  break;
949  gui2::switch_theme(prefs::get().gui2_theme());
950  break;
951  }
952  }
953 }
954 
955 #ifdef _WIN32
956 #define error_exit(res) \
957  do { \
958  if(lg::using_own_console()) { \
959  std::cerr << "Press enter to continue..." << std::endl; \
960  std::cin.get(); \
961  } \
962  return res; \
963  } while(false)
964 #else
965 #define error_exit(res) return res
966 #endif
967 
968 #ifdef __APPLE__
969 extern "C" int wesnoth_main(int argc, char** argv);
970 int wesnoth_main(int argc, char** argv)
971 #else
972 int main(int argc, char** argv)
973 #endif
974 {
976  auto args = read_argv(argc, argv);
977  assert(!args.empty());
978 
979 #ifdef _WIN32
980  _putenv("PANGOCAIRO_BACKEND=fontconfig");
981  _putenv("FONTCONFIG_PATH=fonts");
982 #endif
983 #ifdef __APPLE__
984  // Using setenv with overwrite disabled so we can override this in the
985  // original process environment for research/testing purposes.
986  setenv("PANGOCAIRO_BACKEND", "fontconfig", 0);
987 #endif
988 
989  try {
990  commandline_options cmdline_opts = commandline_options(args);
991  int finished = process_command_args(cmdline_opts);
992 
993  if(finished != -1) {
994 #ifdef _WIN32
995  if(lg::using_own_console()) {
996  std::cerr << "Press enter to continue..." << std::endl;
997  std::cin.get();
998  }
999 #endif
1000  safe_exit(finished);
1001  }
1002 
1003  SDL_SetHint(SDL_HINT_NO_SIGNAL_HANDLERS, "1");
1004  // Is there a reason not to just use SDL_INIT_EVERYTHING?
1005  if(SDL_Init(SDL_INIT_TIMER) < 0) {
1006  PLAIN_LOG << "Couldn't initialize SDL: " << SDL_GetError();
1007  return (1);
1008  }
1009  atexit(SDL_Quit);
1010 
1011  // Mac's touchpad generates touch events too.
1012  // Ignore them until Macs have a touchscreen: https://forums.libsdl.org/viewtopic.php?p=45758
1013 #if defined(__APPLE__) && !defined(__IPHONEOS__)
1014  SDL_EventState(SDL_FINGERMOTION, SDL_DISABLE);
1015  SDL_EventState(SDL_FINGERDOWN, SDL_DISABLE);
1016  SDL_EventState(SDL_FINGERUP, SDL_DISABLE);
1017 #endif
1018 
1019  // declare this here so that it will always be at the front of the event queue.
1020  events::event_context global_context;
1021 
1022  SDL_StartTextInput();
1023 
1024  const int res = do_gameloop(cmdline_opts);
1025  safe_exit(res);
1026  } catch(const boost::program_options::error& e) {
1027  // logging hasn't been initialized by this point
1028  std::cerr << "Error in command line: " << e.what() << std::endl;
1029  std::string error = "Error parsing command line arguments: ";
1030  error += e.what();
1031  SDL_ShowSimpleMessageBox(SDL_MESSAGEBOX_ERROR, "Error", error.c_str(), nullptr);
1032  error_exit(2);
1033  } catch(const video::error& e) {
1034  PLAIN_LOG << "Video system error: " << e.what();
1035  error_exit(1);
1036  } catch(const font::error& e) {
1037  PLAIN_LOG << "Could not initialize fonts.\n\n" << e.what() << "\n\nExiting.";
1038  error_exit(1);
1039  } catch(const config::error& e) {
1040  PLAIN_LOG << e.message;
1041  error_exit(1);
1042  } catch(const video::quit&) {
1043  // just means the game should quit
1044  } catch(const return_to_play_side_exception&) {
1045  PLAIN_LOG << "caught return_to_play_side_exception, please report this bug (quitting)";
1046  } catch(const quit_game_exception&) {
1047  PLAIN_LOG << "caught quit_game_exception (quitting)";
1048  } catch(const wml_exception& e) {
1049  PLAIN_LOG << "WML exception:\nUser message: " << e.user_message << "\nDev message: " << e.dev_message;
1050  error_exit(1);
1051  } catch(const wfl::formula_error& e) {
1052  PLAIN_LOG << e.what() << "\n\nGame will be aborted.";
1053  error_exit(1);
1054  } catch(const sdl::exception& e) {
1055  PLAIN_LOG << e.what();
1056  error_exit(1);
1057  } catch(const game::error& e) {
1058  PLAIN_LOG << "Game error: " << e.what();
1059  error_exit(1);
1060  } catch(const std::bad_alloc&) {
1061  PLAIN_LOG << "Ran out of memory. Aborted.";
1062  error_exit(ENOMEM);
1063 #if !defined(NO_CATCH_AT_GAME_END)
1064  } catch(const std::exception& e) {
1065  // Try to catch unexpected exceptions.
1066  PLAIN_LOG << "Caught general '" << typeid(e).name() << "' exception:\n" << e.what();
1067  error_exit(1);
1068  } catch(const std::string& e) {
1069  PLAIN_LOG << "Caught a string thrown as an exception:\n" << e;
1070  error_exit(1);
1071  } catch(const char* e) {
1072  PLAIN_LOG << "Caught a string thrown as an exception:\n" << e;
1073  error_exit(1);
1074  } catch(...) {
1075  // Ensure that even when we terminate with `throw 42`, the exception
1076  // is caught and all destructors are actually called. (Apparently,
1077  // some compilers will simply terminate without calling destructors if
1078  // the exception isn't caught.)
1079  PLAIN_LOG << "Caught general exception " << utils::get_unknown_exception_type() << ". Terminating.";
1080  error_exit(1);
1081 #endif
1082  }
1083 
1084  return 0;
1085 } // end main
int wesnoth_main(int argc, char **argv)
void refresh_addon_version_info_cache()
Refreshes the per-session cache of add-on's version information structs.
Definition: manager.cpp:381
Used in parsing config file.
Definition: validator.hpp:38
bool nogui
True if –nogui was given on the command line.
utils::optional< std::string > validate_wml
Non-empty if –validate was given on the command line.
bool simple_version
True if –simple-version was given on the command line.
bool report
True if –report was given on the command line.
bool headless_unit_test
True if –unit is used and –showgui is not present.
bool no_log_sanitize
True if –no-log-sanitize was given on the command line.
bool strict_lua
True if –strict-lua was given in the commandline.
utils::optional< std::vector< std::string > > preprocess_defines
Defines that were given to the –preprocess option.
utils::optional< std::string > usercache_dir
Non-empty if –usercache-dir was given on the command line.
utils::optional< std::string > validate_schema
Non-empty if –validate-schema was given on the command line.
utils::optional< std::string > userdata_dir
Non-empty if –userdata-dir was given on the command line.
bool nobanner
True if –nobanner was given on the command line.
utils::optional< std::vector< std::pair< lg::severity, std::string > > > log
Contains parsed arguments of –log-* (e.g.
utils::optional< std::string > generate_spritesheet
Path of which to generate a spritesheet.
utils::optional< std::string > render_image
Image path to render.
utils::optional< std::string > logdomains
Non-empty if –logdomains was given on the command line.
utils::optional< std::string > preprocess_input_macros
Non-empty if –preprocess-input-macros was given on the command line.
bool preprocess
True if –preprocess was given on the command line.
utils::optional< unsigned int > rng_seed
RNG seed specified by –rng-seed option.
std::string diff_left
Files for diffing or patching.
bool data_path
True if –data-path was given on the command line.
bool version
True if –version was given on the command line.
bool allow_insecure
True if –allow-insecure was given in the commandline.
utils::optional< std::string > validate_with
Non-empty if –use-schema was given on the command line.
bool noaddons
True if –noaddons was given on the command line.
utils::optional< std::string > output_file
Output filename for WML diff or preprocessing.
utils::optional< std::string > data_dir
Non-empty if –data-dir was given on the command line.
utils::optional< std::string > preprocess_target
Target (output) path that was given to the –preprocess option.
bool screenshot
True if –screenshot was given on the command line.
bool log_to_file
True if –log-to-file was given on the command line.
bool debug_lua
True if –debug-lua was given in the commandline.
std::vector< std::string > unit_test
Non-empty if –unit was given on the command line.
bool userdata_path
True if –userdata-path was given on the command line.
bool log_precise_timestamps
True if –log-precise was given on the command line.
utils::optional< std::string > preprocess_source_string
String to preprocess.
bool no_log_to_file
True if –no-log-to-file was given on the command line.
utils::optional< std::string > preprocess_output_macros
Non-empty if –preprocess-output-macros was given on the command line.
bool strict_validation
True if –strict-validation was given on the command line.
utils::optional< std::string > preprocess_path
Path to parse that was given to the –preprocess option.
bool help
True if –help was given on the command line.
bool usercache_path
True if –usercache-path was given on the command line.
Class for writing a config out to a file in pieces.
void write(const config &cfg)
A config object defines a single node in a WML file, with access to child nodes.
Definition: config.hpp:158
auto all_children_view() const
In-order iteration over all children.
Definition: config.hpp:796
child_itors child_range(config_key_type key)
Definition: config.cpp:268
void apply_diff(const config &diff, bool track=false)
A function to apply a diff config onto this config object.
Definition: config.cpp:1022
config get_diff(const config &c) const
A function to get the differences between this object, and 'c', as another config object.
Definition: config.cpp:906
@ NO_FORCE_RELOAD
Don't reload if the previous defines equal the new defines.
bool init_game_config(FORCE_RELOAD_CONFIG force_reload)
const game_config_view & game_config() const
optional_const_config optional_child(config_key_type key) const
bool play_multiplayer(mp_mode mode)
static void progress(loading_stage stage=loading_stage::none)
Report what is being loaded to the loading screen.
static void display(const std::function< void()> &f)
@ ok_button
Shows an ok button.
Definition: message.hpp:73
bool show(const unsigned auto_close_time=0)
Shows the window.
This class implements the title screen.
int get_retval()
Definition: window.hpp:402
void play_slice()
Definition: context.cpp:103
std::vector< Reg > reg_vec
Definition: context.hpp:40
std::vector< aReg > areg_vec
Definition: context.hpp:41
void set_callback(const std::string &name, callback_function)
Definition: context.cpp:53
static prefs & get()
Exception used to escape form the ai or ui code to playsingle_controller::play_side.
Realization of serialization/validator.hpp abstract validator.
std::string str() const
Serializes the version number into string form.
Type that can be thrown as an exception to quit to desktop.
Definition: video.hpp:323
std::vector< std::string > read_argv([[maybe_unused]] int argc, [[maybe_unused]] char **argv)
Definitions for the interface to Wesnoth Markup Language (WML).
Declarations for File-IO.
#define VGETTEXT(msgid,...)
Handy wrappers around interpolate_variables_into_string and gettext.
std::size_t i
Definition: function.cpp:1022
Contains the exception interfaces used to signal completion of a scenario, campaign or turn.
Interfaces for manipulating version numbers of engine, add-ons, etc.
static std::string _(const char *str)
Definition: gettext.hpp:103
Standard logging facilities (interface).
#define STREAMING_LOG
Definition: log.hpp:297
#define PLAIN_LOG
Definition: log.hpp:296
auto format_local_timestamp(const std::chrono::system_clock::time_point &time, std::string_view format="%F %T")
Definition: chrono.hpp:62
@ WAIT
Definition: cursor.hpp:28
@ NORMAL
Definition: cursor.hpp:28
void set(CURSOR_TYPE type)
Use the default parameter to reset cursors.
Definition: cursor.cpp:176
void set_main_thread()
Definition: events.cpp:471
std::string get_cache_dir()
Definition: filesystem.cpp:866
filesystem::scoped_istream istream_file(const std::string &fname, bool treat_failure_as_error)
std::string get_user_data_dir()
Definition: filesystem.cpp:856
static bool file_exists(const bfs::path &fpath)
Definition: filesystem.cpp:328
std::string get_exe_dir()
bool is_directory(const std::string &fname)
Returns true if the given file is a directory.
utils::optional< std::string > get_wml_location(const std::string &path, const utils::optional< std::string > &current_dir)
Returns a translated path to the actual file or directory, if it exists.
filesystem::scoped_ostream ostream_file(const std::string &fname, std::ios_base::openmode mode, bool create_directory)
std::string autodetect_game_data_dir(std::string exe_dir)
Try to autodetect the location of the game data dir.
void set_cache_dir(const std::string &newcachedir)
Definition: filesystem.cpp:819
std::string get_logs_dir()
Definition: filesystem.cpp:861
std::unique_ptr< std::ostream > scoped_ostream
Definition: filesystem.hpp:54
bool is_userdata_initialized()
Definition: filesystem.cpp:611
std::string get_intl_dir()
std::string normalize_path(const std::string &fpath, bool normalize_separators, bool resolve_dot_entries)
Returns the absolute path of a file.
void set_user_data_dir(std::string newprefdir)
Definition: filesystem.cpp:723
bool load_font_config()
Definition: font_config.cpp:78
std::string path
Definition: filesystem.cpp:93
bool addon_server_info
Definition: game_config.cpp:79
std::string full_build_report()
Produce a bug report-style info dump.
Definition: build_info.cpp:666
std::string library_versions_report()
Produce a plain-text report of library versions suitable for stdout/stderr.
Definition: build_info.cpp:656
const version_info wesnoth_version(VERSION)
bool allow_insecure
Definition: game_config.cpp:78
std::string title_music
std::string build_arch()
Obtain the processor architecture for this build.
Definition: build_info.cpp:316
std::string optional_features_report()
Produce a plain-text report of features suitable for stdout/stderr.
Definition: build_info.cpp:661
const std::string revision
bool check_migration
Definition: filesystem.cpp:101
void init()
Initializes the GUI subsystems.
Definition: gui.cpp:109
void switch_theme(const std::string &theme_id)
Set and activate the given gui2 theme.
Definition: gui.cpp:150
void show_message(const std::string &title, const std::string &msg, const std::string &button_caption, const bool auto_close, const bool message_use_markup, const bool title_use_markup)
Shows a message to the user.
Definition: message.cpp:148
retval
Default window/dialog return values.
Definition: retval.hpp:30
void build_spritesheet_from(const std::string &entry_point)
config read(std::istream &in, abstract_validator *validator)
Definition: parser.cpp:627
bool using_own_console()
Returns true if a console was allocated by the Wesnoth process.
severity
Definition: log.hpp:82
std::string list_log_domains(const std::string &filter)
Definition: log.cpp:377
bool broke_strict()
Definition: log.cpp:397
void set_log_to_file()
Do the initial redirection to a log file if the logs directory is writable.
Definition: log.cpp:233
void set_log_sanitize(bool sanitize)
toggle log sanitization
Definition: log.cpp:401
utils::optional< bool > log_dir_writable()
Returns the result set by check_log_dir_writable().
Definition: log.cpp:279
void do_console_redirect()
Allocates a console if needed and redirects output to CONOUT.
void precise_timestamps(bool pt)
Definition: log.cpp:304
bool set_log_domain_severity(const std::string &name, severity severity)
Definition: log.cpp:346
void set_strict_severity(severity severity)
Definition: log.cpp:387
void empty_playlist()
Definition: sound.cpp:613
void play_music_config(const config &music_node, bool allow_interrupt_current_track, int i)
Definition: sound.cpp:716
void stop_music()
Definition: sound.cpp:558
void bind_textdomain(const char *domain, const char *directory, const char *)
Definition: gettext.cpp:479
void set_default_textdomain(const char *domain)
Definition: gettext.cpp:487
std::string get_unknown_exception_type()
Utility function for finding the type of thing caught with catch(...).
Definition: general.cpp:23
std::map< std::string, t_string > string_map
static void msg(const char *act, debug_info &i, const char *to="", const char *result="")
Definition: debugger.cpp:109
std::string preprocess_string(const std::string &contents, preproc_map *defines, const std::string &textdomain)
Function to use the WML preprocessor on a string.
void preprocess_resource(const std::string &res_name, preproc_map *defines_map, bool write_cfg, bool write_plain_cfg, const std::string &parent_directory)
filesystem::scoped_istream preprocess_file(const std::string &fname, preproc_map *defines)
Function to use the WML preprocessor on a file.
std::map< std::string, struct preproc_define > preproc_map
One of the realizations of serialization/validator.hpp abstract validator.
Contains a basic exception class for SDL operations.
structure which will hide all current floating labels, and cause floating labels instantiated after i...
Base class for all the errors encountered by the engine.
Definition: exceptions.hpp:29
static preproc_map::value_type read_pair(const config &)
Reports time elapsed at the end of an object scope.
Definition: optimer.hpp:37
An error specifically indicating video subsystem problems.
Definition: video.hpp:316
Helper class, don't construct this directly.
bool strict_validation_enabled
Definition: validator.cpp:21
Some defines: VERSION, PACKAGE, MIN_SAVEGAME_VERSION.
#define PACKAGE
Definition: wesconfig.h:23
static lg::log_domain log_preprocessor("preprocessor")
static void handle_preprocess_string(const commandline_options &cmdline_opts)
Definition: wesnoth.cpp:130
static void safe_exit(int res)
Definition: wesnoth.cpp:124
static int do_gameloop(commandline_options &cmdline_opts)
Setups the game environment and enters the titlescreen or game loops.
Definition: wesnoth.cpp:728
static int handle_validate_command(const std::string &file, abstract_validator &validator, const std::vector< std::string > &defines)
Definition: wesnoth.cpp:304
int main(int argc, char **argv)
Definition: wesnoth.cpp:972
static int process_command_args(commandline_options &cmdline_opts)
Process commandline-arguments.
Definition: wesnoth.cpp:330
static void handle_preprocess_command(const commandline_options &cmdline_opts)
Definition: wesnoth.cpp:187
static void check_fpu()
Definition: wesnoth.cpp:699
#define LOG_PREPROC
Definition: wesnoth.cpp:119
static void init_locale()
I would prefer to setup locale first so that early error messages can get localized,...
Definition: wesnoth.cpp:599
static void handle_lua_script_args(game_launcher *game, commandline_options &)
Handles the lua script command line arguments if present.
Definition: wesnoth.cpp:636
#define error_exit(res)
Definition: wesnoth.cpp:965
static void warn_early_init_failure()
Print an alert and instructions to stderr about early initialization errors.
Definition: wesnoth.cpp:623
#define LOG_GENERAL
Definition: wesnoth.cpp:116
static lg::log_domain log_config("config")
Add a special kind of assert to validate whether the input from WML doesn't contain any problems that...
#define e