|
@@ -39,20 +39,7 @@ namespace LOTRO_DAT {
|
|
MakeDirectories();
|
|
MakeDirectories();
|
|
MakeDictionary();
|
|
MakeDictionary();
|
|
|
|
|
|
- InitLocale(PATCHED, (filename + "patched.dbgm").c_str());
|
|
|
|
- InitLocale(ORIGINAL, (filename + "original.dbgm").c_str());
|
|
|
|
-
|
|
|
|
- FILE *locale = fopen((filename + ".dbgm").c_str(), "r");
|
|
|
|
- if (locale == nullptr)
|
|
|
|
- current_locale_ = ORIGINAL;
|
|
|
|
- else {
|
|
|
|
- auto loc = new char[10];
|
|
|
|
- fscanf(locale, "%s", loc);
|
|
|
|
- if (std::string(loc) == "RU")
|
|
|
|
- current_locale_ = PATCHED;
|
|
|
|
- if (std::string(loc) == "EN")
|
|
|
|
- current_locale_ = ORIGINAL;
|
|
|
|
- }
|
|
|
|
|
|
+ InitLocales();
|
|
|
|
|
|
if (dat_state_ == SUCCESS_DICTIONARY)
|
|
if (dat_state_ == SUCCESS_DICTIONARY)
|
|
dat_state_ = READY;
|
|
dat_state_ = READY;
|
|
@@ -243,7 +230,7 @@ namespace LOTRO_DAT {
|
|
}
|
|
}
|
|
|
|
|
|
// TODO: Write description and make asserts
|
|
// TODO: Write description and make asserts
|
|
- bool DatFile::PatchFile(const SubfileData &data) {
|
|
|
|
|
|
+ bool DatFile::PatchFile(const SubfileData &data, bool rewrite_original) {
|
|
if (dat_state_ < READY) {
|
|
if (dat_state_ < READY) {
|
|
throw DatException("Bad DatFile::PatchFile() - invalid DatFile state!", EXPORT_EXCEPTION);
|
|
throw DatException("Bad DatFile::PatchFile() - invalid DatFile state!", EXPORT_EXCEPTION);
|
|
}
|
|
}
|
|
@@ -258,7 +245,7 @@ namespace LOTRO_DAT {
|
|
|
|
|
|
BinaryData old_data = GetFileData(file);
|
|
BinaryData old_data = GetFileData(file);
|
|
BinaryData patch_data = file->MakeForImport(old_data, data);
|
|
BinaryData patch_data = file->MakeForImport(old_data, data);
|
|
- ApplyFilePatch(dictionary_[file_id], patch_data);
|
|
|
|
|
|
+ ApplyFilePatch(dictionary_[file_id], patch_data, rewrite_original);
|
|
return true;
|
|
return true;
|
|
}
|
|
}
|
|
|
|
|
|
@@ -472,18 +459,18 @@ namespace LOTRO_DAT {
|
|
/// Special functions used by patch process.
|
|
/// Special functions used by patch process.
|
|
/// Shouldn't be used by any external class.
|
|
/// Shouldn't be used by any external class.
|
|
|
|
|
|
- void DatFile::ApplyFilePatch(Subfile *file, const BinaryData &data) {
|
|
|
|
|
|
+ void DatFile::ApplyFilePatch(Subfile *file, const BinaryData &data, bool rewrite_original) {
|
|
if (patched_list.count(file->file_id()) != 0) {
|
|
if (patched_list.count(file->file_id()) != 0) {
|
|
- fprintf(stderr, "Warning: DatFile::ApplyFilePatch - found 2 files in patch with the same file_id. Passing last...\n");
|
|
|
|
|
|
+ fprintf(stderr, "Warning: DatFile::ApplyFilePatch - found 2 files in patch with the same file_id = %lld. Passing last...\n", file->file_id());
|
|
return;
|
|
return;
|
|
}
|
|
}
|
|
- if (current_locale() != PATCHED) {
|
|
|
|
|
|
+ if (current_locale() != PATCHED && !rewrite_original) {
|
|
std::cout << "Changing locale to RU in order to patch file" << std::endl;
|
|
std::cout << "Changing locale to RU in order to patch file" << std::endl;
|
|
SetLocale(PATCHED);
|
|
SetLocale(PATCHED);
|
|
}
|
|
}
|
|
dat_state_ = UPDATED;
|
|
dat_state_ = UPDATED;
|
|
|
|
|
|
- if (orig_dict_.count(file->file_id()) == 0) {
|
|
|
|
|
|
+ if (orig_dict_.count(file->file_id()) == 0 && !rewrite_original) {
|
|
orig_dict_[file->file_id()] = new Subfile(this, dictionary_[file->file_id()]->MakeHeaderData());
|
|
orig_dict_[file->file_id()] = new Subfile(this, dictionary_[file->file_id()]->MakeHeaderData());
|
|
}
|
|
}
|
|
|
|
|
|
@@ -494,7 +481,7 @@ namespace LOTRO_DAT {
|
|
|
|
|
|
file->file_size_ = data.size() - 8;
|
|
file->file_size_ = data.size() - 8;
|
|
|
|
|
|
- if (patch_dict_.count(file->file_id()) == 0 || data.size() > file->block_size()) {
|
|
|
|
|
|
+ if ((patch_dict_.count(file->file_id()) == 0 && !rewrite_original) || data.size() > file->block_size()) {
|
|
file->file_offset_ = journal[0].second;
|
|
file->file_offset_ = journal[0].second;
|
|
file->block_size_ = std::max(data.size(), 256u);
|
|
file->block_size_ = std::max(data.size(), 256u);
|
|
|
|
|
|
@@ -520,8 +507,10 @@ namespace LOTRO_DAT {
|
|
auto file_id = file->file_id();
|
|
auto file_id = file->file_id();
|
|
patched_list.insert(file_id);
|
|
patched_list.insert(file_id);
|
|
|
|
|
|
- patch_dict_.erase(file_id); // Удалили старое значение в русском словаре
|
|
|
|
- patch_dict_[file_id] = new Subfile(this, file->MakeHeaderData()); // Создали новое значение
|
|
|
|
|
|
+ if (!rewrite_original) {
|
|
|
|
+ patch_dict_.erase(file_id); // Удалили старое значение в русском словаре
|
|
|
|
+ patch_dict_[file_id] = new Subfile(this, file->MakeHeaderData()); // Создали новое значение
|
|
|
|
+ }
|
|
|
|
|
|
UpdateFragmentationJournal(journal);
|
|
UpdateFragmentationJournal(journal);
|
|
}
|
|
}
|
|
@@ -564,6 +553,9 @@ namespace LOTRO_DAT {
|
|
return true;
|
|
return true;
|
|
std::cout << "There are some updated files. Rewriting dictionary..." << std::endl << std::flush;
|
|
std::cout << "There are some updated files. Rewriting dictionary..." << std::endl << std::flush;
|
|
|
|
|
|
|
|
+ std::cout << "Updating locales..." << std::endl;
|
|
|
|
+ CommitLocales();
|
|
|
|
+
|
|
auto journal = GetFragmentationJournal();
|
|
auto journal = GetFragmentationJournal();
|
|
if (!patched_list.empty()) {
|
|
if (!patched_list.empty()) {
|
|
journal[0].second = file_size_;
|
|
journal[0].second = file_size_;
|
|
@@ -579,8 +571,7 @@ namespace LOTRO_DAT {
|
|
UpdateSubdirectories();
|
|
UpdateSubdirectories();
|
|
std::cout << "Updated subdirectories..." << std::endl << std::flush;
|
|
std::cout << "Updated subdirectories..." << std::endl << std::flush;
|
|
std::cout << "Changed " << patched_list.size() << " files..." << std::endl << std::flush;
|
|
std::cout << "Changed " << patched_list.size() << " files..." << std::endl << std::flush;
|
|
- std::cout << "Updating locales..." << std::endl;
|
|
|
|
- CommitLocales();
|
|
|
|
|
|
+
|
|
std::cout << "Done!" << std::endl;
|
|
std::cout << "Done!" << std::endl;
|
|
patched_list.clear();
|
|
patched_list.clear();
|
|
dat_state_ = READY;
|
|
dat_state_ = READY;
|
|
@@ -627,30 +618,66 @@ namespace LOTRO_DAT {
|
|
|
|
|
|
// LOCALE MANAGING SECTION
|
|
// LOCALE MANAGING SECTION
|
|
|
|
|
|
- void DatFile::InitLocale(LOCALE locale, const char* filename) {
|
|
|
|
- auto dict = GetLocaleDictReference(locale);
|
|
|
|
- dict->clear();
|
|
|
|
|
|
+ void DatFile::InitLocales() {
|
|
|
|
+ std::cout << "Initialising locales..." << std::endl;
|
|
|
|
+ BinaryData dicts_data = GetFileData(dictionary_[2013266257]);
|
|
|
|
+
|
|
|
|
+ if (dicts_data.size() < 29) {
|
|
|
|
+ fprintf(stderr, "WARNING: DatFile::InitLocales() - locales file is empty.. Initialising locale dicts as empty\n");
|
|
|
|
+ std::cout << "Could't find locales file... Continuing without them" << std::endl;;
|
|
|
|
+ return;
|
|
|
|
+ }
|
|
|
|
|
|
- FILE *dict_file = fopen(filename, "rb");
|
|
|
|
|
|
+ BinaryData hi_data = dicts_data.CutData(14, 29) + BinaryData("\0", 1);
|
|
|
|
+ std::string hi = std::string((char*)(hi_data.data()));
|
|
|
|
+ std::cout << hi << std::endl;
|
|
|
|
|
|
- if (dict_file == nullptr) {
|
|
|
|
- fprintf(stderr, "WARNING!!! DatFile::InitLocale() - cannot open .dat locale file %s\n", filename);
|
|
|
|
|
|
+ if (hi != "Hi from Gi1dor!") {
|
|
|
|
+ fprintf(stderr, "WARNING: DatFile::InitLocales() - Didn't receive 'hi' from Gi1dor... Initialising locale dicts as empty\n");
|
|
|
|
+ std::cout << "Could't init locales' file... Continuing without them" << std::endl;
|
|
return;
|
|
return;
|
|
}
|
|
}
|
|
|
|
|
|
- size_t size;
|
|
|
|
- fread(&size, sizeof(size_t), 1, dict_file);
|
|
|
|
|
|
+ int offset = 29;
|
|
|
|
+ BinaryData current_locale_data = dicts_data.CutData(offset, offset + 4) + BinaryData("\0", 1);
|
|
|
|
+ std::string locale((char*)(current_locale_data.data()));
|
|
|
|
+ offset += 4;
|
|
|
|
+ std::cout << locale << std::endl;
|
|
|
|
+
|
|
|
|
+ if (locale != "PATC" && locale != "ORIG") {
|
|
|
|
+ fprintf(stderr, "WARNING: DatFile::InitLocales() - Incorrect locale... Initialising locale dicts as empty\n");
|
|
|
|
+ std::cout << "Could't recognize locale... Continuing without locales" << std::endl;;
|
|
|
|
+ return;
|
|
|
|
+ }
|
|
|
|
+ current_locale_ = (locale == "PATC" ? PATCHED : ORIGINAL);
|
|
|
|
|
|
- std::cout << "There are " << size << " files in " << std::string(filename) << " dictionary...\n";
|
|
|
|
|
|
+ // 14 bytes for old data
|
|
|
|
+ // 15 bytes for "Hi from Gi1dor"
|
|
|
|
+ // 4 bytes for LOCALE
|
|
|
|
+ // 4 bytes for orig_dict.size()
|
|
|
|
+ // 32 * orig_dict.size() bytes for orig_dict data
|
|
|
|
+ // 4 bytes for patch_dict.size()
|
|
|
|
+ // 32 * patch_dict.size() bytes for patch_dict data
|
|
|
|
|
|
- for (size_t i = 0; i < size; i++) {
|
|
|
|
- BinaryData header(32);
|
|
|
|
- fread(header.data(), unsigned(header.size()), 1, dict_file);
|
|
|
|
- auto file = new Subfile(this, header);
|
|
|
|
- (*dict)[file->file_id()] = file;
|
|
|
|
|
|
+ size_t orig_dict_size = size_t(dicts_data.CutData(offset, offset + 4).ToNumber<4>(0));
|
|
|
|
+ offset += 4;
|
|
|
|
+ for (size_t i = 0; i < orig_dict_size; i++) {
|
|
|
|
+ auto file = new Subfile(this, dicts_data.CutData(offset, offset + 32));
|
|
|
|
+ orig_dict_[file->file_id()] = file;
|
|
|
|
+ offset += 32;
|
|
}
|
|
}
|
|
|
|
|
|
- fclose(dict_file);
|
|
|
|
|
|
+
|
|
|
|
+ size_t patch_dict_size = size_t(dicts_data.CutData(offset, offset + 4).ToNumber<4>(0));
|
|
|
|
+ offset += 4;
|
|
|
|
+ for (size_t i = 0; i < patch_dict_size; i++) {
|
|
|
|
+ auto file = new Subfile(this, dicts_data.CutData(offset, offset + 32));
|
|
|
|
+ patch_dict_[file->file_id()] = file;
|
|
|
|
+ offset += 32;
|
|
|
|
+ }
|
|
|
|
+
|
|
|
|
+ std::cout << "There are " << patch_dict_.size() << " files in patch locale dictionary" << std::endl;
|
|
|
|
+ std::cout << "There are " << orig_dict_.size() << " files in original locale dictionary" << std::endl;
|
|
}
|
|
}
|
|
|
|
|
|
std::unordered_map<long long, Subfile *> *DatFile::GetLocaleDictReference(LOCALE locale) {
|
|
std::unordered_map<long long, Subfile *> *DatFile::GetLocaleDictReference(LOCALE locale) {
|
|
@@ -708,20 +735,6 @@ namespace LOTRO_DAT {
|
|
}
|
|
}
|
|
}
|
|
}
|
|
|
|
|
|
- void DatFile::SaveLocale(LOCALE locale, const char *filename) {
|
|
|
|
- auto dict = GetLocaleDictReference(locale);
|
|
|
|
- FILE *dict_file = fopen(filename, "wb");
|
|
|
|
-
|
|
|
|
- size_t count = size_t(dict->size());
|
|
|
|
- fwrite(&count, sizeof(size_t), 1, dict_file);
|
|
|
|
-
|
|
|
|
- for (auto file : *dict) {
|
|
|
|
- BinaryData header = file.second->MakeHeaderData();
|
|
|
|
- fwrite(header.data(), unsigned(header.size()), 1, dict_file);
|
|
|
|
- }
|
|
|
|
- fclose(dict_file);
|
|
|
|
- }
|
|
|
|
-
|
|
|
|
bool DatFile::CheckIfUpdatedByGame() {
|
|
bool DatFile::CheckIfUpdatedByGame() {
|
|
return false;
|
|
return false;
|
|
}
|
|
}
|
|
@@ -744,20 +757,50 @@ namespace LOTRO_DAT {
|
|
|
|
|
|
void DatFile::CommitLocales() {
|
|
void DatFile::CommitLocales() {
|
|
std::cout << "Commiting locales..." << std::endl;
|
|
std::cout << "Commiting locales..." << std::endl;
|
|
- std::cout << "Saving patched locale..." << std::endl;
|
|
|
|
- SaveLocale(PATCHED, (std::string(filename_) + std::string("patched.dbgm")).c_str());
|
|
|
|
- std::cout << "Saving original locale..." << std::endl;
|
|
|
|
- SaveLocale(ORIGINAL,(std::string(filename_) + std::string("original.dbgm")).c_str());
|
|
|
|
-
|
|
|
|
- std::cout << "Writing current locale" << std::endl;
|
|
|
|
- FILE *locale = fopen((std::string(filename_) + ".dbgm").c_str(), "w");
|
|
|
|
- if (current_locale_ == ORIGINAL)
|
|
|
|
- fprintf(locale, "EN");
|
|
|
|
- else
|
|
|
|
- fprintf(locale, "RU");
|
|
|
|
- fclose(locale);
|
|
|
|
|
|
+ SubfileData data = dictionary_[2013266257]->PrepareForExport(GetFileData(dictionary_[2013266257]));
|
|
|
|
+ data.options["fid"] = "2013266257";
|
|
|
|
+ data.options["ext"] = ".unknown";
|
|
|
|
+
|
|
|
|
+ BinaryData old_data = BinaryData(GetFileData(dictionary_[2013266257u]));
|
|
|
|
+
|
|
|
|
+ // 14 bytes for old data
|
|
|
|
+ // 15 bytes for "Hi from Gi1dor"
|
|
|
|
+ // 4 bytes for LOCALE
|
|
|
|
+ // 4 bytes for orig_dict.size()
|
|
|
|
+ // 32 * orig_dict.size() bytes for orig_dict data
|
|
|
|
+ // 4 bytes for patch_dict.size()
|
|
|
|
+ // 32 * patch_dict.size() bytes for patch_dict data
|
|
|
|
+
|
|
|
|
+ data.binary_data = BinaryData(14 + 15 + 4 + 4 + orig_dict_.size() * 32 + 4 + patch_dict_.size() * 32);
|
|
|
|
+
|
|
|
|
+ size_t current_size = 0;
|
|
|
|
+ data.binary_data.Append(GetFileData(dictionary_[2013266257u]).CutData(0, 14), current_size);
|
|
|
|
+ current_size += 14;
|
|
|
|
+
|
|
|
|
+ data.binary_data.Append(BinaryData("Hi from Gi1dor!", 15), current_size);
|
|
|
|
+ current_size += 15;
|
|
|
|
+
|
|
|
|
+ data.binary_data.Append(BinaryData((current_locale_ == ORIGINAL ? "ORIG" : "PATC"), 4), current_size);
|
|
|
|
+ current_size += 4;
|
|
|
|
+
|
|
|
|
+ data.binary_data.Append(BinaryData::FromNumber<4>(orig_dict_.size()), current_size);
|
|
|
|
+ current_size += 4;
|
|
|
|
+
|
|
|
|
+ for (auto file : orig_dict_) {
|
|
|
|
+ data.binary_data.Append(file.second->MakeHeaderData(), current_size);
|
|
|
|
+ current_size += 32;
|
|
|
|
+ }
|
|
|
|
+
|
|
|
|
+ data.binary_data.Append(BinaryData::FromNumber<4>(patch_dict_.size()), current_size);
|
|
|
|
+ current_size += 4;
|
|
|
|
+
|
|
|
|
+ for (auto file : patch_dict_) {
|
|
|
|
+ data.binary_data.Append(file.second->MakeHeaderData(), current_size);
|
|
|
|
+ current_size += 32;
|
|
|
|
+ }
|
|
|
|
+
|
|
|
|
+ PatchFile(data, true);
|
|
std::cout << "Done!" << std::endl;
|
|
std::cout << "Done!" << std::endl;
|
|
}
|
|
}
|
|
-
|
|
|
|
}
|
|
}
|
|
}
|
|
}
|