#include "content/providers/file_provider.hpp" #include "content/providers/memory_file_provider.hpp" #include #include #include #include #include #include #include #include #include #include #include #include #include #include #if defined(OS_WINDOWS) #include #endif namespace hex::plugin::builtin { static std::mutex s_openCloseMutex; using namespace wolv::literals; std::set FileProvider::s_openedFiles; bool FileProvider::isAvailable() const { return true; } bool FileProvider::isReadable() const { return isAvailable() && m_readable; } bool FileProvider::isWritable() const { return isAvailable() && m_writable; } bool FileProvider::isResizable() const { return isAvailable() && isWritable(); } bool FileProvider::isSavable() const { return m_loadedIntoMemory; } void FileProvider::readRaw(u64 offset, void *buffer, size_t size) { if (m_fileSize == 0 || (offset + size) > m_fileSize || buffer == nullptr || size == 0) return; if (m_loadedIntoMemory) std::memcpy(buffer, m_data.data() + offset, size); else m_file.readBufferAtomic(offset, static_cast(buffer), size); } void FileProvider::writeRaw(u64 offset, const void *buffer, size_t size) { if ((offset + size) > this->getActualSize() || buffer == nullptr || size == 0) return; if (m_loadedIntoMemory) std::memcpy(m_data.data() + offset, buffer, size); else m_file.writeBufferAtomic(offset, static_cast(buffer), size); } void FileProvider::save() { if (m_loadedIntoMemory) { m_ignoreNextChangeEvent = true; m_file.open(); m_file.writeVectorAtomic(0x00, m_data); m_file.setSize(m_data.size()); } else { m_file.flush(); } #if defined(OS_WINDOWS) FILETIME ft; SYSTEMTIME st; if (m_file.isValid()) { GetSystemTime(&st); if (SystemTimeToFileTime(&st, &ft)) { auto fileHandle = HANDLE(_get_osfhandle(_fileno(m_file.getHandle()))); SetFileTime(fileHandle, nullptr, nullptr, &ft); } } #endif if (m_loadedIntoMemory) m_file.close(); Provider::save(); } void FileProvider::saveAs(const std::fs::path &path) { if (path == m_path) this->save(); else Provider::saveAs(path); } void FileProvider::resizeRaw(u64 newSize) { if (m_loadedIntoMemory) m_data.resize(newSize); else m_file.setSize(newSize); m_fileSize = newSize; } u64 FileProvider::getActualSize() const { return m_fileSize; } std::string FileProvider::getName() const { return wolv::util::toUTF8String(m_path.filename()); } std::vector FileProvider::getDataDescription() const { std::vector result; result.emplace_back("hex.builtin.provider.file.path"_lang, wolv::util::toUTF8String(m_path)); result.emplace_back("hex.builtin.provider.file.size"_lang, hex::toByteString(this->getActualSize())); if (m_fileStats.has_value()) { std::string creationTime, accessTime, modificationTime; try { creationTime = hex::format("{:%Y-%m-%d %H:%M:%S}", fmt::localtime(m_fileStats->st_ctime)); } catch (const std::exception&) { creationTime = "???"; } try { accessTime = hex::format("{:%Y-%m-%d %H:%M:%S}", fmt::localtime(m_fileStats->st_atime)); } catch (const std::exception&) { accessTime = "???"; } try { modificationTime = hex::format("{:%Y-%m-%d %H:%M:%S}", fmt::localtime(m_fileStats->st_mtime)); } catch (const std::exception&) { modificationTime = "???"; } result.emplace_back("hex.builtin.provider.file.creation"_lang, creationTime); result.emplace_back("hex.builtin.provider.file.access"_lang, accessTime); result.emplace_back("hex.builtin.provider.file.modification"_lang, modificationTime); } return result; } std::variant FileProvider::queryInformation(const std::string &category, const std::string &argument) { if (category == "file_path") return wolv::io::fs::toNormalizedPathString(m_path); else if (category == "file_name") return wolv::io::fs::toNormalizedPathString(m_path.filename()); else if (category == "file_extension") return wolv::io::fs::toNormalizedPathString(m_path.extension()); else if (category == "creation_time") return m_fileStats->st_ctime; else if (category == "access_time") return m_fileStats->st_atime; else if (category == "modification_time") return m_fileStats->st_mtime; else if (category == "permissions") return m_fileStats->st_mode & 0777; else return Provider::queryInformation(category, argument); } bool FileProvider::handleFilePicker() { return fs::openFileBrowser(fs::DialogMode::Open, {}, [this](const auto &path) { this->setPath(path); }); } std::vector FileProvider::getMenuEntries() { FileProvider::MenuEntry loadMenuItem; if (m_loadedIntoMemory) loadMenuItem = { "hex.builtin.provider.file.menu.direct_access"_lang, ICON_VS_ARROW_SWAP, [this] { this->convertToDirectAccess(); } }; else loadMenuItem = { "hex.builtin.provider.file.menu.into_memory"_lang, ICON_VS_ARROW_SWAP, [this] { this->convertToMemoryFile(); } }; return { { "hex.builtin.provider.file.menu.open_folder"_lang, ICON_VS_FOLDER_OPENED, [this] { fs::openFolderWithSelectionExternal(m_path); } }, { "hex.builtin.provider.file.menu.open_file"_lang, ICON_VS_FILE, [this] { fs::openFileExternal(m_path); } }, loadMenuItem }; } void FileProvider::setPath(const std::fs::path &path) { m_path = path; m_path.make_preferred(); } bool FileProvider::open() { const size_t maxMemoryFileSize = ContentRegistry::Settings::read("hex.builtin.setting.general", "hex.builtin.setting.general.max_mem_file_size", 128_MiB); size_t fileSize = 0x00; { wolv::io::File file(m_path, wolv::io::File::Mode::Read); if (!file.isValid()) { this->setErrorMessage(hex::format("hex.builtin.provider.file.error.open"_lang, m_path.string(), std::system_category().message(file.getOpenError().value_or(0)))); return false; } fileSize = file.getSize(); } const bool directAccess = fileSize >= maxMemoryFileSize; const bool result = open(directAccess); if (result && directAccess) { m_writable = false; ui::PopupQuestion::open("hex.builtin.provider.file.too_large"_lang, [this] { m_writable = false; }, [this] { m_writable = true; RequestUpdateWindowTitle::post(); }); } return result; } bool FileProvider::open(bool directAccess) { m_readable = true; m_writable = true; wolv::io::File file(m_path, wolv::io::File::Mode::Write); if (!file.isValid()) { m_writable = false; file = wolv::io::File(m_path, wolv::io::File::Mode::Read); if (!file.isValid()) { m_readable = false; this->setErrorMessage(hex::format("hex.builtin.provider.file.error.open"_lang, m_path.string(), std::system_category().message(file.getOpenError().value_or(0)))); return false; } ui::ToastInfo::open("hex.builtin.popup.error.read_only"_lang); } std::scoped_lock lock(s_openCloseMutex); m_file = std::move(file); m_fileStats = m_file.getFileInfo(); m_fileSize = m_file.getSize(); // Make sure the current file is not already opened { auto alreadyOpenedFileProvider = std::ranges::find_if(s_openedFiles, [this](const FileProvider *provider) { return provider->m_path == m_path; }); if (alreadyOpenedFileProvider != s_openedFiles.end()) { ImHexApi::Provider::setCurrentProvider(*alreadyOpenedFileProvider); return false; } else { s_openedFiles.insert(this); } } if (m_writable) { if (directAccess) { m_loadedIntoMemory = false; } else { m_data = m_file.readVectorAtomic(0x00, m_fileSize); if (!m_data.empty()) { m_changeTracker = wolv::io::ChangeTracker(m_file); m_changeTracker.startTracking([this]{ this->handleFileChange(); }); m_file.close(); m_loadedIntoMemory = true; } } } return true; } void FileProvider::close() { std::scoped_lock lock(s_openCloseMutex); m_file.close(); m_data.clear(); s_openedFiles.erase(this); m_changeTracker.stopTracking(); } void FileProvider::loadSettings(const nlohmann::json &settings) { Provider::loadSettings(settings); auto pathString = settings.at("path").get(); std::fs::path path = std::u8string(pathString.begin(), pathString.end()); if (auto projectPath = ProjectFile::getPath(); !projectPath.empty()) { std::fs::path fullPath; try { fullPath = std::fs::weakly_canonical(projectPath.parent_path() / path); } catch (const std::fs::filesystem_error &) { fullPath = projectPath.parent_path() / path; } if (!wolv::io::fs::exists(fullPath)) fullPath = path; if (!wolv::io::fs::exists(fullPath)) { this->setErrorMessage(hex::format("hex.builtin.provider.file.error.open"_lang, m_path.string(), std::system_category().message(ENOENT))); } path = std::move(fullPath); } this->setPath(path); } nlohmann::json FileProvider::storeSettings(nlohmann::json settings) const { std::fs::path path; if (m_path.u8string().starts_with(u8"//")) { path = m_path; } else { if (auto projectPath = ProjectFile::getPath(); !projectPath.empty()) path = std::fs::proximate(m_path, projectPath.parent_path()); if (path.empty()) path = m_path; } settings["path"] = wolv::io::fs::toNormalizedPathString(path); return Provider::storeSettings(settings); } std::pair FileProvider::getRegionValidity(u64 address) const { address -= this->getBaseAddress(); if (address < this->getActualSize()) return { Region { this->getBaseAddress() + address, this->getActualSize() - address }, true }; else return { Region::Invalid(), false }; } void FileProvider::convertToMemoryFile() { this->close(); this->open(false); } void FileProvider::convertToDirectAccess() { this->close(); this->open(true); } void FileProvider::handleFileChange() { if (m_ignoreNextChangeEvent) { m_ignoreNextChangeEvent = false; return; } if (m_changeEventAcknowledgementPending) { return; } m_changeEventAcknowledgementPending = true; ui::PopupQuestion::open("hex.builtin.provider.file.reload_changes"_lang, [this] { this->close(); (void)this->open(!m_loadedIntoMemory); getUndoStack().reapply(); m_changeEventAcknowledgementPending = false; }, [this]{ m_changeEventAcknowledgementPending = false; }); } }