2022-09-05 20:59:46 -04:00
|
|
|
//
|
|
|
|
// EuropaTools
|
|
|
|
//
|
2025-01-07 14:17:50 -05:00
|
|
|
// (C) 2021-2025 modeco80 <lily.modeco80@protonmail.ch>
|
2022-09-05 20:59:46 -04:00
|
|
|
//
|
2025-01-07 14:17:50 -05:00
|
|
|
// SPDX-License-Identifier: LGPL-3.0-or-later
|
2022-09-05 20:59:46 -04:00
|
|
|
//
|
|
|
|
|
|
|
|
#ifndef EUROPA_IO_PAKFILE_H
|
|
|
|
#define EUROPA_IO_PAKFILE_H
|
|
|
|
|
2022-09-07 05:07:40 -04:00
|
|
|
#include <cstdint>
|
2022-09-21 03:59:16 -04:00
|
|
|
#include <europa/structs/Pak.hpp>
|
2025-01-06 17:12:58 -05:00
|
|
|
#include <europa/util/Overloaded.hpp>
|
|
|
|
#include <filesystem>
|
|
|
|
#include <stdexcept>
|
|
|
|
#include <variant>
|
2022-09-07 05:07:40 -04:00
|
|
|
#include <vector>
|
2022-09-05 20:59:46 -04:00
|
|
|
|
2022-09-07 05:07:40 -04:00
|
|
|
namespace europa::io {
|
2022-09-05 20:59:46 -04:00
|
|
|
|
2022-09-07 05:07:40 -04:00
|
|
|
struct PakReader;
|
|
|
|
struct PakWriter;
|
2022-09-05 20:59:46 -04:00
|
|
|
|
2025-01-06 17:12:58 -05:00
|
|
|
/// sumtype
|
|
|
|
struct PakFileData {
|
|
|
|
// clang-format off
|
|
|
|
using Variant = std::variant<
|
|
|
|
// File data
|
|
|
|
std::vector<std::uint8_t>,
|
|
|
|
|
|
|
|
// Path
|
|
|
|
std::filesystem::path
|
|
|
|
>;
|
|
|
|
// clang-format on
|
|
|
|
|
|
|
|
static PakFileData InitAsBuffer(std::vector<std::uint8_t>&& buffer) {
|
|
|
|
return PakFileData {
|
|
|
|
.variant_ = Variant(std::move(buffer))
|
|
|
|
};
|
|
|
|
}
|
|
|
|
|
|
|
|
static PakFileData InitAsPath(const std::filesystem::path& path) {
|
|
|
|
return PakFileData {
|
|
|
|
.variant_ = Variant(path)
|
|
|
|
};
|
|
|
|
}
|
|
|
|
|
|
|
|
std::uint32_t GetSize() const {
|
|
|
|
// FIXME: make this just a overloaded lambda
|
|
|
|
struct SizeVisitor {
|
|
|
|
std::uint32_t& size;
|
|
|
|
|
|
|
|
// bleh
|
2025-01-06 19:18:44 -05:00
|
|
|
void operator()(const std::vector<uint8_t>& buffer) {
|
2025-01-06 17:12:58 -05:00
|
|
|
size = static_cast<std::uint32_t>(buffer.size());
|
|
|
|
}
|
|
|
|
|
2025-01-06 19:18:44 -05:00
|
|
|
void operator()(const std::filesystem::path& fsPath) {
|
2025-01-06 17:12:58 -05:00
|
|
|
if(!std::filesystem::exists(fsPath) && !std::filesystem::is_regular_file(fsPath))
|
|
|
|
throw std::runtime_error("invalid path in path file");
|
|
|
|
size = static_cast<std::uint32_t>(std::filesystem::file_size(fsPath));
|
|
|
|
}
|
|
|
|
};
|
|
|
|
|
|
|
|
std::uint32_t size {};
|
2025-01-06 19:18:44 -05:00
|
|
|
std::visit(SizeVisitor { size }, variant_);
|
2025-01-06 17:12:58 -05:00
|
|
|
return size;
|
|
|
|
}
|
|
|
|
|
|
|
|
template <class T>
|
|
|
|
const T* GetIf() const {
|
|
|
|
return std::get_if<T>(&variant_);
|
|
|
|
}
|
|
|
|
|
2025-01-07 14:10:03 -05:00
|
|
|
template <class Visitor>
|
|
|
|
auto Visit(Visitor&& v) const {
|
|
|
|
return std::visit(v, variant_);
|
|
|
|
}
|
|
|
|
|
2025-01-06 17:12:58 -05:00
|
|
|
// private:
|
|
|
|
PakFileData::Variant variant_;
|
|
|
|
};
|
|
|
|
|
2025-01-07 15:13:10 -05:00
|
|
|
/// Repressents a package file. Can either hold a memory buffer of contents
|
|
|
|
/// or a filesystem path (for creating packages).
|
2022-09-07 05:07:40 -04:00
|
|
|
struct PakFile {
|
2025-01-06 17:12:58 -05:00
|
|
|
using DataType = PakFileData;
|
2022-09-05 20:59:46 -04:00
|
|
|
|
2025-01-06 17:12:58 -05:00
|
|
|
template <class T>
|
2025-01-07 13:47:20 -05:00
|
|
|
void InitWithExistingTocEntry(const T& value) {
|
2023-08-01 18:18:40 -04:00
|
|
|
toc = value;
|
|
|
|
}
|
|
|
|
|
|
|
|
void InitAs(structs::PakVersion version) {
|
|
|
|
switch(version) {
|
|
|
|
case structs::PakVersion::Ver3:
|
2025-01-06 17:12:58 -05:00
|
|
|
toc = structs::PakHeader_V3::TocEntry {};
|
2023-08-01 18:18:40 -04:00
|
|
|
break;
|
|
|
|
case structs::PakVersion::Ver4:
|
2025-01-06 17:12:58 -05:00
|
|
|
toc = structs::PakHeader_V4::TocEntry {};
|
2023-08-01 18:18:40 -04:00
|
|
|
break;
|
|
|
|
case structs::PakVersion::Ver5:
|
2025-01-06 17:12:58 -05:00
|
|
|
toc = structs::PakHeader_V5::TocEntry {};
|
2023-08-01 18:18:40 -04:00
|
|
|
break;
|
2025-01-06 19:18:44 -05:00
|
|
|
default:
|
|
|
|
throw std::invalid_argument("Invalid PAK version to initalize TOC entry");
|
|
|
|
break;
|
2023-08-01 18:18:40 -04:00
|
|
|
}
|
|
|
|
}
|
|
|
|
|
2025-01-06 17:12:58 -05:00
|
|
|
bool HasData() const {
|
|
|
|
return fileData.has_value();
|
|
|
|
}
|
|
|
|
|
2022-09-07 05:07:40 -04:00
|
|
|
/**
|
|
|
|
* Get the file data.
|
|
|
|
*/
|
2023-08-01 18:18:40 -04:00
|
|
|
[[nodiscard]] const DataType& GetData() const {
|
2025-01-06 17:12:58 -05:00
|
|
|
if(!fileData.has_value())
|
|
|
|
throw std::runtime_error("no file data to get!");
|
|
|
|
return fileData.value();
|
|
|
|
}
|
|
|
|
|
|
|
|
/// Sets data.
|
|
|
|
void SetData(DataType&& data) {
|
|
|
|
this->fileData = std::move(data);
|
|
|
|
|
|
|
|
// Update the TOC size.
|
|
|
|
std::visit([&](auto& entry) {
|
|
|
|
entry.size = this->fileData.value().GetSize();
|
|
|
|
},
|
|
|
|
toc);
|
|
|
|
}
|
|
|
|
|
|
|
|
/// Purge read file data.
|
|
|
|
void PurgeData() {
|
|
|
|
this->fileData = std::nullopt;
|
2023-08-01 18:18:40 -04:00
|
|
|
}
|
2022-09-05 20:59:46 -04:00
|
|
|
|
2022-09-07 05:07:40 -04:00
|
|
|
/**
|
|
|
|
* Get the TOC entry responsible.
|
|
|
|
*/
|
2025-01-06 17:12:58 -05:00
|
|
|
template <class T>
|
2023-08-01 18:18:40 -04:00
|
|
|
[[nodiscard]] const T& GetTOCEntry() const {
|
|
|
|
return std::get<T>(toc);
|
|
|
|
}
|
|
|
|
|
2025-01-05 20:15:54 -05:00
|
|
|
std::uint32_t GetCreationUnixTime() const {
|
2025-01-06 17:12:58 -05:00
|
|
|
std::uint32_t time {};
|
2025-01-05 20:15:54 -05:00
|
|
|
|
|
|
|
std::visit([&](auto& entry) {
|
|
|
|
time = entry.creationUnixTime;
|
2025-01-06 17:12:58 -05:00
|
|
|
},
|
|
|
|
toc);
|
2025-01-05 20:15:54 -05:00
|
|
|
|
|
|
|
return time;
|
2023-08-01 18:18:40 -04:00
|
|
|
}
|
|
|
|
|
|
|
|
std::uint32_t GetOffset() const {
|
2025-01-06 17:12:58 -05:00
|
|
|
std::uint32_t size {};
|
2023-08-01 18:18:40 -04:00
|
|
|
|
|
|
|
std::visit([&](auto& entry) {
|
|
|
|
size = entry.offset;
|
2025-01-06 17:12:58 -05:00
|
|
|
},
|
|
|
|
toc);
|
2023-08-01 18:18:40 -04:00
|
|
|
|
|
|
|
return size;
|
|
|
|
}
|
|
|
|
|
|
|
|
std::uint32_t GetSize() const {
|
2025-01-06 17:12:58 -05:00
|
|
|
std::uint32_t size {};
|
2023-08-01 18:18:40 -04:00
|
|
|
|
|
|
|
std::visit([&](auto& entry) {
|
|
|
|
size = entry.size;
|
2025-01-06 17:12:58 -05:00
|
|
|
},
|
|
|
|
toc);
|
2022-09-05 20:59:46 -04:00
|
|
|
|
2023-08-01 18:18:40 -04:00
|
|
|
return size;
|
|
|
|
}
|
2022-09-05 20:59:46 -04:00
|
|
|
|
2025-01-07 15:13:10 -05:00
|
|
|
template <class Visitor>
|
|
|
|
auto Visit(Visitor&& cb) {
|
|
|
|
return std::visit(cb, toc);
|
2023-08-01 18:18:40 -04:00
|
|
|
}
|
2022-09-05 20:59:46 -04:00
|
|
|
|
2022-09-07 05:07:40 -04:00
|
|
|
private:
|
|
|
|
friend PakReader;
|
|
|
|
friend PakWriter;
|
2022-09-05 20:59:46 -04:00
|
|
|
|
2025-01-06 17:12:58 -05:00
|
|
|
std::optional<PakFileData> fileData;
|
2023-08-01 18:18:40 -04:00
|
|
|
structs::PakTocEntryVariant toc;
|
2022-09-07 05:07:40 -04:00
|
|
|
};
|
2022-09-05 20:59:46 -04:00
|
|
|
|
2022-09-07 05:07:40 -04:00
|
|
|
} // namespace europa::io
|
2022-09-05 20:59:46 -04:00
|
|
|
|
2022-09-07 05:07:40 -04:00
|
|
|
#endif // EUROPA_IO_PAKFILE_H
|