SubDirectory.cpp 9.6 KB

123456789101112131415161718192021222324252627282930313233343536373839404142434445464748495051525354555657585960616263646566676869707172737475767778798081828384858687888990919293949596979899100101102103104105106107108109110111112113114115116117118119120121122123124125126127128129130131132133134135136137138139140141142143144145146147148149150151152153154155156157158159160161162163164165166167168169170171172173174175176177178179180181182183184185186187188189190191192193194195196197198199200201202203204205206207208209210211212213214215216217218219220221222223224225226227228229230231
  1. //
  2. // Created by Иван_Архипов on 07.11.2017.
  3. //
  4. #include "SubDirectory.h"
  5. #include "DatFile.h"
  6. #include "SubFile.h"
  7. #include "BinaryData.h"
  8. #include "SubfileData.h"
  9. #include "EasyLogging++/easylogging++.h"
  10. #include "DatOperationResult.h"
  11. #include "Subfiles/TextSubFile.h"
  12. #include "Subfiles/DdsSubFile.h"
  13. #include "Subfiles/FontSubFile.h"
  14. #include "Subfiles/JpgSubFile.h"
  15. #include "Subfiles/OggSubFile.h"
  16. #include "Subfiles/WavSubFile.h"
  17. #include "Subfiles/UnknownSubFile.h"
  18. namespace LOTRO_DAT {
  19. std::unordered_set<long long> SubDirectory::visited_subdirectories_ = std::unordered_set<long long>();
  20. std::unordered_set<long long> SubDirectory::visited_subfiles_ = std::unordered_set<long long>();
  21. std::set<std::shared_ptr<SubDirectory>, SubDirectory::SubDirectoryOffsetComp> SubDirectory::subdir_init_queue_
  22. = std::set<std::shared_ptr<SubDirectory>, SubDirectory::SubDirectoryOffsetComp>();
  23. std::set<std::shared_ptr<SubDirectory>, SubDirectory::SubDirectoryOffsetComp> SubDirectory::subfile_init_queue_
  24. = std::set<std::shared_ptr<SubDirectory>, SubDirectory::SubDirectoryOffsetComp>();
  25. SubDirectory::~SubDirectory() {
  26. subfiles_.clear();
  27. subdirs_.clear();
  28. }
  29. SubDirectory::SubDirectory(long long offset, DatFile &dat, long long max_subdirs) :
  30. dat_(dat), offset_(offset), max_subdirs_(max_subdirs) {
  31. }
  32. bool SubDirectory::MakeSubDirectories() {
  33. BinaryData data(1024);
  34. dat_.getIO().ReadData(data, 63 * 8, offset_);
  35. if (data.Empty()) {
  36. LOG(ERROR) << "(READ ERROR) Incorrect directory at offset " << offset_;
  37. return false;
  38. }
  39. if (data.ToNumber<4>(0) != 0 || data.ToNumber<4>(4) != 0) {
  40. LOG(WARNING) << "first 8 bytes are not equal to 0 at offset " << offset_;
  41. return false;
  42. }
  43. for (unsigned int i = 8; i < 63 * 8; i += 8) {
  44. if (data.ToNumber<4>(i) == 0 || data.ToNumber<4>(i + 4) == 0)
  45. break;
  46. // if (visited_subdirectories_.count(data.ToNumber<4>(i + 4)) > 0) {
  47. // LOG(WARNING) << "Visiting subdirectory at offset " << data.ToNumber<4>(i + 4) << " more than one time. Passing.";
  48. // continue;
  49. // }
  50. visited_subdirectories_.insert(data.ToNumber<4>(i + 4));
  51. std::shared_ptr<SubDirectory> subdir = std::make_shared<SubDirectory>(data.ToNumber<4>(i + 4), dat_);
  52. //if (subdir->subfiles_.empty() && subdir->subdirs_.empty()) {
  53. // LOG(WARNING) << "Sub-subdirectory is empty or made empty... Dictionary offset = " << offset_ + i << "; Passing others";
  54. // break;
  55. //} else {
  56. subdirs_.push_back(subdir);
  57. subdir_init_queue_.insert(subdir);
  58. //}
  59. }
  60. return true;
  61. }
  62. bool SubDirectory::MakeSubFiles() {
  63. BinaryData data = BinaryData(4);
  64. dat_.getIO().ReadData(data, 4, offset_ + 63 * 8);
  65. if (data.Empty()) {
  66. LOG(ERROR) << "(READ ERROR) Incorrect directory at offset " << offset_;
  67. return false;
  68. }
  69. auto subfiles_number = data.ToNumber<4>(0);
  70. if (subfiles_number >= 64) {
  71. LOG(ERROR) << "Incorrect directory (subfiles_num >= 64) at offset " << offset_;
  72. return false;
  73. }
  74. subfiles_.resize(unsigned(subfiles_number), nullptr);
  75. BinaryData headers(32 * unsigned(subfiles_number));
  76. dat_.getIO().ReadData(headers, 32 * subfiles_number, offset_ + 63 * 8 + 4);
  77. // LOG(INFO) << "============== DIRECTORY offset = " << offset_ << "=======================";
  78. for (int i = 0; i < subfiles_number; i++) {
  79. BinaryData header = headers.CutData(32 * i, 32 * (i + 1));
  80. if (header.Empty()) {
  81. LOG(ERROR) << "(READ ERROR) Incorrect directory (unable to read subfile data) at offset " << offset_;
  82. return false;
  83. }
  84. if (header.ToNumber<4>(20) == 0 || header.ToNumber<4>(28) != 0)
  85. continue;
  86. if (visited_subfiles_.count(header.ToNumber<4>(4))) {
  87. LOG(ERROR) << "Found duplicate file with id " << header.ToNumber<4>(4);
  88. break;
  89. }
  90. subfiles_[i] = MakeSubfile(
  91. offset_ + 63 * 8 + 4 + 32 * i,
  92. header.ToNumber<4>(0), // unknown1
  93. header.ToNumber<4>(4), // file_id
  94. header.ToNumber<4>(8), // file_offset
  95. header.ToNumber<4>(12), // file_size
  96. header.ToNumber<4>(16), // timestamp
  97. header.ToNumber<4>(20), // version
  98. header.ToNumber<4>(24), // block_size
  99. header.ToNumber<4>(28) // unknown2 - must be zero??
  100. );
  101. // LOG(INFO) << subfiles_[i]->file_id();
  102. //if (dat_.CorrectSubfile(subfiles_[i])) {
  103. visited_subfiles_.insert(subfiles_[i]->file_id());
  104. // } else {
  105. // LOG(WARNING) << "Incorrect SubFile in directory at offset " << offset_ + 63 * 8 + 4 + 32 * i << " (id = " << subfiles_[i]->file_id() << ");";
  106. // subfiles_[i] = nullptr;
  107. // break;
  108. // }
  109. }
  110. return true;
  111. }
  112. void SubDirectory::MakeDictionary(std::map<long long, std::shared_ptr<SubFile> > &dict) {
  113. for (const std::shared_ptr<SubFile> &file: subfiles_) {
  114. if (!file)
  115. continue;
  116. if (dict.count(file->file_id() != 0)) {
  117. LOG(WARNING) << "Found multiple instances of file " << file->file_id() << " at dictionary offset "
  118. << file->dictionary_offset() << ". Base offset = " << dict[file->file_id()]->dictionary_offset();
  119. //if (!dat_.CorrectSubfile(file) || dat_.CorrectSubfile(dict[file->file_id_]))
  120. // continue;
  121. }
  122. dict[file->file_id()] = file;
  123. }
  124. for (const auto &dir : subdirs_) {
  125. dir->MakeDictionary(dict);
  126. }
  127. }
  128. std::shared_ptr<SubFile> SubDirectory::MakeSubfile(long long dictionary_offset, long long unknown1, long long file_id,
  129. long long file_offset, long long file_size, long long timestamp,
  130. long long version, long long block_size, long long unknown2) {
  131. FILE_TYPE type = GetSubfileType(file_id, file_offset);
  132. switch (type) {
  133. case TEXT:
  134. return std::dynamic_pointer_cast<SubFile>(std::make_shared<TextSubFile>(dat_, dictionary_offset, unknown1, file_id, file_offset, file_size, timestamp, version, block_size, unknown2));
  135. case JPG:
  136. return std::dynamic_pointer_cast<SubFile>(std::make_shared<JpgSubFile>(dat_, dictionary_offset, unknown1, file_id, file_offset, file_size, timestamp, version, block_size, unknown2));
  137. case DDS:
  138. return std::dynamic_pointer_cast<SubFile>(std::make_shared<DdsSubFile>(dat_, dictionary_offset, unknown1, file_id, file_offset, file_size, timestamp, version, block_size, unknown2));
  139. case WAV:
  140. return std::dynamic_pointer_cast<SubFile>(std::make_shared<WavSubFile>(dat_, dictionary_offset, unknown1, file_id, file_offset, file_size, timestamp, version, block_size, unknown2));
  141. case OGG:
  142. return std::dynamic_pointer_cast<SubFile>(std::make_shared<OggSubFile>(dat_, dictionary_offset, unknown1, file_id, file_offset, file_size, timestamp, version, block_size, unknown2));
  143. case FONT:
  144. return std::dynamic_pointer_cast<SubFile>(std::make_shared<FontSubFile>(dat_, dictionary_offset, unknown1, file_id, file_offset, file_size, timestamp, version, block_size, unknown2));
  145. case UNKNOWN:
  146. return std::dynamic_pointer_cast<SubFile>(std::make_shared<UnknownSubFile>(dat_, dictionary_offset, unknown1, file_id, file_offset, file_size, timestamp, version, block_size, unknown2));
  147. }
  148. LOG(ERROR) << "Incorrect file type..";
  149. return std::dynamic_pointer_cast<SubFile>(std::make_shared<UnknownSubFile>(dat_, dictionary_offset, unknown1, file_id, file_offset, file_size, timestamp, version, block_size, unknown2));
  150. }
  151. FILE_TYPE SubDirectory::GetSubfileType(long long file_id, long long file_offset) const {
  152. // Text check based on file_id
  153. if (((unsigned long long) file_id >> 24ull) == 0x25ull)
  154. return TEXT;
  155. // Font check based on file_id
  156. if (((unsigned long long) file_id >> 24ull) == 0x42ull)
  157. return FONT;
  158. BinaryData header(64);
  159. dat_.getIO().ReadData(header, 64, (unsigned) file_offset + 8);
  160. if (header.Empty()) {
  161. LOG(ERROR) << "Unable to read file header. file_id = " << file_id << ", offset = " << offset_;
  162. return UNKNOWN;
  163. }
  164. // jpeg / dds check
  165. if (((unsigned long long) file_id >> 24ull) == 0x41ull) {
  166. long long soi = header.ToNumber<2>(24);
  167. long long marker = header.ToNumber<2>(26);
  168. //auto markerSize = header.ToNumber<short>(28);
  169. //auto four = header.ToNumber<int>(30);
  170. if ((soi == 0xD8FFll && marker == 0xE0FFll) || marker == 0xE1FFll)
  171. return JPG;
  172. return DDS;
  173. }
  174. // Ogg and Wav check
  175. if (header[8] == 0x4F && header[9] == 0x67 && header[10] == 0x67 && header[11] == 0x53)
  176. return OGG;
  177. if (header[8] == 0x52 && header[9] == 0x49 && header[10] == 0x46 && header[11] == 0x46)
  178. return WAV;
  179. return UNKNOWN;
  180. }
  181. void SubDirectory::clear() {
  182. subfiles_.clear();
  183. subdirs_.clear();
  184. }
  185. };