DatFile.cpp 19 KB

123456789101112131415161718192021222324252627282930313233343536373839404142434445464748495051525354555657585960616263646566676869707172737475767778798081828384858687888990919293949596979899100101102103104105106107108109110111112113114115116117118119120121122123124125126127128129130131132133134135136137138139140141142143144145146147148149150151152153154155156157158159160161162163164165166167168169170171172173174175176177178179180181182183184185186187188189190191192193194195196197198199200201202203204205206207208209210211212213214215216217218219220221222223224225226227228229230231232233234235236237238239240241242243244245246247248249250251252253254255256257258259260261262263264265266267268269270271272273274275276277278279280281282283284285286287288289290291292293294295296297298299300301302303304305306307308309310311312313314315316317318319320321322323324325326327328329330331332333334335336337338339340341342343344345346347348349350351352353354355356357358359360361362363364365366367368369370371372373374375376377378379380381382383384385386387388389390391392393394395396397398399400401402403404405406407408409410411412413414415416417418419420421422423424425426427428429430431432433434435436437438439440441442443444445446447448449450451452453454455456457458459460461462463464465466467468469470471472473474475476477478479480481482483484485486487488489490491492493494495496497498499500501502503504505506507508509510511512513514515516517518519520521522
  1. //
  2. // Created by Иван_Архипов on 31.10.2017.
  3. //
  4. #include "DatFile.h"
  5. #include "BinaryData.h"
  6. #include "Common/DatException.h"
  7. #include "SubDirectory.h"
  8. #include "Subfile.h"
  9. #include <locale>
  10. extern "C++"
  11. {
  12. namespace LOTRO_DAT {
  13. DatFile::DatFile() {
  14. dat_state_ = CLOSED;
  15. patched_ = false;
  16. }
  17. DatFile::DatFile(const char *filename, int dat_id) {
  18. dat_id_ = dat_id;
  19. dat_state_ = CLOSED;
  20. patched_ = false;
  21. OpenDatFile(filename);
  22. ReadSuperBlock();
  23. MakeDirectories();
  24. try {
  25. MakeDictionary();
  26. } catch (std::exception &e) {
  27. fprintf(stderr, "Caught %s exception.", e.what());
  28. fprintf(stderr, "Unable to make dictionary!! Unable to init DatFile!!!");
  29. return;
  30. }
  31. if (dat_state_ == SUCCESS_DICTIONARY)
  32. dat_state_ = READY;
  33. else
  34. throw DatException("Bad DatFile initialization! Not all init states were successfully passed!",
  35. INIT_EXCEPTION);
  36. }
  37. DatFile::~DatFile() {
  38. if (patched_) {
  39. std::cout << "There are some updated files. Rewriting dictionary..." << std::endl << std::flush;
  40. UpdateHeader();
  41. std::cout << "Updated header..." << std::endl << std::flush;
  42. UpdateSubdirectories();
  43. std::cout << "Updated subdirectories..." << std::endl << std::flush;
  44. }
  45. if (file_handler_ != nullptr)
  46. fclose(file_handler_);
  47. delete file_handler_;
  48. delete root_directory_;
  49. }
  50. /// Extracts file with file_id.
  51. /// If path is undefined then it will be recognised as current working directory
  52. /// Output file path consists of "path + file_id + file_extension";
  53. /// NOTICE: The directory, mentioned in "std::string path" variable SHOULD BE ALREADY CREATED;
  54. /// Otherwise DatException() will be thrown.
  55. /// Returns true, if file was successfully extracted;
  56. /// Throws DatException() if undefined behaviour happened
  57. bool DatFile::ExtractFile(long long file_id, const std::string &path) {
  58. if (dat_state_ != READY) {
  59. throw DatException("Bad DatFile::ExtractFile() - invalid DatFile state!", EXPORT_EXCEPTION);
  60. }
  61. BinaryData file_data;
  62. try {
  63. file_data = GetFileData(dictionary_[file_id], 8);
  64. } catch (std::exception &e) {
  65. fprintf(stderr, "Caught %s exception.", e.what());
  66. fprintf(stderr, "Unable to extract file due to uncaught exception while getting file data. Passing...\n");
  67. return false;
  68. }
  69. long long export_size = 0;
  70. std::vector<BinaryData> binary_data;
  71. std::vector<std::u16string> text_data;
  72. std::vector<YAML::Node> options;
  73. try {
  74. dictionary_[file_id]->PrepareForExport(file_data, export_size, binary_data, text_data, options);
  75. } catch (std::exception &e) {
  76. fprintf(stderr, "Caught %s exception.", e.what());
  77. fprintf(stderr, "Unable to extract file due to uncaught exception while preparing file for export. Passing...\n");
  78. return false;
  79. }
  80. for (int i = 0; i < export_size; ++i) {
  81. binary_data[i].WriteToFile(path + "_" + std::to_string(i) + options[i]["ext"].as<std::string>());
  82. }
  83. return true;
  84. }
  85. /// Extracts file with file_id to database "db".
  86. /// DATABASE SHOULD BE ALREADY CREATED; Otherwise DatException will be called.
  87. /// NOTICE: The directory, mentioned in "std::string path" variable SHOULD BE ALREADY CREATED;
  88. /// Otherwise DatException() will be thrown.
  89. /// Returns true, if file was successfully extracted;
  90. /// Throws DatException() if undefined behaviour happened
  91. bool DatFile::ExtractFile(long long file_id, Database *db) {
  92. if (dat_state_ != READY) {
  93. throw DatException("Bad DatFile::ExtractFile() - invalid DatFile state!", EXPORT_EXCEPTION);
  94. }
  95. BinaryData file_data;
  96. try {
  97. file_data = GetFileData(dictionary_[file_id], 8);
  98. } catch (std::exception &e) {
  99. fprintf(stderr, "Caught %s exception.", e.what());
  100. fprintf(stderr, "Unable to extract file due to uncaught exception while getting file data. Passing...\n");
  101. return false;
  102. }
  103. long long export_size = 0;
  104. std::vector<BinaryData> binary_data;
  105. std::vector<std::u16string> text_data;
  106. std::vector<YAML::Node> options;
  107. try {
  108. dictionary_[file_id]->PrepareForExport(file_data, export_size, binary_data, text_data, options);
  109. } catch (std::exception &e) {
  110. fprintf(stderr, "Caught %s exception.", e.what());
  111. fprintf(stderr, "Unable to extract file due to uncaught exception while preparing file for export. Passing...\n");
  112. return false;
  113. }
  114. for (int i = 0; i < export_size; ++i) {
  115. std::stringstream option;
  116. option << options[i];
  117. try {
  118. db->PushFile(binary_data[i], text_data[i], option.str());
  119. } catch (std::exception &e) {
  120. fprintf(stderr, "Caught %s exception.", e.what());
  121. printf("Caught %s exception.", e.what());
  122. fflush(stdout);
  123. fprintf(stderr, "Unable to put file or it's part to database. Continuing without this part. Database may be not complete\n");
  124. }
  125. }
  126. return true;
  127. }
  128. /// Extracts all files with specific type to "path + type + file_id + file_part + extension" files;
  129. /// If path is undefined then it will be recognised as current working directory
  130. /// NOTICE: The directory, mentioned in "std::string path" variable SHOULD BE ALREADY CREATED;
  131. /// Otherwise DatException() will be thrown.
  132. /// Returns number of successfully extracted files
  133. /// Throws DatException() if undefined behaviour happened
  134. int DatFile::ExtractAllFilesByType(FILE_TYPE type, std::string path) {
  135. if (dat_state_ != READY) {
  136. throw DatException("Bad DatFile::ExtractAllFilesByType() - invalid DatFile state!", EXPORT_EXCEPTION);
  137. }
  138. int success = 0;
  139. for (auto i : dictionary_) {
  140. FILE_TYPE file_type = i.second->FileType();
  141. if (file_type == type) {
  142. success += ExtractFile(i.second->file_id(), (path + std::to_string(i.second->file_id())));
  143. }
  144. }
  145. return success;
  146. }
  147. /// Extracts all files with specific type to database "db";
  148. /// DATABASE SHOULD BE ALREADY CREATED; Otherwise DatException will be called.
  149. /// Returns number of successfully extracted files
  150. /// Throws DatException() if undefined behaviour happened
  151. int DatFile::ExtractAllFilesByType(FILE_TYPE type, Database *db) {
  152. if (dat_state_ != READY) {
  153. throw DatException("Bad DatFile::ExtractAllFilesByType() - invalid DatFile state!", EXPORT_EXCEPTION);
  154. }
  155. int success = 0;
  156. for (auto i : dictionary_) {
  157. FILE_TYPE file_type = i.second->FileType();
  158. if (file_type == type) {
  159. success += ExtractFile(i.second->file_id(), db);
  160. }
  161. }
  162. return success;
  163. }
  164. /// TODO: Write desctiption
  165. bool DatFile::PatchFile(const char *filename, YAML::Node options, long long dat_id) {
  166. if (dat_id != dat_id_)
  167. return false;
  168. BinaryData data;
  169. data.ReadFromFile(filename);
  170. auto file_id = options["fid"].as<long long>();
  171. BinaryData old_data = GetFileData(dictionary_[file_id]);
  172. data = dictionary_[file_id]->MakeForImport(old_data, data, u"", options);
  173. try {
  174. ApplyFilePatch(dictionary_[file_id], data);
  175. } catch (std::exception &e) {
  176. fprintf(stderr, "Caught %s exception.", e.what());
  177. fprintf(stderr,
  178. "Some errors happened while patching file with id = %lld. Continuing process without this file..\n"
  179. "WARNING: DAT FILE CAN BE CORRUPTED!\n", file_id);
  180. printf("Some errors happened while patching file with id = %lld. Continuing process without this file..\n"
  181. "WARNING: DAT FILE CAN BE CORRUPTED!\n", file_id);
  182. fflush(stdout);
  183. return false;
  184. }
  185. return true;
  186. }
  187. /// TODO: Write description
  188. bool DatFile::PatchDatabase(Database *db) {
  189. return false;
  190. }
  191. /// DatFile::WriteUnorderedDictionary(...);
  192. /// Prints list of all found files with some information about them to file.
  193. /// Gets std::string path - path to directory, where the file will be written with name "dict.txt"
  194. void DatFile::WriteUnorderedDictionary(std::string path) const {
  195. FILE *f;
  196. fopen_s(&f, (path + "dict.txt").c_str(), "w");
  197. fprintf(f, "file_id offset size size2 extension\n");
  198. for (auto i : dictionary_) {
  199. fprintf(f, "%lld %lld %lld %lld %s\n", i.second->file_id(), i.second->file_offset(), i.second->file_size(),
  200. i.second->block_size(), i.second->Extension().c_str());
  201. }
  202. fclose(f);
  203. }
  204. /// DatFile::files_number();
  205. /// Returns amount of files, found in dictionaries of DatFile. Some if them may be empty or erased.
  206. long long DatFile::files_number() const {
  207. return dictionary_.size();
  208. }
  209. /// DatFile::GetFileData()
  210. /// Returns BinaryData, which contains of subfile data, made from parts of file in DatFile
  211. BinaryData DatFile::GetFileData(const Subfile *file, long long int offset) {
  212. BinaryData mfile_id(4);
  213. ReadData(mfile_id, 4, file->file_offset() + 8);
  214. if (file->file_id() != mfile_id.ToNumber<4>(0))
  215. throw DatException("Bad DatFile::GetFileData() - file_id in Subfile doesn't match to file_id in DatFile.", READ_EXCEPTION);
  216. BinaryData data((unsigned)(file->file_size()));
  217. if (file->block_size() >= file->file_size() + 8) {
  218. ReadData(data, file->file_size(), file->file_offset() + offset);
  219. return data;
  220. }
  221. BinaryData fragments_count(4);
  222. ReadData(fragments_count, 4, file->file_offset());
  223. long long fragments_number = fragments_count.ToNumber<4>(0);
  224. long long current_block_size = file->block_size() - offset - 8 * fragments_number;
  225. ReadData(data, current_block_size , file->file_offset() + offset);
  226. BinaryData FragmentsDictionary(8 * unsigned(fragments_number));
  227. ReadData(FragmentsDictionary, 8 * unsigned(fragments_number), file->file_offset() + file->block_size() - 8 * fragments_number);
  228. for (long long i = 0; i < fragments_number; i++) {
  229. long long fragment_size = FragmentsDictionary.ToNumber<4>(8 * i);
  230. long long fragment_offset = FragmentsDictionary.ToNumber<4>(8 * i + 4);
  231. ReadData(data, std::min(fragment_size, file->file_size() - current_block_size), fragment_offset, current_block_size );
  232. current_block_size += fragment_size;
  233. }
  234. return data;
  235. }
  236. /// DatFile constants' getters.
  237. long long DatFile::constant1() const {
  238. return constant1_;
  239. }
  240. long long DatFile::constant2() const {
  241. return constant2_;
  242. }
  243. long long DatFile::file_size() const {
  244. return file_size_;
  245. }
  246. long long DatFile::version1() const {
  247. return version1_;
  248. }
  249. long long DatFile::version2() const {
  250. return version2_;
  251. }
  252. /// DatFile special functions for opening and reading/writing raw data.
  253. /// Shouldn't be used by any external classes except Subfile and Subdirectory.
  254. void DatFile::OpenDatFile(const char *dat_name) {
  255. if (dat_state_ != CLOSED)
  256. throw DatException("Bad initialisation of DatFile - current DatFile isn't in correct state!",
  257. INIT_EXCEPTION);
  258. fopen_s(&file_handler_, dat_name, "r+b");
  259. if (file_handler_ == nullptr) {
  260. std::string err = "Bad DatFile::OpenDatFile. Unable to open file ";
  261. err += dat_name;
  262. throw DatException(err.c_str(), INIT_EXCEPTION);
  263. }
  264. fseek(file_handler_, 0, SEEK_END);
  265. file_size_ = ftell(file_handler_);
  266. fseek(file_handler_, 0, SEEK_SET);
  267. dat_state_ = SUCCESS_OPENED;
  268. }
  269. void DatFile::ReadSuperBlock() {
  270. if (dat_state_ != SUCCESS_OPENED)
  271. throw DatException("Bad DatFile::ReadSuperBlock() - DatFile isn't in valid state!", INIT_EXCEPTION);
  272. BinaryData data(1024);
  273. ReadData(data, 1024);
  274. constant1_ = data.ToNumber<4>(0x100);
  275. constant2_ = data.ToNumber<4>(0x140);
  276. version1_ = data.ToNumber<4>(0x14C);
  277. version2_ = data.ToNumber<4>(0x150);
  278. fragmentation_journal_offset_ = data.ToNumber<4>(0x154);
  279. root_directory_offset_ = data.ToNumber<4>(0x160);
  280. auto size1 = data.ToNumber<4>(0x148);
  281. if (constant1_ != 0x4C5000)
  282. throw DatException(
  283. "Bad DatFile::ReadSuperBlock - variable at position 0x100 is not equal to .dat file constant!",
  284. INIT_EXCEPTION);
  285. if (constant2_ != 0x5442)
  286. throw DatException(
  287. "Bad DatFile::ReadSuperBlock - variable at position 0x140 is not equal to .dat file constant!",
  288. INIT_EXCEPTION);
  289. if (file_size_ != size1)
  290. throw DatException(
  291. "Bad DatFile::ReadSuperBlock - variable at 0x148 position is not equal to .dat file size!",
  292. INIT_EXCEPTION);
  293. dat_state_ = SUCCESS_SUPERBLOCK;
  294. }
  295. void DatFile::MakeDirectories() {
  296. if (dat_state_ != SUCCESS_SUPERBLOCK)
  297. throw DatException("Bad DatFile::MakeDirectories() - DatFile isn't in valid state!", INIT_EXCEPTION);
  298. root_directory_ = new SubDirectory((unsigned) root_directory_offset_, this);
  299. dat_state_ = SUCCESS_DIRECTORIES;
  300. }
  301. void DatFile::MakeDictionary() {
  302. if (dat_state_ != SUCCESS_DIRECTORIES)
  303. throw DatException("Bad DatFile::MakeDictionary() - DatFile isn't in valid state!", INIT_EXCEPTION);
  304. try {
  305. root_directory_->MakeDictionary(dictionary_);
  306. } catch (std::exception &e) {
  307. fprintf(stderr, "Caught %s exception.", e.what());
  308. fprintf(stderr, "Bad DatFile::MakeDictionary() - File is corrupted?\n");
  309. return;
  310. }
  311. dat_state_ = SUCCESS_DICTIONARY;
  312. }
  313. void DatFile::ReadData(BinaryData &data, long long size, long long offset, long long data_offset) {
  314. if (dat_state_ == CLOSED)
  315. throw DatException("Bad DatFile::ReadData() - DatFile isn't in valid state!", READ_EXCEPTION);
  316. if (data_offset + size > data.size()) {
  317. std::string err = "Bad DatFile::ReadData - trying to read more than BinaryData size\n";
  318. err += std::string("Reading ") + std::to_string(size) + std::string(" bytes from ")
  319. + std::to_string(offset) + std::string(" position in dat file.");
  320. throw DatException(err.c_str(), READ_EXCEPTION);
  321. }
  322. if (offset + size > file_size()) {
  323. std::string err = "Bad DatFile::ReadData - trying to read more than DatFile size elapsed\n";
  324. err += std::string("Reading ") + std::to_string(size) + std::string(" bytes from ")
  325. + std::to_string(offset) + std::string(" position in dat file.");
  326. throw DatException(err.c_str(), READ_EXCEPTION);
  327. }
  328. _fseeki64(file_handler_, offset, SEEK_SET);
  329. fread(data.data() + data_offset, unsigned(size), 1, file_handler_);
  330. data.CheckCompression();
  331. }
  332. void DatFile::WriteData(const BinaryData &data, long long size, long long offset, long long data_offset) {
  333. if (dat_state_ != READY)
  334. throw DatException("Bad DatFile::WriteData() - DatFile isn't in valid state!", WRITE_EXCEPTION);
  335. _fseeki64(file_handler_, offset, SEEK_SET);
  336. if (data_offset + size > data.size())
  337. throw DatException("Bad DatFile::WriteData - trying to write more than BinaryData size", WRITE_EXCEPTION);
  338. fwrite(data.data() + data_offset, unsigned(size), 1, file_handler_);
  339. }
  340. /// Special functions used by patch process.
  341. /// Shouldn't be used by any external class.
  342. void DatFile::ApplyFilePatch(Subfile *file, const BinaryData &data) {
  343. if (patched_list.count(file->file_id()) != 0) {
  344. return;
  345. }
  346. auto journal = GetFragmentationJournal();
  347. file->file_size_ = data.size() - 8;
  348. file->fragments_count_ = 0;
  349. if (data.size() > file->block_size()) {
  350. file->file_offset_ = journal[0].second;
  351. file->block_size_ = data.size();
  352. journal[0].second += data.size();
  353. if (journal[0].first < data.size()) {
  354. UpdateFileSize();
  355. journal[0].first = 0;
  356. std::cout << "File size updated - new size is " << file_size_ << std::endl;
  357. } else {
  358. journal[0].first -= data.size();
  359. }
  360. }
  361. BinaryData fragments_count(4);
  362. fragments_count.FromNumber<4>(file->fragments_count_);
  363. BinaryData file_data = fragments_count + data.CutData(4);
  364. if (file->file_id() != file_data.ToNumber<4>(8))
  365. throw DatException("Bad DatFile::ApplyFilePatch() - Created data's file_id doesn't match to original! "
  366. "Patch wasn't written to .dat file");
  367. WriteData(file_data, file_data.size(), file->file_offset());
  368. patched_list[file->file_id()] = new BinaryData(file->MakeHeaderData());
  369. UpdateFragmentationJournal(journal);
  370. }
  371. void DatFile::UpdateSubdirectories() {
  372. root_directory_->UpdateDirectories(patched_list);
  373. }
  374. std::vector<std::pair<long long, long long> > DatFile::GetFragmentationJournal() {
  375. BinaryData data(8);
  376. std::cout << "journal offset: " << fragmentation_journal_offset_ << std::endl;
  377. ReadData(data, 8, fragmentation_journal_offset_ + 8);
  378. std::vector<std::pair<long long, long long> > result;
  379. result.emplace_back(std::make_pair(data.ToNumber<4>(0), data.ToNumber<4>(4)));
  380. return result;
  381. }
  382. void DatFile::UpdateHeader() {
  383. BinaryData data(4);
  384. data.FromNumber<4>(constant1_);
  385. WriteData(data, 4, 0x100);
  386. data.FromNumber<4>(constant2_);
  387. WriteData(data, 4, 0x140);
  388. data.FromNumber<4>(file_size_);
  389. WriteData(data, 4, 0x148);
  390. data.FromNumber<4>(version1_);
  391. WriteData(data, 4, 0x14C);
  392. data.FromNumber<4>(version2_);
  393. WriteData(data, 4, 0x150);
  394. data.FromNumber<4>(fragmentation_journal_offset_);
  395. WriteData(data, 4, 0x154);
  396. data.FromNumber<4>(root_directory_offset_);
  397. WriteData(data, 4, 0x160);
  398. }
  399. void DatFile::UpdateFragmentationJournal(const std::vector<std::pair<long long, long long> > &journal) {
  400. for (unsigned i = 0; i < journal.size(); i++) {
  401. long long size = journal[i].first;
  402. long long offset = journal[i].second;
  403. BinaryData data(4);
  404. data.FromNumber<4>(size);
  405. WriteData(data, 4, fragmentation_journal_offset_ + 8 * (i + 1));
  406. data.FromNumber<4>(offset);
  407. WriteData(data, 4, fragmentation_journal_offset_ + 8 * (i + 1) + 4);
  408. }
  409. }
  410. void DatFile::UpdateFileSize() {
  411. _fseeki64(file_handler_, 0, SEEK_END);
  412. file_size_ = _ftelli64(file_handler_);
  413. }
  414. }
  415. }