2
0

BinaryDeserializer.cpp 2.9 KB

123456789101112131415161718192021222324252627282930313233343536373839404142434445464748495051525354555657585960616263646566676869707172737475767778798081828384858687888990919293949596979899100101102103104105106
  1. /*
  2. * BinaryDeserializer.cpp, part of VCMI engine
  3. *
  4. * Authors: listed in file AUTHORS in main folder
  5. *
  6. * License: GNU General Public License v2.0 or later
  7. * Full text of license available in license.txt file, in main folder
  8. *
  9. */
  10. #include "StdInc.h"
  11. #include "BinaryDeserializer.h"
  12. #include "../filesystem/FileStream.h"
  13. #include "../registerTypes/RegisterTypes.h"
  14. VCMI_LIB_NAMESPACE_BEGIN
  15. extern template void registerTypes<BinaryDeserializer>(BinaryDeserializer & s);
  16. CLoadFile::CLoadFile(const boost::filesystem::path & fname, int minimalVersion)
  17. : serializer(this)
  18. {
  19. registerTypes(serializer);
  20. openNextFile(fname, minimalVersion);
  21. }
  22. //must be instantiated in .cpp file for access to complete types of all member fields
  23. CLoadFile::~CLoadFile() = default;
  24. int CLoadFile::read(void * data, unsigned size)
  25. {
  26. sfile->read(reinterpret_cast<char *>(data), size);
  27. return size;
  28. }
  29. void CLoadFile::openNextFile(const boost::filesystem::path & fname, int minimalVersion)
  30. {
  31. assert(!serializer.reverseEndianess);
  32. assert(minimalVersion <= SERIALIZATION_VERSION);
  33. try
  34. {
  35. fName = fname.string();
  36. sfile = std::make_unique<FileStream>(fname, std::ios::in | std::ios::binary);
  37. sfile->exceptions(std::ifstream::failbit | std::ifstream::badbit); //we throw a lot anyway
  38. if(!(*sfile))
  39. THROW_FORMAT("Error: cannot open to read %s!", fName);
  40. //we can read
  41. char buffer[4];
  42. sfile->read(buffer, 4);
  43. if(std::memcmp(buffer, "VCMI", 4) != 0)
  44. THROW_FORMAT("Error: not a VCMI file(%s)!", fName);
  45. serializer & serializer.fileVersion;
  46. if(serializer.fileVersion < minimalVersion)
  47. THROW_FORMAT("Error: too old file format (%s)!", fName);
  48. if(serializer.fileVersion > SERIALIZATION_VERSION)
  49. {
  50. logGlobal->warn("Warning format version mismatch: found %d when current is %d! (file %s)\n", serializer.fileVersion, SERIALIZATION_VERSION , fName);
  51. auto * versionptr = reinterpret_cast<char *>(&serializer.fileVersion);
  52. std::reverse(versionptr, versionptr + 4);
  53. logGlobal->warn("Version number reversed is %x, checking...", serializer.fileVersion);
  54. if(serializer.fileVersion == SERIALIZATION_VERSION)
  55. {
  56. logGlobal->warn("%s seems to have different endianness! Entering reversing mode.", fname.string());
  57. serializer.reverseEndianess = true;
  58. }
  59. else
  60. THROW_FORMAT("Error: too new file format (%s)!", fName);
  61. }
  62. }
  63. catch(...)
  64. {
  65. clear(); //if anything went wrong, we delete file and rethrow
  66. throw;
  67. }
  68. }
  69. void CLoadFile::reportState(vstd::CLoggerBase * out)
  70. {
  71. out->debug("CLoadFile");
  72. if(!!sfile && *sfile)
  73. out->debug("\tOpened %s Position: %d", fName, sfile->tellg());
  74. }
  75. void CLoadFile::clear()
  76. {
  77. sfile = nullptr;
  78. fName.clear();
  79. serializer.fileVersion = 0;
  80. }
  81. void CLoadFile::checkMagicBytes(const std::string &text)
  82. {
  83. std::string loaded = text;
  84. read((void *)loaded.data(), static_cast<unsigned int>(text.length()));
  85. if(loaded != text)
  86. throw std::runtime_error("Magic bytes doesn't match!");
  87. }
  88. VCMI_LIB_NAMESPACE_END