The Battle for Wesnoth  1.19.10+dev
save_index.cpp
Go to the documentation of this file.
1 /*
2  Copyright (C) 2003 - 2025
3  by Jörg Hinrichs, 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 "save_index.hpp"
17 
18 #include "config.hpp"
19 #include "filesystem.hpp"
20 #include "format_time_summary.hpp"
21 #include "game_errors.hpp"
22 #include "gettext.hpp"
23 #include "log.hpp"
25 #include "serialization/chrono.hpp"
26 #include "serialization/parser.hpp"
27 #include "team.hpp"
28 #include "utils/general.hpp"
29 
30 #include <boost/algorithm/string/replace.hpp>
31 #include <boost/iostreams/filter/gzip.hpp>
32 
33 static lg::log_domain log_engine("engine");
34 #define LOG_SAVE LOG_STREAM(info, log_engine)
35 #define ERR_SAVE LOG_STREAM(err, log_engine)
36 
37 static lg::log_domain log_enginerefac("enginerefac");
38 #define LOG_RG LOG_STREAM(info, log_enginerefac)
39 
40 namespace savegame
41 {
43 
44 void save_index_class::rebuild(const std::string& name)
45 {
46  auto modified = filesystem::file_modified_time(dir_ + "/" + name);
47  rebuild(name, modified);
48 }
49 
50 void save_index_class::rebuild(const std::string& name, const std::chrono::system_clock::time_point& modified)
51 {
52  log_scope("load_summary_from_file");
53 
54  config& summary = data(name);
55 
56  try {
57  config full;
58  std::string dummy;
59  read_save_file(dir_, name, full, &dummy);
60 
61  extract_summary_from_config(full, summary);
62  } catch(const game::load_game_failed&) {
63  summary["corrupt"] = true;
64  }
65 
66  summary["mod_time"] = chrono::serialize_timestamp(modified);
68 }
69 
70 void save_index_class::remove(const std::string& name)
71 {
72  config& root = data();
73  root.remove_children("save", [&name](const config& d) { return name == d["save"]; });
75 }
76 
77 void save_index_class::set_modified(const std::string& name, const std::chrono::system_clock::time_point& modified)
78 {
79  modified_[name] = modified;
80 }
81 
82 config& save_index_class::get(const std::string& name)
83 {
84  config& result = data(name);
85  const auto& m = modified_[name];
86 
87  config::attribute_value& mod_time = result["mod_time"];
88  if(mod_time.empty() || chrono::parse_timestamp(mod_time) != m) {
89  rebuild(name, m);
90  }
91 
92  return result;
93 }
94 
95 const std::string& save_index_class::dir() const
96 {
97  return dir_;
98 }
99 
101 {
102  config &root = data();
103 
104  std::vector<std::string> filenames;
105  filesystem::get_files_in_dir(dir(), &filenames);
106 
107  if(root.all_children_count() > filenames.size()) {
108  root.remove_children("save", [&filenames](const config& d)
109  {
110  return std::find(filenames.begin(), filenames.end(), d["save"]) == filenames.end();
111  }
112  );
113  }
114 }
115 
117 {
118  log_scope("write_save_index()");
119 
120  if(read_only_) {
121  LOG_SAVE << "no-op: read_only instance";
122  return;
123  }
124 
125  if(clean_up_index_) {
126  clean_up_index();
127  clean_up_index_ = false;
128  }
129 
130  try {
132 
133  if(prefs::get().save_compression_format() != compression::format::none) {
134  // TODO: maybe allow writing this using bz2 too?
135  io::write_gz(*stream, data());
136  } else {
137  io::write(*stream, data());
138  }
139  } catch(const filesystem::io_exception& e) {
140  ERR_SAVE << "error writing to save index file: '" << e.what() << "'";
141  }
142 }
143 
144 save_index_class::save_index_class(const std::string& dir)
145  : loaded_(false)
146  , data_()
147  , modified_()
148  , dir_(dir)
149  , read_only_(true)
150  , clean_up_index_(true)
151 {
152 }
153 
156 {
157  read_only_ = false;
158 }
159 
160 config& save_index_class::data(const std::string& name)
161 {
162  config& cfg = data();
163  if(auto sv = cfg.find_child("save", "save", name)) {
165  return *sv;
166  }
167 
168  config& res = cfg.add_child("save");
169  res["save"] = name;
170  return res;
171 }
172 
174 {
175  const std::string si_file = filesystem::get_save_index_file();
176 
177  // Don't try to load the file if it doesn't exist.
178  if(loaded_ == false && filesystem::file_exists(si_file)) {
179  try {
181  try {
182  data_ = io::read_gz(*stream);
183  } catch(const boost::iostreams::gzip_error&) {
184  stream->seekg(0);
185  data_ = io::read(*stream);
186  }
187  } catch(const filesystem::io_exception& e) {
188  ERR_SAVE << "error reading save index: '" << e.what() << "'";
189  } catch(const config::error& e) {
190  ERR_SAVE << "error parsing save index config file:\n" << e.message;
191  data_.clear();
192  }
193 
194  loaded_ = true;
195  }
196 
197  return data_;
198 }
199 
201 {
202  for(config& leader : data.child_range("leader")) {
203  std::string leader_image = leader["leader_image"];
204  boost::algorithm::replace_all(leader_image, "\\", "/");
205 
206  leader["leader_image"] = leader_image;
207  }
208 }
209 
210 std::shared_ptr<save_index_class> save_index_class::default_saves_dir()
211 {
212  static auto instance = std::make_shared<save_index_class>(create_for_default_saves_dir::yes);
213  return instance;
214 }
215 
216 /** Get a list of available saves. */
217 std::vector<save_info> save_index_class::get_saves_list(const std::string* filter)
218 {
219  create_save_info creator(shared_from_this());
220 
221  std::vector<std::string> filenames;
222  filesystem::get_files_in_dir(dir(), &filenames);
223 
224  utils::erase_if(filenames, [filter](const std::string& filename) {
225  // Steam documentation indicates games can ignore their auto-generated 'steam_autocloud.vdf'.
226  // Reference: https://partner.steamgames.com/doc/features/cloud (under Steam Auto-Cloud section as of September 2021)
227  static const std::vector<std::string> to_ignore {"steam_autocloud.vdf"};
228 
229  if(std::find(to_ignore.begin(), to_ignore.end(), filename) != to_ignore.end()) {
230  return true;
231  } else if(filter) {
232  return filename.end() == std::search(filename.begin(), filename.end(), filter->begin(), filter->end());
233  }
234 
235  return false;
236  });
237 
238  std::vector<save_info> result;
239  std::transform(filenames.begin(), filenames.end(), std::back_inserter(result), creator);
240  std::sort(result.begin(), result.end(), save_info_less_time());
241 
242  return result;
243 }
244 
246 {
247  return save_index_->get(name());
248 }
249 
250 std::string save_info::format_time_local() const
251 {
252  const std::string format = prefs::get().use_twelve_hour_clock_format()
253  // TRANSLATORS: Day of week + month + day of month + year + 12-hour time, eg 'Tue Nov 02 2021, 1:59 PM'.
254  // Format for your locale.
255  ? _("%a %b %d %Y, %I:%M %p")
256  // TRANSLATORS: Day of week + month + day of month + year + 24-hour time, eg 'Tue Nov 02 2021, 13:59'.
257  // Format for your locale.
258  : _("%a %b %d %Y, %H:%M");
259 
261 }
262 
264 {
266 }
267 
269 {
270  // This translatable string must be same one as in replay_savegame::create_initial_filename.
271  // TODO: we really shouldn't be relying on translatable strings like this, especially since
272  // old savefiles may have been created in a different language than the current UI language
273  const std::string replay_str = " " + _("replay");
274  if(a.modified() > b.modified()) {
275  return true;
276  } else if(a.modified() < b.modified()) {
277  return false;
278  } else if(a.name().find(replay_str) == std::string::npos && b.name().find(replay_str) != std::string::npos) {
279  // Special funky case; for files created in the same second,
280  // a replay file sorts less than a non-replay file. Prevents
281  // a timing-dependent bug where it may look like, at the end
282  // of a scenario, the replay and the autosave for the next
283  // scenario are displayed in the wrong order.
284  return true;
285  } else if(a.name().find(replay_str) != std::string::npos && b.name().find(replay_str) == std::string::npos) {
286  return false;
287  } else {
288  return a.name() > b.name();
289  }
290 }
291 
292 static filesystem::scoped_istream find_save_file(const std::string& dir,
293  const std::string& name, const std::vector<std::string>& suffixes)
294 {
295  for(const std::string& suf : suffixes) {
296  filesystem::scoped_istream file_stream =
297  filesystem::istream_file(dir + "/" + name + suf);
298 
299  if(!file_stream->fail()) {
300  return file_stream;
301  }
302  }
303 
304  LOG_SAVE << "Could not open supplied filename '" << name << "'";
305  throw game::load_game_failed();
306 }
307 
308 void read_save_file(const std::string& dir, const std::string& name, config& cfg, std::string* error_log)
309 {
310  static const std::vector<std::string> suffixes{"", ".gz", ".bz2"};
311  filesystem::scoped_istream file_stream = find_save_file(dir, name, suffixes);
312 
313  try {
314  /*
315  * Test the modified name, since it might use a .gz
316  * file even when not requested.
317  */
318  if(filesystem::is_gzip_file(name)) {
319  cfg = io::read_gz(*file_stream);
320  } else if(filesystem::is_bzip2_file(name)) {
321  cfg = io::read_bz2(*file_stream);
322  } else {
323  cfg = io::read(*file_stream);
324  }
325  } catch(const std::ios_base::failure& e) {
326  LOG_SAVE << e.what();
327 
328  if(error_log) {
329  *error_log += e.what();
330  }
331  throw game::load_game_failed();
332  } catch(const config::error& err) {
333  LOG_SAVE << err.message;
334 
335  if(error_log) {
336  *error_log += err.message;
337  }
338 
339  throw game::load_game_failed();
340  }
341 
342  if(cfg.empty()) {
343  LOG_SAVE << "Could not parse file data into config";
344  throw game::load_game_failed();
345  }
346 }
347 
348 void save_index_class::delete_old_auto_saves(const int autosavemax, const int infinite_auto_saves)
349 {
350  log_scope("delete_old_auto_saves()");
351  if(read_only_) {
352  LOG_SAVE << "no-op: read_only instance";
353  return;
354  }
355 
356  const std::string auto_save = _("Auto-Save");
357 
358  int countdown = autosavemax;
359  if(countdown == infinite_auto_saves) {
360  return;
361  }
362 
363  std::vector<save_info> games = get_saves_list(&auto_save);
364  for(std::vector<save_info>::iterator i = games.begin(); i != games.end(); ++i) {
365  if(countdown-- <= 0) {
366  LOG_SAVE << "Deleting savegame '" << i->name() << "'";
367  delete_game(i->name());
368  }
369  }
370 }
371 
372 void save_index_class::delete_game(const std::string& name)
373 {
374  if(read_only_) {
375  log_scope("delete_game()");
376  LOG_SAVE << "no-op: read_only instance";
377  return;
378  }
379 
380  filesystem::delete_file(dir() + "/" + name);
381  remove(name);
382 }
383 
384 create_save_info::create_save_info(const std::shared_ptr<save_index_class>& manager)
385  : manager_(manager)
386 {
387 }
388 
390 {
391  auto modified = filesystem::file_modified_time(manager_->dir() + "/" + filename);
392  manager_->set_modified(filename, modified);
393  return save_info(filename, manager_, modified);
394 }
395 
396 void extract_summary_from_config(config& cfg_save, config& cfg_summary)
397 {
398  auto cfg_snapshot = cfg_save.optional_child("snapshot");
399 
400  // Servergenerated replays contain [scenario] and no [replay_start]
401  auto cfg_replay_start = cfg_save.has_child("replay_start")
402  ? cfg_save.optional_child("replay_start")
403  : cfg_save.optional_child("scenario");
404 
405  auto cfg_replay = cfg_save.optional_child("replay");
406  const bool has_replay = cfg_replay && !cfg_replay->empty();
407  const bool has_snapshot = cfg_snapshot && cfg_snapshot->has_child("side");
408 
409  cfg_summary["replay"] = has_replay;
410  cfg_summary["snapshot"] = has_snapshot;
411 
412  cfg_summary["label"] = cfg_save["label"];
413  cfg_summary["campaign_type"] = cfg_save["campaign_type"];
414 
415  if(cfg_save.has_child("carryover_sides_start")) {
416  cfg_summary["scenario"] = cfg_save.mandatory_child("carryover_sides_start")["next_scenario"];
417  } else {
418  cfg_summary["scenario"] = cfg_save["scenario"];
419  }
420 
421  cfg_summary["difficulty"] = cfg_save["difficulty"];
422  cfg_summary["random_mode"] = cfg_save["random_mode"];
423 
424  cfg_summary["active_mods"] = cfg_save.child_or_empty("multiplayer")["active_mods"];
425  cfg_summary["campaign"] = cfg_save["campaign"];
426  cfg_summary["version"] = cfg_save["version"];
427  cfg_summary["corrupt"] = "";
428 
429  if(has_snapshot) {
430  cfg_summary["turn"] = cfg_snapshot["turn_at"];
431  if(cfg_snapshot["turns"] != "-1") {
432  cfg_summary["turn"] = cfg_summary["turn"].str() + "/" + cfg_snapshot["turns"].str();
433  }
434  }
435 
436  // Ensure we don't get duplicate [leader] tags
437  cfg_summary.clear_children("leader");
438 
439  // Find the human leaders so we can display their icons and names in the load menu.
440  config leader_config;
441 
442  bool shrouded = false;
443 
444  if(auto snapshot = (has_snapshot ? cfg_snapshot : cfg_replay_start)) {
445  for(const config& side : snapshot->child_range("side")) {
446  std::string leader;
447  std::string leader_image;
448  std::string leader_image_tc_modifier;
449  std::string leader_name;
450  int gold = side["gold"].to_int();
451  int units = 0, recall_units = 0;
452 
453  if(side["controller"] != side_controller::human) {
454  continue;
455  }
456 
457  if(side["shroud"].to_bool()) {
458  shrouded = true;
459  }
460 
461  for(const config& u : side.child_range("unit")) {
462  if(u.has_attribute("x") && u.has_attribute("y")) {
463  units++;
464  } else {
465  recall_units++;
466  }
467 
468  // Only take the first leader
469  if(!leader.empty() || !u["canrecruit"].to_bool()) {
470  continue;
471  }
472 
473  const std::string tc_color = team::get_side_color_id_from_config(side);
474 
475  // Don't count it among the troops
476  units--;
477  leader = u["id"].str();
478  leader_name = u["name"].str();
479  leader_image = u["image"].str();
480  leader_image_tc_modifier = "~RC(" + u["flag_rgb"].str() + ">" + tc_color + ")";
481  }
482 
483  // We need a binary path-independent path to the leader image here so it can be displayed
484  // for campaign-specific units even when the campaign isn't loaded yet.
485  auto leader_image_path = filesystem::get_independent_binary_file_path("images", leader_image);
486 
487  // If the image path was found, we append the leader TC modifier. If it's not (such as in
488  // the case where the binary path hasn't been loaded yet, perhaps due to save_index being
489  // deleted), the unaltered image path is used and will be parsed by get_independent_binary_file_path
490  // at runtime.
491  if(leader_image_path) {
492  leader_image = leader_image_path.value() + leader_image_tc_modifier;
493  }
494 
495  leader_config["leader"] = leader;
496  leader_config["leader_name"] = leader_name;
497  leader_config["leader_image"] = leader_image;
498  leader_config["leader_image_tc_modifier"] = leader_image_tc_modifier;
499  leader_config["gold"] = gold;
500  leader_config["units"] = units;
501  leader_config["recall_units"] = recall_units;
502 
503  cfg_summary.add_child("leader", leader_config);
504  }
505  }
506 
507  if(!shrouded) {
508  if(has_snapshot) {
509  if(!cfg_snapshot->find_child("side", "shroud", "yes") && cfg_snapshot->has_attribute("map_data")) {
510  cfg_summary["map_data"] = cfg_snapshot["map_data"].str();
511  } else {
512  ERR_SAVE << "Not saving map because there is shroud";
513  }
514  } else if(has_replay) {
515  if(!cfg_replay_start->find_child("side", "shroud", "yes") && cfg_replay_start->has_attribute("map_data")) {
516  cfg_summary["map_data"] = cfg_replay_start["map_data"];
517  } else {
518  ERR_SAVE << "Not saving map because there is shroud";
519  }
520  }
521  }
522 }
523 
524 } // end namespace savegame
static auto & dummy
Variant for storing WML attributes.
bool empty() const
Tests for an attribute that either was never set or was set to "".
A config object defines a single node in a WML file, with access to child nodes.
Definition: config.hpp:158
const config & child_or_empty(config_key_type key) const
Returns the first child with the given key, or an empty config if there is none.
Definition: config.cpp:390
config & mandatory_child(config_key_type key, int n=0)
Returns the nth child with the given key, or throws an error if there is none.
Definition: config.cpp:362
optional_config_impl< config > find_child(config_key_type key, const std::string &name, const std::string &value)
Returns the first child of tag key with a name attribute containing value.
Definition: config.cpp:780
void clear_children(T... keys)
Definition: config.hpp:602
bool has_child(config_key_type key) const
Determine whether a config has a child or not.
Definition: config.cpp:312
void remove_children(config_key_type key, const std::function< bool(const config &)> &p={})
Removes all children with tag key for which p returns true.
Definition: config.cpp:650
child_itors child_range(config_key_type key)
Definition: config.cpp:268
std::size_t all_children_count() const
Definition: config.cpp:302
bool empty() const
Definition: config.cpp:845
void clear()
Definition: config.cpp:824
optional_config_impl< config > optional_child(config_key_type key, int n=0)
Equivalent to mandatory_child, but returns an empty optional if the nth child was not found.
Definition: config.cpp:380
config & add_child(config_key_type key)
Definition: config.cpp:436
static prefs & get()
bool use_twelve_hour_clock_format()
std::shared_ptr< save_index_class > manager_
Definition: save_index.hpp:79
save_info operator()(const std::string &filename) const
Definition: save_index.cpp:389
create_save_info(const std::shared_ptr< save_index_class > &)
Definition: save_index.cpp:384
void clean_up_index()
Deletes non-existent save files from the index.
Definition: save_index.cpp:100
create_for_default_saves_dir
Syntatic sugar for choosing which constructor to use.
Definition: save_index.hpp:90
static std::shared_ptr< save_index_class > default_saves_dir()
Returns an instance for managing saves in filesystem::get_saves_dir()
Definition: save_index.cpp:210
std::vector< save_info > get_saves_list(const std::string *filter=nullptr)
Get a list of available saves.
Definition: save_index.cpp:217
void remove(const std::string &name)
Delete a savegame from the index, without deleting the underlying file.
Definition: save_index.cpp:70
bool read_only_
The instance for default_saves_dir() writes a cache file.
Definition: save_index.hpp:141
void write_save_index()
Sync to disk, no-op if read_only_ is set.
Definition: save_index.cpp:116
const std::string & dir() const
Definition: save_index.cpp:95
config & get(const std::string &name)
Definition: save_index.cpp:82
void set_modified(const std::string &name, const std::chrono::system_clock::time_point &modified)
Definition: save_index.cpp:77
const std::string dir_
Definition: save_index.hpp:136
void delete_old_auto_saves(const int autosavemax, const int infinite_auto_saves)
Delete autosaves that are no longer needed (according to the autosave policy in the preferences).
Definition: save_index.cpp:348
void delete_game(const std::string &name)
Delete a savegame, including deleting the underlying file.
Definition: save_index.cpp:372
void rebuild(const std::string &name)
Definition: save_index.cpp:44
std::map< std::string, std::chrono::system_clock::time_point > modified_
Definition: save_index.hpp:135
save_index_class(const std::string &dir)
Constructor for a read-only instance.
Definition: save_index.cpp:144
bool clean_up_index_
Flag to only run the clean_up_index method once.
Definition: save_index.hpp:143
static void fix_leader_image_path(config &data)
Definition: save_index.cpp:200
Filename and modification date for a file list.
Definition: save_index.hpp:28
const std::string & name() const
Definition: save_index.hpp:42
const config & summary() const
Definition: save_index.cpp:245
std::string format_time_local() const
Definition: save_index.cpp:250
std::string format_time_summary() const
Definition: save_index.cpp:263
std::shared_ptr< save_index_class > save_index_
Definition: save_index.hpp:58
const auto & modified() const
Definition: save_index.hpp:47
static std::string get_side_color_id_from_config(const config &cfg)
Definition: team.cpp:1000
Definitions for the interface to Wesnoth Markup Language (WML).
Declarations for File-IO.
std::size_t i
Definition: function.cpp:1022
static std::string _(const char *str)
Definition: gettext.hpp:103
Standard logging facilities (interface).
#define log_scope(description)
Definition: log.hpp:275
auto serialize_timestamp(const std::chrono::system_clock::time_point &time)
Definition: chrono.hpp:57
auto parse_timestamp(long long val)
Definition: chrono.hpp:47
auto format_local_timestamp(const std::chrono::system_clock::time_point &time, std::string_view format="%F %T")
Definition: chrono.hpp:62
bool is_bzip2_file(const std::string &filename)
Returns true if the file ends with '.bz2'.
filesystem::scoped_istream istream_file(const std::string &fname, bool treat_failure_as_error)
void get_files_in_dir(const std::string &dir, std::vector< std::string > *files, std::vector< std::string > *dirs, name_mode mode, filter_mode filter, reorder_mode reorder, file_tree_checksum *checksum)
Get a list of all files and/or directories in a given directory.
Definition: filesystem.cpp:450
std::chrono::system_clock::time_point file_modified_time(const bfs::path &path)
bool delete_file(const std::string &filename)
bool is_gzip_file(const std::string &filename)
Returns true if the file ends with '.gz'.
static bool file_exists(const bfs::path &fpath)
Definition: filesystem.cpp:328
std::string get_saves_dir()
std::string get_save_index_file()
filesystem::scoped_ostream ostream_file(const std::string &fname, std::ios_base::openmode mode, bool create_directory)
std::unique_ptr< std::istream > scoped_istream
Definition: filesystem.hpp:53
std::unique_ptr< std::ostream > scoped_ostream
Definition: filesystem.hpp:54
utils::optional< std::string > get_independent_binary_file_path(const std::string &type, const std::string &filename)
Returns an asset path to filename for binary path-independent use in saved games.
config read(std::istream &in, abstract_validator *validator)
Definition: parser.cpp:627
config read_bz2(std::istream &file, abstract_validator *validator)
Might throw a std::ios_base::failure especially bzip2_error.
Definition: parser.cpp:689
void write(std::ostream &out, const configr_of &cfg, unsigned int level)
Definition: parser.cpp:764
config read_gz(std::istream &file, abstract_validator *validator)
Might throw a std::ios_base::failure especially a gzip_error.
Definition: parser.cpp:683
void write_gz(std::ostream &out, const configr_of &cfg)
Definition: parser.cpp:783
logger & err()
Definition: log.cpp:306
void extract_summary_from_config(config &, config &)
Definition: save_index.cpp:396
static filesystem::scoped_istream find_save_file(const std::string &dir, const std::string &name, const std::vector< std::string > &suffixes)
Definition: save_index.cpp:292
void read_save_file(const std::string &dir, const std::string &name, config &cfg, std::string *error_log)
Read the complete config information out of a savefile.
Definition: save_index.cpp:308
constexpr auto transform
Definition: ranges.hpp:41
constexpr auto filter
Definition: ranges.hpp:38
std::string format_time_summary(const std::chrono::system_clock::time_point &t)
void erase_if(Container &container, const Predicate &predicate)
Convenience wrapper for using std::remove_if on a container.
Definition: general.hpp:106
auto * find(Container &container, const Value &value)
Convenience wrapper for using find on a container without needing to comare to end()
Definition: general.hpp:140
std::string::const_iterator iterator
Definition: tokenizer.hpp:25
std::string_view data
Definition: picture.cpp:178
#define LOG_SAVE
Definition: save_index.cpp:34
static lg::log_domain log_engine("engine")
static lg::log_domain log_enginerefac("enginerefac")
#define ERR_SAVE
Definition: save_index.cpp:35
std::string filename
Filename.
An exception object used when an IO error occurs.
Definition: filesystem.hpp:67
Error used when game loading fails.
Definition: game_errors.hpp:31
A structure for comparing to save_info objects based on their modified time.
Definition: save_index.hpp:67
bool operator()(const save_info &a, const save_info &b) const
Definition: save_index.cpp:268
#define d
#define e
#define b