2022-08-12 15:11:27 +02:00
|
|
|
#include "content/providers/intel_hex_provider.hpp"
|
|
|
|
|
|
|
|
#include <cstring>
|
|
|
|
|
2023-11-21 14:38:01 +01:00
|
|
|
#include <hex/api/localization_manager.hpp>
|
2022-08-12 15:11:27 +02:00
|
|
|
#include <hex/helpers/utils.hpp>
|
|
|
|
#include <hex/helpers/fmt.hpp>
|
|
|
|
|
|
|
|
#include <nlohmann/json.hpp>
|
|
|
|
|
2023-03-12 18:27:29 +01:00
|
|
|
#include <wolv/io/file.hpp>
|
2024-05-10 23:01:36 +02:00
|
|
|
#include <wolv/utils/expected.hpp>
|
2023-03-12 18:43:05 +01:00
|
|
|
#include <wolv/utils/string.hpp>
|
2023-03-12 18:27:29 +01:00
|
|
|
|
2022-11-07 00:04:47 +01:00
|
|
|
namespace hex::plugin::builtin {
|
2022-08-12 15:11:27 +02:00
|
|
|
|
|
|
|
namespace intel_hex {
|
|
|
|
|
|
|
|
u8 parseHexDigit(char c) {
|
|
|
|
if (c >= '0' && c <= '9')
|
|
|
|
return c - '0';
|
|
|
|
else if (c >= 'A' && c <= 'F')
|
|
|
|
return c - 'A' + 10;
|
|
|
|
else if (c >= 'a' && c <= 'f')
|
|
|
|
return c - 'a' + 10;
|
|
|
|
else
|
|
|
|
throw std::runtime_error("Failed to parse hex digit");
|
|
|
|
}
|
|
|
|
|
2024-05-10 23:01:36 +02:00
|
|
|
wolv::util::Expected<std::map<u64, std::vector<u8>>, std::string> parseIntelHex(const std::string &string) {
|
2022-08-12 15:11:27 +02:00
|
|
|
std::map<u64, std::vector<u8>> result;
|
|
|
|
|
|
|
|
u8 checksum = 0x00;
|
|
|
|
u64 offset = 0x00;
|
|
|
|
|
2023-11-10 20:47:08 +01:00
|
|
|
u8 byteCount = 0x00;
|
2022-08-12 15:11:27 +02:00
|
|
|
u32 segmentAddress = 0x0000'0000;
|
|
|
|
u32 extendedLinearAddress = 0x0000'0000;
|
|
|
|
u16 address = 0x0000;
|
|
|
|
std::vector<u8> data;
|
|
|
|
|
|
|
|
enum class RecordType {
|
|
|
|
Data = 0x00,
|
|
|
|
EndOfFile = 0x01,
|
|
|
|
ExtendedSegmentAddress = 0x02,
|
|
|
|
StartSegmentAddress = 0x03,
|
|
|
|
ExtendedLinearAddress = 0x04,
|
|
|
|
StartLinearAddress = 0x05
|
|
|
|
} recordType;
|
|
|
|
|
2023-11-10 20:47:08 +01:00
|
|
|
auto c = [&] {
|
2023-11-17 15:01:57 +01:00
|
|
|
while (offset < string.length() && std::isspace(string[offset]))
|
2022-08-12 15:11:27 +02:00
|
|
|
offset++;
|
|
|
|
|
|
|
|
if (offset >= string.length())
|
|
|
|
throw std::runtime_error("Unexpected end of file");
|
|
|
|
|
|
|
|
return string[offset++];
|
|
|
|
};
|
|
|
|
|
2023-12-27 16:33:49 +01:00
|
|
|
auto parseValue = [&](u8 count) {
|
2022-08-12 15:11:27 +02:00
|
|
|
u64 value = 0x00;
|
2023-12-27 16:33:49 +01:00
|
|
|
for (u8 i = 0; i < count; i++) {
|
2022-08-12 15:11:27 +02:00
|
|
|
u8 byte = (parseHexDigit(c()) << 4) | parseHexDigit(c());
|
|
|
|
value <<= 8;
|
|
|
|
value |= byte;
|
|
|
|
|
|
|
|
checksum += byte;
|
|
|
|
}
|
|
|
|
|
|
|
|
|
|
|
|
return value;
|
|
|
|
};
|
|
|
|
|
|
|
|
bool endOfFile = false;
|
|
|
|
try {
|
|
|
|
while (offset < string.length()) {
|
|
|
|
// Parse start code
|
|
|
|
if (c() != ':')
|
|
|
|
return { };
|
|
|
|
|
|
|
|
checksum = 0x00;
|
|
|
|
|
|
|
|
if (endOfFile)
|
|
|
|
throw std::runtime_error("Unexpected end of file");
|
|
|
|
|
|
|
|
// Parse byte count
|
|
|
|
byteCount = parseValue(1);
|
|
|
|
|
|
|
|
// Parse address
|
|
|
|
address = parseValue(2);
|
|
|
|
|
|
|
|
// Parse record type
|
|
|
|
recordType = static_cast<RecordType>(parseValue(1));
|
|
|
|
|
|
|
|
data.clear();
|
|
|
|
for (u32 i = 0; i < byteCount; i++) {
|
|
|
|
data.push_back(parseValue(1));
|
|
|
|
}
|
|
|
|
|
|
|
|
parseValue(1);
|
|
|
|
if (!data.empty() && checksum != 0x00)
|
|
|
|
throw std::runtime_error("Checksum mismatch");
|
|
|
|
|
|
|
|
while (std::isspace(string[offset]) && offset < string.length())
|
|
|
|
offset++;
|
|
|
|
|
|
|
|
// Construct region
|
|
|
|
switch (recordType) {
|
|
|
|
case RecordType::Data: {
|
|
|
|
result[extendedLinearAddress | (segmentAddress + address)] = data;
|
|
|
|
break;
|
|
|
|
}
|
|
|
|
case RecordType::EndOfFile: {
|
|
|
|
endOfFile = true;
|
|
|
|
break;
|
|
|
|
}
|
|
|
|
case RecordType::ExtendedSegmentAddress: {
|
|
|
|
if (byteCount != 2)
|
|
|
|
throw std::runtime_error("Unexpected byte count");
|
|
|
|
|
|
|
|
segmentAddress = (data[0] << 8 | data[1]) * 16;
|
|
|
|
break;
|
|
|
|
}
|
|
|
|
case RecordType::StartSegmentAddress: {
|
|
|
|
if (byteCount != 4)
|
|
|
|
throw std::runtime_error("Unexpected byte count");
|
|
|
|
|
|
|
|
// Can be safely ignored
|
|
|
|
break;
|
|
|
|
}
|
|
|
|
case RecordType::ExtendedLinearAddress: {
|
|
|
|
if (byteCount != 2)
|
|
|
|
throw std::runtime_error("Unexpected byte count");
|
|
|
|
|
|
|
|
extendedLinearAddress = (data[0] << 8 | data[1]) << 16;
|
|
|
|
break;
|
|
|
|
}
|
|
|
|
case RecordType::StartLinearAddress: {
|
|
|
|
if (byteCount != 4)
|
|
|
|
throw std::runtime_error("Unexpected byte count");
|
|
|
|
|
|
|
|
// Can be safely ignored
|
|
|
|
break;
|
|
|
|
}
|
|
|
|
}
|
2023-01-26 11:05:51 +01:00
|
|
|
|
|
|
|
while (std::isspace(string[offset]) && offset < string.length())
|
|
|
|
offset++;
|
2022-08-12 15:11:27 +02:00
|
|
|
}
|
|
|
|
|
2024-05-10 23:01:36 +02:00
|
|
|
} catch (const std::runtime_error &e) {
|
|
|
|
return wolv::util::Unexpected<std::string>(e.what());
|
2022-08-12 15:11:27 +02:00
|
|
|
}
|
|
|
|
|
|
|
|
return result;
|
|
|
|
}
|
|
|
|
|
|
|
|
}
|
|
|
|
|
2022-08-17 16:28:44 +02:00
|
|
|
void IntelHexProvider::setBaseAddress(u64 address) {
|
|
|
|
auto oldBase = this->getBaseAddress();
|
|
|
|
|
2023-12-19 13:10:25 +01:00
|
|
|
auto regions = m_data.overlapping({ oldBase, oldBase + this->getActualSize() });
|
2022-08-17 16:28:44 +02:00
|
|
|
|
2023-12-19 13:10:25 +01:00
|
|
|
decltype(m_data) newIntervals;
|
2023-05-11 09:27:23 +02:00
|
|
|
for (auto &[interval, data] : regions) {
|
|
|
|
newIntervals.insert({ interval.start - oldBase + address, interval.end - oldBase + address }, *data);
|
|
|
|
}
|
2023-12-19 13:10:25 +01:00
|
|
|
m_data = newIntervals;
|
2022-08-17 16:28:44 +02:00
|
|
|
|
|
|
|
Provider::setBaseAddress(address);
|
|
|
|
}
|
|
|
|
|
2022-08-12 15:11:27 +02:00
|
|
|
void IntelHexProvider::readRaw(u64 offset, void *buffer, size_t size) {
|
2023-12-19 13:10:25 +01:00
|
|
|
auto intervals = m_data.overlapping({ offset, (offset + size) - 1 });
|
2022-08-12 15:11:27 +02:00
|
|
|
|
|
|
|
std::memset(buffer, 0x00, size);
|
2023-11-10 20:47:08 +01:00
|
|
|
auto bytes = static_cast<u8*>(buffer);
|
2023-05-11 09:27:23 +02:00
|
|
|
for (const auto &[interval, data] : intervals) {
|
|
|
|
for (u32 i = std::max(interval.start, offset); i <= interval.end && (i - offset) < size; i++) {
|
|
|
|
bytes[i - offset] = (*data)[i - interval.start];
|
2022-08-12 15:11:27 +02:00
|
|
|
}
|
|
|
|
}
|
|
|
|
}
|
|
|
|
|
|
|
|
void IntelHexProvider::writeRaw(u64 offset, const void *buffer, size_t size) {
|
2024-12-14 21:35:54 +01:00
|
|
|
std::ignore = offset;
|
|
|
|
std::ignore = buffer;
|
|
|
|
std::ignore = size;
|
2022-08-12 15:11:27 +02:00
|
|
|
}
|
|
|
|
|
2023-12-07 12:06:26 +01:00
|
|
|
u64 IntelHexProvider::getActualSize() const {
|
2023-12-19 13:10:25 +01:00
|
|
|
return m_dataSize;
|
2022-08-12 15:11:27 +02:00
|
|
|
}
|
|
|
|
|
|
|
|
bool IntelHexProvider::open() {
|
2023-12-19 13:10:25 +01:00
|
|
|
auto file = wolv::io::File(m_sourceFilePath, wolv::io::File::Mode::Read);
|
2024-05-10 23:01:36 +02:00
|
|
|
if (!file.isValid()) {
|
|
|
|
this->setErrorMessage(hex::format("hex.builtin.provider.file.error.open"_lang, m_sourceFilePath.string(), ::strerror(errno)));
|
2022-08-12 15:11:27 +02:00
|
|
|
return false;
|
2024-05-10 23:01:36 +02:00
|
|
|
}
|
2022-08-12 15:11:27 +02:00
|
|
|
|
|
|
|
auto data = intel_hex::parseIntelHex(file.readString());
|
2024-05-10 23:01:36 +02:00
|
|
|
if (!data.has_value()) {
|
|
|
|
this->setErrorMessage(data.error());
|
2022-08-12 15:11:27 +02:00
|
|
|
return false;
|
2024-05-10 23:01:36 +02:00
|
|
|
}
|
2022-08-12 15:11:27 +02:00
|
|
|
|
|
|
|
u64 maxAddress = 0x00;
|
2024-05-10 23:02:39 +02:00
|
|
|
for (auto &[address, bytes] : data.value()) {
|
2022-08-12 15:11:27 +02:00
|
|
|
auto endAddress = (address + bytes.size()) - 1;
|
2023-12-19 13:10:25 +01:00
|
|
|
m_data.emplace({ address, endAddress }, std::move(bytes));
|
2022-08-12 15:11:27 +02:00
|
|
|
|
|
|
|
if (endAddress > maxAddress)
|
|
|
|
maxAddress = endAddress;
|
|
|
|
}
|
2023-05-07 23:27:43 +02:00
|
|
|
|
2023-12-19 13:10:25 +01:00
|
|
|
m_dataSize = maxAddress + 1;
|
|
|
|
m_dataValid = true;
|
2022-08-12 15:11:27 +02:00
|
|
|
|
2022-09-04 11:16:20 +02:00
|
|
|
return true;
|
2022-08-12 15:11:27 +02:00
|
|
|
}
|
|
|
|
|
|
|
|
void IntelHexProvider::close() {
|
|
|
|
|
|
|
|
}
|
|
|
|
|
|
|
|
[[nodiscard]] std::string IntelHexProvider::getName() const {
|
2023-12-19 13:10:25 +01:00
|
|
|
return hex::format("hex.builtin.provider.intel_hex.name"_lang, wolv::util::toUTF8String(m_sourceFilePath.filename()));
|
2022-08-12 15:11:27 +02:00
|
|
|
}
|
|
|
|
|
2023-06-04 10:42:11 +02:00
|
|
|
[[nodiscard]] std::vector<IntelHexProvider::Description> IntelHexProvider::getDataDescription() const {
|
|
|
|
std::vector<Description> result;
|
|
|
|
|
2023-12-19 13:10:25 +01:00
|
|
|
result.emplace_back("hex.builtin.provider.file.path"_lang, wolv::util::toUTF8String(m_sourceFilePath));
|
2023-06-04 10:42:11 +02:00
|
|
|
result.emplace_back("hex.builtin.provider.file.size"_lang, hex::toByteString(this->getActualSize()));
|
|
|
|
|
|
|
|
return result;
|
|
|
|
}
|
|
|
|
|
2022-08-12 15:11:27 +02:00
|
|
|
bool IntelHexProvider::handleFilePicker() {
|
2023-01-12 08:20:15 +01:00
|
|
|
auto picked = fs::openFileBrowser(fs::DialogMode::Open, {
|
|
|
|
{ "Intel Hex File", "hex" },
|
|
|
|
{ "Intel Hex File", "h86" },
|
|
|
|
{ "Intel Hex File", "hxl" },
|
|
|
|
{ "Intel Hex File", "hxh" },
|
|
|
|
{ "Intel Hex File", "obl" },
|
|
|
|
{ "Intel Hex File", "obh" },
|
|
|
|
{ "Intel Hex File", "mcs" },
|
|
|
|
{ "Intel Hex File", "ihex" },
|
|
|
|
{ "Intel Hex File", "ihe" },
|
|
|
|
{ "Intel Hex File", "ihx" },
|
|
|
|
{ "Intel Hex File", "a43" },
|
|
|
|
{ "Intel Hex File", "a90" }
|
|
|
|
}, [this](const std::fs::path &path) {
|
2023-12-19 13:10:25 +01:00
|
|
|
m_sourceFilePath = path;
|
2023-01-12 08:20:15 +01:00
|
|
|
}
|
|
|
|
);
|
|
|
|
|
2022-08-12 15:11:27 +02:00
|
|
|
if (!picked)
|
|
|
|
return false;
|
2023-12-19 13:10:25 +01:00
|
|
|
if (!wolv::io::fs::isRegularFile(m_sourceFilePath))
|
2022-08-12 15:11:27 +02:00
|
|
|
return false;
|
|
|
|
|
|
|
|
return true;
|
|
|
|
}
|
|
|
|
|
|
|
|
std::pair<Region, bool> IntelHexProvider::getRegionValidity(u64 address) const {
|
2023-12-19 13:10:25 +01:00
|
|
|
auto intervals = m_data.overlapping({ address, address });
|
2023-05-11 09:27:23 +02:00
|
|
|
if (intervals.empty()) {
|
2022-08-14 14:45:18 +02:00
|
|
|
return Provider::getRegionValidity(address);
|
|
|
|
}
|
2022-08-12 15:11:27 +02:00
|
|
|
|
2023-12-19 13:10:25 +01:00
|
|
|
decltype(m_data)::Interval closestInterval = { 0, 0 };
|
2023-05-11 09:27:23 +02:00
|
|
|
for (const auto &[interval, data] : intervals) {
|
2023-06-03 15:53:55 +02:00
|
|
|
if (interval.start <= closestInterval.end)
|
2023-05-11 09:27:23 +02:00
|
|
|
closestInterval = interval;
|
2022-08-12 15:11:27 +02:00
|
|
|
}
|
2023-05-11 09:27:23 +02:00
|
|
|
return { Region { closestInterval.start, (closestInterval.end - closestInterval.start) + 1}, true };
|
2023-05-07 23:27:43 +02:00
|
|
|
|
2022-08-12 15:11:27 +02:00
|
|
|
}
|
|
|
|
|
2022-08-14 10:07:45 +02:00
|
|
|
void IntelHexProvider::loadSettings(const nlohmann::json &settings) {
|
|
|
|
Provider::loadSettings(settings);
|
|
|
|
|
2023-05-16 11:33:00 +02:00
|
|
|
auto path = settings.at("path").get<std::string>();
|
2023-12-19 13:10:25 +01:00
|
|
|
m_sourceFilePath = std::u8string(path.begin(), path.end());
|
2022-08-14 10:07:45 +02:00
|
|
|
}
|
|
|
|
|
|
|
|
nlohmann::json IntelHexProvider::storeSettings(nlohmann::json settings) const {
|
2024-05-08 21:30:20 +02:00
|
|
|
settings["path"] = wolv::io::fs::toNormalizedPathString(m_sourceFilePath);
|
2022-08-14 10:07:45 +02:00
|
|
|
|
|
|
|
return Provider::storeSettings(settings);
|
|
|
|
}
|
|
|
|
|
2022-08-12 15:11:27 +02:00
|
|
|
}
|