2021-01-13 17:28:27 +01:00
|
|
|
#include <hex/providers/provider.hpp>
|
2020-12-22 18:10:01 +01:00
|
|
|
|
|
|
|
#include <hex.hpp>
|
|
|
|
|
|
|
|
#include <cmath>
|
|
|
|
#include <map>
|
|
|
|
#include <optional>
|
|
|
|
#include <string>
|
|
|
|
|
|
|
|
namespace hex::prv {
|
|
|
|
|
|
|
|
Provider::Provider() {
|
|
|
|
this->m_patches.emplace_back();
|
|
|
|
}
|
|
|
|
|
2021-02-04 01:14:05 +01:00
|
|
|
Provider::~Provider() {
|
|
|
|
for (auto &overlay : this->m_overlays)
|
|
|
|
this->deleteOverlay(overlay);
|
|
|
|
}
|
|
|
|
|
2021-03-21 14:50:47 +01:00
|
|
|
void Provider::read(u64 offset, void *buffer, size_t size, bool overlays) {
|
2020-12-22 18:10:01 +01:00
|
|
|
this->readRaw(offset, buffer, size);
|
|
|
|
}
|
|
|
|
|
|
|
|
void Provider::write(u64 offset, const void *buffer, size_t size) {
|
|
|
|
this->writeRaw(offset, buffer, size);
|
|
|
|
}
|
|
|
|
|
2021-03-21 14:50:47 +01:00
|
|
|
void Provider::applyOverlays(u64 offset, void *buffer, size_t size) {
|
|
|
|
for (auto &overlay : this->m_overlays) {
|
|
|
|
auto overlayOffset = overlay->getAddress();
|
|
|
|
auto overlaySize = overlay->getSize();
|
|
|
|
|
|
|
|
s128 overlapMin = std::max(offset, overlayOffset);
|
|
|
|
s128 overlapMax = std::min(offset + size, overlayOffset + overlaySize);
|
|
|
|
if (overlapMax > overlapMin)
|
|
|
|
std::memcpy(static_cast<u8*>(buffer) + std::max<s128>(0, overlapMin - offset), overlay->getData().data() + std::max<s128>(0, overlapMin - overlayOffset), overlapMax - overlapMin);
|
|
|
|
}
|
|
|
|
}
|
|
|
|
|
2020-12-22 18:10:01 +01:00
|
|
|
|
|
|
|
std::map<u64, u8>& Provider::getPatches() {
|
|
|
|
return this->m_patches.back();
|
|
|
|
}
|
|
|
|
|
|
|
|
void Provider::applyPatches() {
|
|
|
|
for (auto &[patchAddress, patch] : this->m_patches.back())
|
|
|
|
this->writeRaw(patchAddress, &patch, 1);
|
|
|
|
}
|
|
|
|
|
2021-01-30 22:39:06 +01:00
|
|
|
|
|
|
|
Overlay* Provider::newOverlay() {
|
|
|
|
return this->m_overlays.emplace_back(new Overlay());
|
|
|
|
}
|
|
|
|
|
|
|
|
void Provider::deleteOverlay(Overlay *overlay) {
|
|
|
|
this->m_overlays.erase(std::find(this->m_overlays.begin(), this->m_overlays.end(), overlay));
|
|
|
|
delete overlay;
|
|
|
|
}
|
|
|
|
|
|
|
|
const std::list<Overlay*>& Provider::getOverlays() {
|
|
|
|
return this->m_overlays;
|
|
|
|
}
|
|
|
|
|
|
|
|
|
2020-12-22 18:10:01 +01:00
|
|
|
u32 Provider::getPageCount() {
|
|
|
|
return std::ceil(this->getActualSize() / double(PageSize));
|
|
|
|
}
|
|
|
|
|
|
|
|
u32 Provider::getCurrentPage() const {
|
|
|
|
return this->m_currPage;
|
|
|
|
}
|
|
|
|
|
|
|
|
void Provider::setCurrentPage(u32 page) {
|
|
|
|
if (page < getPageCount())
|
|
|
|
this->m_currPage = page;
|
|
|
|
}
|
|
|
|
|
|
|
|
|
2021-01-11 13:50:04 +01:00
|
|
|
void Provider::setBaseAddress(u64 address) {
|
|
|
|
this->m_baseAddress = address;
|
|
|
|
}
|
|
|
|
|
|
|
|
u64 Provider::getBaseAddress() {
|
|
|
|
return this->m_baseAddress + PageSize * this->m_currPage;
|
2020-12-22 18:10:01 +01:00
|
|
|
}
|
|
|
|
|
|
|
|
size_t Provider::getSize() {
|
|
|
|
return std::min(this->getActualSize() - PageSize * this->m_currPage, PageSize);
|
|
|
|
}
|
|
|
|
|
|
|
|
std::optional<u32> Provider::getPageOfAddress(u64 address) {
|
|
|
|
u32 page = std::floor(address / double(PageSize));
|
|
|
|
|
|
|
|
if (page >= this->getPageCount())
|
|
|
|
return { };
|
|
|
|
|
|
|
|
return page;
|
|
|
|
}
|
|
|
|
|
|
|
|
}
|