#pragma once #include #include "providers/provider.hpp" #include "helpers/utils.hpp" #include "lang/pattern_data.hpp" #include "ast_node.hpp" #include #include #include #include namespace hex::lang { class Evaluator { public: enum ConsoleLogLevel { Debug, Info, Warning, Error }; struct Function { constexpr static u32 UnlimitedParameters = 0xFFFF'FFFF; constexpr static u32 MoreParametersThan = 0x8000'0000; constexpr static u32 LessParametersThan = 0x4000'0000; constexpr static u32 NoParameters = 0x0000'0000; u32 parameterCount; std::function)> func; }; Evaluator(prv::Provider* &provider, std::endian defaultDataEndian); std::optional> evaluate(const std::vector& ast); const auto& getConsoleLog() { return this->m_consoleLog; } private: std::map m_types; prv::Provider* &m_provider; std::endian m_defaultDataEndian; u64 m_currOffset = 0; std::vector m_endianStack; std::vector m_globalMembers; std::vector*> m_currMembers; std::map m_functions; std::vector> m_consoleLog; using EvaluateError = std::string; void emmitDebugInfo(std::string_view message) { this->m_consoleLog.emplace_back(ConsoleLogLevel::Debug, "[-] " + std::string(message)); } void emmitInfo(std::string_view message) { this->m_consoleLog.emplace_back(ConsoleLogLevel::Info, "[i] " + std::string(message)); } void emmitWaring(std::string_view message) { this->m_consoleLog.emplace_back(ConsoleLogLevel::Warning, "[*] " + std::string(message)); } [[noreturn]] static void throwEvaluateError(std::string_view message) { throw EvaluateError("[!] " + std::string(message)); } [[nodiscard]] std::endian getCurrentEndian() const { return this->m_endianStack.back(); } void addFunction(std::string_view name, u32 parameterCount, std::function)> func) { if (this->m_functions.contains(name.data())) throwEvaluateError(hex::format("redefinition of function '%s'", name.data())); this->m_functions[name.data()] = { parameterCount, func }; } ASTNodeIntegerLiteral* evaluateScopeResolution(ASTNodeScopeResolution *node); ASTNodeIntegerLiteral* evaluateRValue(ASTNodeRValue *node); ASTNodeIntegerLiteral* evaluateFunctionCall(ASTNodeFunctionCall *node); ASTNodeIntegerLiteral* evaluateOperator(ASTNodeIntegerLiteral *left, ASTNodeIntegerLiteral *right, Token::Operator op); ASTNodeIntegerLiteral* evaluateOperand(ASTNode *node); ASTNodeIntegerLiteral* evaluateTernaryExpression(ASTNodeTernaryExpression *node); ASTNodeIntegerLiteral* evaluateMathematicalExpression(ASTNodeNumericExpression *node); PatternData* evaluateBuiltinType(ASTNodeBuiltinType *node); void evaluateMember(ASTNode *node, std::vector &currMembers, bool increaseOffset); PatternData* evaluateStruct(ASTNodeStruct *node); PatternData* evaluateUnion(ASTNodeUnion *node); PatternData* evaluateEnum(ASTNodeEnum *node); PatternData* evaluateBitfield(ASTNodeBitfield *node); PatternData* evaluateType(ASTNodeTypeDecl *node); PatternData* evaluateVariable(ASTNodeVariableDecl *node); PatternData* evaluateArray(ASTNodeArrayVariableDecl *node); PatternData* evaluatePointer(ASTNodePointerVariableDecl *node); template T* asType(ASTNode *param) { if (auto evaluatedParam = dynamic_cast(param); evaluatedParam != nullptr) return evaluatedParam; else throwEvaluateError("function got wrong type of parameter"); } #define BUILTIN_FUNCTION(name) ASTNodeIntegerLiteral* TOKEN_CONCAT(builtin_, name)(std::vector params) BUILTIN_FUNCTION(findSequence); BUILTIN_FUNCTION(readUnsigned); BUILTIN_FUNCTION(readSigned); BUILTIN_FUNCTION(assert); BUILTIN_FUNCTION(warnAssert); BUILTIN_FUNCTION(print); #undef BUILTIN_FUNCTION }; }