2021-01-13 17:28:27 +01:00
|
|
|
#include <hex/providers/provider.hpp>
|
2020-12-22 18:10:01 +01:00
|
|
|
|
|
|
|
#include <hex.hpp>
|
2023-11-18 14:50:43 +01:00
|
|
|
#include <hex/api/event_manager.hpp>
|
2020-12-22 18:10:01 +01:00
|
|
|
|
|
|
|
#include <cmath>
|
2021-08-29 22:15:18 +02:00
|
|
|
#include <cstring>
|
2020-12-22 18:10:01 +01:00
|
|
|
#include <optional>
|
2022-04-17 16:57:30 +02:00
|
|
|
|
2023-01-24 09:07:11 +01:00
|
|
|
#include <hex/helpers/magic.hpp>
|
2023-03-17 08:15:43 +01:00
|
|
|
#include <wolv/io/file.hpp>
|
2023-12-05 14:32:28 +01:00
|
|
|
#include <wolv/literals.hpp>
|
2023-01-24 09:07:11 +01:00
|
|
|
|
2023-11-30 14:40:07 +01:00
|
|
|
#include <nlohmann/json.hpp>
|
|
|
|
|
2020-12-22 18:10:01 +01:00
|
|
|
namespace hex::prv {
|
|
|
|
|
2023-12-05 14:32:28 +01:00
|
|
|
using namespace wolv::literals;
|
|
|
|
|
2023-07-26 13:50:51 +02:00
|
|
|
namespace {
|
|
|
|
|
|
|
|
u32 s_idCounter = 0;
|
|
|
|
|
|
|
|
}
|
|
|
|
|
2022-08-08 21:23:52 +02:00
|
|
|
|
2023-11-25 12:43:48 +01:00
|
|
|
Provider::Provider() : m_undoRedoStack(this), m_id(s_idCounter++) {
|
|
|
|
|
2020-12-22 18:10:01 +01:00
|
|
|
}
|
|
|
|
|
2021-02-04 01:14:05 +01:00
|
|
|
Provider::~Provider() {
|
2023-12-19 13:10:25 +01:00
|
|
|
m_overlays.clear();
|
2023-09-03 16:18:29 +02:00
|
|
|
|
|
|
|
if (auto selection = ImHexApi::HexEditor::getSelection(); selection.has_value() && selection->provider == this)
|
2023-12-08 10:29:44 +01:00
|
|
|
EventRegionSelected::post(ImHexApi::HexEditor::ProviderRegion { { 0x00, 0x00 }, nullptr });
|
2021-02-04 01:14:05 +01:00
|
|
|
}
|
|
|
|
|
2021-03-21 14:50:47 +01:00
|
|
|
void Provider::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);
|
2023-12-02 18:42:57 +01:00
|
|
|
|
|
|
|
if (overlays)
|
|
|
|
this->applyOverlays(offset, buffer, size);
|
2021-04-16 21:50:15 +02:00
|
|
|
}
|
|
|
|
|
2020-12-22 18:10:01 +01:00
|
|
|
void Provider::write(u64 offset, const void *buffer, size_t size) {
|
2024-02-23 17:59:37 +01:00
|
|
|
if (!this->isWritable())
|
|
|
|
return;
|
|
|
|
|
2023-12-08 10:29:44 +01:00
|
|
|
EventProviderDataModified::post(this, offset, size, static_cast<const u8*>(buffer));
|
2022-08-08 21:23:52 +02:00
|
|
|
this->markDirty();
|
2021-04-16 21:50:15 +02:00
|
|
|
}
|
|
|
|
|
2023-04-06 17:36:28 +02:00
|
|
|
void Provider::save() {
|
2024-02-23 17:59:37 +01:00
|
|
|
if (!this->isWritable())
|
|
|
|
return;
|
|
|
|
|
2023-12-08 10:29:44 +01:00
|
|
|
EventProviderSaved::post(this);
|
2023-04-06 17:36:28 +02:00
|
|
|
}
|
2022-03-27 00:01:28 +01:00
|
|
|
void Provider::saveAs(const std::fs::path &path) {
|
2023-03-17 08:15:43 +01:00
|
|
|
wolv::io::File file(path, wolv::io::File::Mode::Create);
|
|
|
|
|
|
|
|
if (file.isValid()) {
|
2023-12-05 14:32:28 +01:00
|
|
|
std::vector<u8> buffer(std::min<size_t>(2_MiB, this->getActualSize()), 0x00);
|
2023-03-17 08:38:38 +01:00
|
|
|
size_t bufferSize = 0;
|
2023-03-17 08:15:43 +01:00
|
|
|
|
|
|
|
for (u64 offset = 0; offset < this->getActualSize(); offset += bufferSize) {
|
2023-12-05 14:32:28 +01:00
|
|
|
bufferSize = std::min<size_t>(buffer.size(), this->getActualSize() - offset);
|
2023-03-17 08:15:43 +01:00
|
|
|
|
2023-12-05 14:32:28 +01:00
|
|
|
this->read(this->getBaseAddress() + offset, buffer.data(), bufferSize, true);
|
2023-03-23 11:23:07 +01:00
|
|
|
file.writeBuffer(buffer.data(), bufferSize);
|
2023-03-17 08:15:43 +01:00
|
|
|
}
|
2023-04-06 17:36:28 +02:00
|
|
|
|
2023-12-08 10:29:44 +01:00
|
|
|
EventProviderSaved::post(this);
|
2023-03-17 08:15:43 +01:00
|
|
|
}
|
2022-03-27 00:01:28 +01:00
|
|
|
}
|
2021-08-21 13:53:50 +02:00
|
|
|
|
2024-02-26 20:51:08 +01:00
|
|
|
bool Provider::resize(u64 newSize) {
|
|
|
|
if (newSize >> 63) {
|
|
|
|
log::error("new provider size is very large ({}). Is it a negative number ?", newSize);
|
|
|
|
return false;
|
|
|
|
}
|
2023-11-25 12:43:48 +01:00
|
|
|
i64 difference = newSize - this->getActualSize();
|
|
|
|
|
|
|
|
if (difference > 0)
|
2023-12-08 10:29:44 +01:00
|
|
|
EventProviderDataInserted::post(this, this->getActualSize(), difference);
|
2023-11-25 12:43:48 +01:00
|
|
|
else if (difference < 0)
|
2023-12-15 20:59:58 +01:00
|
|
|
EventProviderDataRemoved::post(this, this->getActualSize() + difference, -difference);
|
2022-08-08 21:23:52 +02:00
|
|
|
|
|
|
|
this->markDirty();
|
2024-02-26 20:51:08 +01:00
|
|
|
return true;
|
2022-03-27 00:01:28 +01:00
|
|
|
}
|
2022-01-20 23:24:26 +01:00
|
|
|
|
2023-12-08 10:29:44 +01:00
|
|
|
void Provider::insert(u64 offset, u64 size) {
|
|
|
|
EventProviderDataInserted::post(this, offset, size);
|
2022-08-08 21:23:52 +02:00
|
|
|
|
|
|
|
this->markDirty();
|
2022-01-20 23:24:26 +01:00
|
|
|
}
|
2021-07-27 21:07:36 +02:00
|
|
|
|
2023-12-08 10:29:44 +01:00
|
|
|
void Provider::remove(u64 offset, u64 size) {
|
|
|
|
EventProviderDataRemoved::post(this, offset, size);
|
2022-08-08 21:23:52 +02:00
|
|
|
|
|
|
|
this->markDirty();
|
2022-06-17 15:21:56 +03:00
|
|
|
}
|
|
|
|
|
2024-03-10 14:31:39 +01:00
|
|
|
void Provider::insertRaw(u64 offset, u64 size) {
|
|
|
|
auto oldSize = this->getActualSize();
|
|
|
|
this->resizeRaw(oldSize + size);
|
|
|
|
|
|
|
|
std::vector<u8> buffer(0x1000);
|
|
|
|
const std::vector<u8> zeroBuffer(0x1000);
|
|
|
|
|
|
|
|
auto position = oldSize;
|
|
|
|
while (position > offset) {
|
|
|
|
const auto readSize = std::min<size_t>(position - offset, buffer.size());
|
|
|
|
|
|
|
|
position -= readSize;
|
|
|
|
|
|
|
|
this->readRaw(position, buffer.data(), readSize);
|
|
|
|
this->writeRaw(position, zeroBuffer.data(), readSize);
|
|
|
|
this->writeRaw(position + size, buffer.data(), readSize);
|
|
|
|
}
|
|
|
|
}
|
|
|
|
|
|
|
|
void Provider::removeRaw(u64 offset, u64 size) {
|
|
|
|
if (offset > this->getActualSize() || size == 0)
|
|
|
|
return;
|
|
|
|
|
|
|
|
if ((offset + size) > this->getActualSize())
|
|
|
|
size = this->getActualSize() - offset;
|
|
|
|
|
|
|
|
auto oldSize = this->getActualSize();
|
|
|
|
|
|
|
|
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->resizeRaw(newSize);
|
|
|
|
}
|
|
|
|
|
|
|
|
|
|
|
|
|
2023-11-10 20:47:08 +01:00
|
|
|
void Provider::applyOverlays(u64 offset, void *buffer, size_t size) const {
|
2023-12-19 13:10:25 +01:00
|
|
|
for (auto &overlay : m_overlays) {
|
2021-03-21 14:50:47 +01:00
|
|
|
auto overlayOffset = overlay->getAddress();
|
2022-02-01 22:09:44 +01:00
|
|
|
auto overlaySize = overlay->getSize();
|
2021-03-21 14:50:47 +01:00
|
|
|
|
2022-01-22 22:37:52 +01:00
|
|
|
i128 overlapMin = std::max(offset, overlayOffset);
|
|
|
|
i128 overlapMax = std::min(offset + size, overlayOffset + overlaySize);
|
2022-01-24 20:53:17 +01:00
|
|
|
if (overlapMax > overlapMin)
|
|
|
|
std::memcpy(static_cast<u8 *>(buffer) + std::max<i128>(0, overlapMin - offset), overlay->getData().data() + std::max<i128>(0, overlapMin - overlayOffset), overlapMax - overlapMin);
|
2021-03-21 14:50:47 +01:00
|
|
|
}
|
|
|
|
}
|
|
|
|
|
2022-01-24 20:53:17 +01:00
|
|
|
Overlay *Provider::newOverlay() {
|
2023-12-19 13:10:25 +01:00
|
|
|
return m_overlays.emplace_back(std::make_unique<Overlay>()).get();
|
2021-01-30 22:39:06 +01:00
|
|
|
}
|
|
|
|
|
|
|
|
void Provider::deleteOverlay(Overlay *overlay) {
|
2023-12-19 13:10:25 +01:00
|
|
|
m_overlays.remove_if([overlay](const auto &item) {
|
2023-08-25 23:54:39 +02:00
|
|
|
return item.get() == overlay;
|
|
|
|
});
|
2021-01-30 22:39:06 +01:00
|
|
|
}
|
|
|
|
|
2023-11-22 08:26:31 +01:00
|
|
|
const std::list<std::unique_ptr<Overlay>> &Provider::getOverlays() const {
|
2023-12-19 13:10:25 +01:00
|
|
|
return m_overlays;
|
2021-01-30 22:39:06 +01:00
|
|
|
}
|
|
|
|
|
|
|
|
|
2023-12-08 10:29:44 +01:00
|
|
|
u64 Provider::getPageSize() const {
|
2023-12-19 13:10:25 +01:00
|
|
|
return m_pageSize;
|
2023-05-23 11:34:30 +02:00
|
|
|
}
|
|
|
|
|
2023-12-08 10:29:44 +01:00
|
|
|
void Provider::setPageSize(u64 pageSize) {
|
2023-05-23 11:34:30 +02:00
|
|
|
if (pageSize > MaxPageSize)
|
|
|
|
pageSize = MaxPageSize;
|
2023-11-08 11:14:56 +01:00
|
|
|
if (pageSize == 0)
|
|
|
|
return;
|
2023-05-23 11:34:30 +02:00
|
|
|
|
2023-12-19 13:10:25 +01:00
|
|
|
m_pageSize = pageSize;
|
2023-05-23 11:34:30 +02:00
|
|
|
}
|
|
|
|
|
2021-09-21 02:29:54 +02:00
|
|
|
u32 Provider::getPageCount() const {
|
2023-05-23 11:34:30 +02:00
|
|
|
return (this->getActualSize() / this->getPageSize()) + (this->getActualSize() % this->getPageSize() != 0 ? 1 : 0);
|
2020-12-22 18:10:01 +01:00
|
|
|
}
|
|
|
|
|
|
|
|
u32 Provider::getCurrentPage() const {
|
2023-12-19 13:10:25 +01:00
|
|
|
return m_currPage;
|
2020-12-22 18:10:01 +01:00
|
|
|
}
|
|
|
|
|
|
|
|
void Provider::setCurrentPage(u32 page) {
|
|
|
|
if (page < getPageCount())
|
2023-12-19 13:10:25 +01:00
|
|
|
m_currPage = page;
|
2020-12-22 18:10:01 +01:00
|
|
|
}
|
|
|
|
|
|
|
|
|
2021-01-11 13:50:04 +01:00
|
|
|
void Provider::setBaseAddress(u64 address) {
|
2023-12-19 13:10:25 +01:00
|
|
|
m_baseAddress = address;
|
2022-08-08 21:23:52 +02:00
|
|
|
this->markDirty();
|
2021-01-11 13:50:04 +01:00
|
|
|
}
|
|
|
|
|
2021-09-21 02:29:54 +02:00
|
|
|
u64 Provider::getBaseAddress() const {
|
2023-12-19 13:10:25 +01:00
|
|
|
return m_baseAddress;
|
2021-12-09 21:10:24 +01:00
|
|
|
}
|
|
|
|
|
|
|
|
u64 Provider::getCurrentPageAddress() const {
|
2023-05-23 11:34:30 +02:00
|
|
|
return this->getPageSize() * this->getCurrentPage();
|
2020-12-22 18:10:01 +01:00
|
|
|
}
|
|
|
|
|
2023-12-07 13:02:12 +01:00
|
|
|
u64 Provider::getSize() const {
|
2023-12-19 13:10:25 +01:00
|
|
|
return std::min<u64>(this->getActualSize() - this->getPageSize() * m_currPage, this->getPageSize());
|
2020-12-22 18:10:01 +01:00
|
|
|
}
|
|
|
|
|
2021-09-21 02:29:54 +02:00
|
|
|
std::optional<u32> Provider::getPageOfAddress(u64 address) const {
|
2023-05-23 11:34:30 +02:00
|
|
|
u32 page = std::floor((address - this->getBaseAddress()) / double(this->getPageSize()));
|
2020-12-22 18:10:01 +01:00
|
|
|
|
|
|
|
if (page >= this->getPageCount())
|
2022-01-30 12:43:43 +01:00
|
|
|
return std::nullopt;
|
2020-12-22 18:10:01 +01:00
|
|
|
|
|
|
|
return page;
|
|
|
|
}
|
|
|
|
|
2023-09-08 21:59:27 +02:00
|
|
|
std::vector<Provider::Description> Provider::getDataDescription() const {
|
|
|
|
return { };
|
|
|
|
}
|
|
|
|
|
2021-03-26 21:43:24 +01:00
|
|
|
void Provider::undo() {
|
2023-12-19 13:10:25 +01:00
|
|
|
m_undoRedoStack.undo();
|
2021-03-26 21:43:24 +01:00
|
|
|
}
|
|
|
|
|
|
|
|
void Provider::redo() {
|
2023-12-19 13:10:25 +01:00
|
|
|
m_undoRedoStack.redo();
|
2021-03-26 21:43:24 +01:00
|
|
|
}
|
|
|
|
|
2021-09-21 02:29:54 +02:00
|
|
|
bool Provider::canUndo() const {
|
2023-12-19 13:10:25 +01:00
|
|
|
return m_undoRedoStack.canUndo();
|
2021-03-26 21:43:24 +01:00
|
|
|
}
|
|
|
|
|
2021-09-21 02:29:54 +02:00
|
|
|
bool Provider::canRedo() const {
|
2023-12-19 13:10:25 +01:00
|
|
|
return m_undoRedoStack.canRedo();
|
2021-03-26 21:43:24 +01:00
|
|
|
}
|
|
|
|
|
2022-08-12 15:11:27 +02:00
|
|
|
bool Provider::hasFilePicker() const {
|
|
|
|
return false;
|
|
|
|
}
|
|
|
|
|
|
|
|
bool Provider::handleFilePicker() {
|
|
|
|
return false;
|
|
|
|
}
|
2021-12-12 00:41:44 +01:00
|
|
|
|
|
|
|
bool Provider::hasLoadInterface() const {
|
|
|
|
return false;
|
|
|
|
}
|
|
|
|
|
|
|
|
bool Provider::hasInterface() const {
|
|
|
|
return false;
|
|
|
|
}
|
|
|
|
|
2023-03-16 16:48:15 +01:00
|
|
|
bool Provider::drawLoadInterface() {
|
|
|
|
return true;
|
2021-12-12 00:41:44 +01:00
|
|
|
}
|
|
|
|
|
|
|
|
void Provider::drawInterface() {
|
|
|
|
}
|
|
|
|
|
2022-08-08 21:23:52 +02:00
|
|
|
nlohmann::json Provider::storeSettings(nlohmann::json settings) const {
|
2022-08-14 10:07:45 +02:00
|
|
|
settings["displayName"] = this->getName();
|
|
|
|
settings["type"] = this->getTypeName();
|
|
|
|
|
2023-12-19 13:10:25 +01:00
|
|
|
settings["baseAddress"] = m_baseAddress;
|
|
|
|
settings["currPage"] = m_currPage;
|
2022-08-08 21:23:52 +02:00
|
|
|
|
|
|
|
return settings;
|
|
|
|
}
|
|
|
|
|
|
|
|
void Provider::loadSettings(const nlohmann::json &settings) {
|
2023-12-19 13:10:25 +01:00
|
|
|
m_baseAddress = settings["baseAddress"];
|
|
|
|
m_currPage = settings["currPage"];
|
2022-08-08 21:23:52 +02:00
|
|
|
}
|
|
|
|
|
2022-08-10 09:26:48 +02:00
|
|
|
std::pair<Region, bool> Provider::getRegionValidity(u64 address) const {
|
2023-12-27 11:31:25 +01:00
|
|
|
u64 absoluteAddress = address - this->getBaseAddress();
|
|
|
|
|
|
|
|
if (absoluteAddress < this->getActualSize())
|
|
|
|
return { Region { this->getBaseAddress() + absoluteAddress, this->getActualSize() - absoluteAddress }, true };
|
|
|
|
|
2022-08-10 09:26:48 +02:00
|
|
|
|
|
|
|
bool insideValidRegion = false;
|
|
|
|
|
|
|
|
std::optional<u64> nextRegionAddress;
|
2023-12-19 13:10:25 +01:00
|
|
|
for (const auto &overlay : m_overlays) {
|
2022-08-10 09:26:48 +02:00
|
|
|
Region overlayRegion = { overlay->getAddress(), overlay->getSize() };
|
|
|
|
if (!nextRegionAddress.has_value() || overlay->getAddress() < nextRegionAddress) {
|
|
|
|
nextRegionAddress = overlayRegion.getStartAddress();
|
|
|
|
}
|
|
|
|
|
|
|
|
if (Region { address, 1 }.overlaps(overlayRegion)) {
|
|
|
|
insideValidRegion = true;
|
|
|
|
}
|
|
|
|
}
|
|
|
|
|
|
|
|
if (!nextRegionAddress.has_value())
|
2022-08-14 22:38:01 +02:00
|
|
|
return { Region::Invalid(), false };
|
2022-08-10 09:26:48 +02:00
|
|
|
else
|
|
|
|
return { Region { address, *nextRegionAddress - address }, insideValidRegion };
|
|
|
|
}
|
|
|
|
|
2022-09-22 09:05:09 +02:00
|
|
|
|
|
|
|
u32 Provider::getID() const {
|
2023-12-19 13:10:25 +01:00
|
|
|
return m_id;
|
2022-09-22 09:05:09 +02:00
|
|
|
}
|
|
|
|
|
|
|
|
void Provider::setID(u32 id) {
|
2023-12-19 13:10:25 +01:00
|
|
|
m_id = id;
|
2022-09-22 09:05:09 +02:00
|
|
|
if (id > s_idCounter)
|
|
|
|
s_idCounter = id + 1;
|
|
|
|
}
|
|
|
|
|
2023-01-24 09:07:11 +01:00
|
|
|
|
|
|
|
[[nodiscard]] std::variant<std::string, i128> Provider::queryInformation(const std::string &category, const std::string &) {
|
|
|
|
if (category == "mime")
|
|
|
|
return magic::getMIMEType(this);
|
|
|
|
else if (category == "description")
|
|
|
|
return magic::getDescription(this);
|
|
|
|
else if (category == "provider_type")
|
|
|
|
return this->getTypeName();
|
|
|
|
else
|
|
|
|
return 0;
|
|
|
|
}
|
|
|
|
|
2023-07-05 20:49:57 +02:00
|
|
|
[[nodiscard]] bool Provider::isDumpable() const {
|
|
|
|
return true;
|
|
|
|
}
|
|
|
|
|
2021-12-22 07:36:26 -05:00
|
|
|
}
|