#include #include #include #include #include #include namespace hex::plugin::builtin { #define LITERAL_COMPARE(literal, cond) std::visit([&](auto &&literal) { return (cond) != 0; }, literal) #define AS_TYPE(type, value) ctx.template asType(value) void registerPatternLanguageFunctions() { using namespace hex::lang; /* findSequence(occurrenceIndex, byte...) */ ContentRegistry::PatternLanguageFunctions::add("findSequence", ContentRegistry::PatternLanguageFunctions::MoreParametersThan | 1, [](auto &ctx, auto params) { auto& occurrenceIndex = AS_TYPE(ASTNodeIntegerLiteral, params[0])->getValue(); std::vector sequence; for (u32 i = 1; i < params.size(); i++) { sequence.push_back(std::visit([&](auto &&value) -> u8 { if (value <= 0xFF) return value; else ctx.getConsole().abortEvaluation("sequence bytes need to fit into 1 byte"); }, AS_TYPE(ASTNodeIntegerLiteral, params[i])->getValue())); } std::vector bytes(sequence.size(), 0x00); u32 occurrences = 0; for (u64 offset = 0; offset < SharedData::currentProvider->getSize() - sequence.size(); offset++) { SharedData::currentProvider->read(offset, bytes.data(), bytes.size()); if (bytes == sequence) { if (LITERAL_COMPARE(occurrenceIndex, occurrences < occurrenceIndex)) { occurrences++; continue; } return new ASTNodeIntegerLiteral({ Token::ValueType::Unsigned64Bit, offset }); } } ctx.getConsole().abortEvaluation("failed to find sequence"); }); /* readUnsigned(address, size) */ ContentRegistry::PatternLanguageFunctions::add("readUnsigned", 2, [](auto &ctx, auto params) { auto address = AS_TYPE(ASTNodeIntegerLiteral, params[0])->getValue(); auto size = AS_TYPE(ASTNodeIntegerLiteral, params[1])->getValue(); if (LITERAL_COMPARE(address, address >= SharedData::currentProvider->getActualSize())) ctx.getConsole().abortEvaluation("address out of range"); return std::visit([&](auto &&address, auto &&size) { if (size <= 0 || size > 16) ctx.getConsole().abortEvaluation("invalid read size"); u8 value[(u8)size]; SharedData::currentProvider->read(address, value, size); switch ((u8)size) { case 1: return new ASTNodeIntegerLiteral({ Token::ValueType::Unsigned8Bit, *reinterpret_cast(value) }); case 2: return new ASTNodeIntegerLiteral({ Token::ValueType::Unsigned16Bit, *reinterpret_cast(value) }); case 4: return new ASTNodeIntegerLiteral({ Token::ValueType::Unsigned32Bit, *reinterpret_cast(value) }); case 8: return new ASTNodeIntegerLiteral({ Token::ValueType::Unsigned64Bit, *reinterpret_cast(value) }); case 16: return new ASTNodeIntegerLiteral({ Token::ValueType::Unsigned128Bit, *reinterpret_cast(value) }); default: ctx.getConsole().abortEvaluation("invalid read size"); } }, address, size); }); /* readSigned(address, size) */ ContentRegistry::PatternLanguageFunctions::add("readSigned", 2, [](auto &ctx, auto params) { auto address = AS_TYPE(ASTNodeIntegerLiteral, params[0])->getValue(); auto size = AS_TYPE(ASTNodeIntegerLiteral, params[1])->getValue(); if (LITERAL_COMPARE(address, address >= SharedData::currentProvider->getActualSize())) ctx.getConsole().abortEvaluation("address out of range"); return std::visit([&](auto &&address, auto &&size) { if (size <= 0 || size > 16) ctx.getConsole().abortEvaluation("invalid read size"); u8 value[(u8)size]; SharedData::currentProvider->read(address, value, size); switch ((u8)size) { case 1: return new ASTNodeIntegerLiteral({ Token::ValueType::Signed8Bit, *reinterpret_cast(value) }); case 2: return new ASTNodeIntegerLiteral({ Token::ValueType::Signed16Bit, *reinterpret_cast(value) }); case 4: return new ASTNodeIntegerLiteral({ Token::ValueType::Signed32Bit, *reinterpret_cast(value) }); case 8: return new ASTNodeIntegerLiteral({ Token::ValueType::Signed64Bit, *reinterpret_cast(value) }); case 16: return new ASTNodeIntegerLiteral({ Token::ValueType::Signed128Bit, *reinterpret_cast(value) }); default: ctx.getConsole().abortEvaluation("invalid read size"); } }, address, size); }); /* assert(condition, message) */ ContentRegistry::PatternLanguageFunctions::add("assert", 2, [](auto &ctx, auto params) { auto condition = AS_TYPE(ASTNodeIntegerLiteral, params[0])->getValue(); auto message = AS_TYPE(ASTNodeStringLiteral, params[1])->getString(); if (LITERAL_COMPARE(condition, condition == 0)) ctx.getConsole().abortEvaluation(hex::format("assert failed \"{0}\"", message.data())); return nullptr; }); /* warnAssert(condition, message) */ ContentRegistry::PatternLanguageFunctions::add("warnAssert", 2, [](auto ctx, auto params) { auto condition = AS_TYPE(ASTNodeIntegerLiteral, params[0])->getValue(); auto message = AS_TYPE(ASTNodeStringLiteral, params[1])->getString(); if (LITERAL_COMPARE(condition, condition == 0)) ctx.getConsole().log(LogConsole::Level::Warning, hex::format("assert failed \"{0}\"", message.data())); return nullptr; }); /* print(values...) */ ContentRegistry::PatternLanguageFunctions::add("print", ContentRegistry::PatternLanguageFunctions::MoreParametersThan | 0, [](auto &ctx, auto params) { std::string message; for (auto& param : params) { if (auto integerLiteral = dynamic_cast(param); integerLiteral != nullptr) { std::visit([&](auto &&value) { switch (integerLiteral->getType()) { case lang::Token::ValueType::Character: message += (char)value; break; case lang::Token::ValueType::Boolean: message += value == 0 ? "false" : "true"; break; case lang::Token::ValueType::Unsigned8Bit: case lang::Token::ValueType::Unsigned16Bit: case lang::Token::ValueType::Unsigned32Bit: case lang::Token::ValueType::Unsigned64Bit: case lang::Token::ValueType::Unsigned128Bit: message += std::to_string(static_cast(value)); break; case lang::Token::ValueType::Signed8Bit: case lang::Token::ValueType::Signed16Bit: case lang::Token::ValueType::Signed32Bit: case lang::Token::ValueType::Signed64Bit: case lang::Token::ValueType::Signed128Bit: message += std::to_string(static_cast(value)); break; default: message += "< Custom Type >"; } }, integerLiteral->getValue()); } else if (auto stringLiteral = dynamic_cast(param); stringLiteral != nullptr) message += stringLiteral->getString(); } ctx.getConsole().log(LogConsole::Level::Info, message); return nullptr; }); /* alignTo(alignment, value) */ ContentRegistry::PatternLanguageFunctions::add("alignTo", 2, [](auto &ctx, auto params) -> ASTNode* { auto alignment = AS_TYPE(ASTNodeIntegerLiteral, params[0])->getValue(); auto value = AS_TYPE(ASTNodeIntegerLiteral, params[1])->getValue(); auto result = std::visit([](auto &&alignment, auto &&value) { u64 remainder = u64(value) % u64(alignment); return remainder != 0 ? u64(value) + (u64(alignment) - remainder) : u64(value); }, alignment, value); return new ASTNodeIntegerLiteral({ Token::ValueType::Unsigned64Bit, u64(result) }); }); /* dataSize() */ ContentRegistry::PatternLanguageFunctions::add("dataSize", ContentRegistry::PatternLanguageFunctions::NoParameters, [](auto &ctx, auto params) -> ASTNode* { return new ASTNodeIntegerLiteral({ Token::ValueType::Unsigned64Bit, u64(SharedData::currentProvider->getActualSize()) }); }); } }