|
@@ -30,51 +30,42 @@ namespace LOTRO_DAT {
|
|
* \return Возвращает result = ERROR и value = BinaryData, если файл не найден или информаци в словаре некорректная
|
|
* \return Возвращает result = ERROR и value = BinaryData, если файл не найден или информаци в словаре некорректная
|
|
*/
|
|
*/
|
|
|
|
|
|
- DatOperationResult<BinaryData> DatFileSystem::GetFileData(long long file_id, long long int offset) {
|
|
|
|
|
|
+ DatOperationResult<BinaryData> DatFileSystem::GetFileData(const SubFile& file, long long int offset) {
|
|
if (!dat)
|
|
if (!dat)
|
|
return DatOperationResult<BinaryData>(BinaryData(), ERROR,
|
|
return DatOperationResult<BinaryData>(BinaryData(), ERROR,
|
|
"DatFileSystem error: no connection with Dat (dat is nullptr)");
|
|
"DatFileSystem error: no connection with Dat (dat is nullptr)");
|
|
|
|
|
|
-
|
|
|
|
- auto getfile_operation = GetFile(file_id);
|
|
|
|
- if (getfile_operation.result == ERROR)
|
|
|
|
- return DatOperationResult<BinaryData>(BinaryData(), ERROR,
|
|
|
|
- "DATFSGETFILEDATA: no file with id = " + std::to_string(file_id));
|
|
|
|
-
|
|
|
|
- auto& file = getfile_operation.value;
|
|
|
|
-
|
|
|
|
-
|
|
|
|
BinaryData mfile_id(20);
|
|
BinaryData mfile_id(20);
|
|
auto operation = CheckCorrectSubfile(file);
|
|
auto operation = CheckCorrectSubfile(file);
|
|
if (operation.result == ERROR || !operation.value)
|
|
if (operation.result == ERROR || !operation.value)
|
|
return DatOperationResult<BinaryData>(BinaryData(), ERROR,
|
|
return DatOperationResult<BinaryData>(BinaryData(), ERROR,
|
|
- "DATFSGETFILEDATA: Incorrect file" + std::to_string(file_id));
|
|
|
|
|
|
+ "DATFSGETFILEDATA: Incorrect file" + std::to_string(file.file_id()));
|
|
|
|
|
|
|
|
|
|
- BinaryData data((unsigned) (file->file_size() + (8 - offset)));
|
|
|
|
- if (file->block_size() >= file->file_size() + 8) {
|
|
|
|
- dat->GetIO().ReadData(data, file->file_size() + (8 - offset), file->file_offset() + offset);
|
|
|
|
|
|
+ BinaryData data((unsigned) (file.file_size() + (8 - offset)));
|
|
|
|
+ if (file.block_size() >= file.file_size() + 8) {
|
|
|
|
+ dat->GetIO().ReadData(data, file.file_size() + (8 - offset), file.file_offset() + offset);
|
|
return DatOperationResult<BinaryData>(data, SUCCESS);
|
|
return DatOperationResult<BinaryData>(data, SUCCESS);
|
|
}
|
|
}
|
|
|
|
|
|
BinaryData fragments_count(4);
|
|
BinaryData fragments_count(4);
|
|
- dat->GetIO().ReadData(fragments_count, 4, file->file_offset());
|
|
|
|
|
|
+ dat->GetIO().ReadData(fragments_count, 4, file.file_offset());
|
|
|
|
|
|
long long fragments_number = fragments_count.ToNumber<4>(0);
|
|
long long fragments_number = fragments_count.ToNumber<4>(0);
|
|
|
|
|
|
- long long current_block_size = file->block_size() - offset - 8 * fragments_number;
|
|
|
|
|
|
+ long long current_block_size = file.block_size() - offset - 8 * fragments_number;
|
|
|
|
|
|
- dat->GetIO().ReadData(data, current_block_size, file->file_offset() + offset);
|
|
|
|
|
|
+ dat->GetIO().ReadData(data, current_block_size, file.file_offset() + offset);
|
|
|
|
|
|
BinaryData FragmentsDictionary(8 * unsigned(fragments_number));
|
|
BinaryData FragmentsDictionary(8 * unsigned(fragments_number));
|
|
dat->GetIO().ReadData(FragmentsDictionary, 8 * unsigned(fragments_number),
|
|
dat->GetIO().ReadData(FragmentsDictionary, 8 * unsigned(fragments_number),
|
|
- file->file_offset() + file->block_size() - 8 * fragments_number);
|
|
|
|
|
|
+ file.file_offset() + file.block_size() - 8 * fragments_number);
|
|
|
|
|
|
|
|
|
|
for (long long i = 0; i < fragments_number; i++) {
|
|
for (long long i = 0; i < fragments_number; i++) {
|
|
long long fragment_size = FragmentsDictionary.ToNumber<4>(8 * i);
|
|
long long fragment_size = FragmentsDictionary.ToNumber<4>(8 * i);
|
|
long long fragment_offset = FragmentsDictionary.ToNumber<4>(8 * i + 4);
|
|
long long fragment_offset = FragmentsDictionary.ToNumber<4>(8 * i + 4);
|
|
- dat->GetIO().ReadData(data, std::min(fragment_size, file->file_size() - current_block_size),
|
|
|
|
|
|
+ dat->GetIO().ReadData(data, std::min(fragment_size, file.file_size() - current_block_size),
|
|
fragment_offset,
|
|
fragment_offset,
|
|
current_block_size);
|
|
current_block_size);
|
|
current_block_size += fragment_size;
|
|
current_block_size += fragment_size;
|
|
@@ -144,20 +135,20 @@ namespace LOTRO_DAT {
|
|
* \return Объект DatOperationResult<bool> для которого value = true, если файл корректный, и false - иначе.
|
|
* \return Объект DatOperationResult<bool> для которого value = true, если файл корректный, и false - иначе.
|
|
*/
|
|
*/
|
|
|
|
|
|
- DatOperationResult<bool> DatFileSystem::CheckCorrectSubfile(const std::shared_ptr<SubFile> &file) {
|
|
|
|
|
|
+ DatOperationResult<bool> DatFileSystem::CheckCorrectSubfile(const SubFile& file) {
|
|
if (!dat)
|
|
if (!dat)
|
|
return DatOperationResult<bool>(false, ERROR,
|
|
return DatOperationResult<bool>(false, ERROR,
|
|
"DATFSCORRECTSUBFILE: no connection with Dat (dat is nullptr)");
|
|
"DATFSCORRECTSUBFILE: no connection with Dat (dat is nullptr)");
|
|
- if (file->file_size() < 16)
|
|
|
|
|
|
+ if (file.file_size() < 16)
|
|
return DatOperationResult<bool>(false, SUCCESS);
|
|
return DatOperationResult<bool>(false, SUCCESS);
|
|
|
|
|
|
BinaryData mfile_id(20);
|
|
BinaryData mfile_id(20);
|
|
- auto operation = dat->GetIO().ReadData(mfile_id, 20, file->file_offset() + 8);
|
|
|
|
|
|
+ auto operation = dat->GetIO().ReadData(mfile_id, 20, file.file_offset() + 8);
|
|
if (operation.result == ERROR)
|
|
if (operation.result == ERROR)
|
|
return DatOperationResult<bool>(false, ERROR, "DATFSCORRECTSUBFILE: cannot read file header data");
|
|
return DatOperationResult<bool>(false, ERROR, "DATFSCORRECTSUBFILE: cannot read file header data");
|
|
|
|
|
|
- return DatOperationResult<bool>((mfile_id.CheckCompression() || file->file_id() == mfile_id.ToNumber<4>(0)) &&
|
|
|
|
- file->file_size() < 50ll * 1024ll * 1024ll, SUCCESS);
|
|
|
|
|
|
+ return DatOperationResult<bool>((mfile_id.CheckCompression() || file.file_id() == mfile_id.ToNumber<4>(0)) &&
|
|
|
|
+ file.file_size() < 50ll * 1024ll * 1024ll, SUCCESS);
|
|
}
|
|
}
|
|
|
|
|
|
|
|
|
|
@@ -317,14 +308,14 @@ namespace LOTRO_DAT {
|
|
if (!initialised_file)
|
|
if (!initialised_file)
|
|
return DatOperationResult<>(ERROR, "DATFSINITSUBFILE: initialised subfile pointer is empty");
|
|
return DatOperationResult<>(ERROR, "DATFSINITSUBFILE: initialised subfile pointer is empty");
|
|
|
|
|
|
- if (!CheckCorrectSubfile(initialised_file).value)
|
|
|
|
|
|
+ if (!CheckCorrectSubfile(*initialised_file).value)
|
|
return DatOperationResult<>(ERROR, "DATFSINITSUBFILE: initialised file " + std::to_string(file.file_id()) + " is incorrect");
|
|
return DatOperationResult<>(ERROR, "DATFSINITSUBFILE: initialised file " + std::to_string(file.file_id()) + " is incorrect");
|
|
|
|
|
|
if (dictionary_.count(file_id) > 0) {
|
|
if (dictionary_.count(file_id) > 0) {
|
|
LOG(WARNING) << "Dublicate files id = " << file_id << "dictionary offsets = "
|
|
LOG(WARNING) << "Dublicate files id = " << file_id << "dictionary offsets = "
|
|
<< dictionary_[file_id]->dictionary_offset() << " and "
|
|
<< dictionary_[file_id]->dictionary_offset() << " and "
|
|
<< initialised_file->dictionary_offset();
|
|
<< initialised_file->dictionary_offset();
|
|
- if (CheckCorrectSubfile(dictionary_[file_id]).value)
|
|
|
|
|
|
+ if (CheckCorrectSubfile(*dictionary_[file_id]).value)
|
|
LOG(ERROR) << " FILE IN DICTIONARY IS ALREADY CORRECT!";
|
|
LOG(ERROR) << " FILE IN DICTIONARY IS ALREADY CORRECT!";
|
|
|
|
|
|
dictionary_[file_id] = initialised_file;
|
|
dictionary_[file_id] = initialised_file;
|