2021-12-07 22:47:41 +01:00
|
|
|
#include "content/providers/file_provider.hpp"
|
2020-11-11 09:18:35 +01:00
|
|
|
|
2021-01-03 15:09:12 +01:00
|
|
|
#include <cstring>
|
2020-11-15 16:06:10 +01:00
|
|
|
|
2022-08-08 21:23:52 +02:00
|
|
|
#include <hex/api/imhex_api.hpp>
|
2022-02-01 18:09:40 +01:00
|
|
|
#include <hex/api/localization.hpp>
|
2023-01-11 23:31:25 +01:00
|
|
|
#include <hex/api/project_file_manager.hpp>
|
|
|
|
|
2021-08-29 22:15:18 +02:00
|
|
|
#include <hex/helpers/utils.hpp>
|
2022-02-01 18:09:40 +01:00
|
|
|
#include <hex/helpers/fmt.hpp>
|
2022-08-08 21:23:52 +02:00
|
|
|
|
2023-03-12 18:43:05 +01:00
|
|
|
#include <wolv/utils/string.hpp>
|
|
|
|
|
2022-08-08 21:23:52 +02:00
|
|
|
#include <nlohmann/json.hpp>
|
2020-11-15 16:06:10 +01:00
|
|
|
|
2022-11-07 00:04:47 +01:00
|
|
|
namespace hex::plugin::builtin {
|
2021-12-07 22:47:41 +01:00
|
|
|
|
2021-09-21 02:29:54 +02:00
|
|
|
bool FileProvider::isAvailable() const {
|
2023-03-17 11:31:50 +01:00
|
|
|
return true;
|
2020-11-11 09:18:35 +01:00
|
|
|
}
|
|
|
|
|
2021-09-21 02:29:54 +02:00
|
|
|
bool FileProvider::isReadable() const {
|
2020-11-11 10:47:02 +01:00
|
|
|
return isAvailable() && this->m_readable;
|
2020-11-11 09:18:35 +01:00
|
|
|
}
|
|
|
|
|
2021-09-21 02:29:54 +02:00
|
|
|
bool FileProvider::isWritable() const {
|
2020-11-11 10:47:02 +01:00
|
|
|
return isAvailable() && this->m_writable;
|
2020-11-11 09:18:35 +01:00
|
|
|
}
|
|
|
|
|
2021-09-21 02:29:54 +02:00
|
|
|
bool FileProvider::isResizable() const {
|
2022-09-03 23:08:40 +02:00
|
|
|
return isAvailable() && isWritable();
|
2021-07-27 21:07:36 +02:00
|
|
|
}
|
|
|
|
|
2021-09-21 02:29:54 +02:00
|
|
|
bool FileProvider::isSavable() const {
|
2021-08-21 13:53:50 +02:00
|
|
|
return !this->getPatches().empty();
|
|
|
|
}
|
|
|
|
|
2020-11-11 09:18:35 +01:00
|
|
|
|
2021-03-21 14:50:47 +01:00
|
|
|
void FileProvider::read(u64 offset, void *buffer, size_t size, bool overlays) {
|
2021-12-09 21:10:24 +01:00
|
|
|
this->readRaw(offset - this->getBaseAddress(), buffer, size);
|
2020-11-27 09:09:48 +01:00
|
|
|
|
2023-04-12 19:50:03 +02:00
|
|
|
if (overlays) [[likely]] {
|
|
|
|
for (const auto&[patchOffset, patchData] : getPatches()) {
|
|
|
|
if (patchOffset >= offset && patchOffset <= (offset + size))
|
|
|
|
reinterpret_cast<u8 *>(buffer)[patchOffset - offset] = patchData;
|
2023-03-26 12:48:18 +02:00
|
|
|
}
|
2021-03-21 14:50:47 +01:00
|
|
|
|
|
|
|
this->applyOverlays(offset, buffer, size);
|
2023-03-07 16:03:34 +01:00
|
|
|
}
|
2020-11-11 09:18:35 +01:00
|
|
|
}
|
|
|
|
|
2020-11-27 13:44:52 +01:00
|
|
|
void FileProvider::write(u64 offset, const void *buffer, size_t size) {
|
2021-12-09 21:10:24 +01:00
|
|
|
if ((offset - this->getBaseAddress()) > (this->getActualSize() - size) || buffer == nullptr || size == 0)
|
2020-11-11 09:18:35 +01:00
|
|
|
return;
|
|
|
|
|
2022-02-15 21:50:02 +01:00
|
|
|
addPatch(offset, buffer, size, true);
|
2020-11-27 09:09:48 +01:00
|
|
|
}
|
|
|
|
|
|
|
|
void FileProvider::readRaw(u64 offset, void *buffer, size_t size) {
|
2023-01-30 10:26:37 +01:00
|
|
|
if (offset > (this->getActualSize() - size) || buffer == nullptr || size == 0)
|
2020-11-27 09:09:48 +01:00
|
|
|
return;
|
|
|
|
|
2023-03-17 11:31:50 +01:00
|
|
|
auto &file = this->getFile();
|
|
|
|
file.seek(offset);
|
|
|
|
file.readBuffer(reinterpret_cast<u8*>(buffer), size);
|
2020-11-11 09:18:35 +01:00
|
|
|
}
|
|
|
|
|
2020-11-27 13:44:52 +01:00
|
|
|
void FileProvider::writeRaw(u64 offset, const void *buffer, size_t size) {
|
2021-12-09 21:10:24 +01:00
|
|
|
if ((offset + size) > this->getActualSize() || buffer == nullptr || size == 0)
|
2020-11-27 09:09:48 +01:00
|
|
|
return;
|
|
|
|
|
2023-03-17 11:43:50 +01:00
|
|
|
std::scoped_lock lock(this->m_writeMutex);
|
2023-03-12 18:27:29 +01:00
|
|
|
wolv::io::File writeFile(this->m_path, wolv::io::File::Mode::Write);
|
2023-02-17 10:26:09 +01:00
|
|
|
if (!writeFile.isValid())
|
|
|
|
return;
|
|
|
|
|
|
|
|
writeFile.seek(offset);
|
2023-03-23 11:23:07 +01:00
|
|
|
writeFile.writeBuffer(reinterpret_cast<const u8*>(buffer), size);
|
2023-04-01 11:18:03 +02:00
|
|
|
|
|
|
|
this->invalidateFiles();
|
2020-11-27 09:09:48 +01:00
|
|
|
}
|
2021-07-27 21:07:36 +02:00
|
|
|
|
2021-08-21 13:53:50 +02:00
|
|
|
void FileProvider::save() {
|
|
|
|
this->applyPatches();
|
2023-04-06 17:36:28 +02:00
|
|
|
Provider::save();
|
2021-08-21 13:53:50 +02:00
|
|
|
}
|
|
|
|
|
2023-03-21 15:42:10 +01:00
|
|
|
void FileProvider::saveAs(const std::fs::path &path) {
|
|
|
|
if (path == this->m_path)
|
|
|
|
this->save();
|
|
|
|
else
|
|
|
|
Provider::saveAs(path);
|
|
|
|
}
|
|
|
|
|
2022-01-20 23:24:26 +01:00
|
|
|
void FileProvider::resize(size_t newSize) {
|
2021-07-27 22:46:37 +02:00
|
|
|
this->close();
|
2021-07-27 21:07:36 +02:00
|
|
|
|
|
|
|
{
|
2023-03-12 18:27:29 +01:00
|
|
|
wolv::io::File file(this->m_path, wolv::io::File::Mode::Write);
|
2022-01-20 23:24:26 +01:00
|
|
|
|
|
|
|
file.setSize(newSize);
|
2021-07-27 21:07:36 +02:00
|
|
|
}
|
|
|
|
|
2022-03-27 00:01:28 +01:00
|
|
|
(void)this->open();
|
2021-07-27 21:07:36 +02:00
|
|
|
}
|
|
|
|
|
2022-01-20 23:24:26 +01:00
|
|
|
void FileProvider::insert(u64 offset, size_t size) {
|
|
|
|
auto oldSize = this->getActualSize();
|
|
|
|
this->resize(oldSize + size);
|
|
|
|
|
|
|
|
std::vector<u8> buffer(0x1000);
|
|
|
|
const std::vector<u8> zeroBuffer(0x1000);
|
|
|
|
|
|
|
|
auto position = oldSize;
|
|
|
|
while (position > offset) {
|
2022-06-17 15:21:56 +03:00
|
|
|
const auto readSize = std::min<size_t>(position - offset, buffer.size());
|
2022-01-20 23:24:26 +01:00
|
|
|
|
|
|
|
position -= readSize;
|
|
|
|
|
|
|
|
this->readRaw(position, buffer.data(), readSize);
|
|
|
|
this->writeRaw(position, zeroBuffer.data(), readSize);
|
|
|
|
this->writeRaw(position + size, buffer.data(), readSize);
|
|
|
|
}
|
|
|
|
|
|
|
|
Provider::insert(offset, size);
|
|
|
|
}
|
|
|
|
|
2023-04-01 11:18:03 +02:00
|
|
|
void FileProvider::invalidateFiles() {
|
|
|
|
for(auto & [threadId, file] : this->m_files){
|
|
|
|
file.close();
|
|
|
|
}
|
|
|
|
this->m_files.clear();
|
|
|
|
}
|
|
|
|
|
2022-06-17 15:21:56 +03:00
|
|
|
void FileProvider::remove(u64 offset, size_t size) {
|
|
|
|
auto oldSize = this->getActualSize();
|
|
|
|
this->resize(oldSize + size);
|
|
|
|
|
|
|
|
std::vector<u8> buffer(0x1000);
|
|
|
|
|
|
|
|
const auto newSize = oldSize - size;
|
|
|
|
auto position = offset;
|
|
|
|
while (position < newSize) {
|
|
|
|
const auto readSize = std::min<size_t>(newSize - position, buffer.size());
|
|
|
|
|
|
|
|
this->readRaw(position + size, buffer.data(), readSize);
|
|
|
|
this->writeRaw(position, buffer.data(), readSize);
|
|
|
|
|
|
|
|
position += readSize;
|
|
|
|
}
|
|
|
|
|
|
|
|
this->resize(newSize);
|
|
|
|
|
|
|
|
Provider::insert(offset, size);
|
2022-10-21 12:01:28 +02:00
|
|
|
Provider::remove(offset, size);
|
2022-06-17 15:21:56 +03:00
|
|
|
}
|
|
|
|
|
2021-09-21 02:29:54 +02:00
|
|
|
size_t FileProvider::getActualSize() const {
|
2023-03-17 11:43:50 +01:00
|
|
|
return this->m_sizeFile.getSize();
|
2020-11-11 09:18:35 +01:00
|
|
|
}
|
|
|
|
|
2021-09-21 02:29:54 +02:00
|
|
|
std::string FileProvider::getName() const {
|
2023-03-12 18:43:05 +01:00
|
|
|
return wolv::util::toUTF8String(this->m_path.filename());
|
2021-09-21 02:29:54 +02:00
|
|
|
}
|
|
|
|
|
2023-01-24 09:07:11 +01:00
|
|
|
std::vector<std::pair<std::string, std::string>> FileProvider::getDataDescription() const {
|
2020-11-15 16:06:10 +01:00
|
|
|
std::vector<std::pair<std::string, std::string>> result;
|
|
|
|
|
2023-03-12 18:43:05 +01:00
|
|
|
result.emplace_back("hex.builtin.provider.file.path"_lang, wolv::util::toUTF8String(this->m_path));
|
2021-02-22 13:07:25 +01:00
|
|
|
result.emplace_back("hex.builtin.provider.file.size"_lang, hex::toByteString(this->getActualSize()));
|
2020-11-15 16:06:10 +01:00
|
|
|
|
2023-02-17 10:26:09 +01:00
|
|
|
if (this->m_fileStats.has_value()) {
|
|
|
|
result.emplace_back("hex.builtin.provider.file.creation"_lang, hex::format("{:%Y-%m-%d %H:%M:%S}", fmt::localtime(this->m_fileStats->st_ctime)));
|
|
|
|
result.emplace_back("hex.builtin.provider.file.access"_lang, hex::format("{:%Y-%m-%d %H:%M:%S}", fmt::localtime(this->m_fileStats->st_atime)));
|
|
|
|
result.emplace_back("hex.builtin.provider.file.modification"_lang, hex::format("{:%Y-%m-%d %H:%M:%S}", fmt::localtime(this->m_fileStats->st_mtime)));
|
2020-11-15 16:06:10 +01:00
|
|
|
}
|
|
|
|
|
|
|
|
return result;
|
|
|
|
}
|
|
|
|
|
2023-01-24 09:07:11 +01:00
|
|
|
std::variant<std::string, i128> FileProvider::queryInformation(const std::string &category, const std::string &argument) {
|
|
|
|
if (category == "file_path")
|
2023-03-12 18:43:05 +01:00
|
|
|
return wolv::util::toUTF8String(this->m_path);
|
2023-01-24 09:07:11 +01:00
|
|
|
else if (category == "file_name")
|
2023-03-12 18:43:05 +01:00
|
|
|
return wolv::util::toUTF8String(this->m_path.filename());
|
2023-01-24 09:07:11 +01:00
|
|
|
else if (category == "file_extension")
|
2023-03-12 18:43:05 +01:00
|
|
|
return wolv::util::toUTF8String(this->m_path.extension());
|
2023-01-24 09:07:11 +01:00
|
|
|
else if (category == "creation_time")
|
2023-02-17 10:26:09 +01:00
|
|
|
return this->m_fileStats->st_ctime;
|
2023-01-24 09:07:11 +01:00
|
|
|
else if (category == "access_time")
|
2023-02-17 10:26:09 +01:00
|
|
|
return this->m_fileStats->st_atime;
|
2023-01-24 09:07:11 +01:00
|
|
|
else if (category == "modification_time")
|
2023-02-17 10:26:09 +01:00
|
|
|
return this->m_fileStats->st_mtime;
|
2023-01-24 09:07:11 +01:00
|
|
|
else if (category == "permissions")
|
2023-02-17 10:26:09 +01:00
|
|
|
return this->m_fileStats->st_mode & 0777;
|
2023-01-24 09:07:11 +01:00
|
|
|
else
|
|
|
|
return Provider::queryInformation(category, argument);
|
|
|
|
}
|
|
|
|
|
2022-08-12 15:11:27 +02:00
|
|
|
bool FileProvider::handleFilePicker() {
|
|
|
|
return fs::openFileBrowser(fs::DialogMode::Open, {}, [this](const auto &path) {
|
|
|
|
this->setPath(path);
|
|
|
|
});
|
|
|
|
}
|
|
|
|
|
2022-03-04 11:36:37 +01:00
|
|
|
void FileProvider::setPath(const std::fs::path &path) {
|
2021-12-07 22:47:41 +01:00
|
|
|
this->m_path = path;
|
2021-12-12 00:41:44 +01:00
|
|
|
}
|
|
|
|
|
|
|
|
bool FileProvider::open() {
|
2021-07-27 21:07:36 +02:00
|
|
|
this->m_readable = true;
|
|
|
|
this->m_writable = true;
|
|
|
|
|
2023-03-12 18:27:29 +01:00
|
|
|
wolv::io::File file(this->m_path, wolv::io::File::Mode::Read);
|
2023-02-17 10:26:09 +01:00
|
|
|
if (!file.isValid()) {
|
|
|
|
this->m_writable = false;
|
|
|
|
this->m_readable = false;
|
|
|
|
return false;
|
|
|
|
}
|
2022-08-08 21:23:52 +02:00
|
|
|
|
2023-02-17 10:26:09 +01:00
|
|
|
this->m_fileStats = file.getFileInfo();
|
2023-03-21 09:47:42 +01:00
|
|
|
this->m_sizeFile = file.clone();
|
|
|
|
|
|
|
|
this->m_files.emplace(std::this_thread::get_id(), std::move(file));
|
2022-08-08 21:23:52 +02:00
|
|
|
|
2022-09-04 11:16:20 +02:00
|
|
|
return true;
|
2022-08-08 21:23:52 +02:00
|
|
|
}
|
|
|
|
|
|
|
|
void FileProvider::close() {
|
2021-07-27 21:07:36 +02:00
|
|
|
|
2022-08-08 21:23:52 +02:00
|
|
|
}
|
2021-12-12 00:41:44 +01:00
|
|
|
|
2023-03-17 11:31:50 +01:00
|
|
|
wolv::io::File& FileProvider::getFile() {
|
|
|
|
|
2023-03-21 09:47:42 +01:00
|
|
|
auto threadId = std::this_thread::get_id();
|
|
|
|
if (!this->m_files.contains(threadId)) {
|
|
|
|
std::scoped_lock lock(this->m_fileAccessMutex);
|
|
|
|
if (!this->m_files.contains(threadId))
|
|
|
|
this->m_files.emplace(threadId, this->m_sizeFile.clone());
|
|
|
|
}
|
2023-03-17 11:31:50 +01:00
|
|
|
|
2023-03-21 09:47:42 +01:00
|
|
|
return this->m_files[threadId];
|
2023-03-17 11:31:50 +01:00
|
|
|
}
|
|
|
|
|
2022-08-08 21:23:52 +02:00
|
|
|
void FileProvider::loadSettings(const nlohmann::json &settings) {
|
|
|
|
Provider::loadSettings(settings);
|
2022-04-17 16:57:30 +02:00
|
|
|
|
2023-01-11 23:31:25 +01:00
|
|
|
auto pathString = settings["path"].get<std::string>();
|
|
|
|
std::fs::path path = std::u8string(pathString.begin(), pathString.end());
|
|
|
|
|
|
|
|
if (auto projectPath = ProjectFile::getPath(); !projectPath.empty())
|
2023-03-17 11:43:50 +01:00
|
|
|
this->setPath(std::fs::weakly_canonical(projectPath.parent_path() / path));
|
2023-01-11 23:31:25 +01:00
|
|
|
else
|
|
|
|
this->setPath(path);
|
2021-07-27 21:07:36 +02:00
|
|
|
}
|
|
|
|
|
2022-08-08 21:23:52 +02:00
|
|
|
nlohmann::json FileProvider::storeSettings(nlohmann::json settings) const {
|
2023-01-19 11:09:24 +01:00
|
|
|
std::string path;
|
2023-01-11 23:31:25 +01:00
|
|
|
if (auto projectPath = ProjectFile::getPath(); !projectPath.empty())
|
2023-03-12 18:43:05 +01:00
|
|
|
path = wolv::util::toUTF8String(std::fs::proximate(this->m_path, projectPath.parent_path()));
|
2023-01-19 11:09:24 +01:00
|
|
|
if (path.empty())
|
2023-03-12 18:43:05 +01:00
|
|
|
path = wolv::util::toUTF8String(this->m_path);
|
2023-01-19 11:09:24 +01:00
|
|
|
|
|
|
|
settings["path"] = path;
|
2022-08-08 21:23:52 +02:00
|
|
|
|
|
|
|
return Provider::storeSettings(settings);
|
2021-07-27 21:07:36 +02:00
|
|
|
}
|
|
|
|
|
2022-08-14 09:38:38 +02:00
|
|
|
std::pair<Region, bool> FileProvider::getRegionValidity(u64 address) const {
|
2022-08-17 16:28:44 +02:00
|
|
|
address -= this->getBaseAddress();
|
|
|
|
|
2022-08-14 09:38:38 +02:00
|
|
|
if (address < this->getActualSize())
|
2022-08-17 16:28:44 +02:00
|
|
|
return { Region { this->getBaseAddress() + address, this->getActualSize() - address }, true };
|
2022-08-14 09:38:38 +02:00
|
|
|
else
|
|
|
|
return { Region::Invalid(), false };
|
|
|
|
}
|
|
|
|
|
2021-12-22 07:36:26 -05:00
|
|
|
}
|