#include #include #include #include #include #include #include #include #include #include namespace hex::pl { class PatternData; PatternLanguage::PatternLanguage() { this->m_preprocessor = new Preprocessor(); this->m_lexer = new Lexer(); this->m_parser = new Parser(); this->m_validator = new Validator(); this->m_evaluator = new Evaluator(); this->m_preprocessor->addPragmaHandler("endian", [this](std::string value) { if (value == "big") { this->m_defaultEndian = std::endian::big; return true; } else if (value == "little") { this->m_defaultEndian = std::endian::little; return true; } else if (value == "native") { this->m_defaultEndian = std::endian::native; return true; } else return false; }); this->m_preprocessor->addPragmaHandler("eval_depth", [this](std::string value) { auto limit = strtol(value.c_str(), nullptr, 0); if (limit <= 0) return false; this->m_evalDepth = limit; return true; }); this->m_preprocessor->addPragmaHandler("array_limit", [this](std::string value) { auto limit = strtol(value.c_str(), nullptr, 0); if (limit <= 0) return false; this->m_arrayLimit = limit; return true; }); this->m_preprocessor->addPragmaHandler("base_address", [](std::string value) { auto baseAddress = strtoull(value.c_str(), nullptr, 0); ImHexApi::Provider::get()->setBaseAddress(baseAddress); return true; }); this->m_preprocessor->addDefaultPragmaHandlers(); } PatternLanguage::~PatternLanguage() { delete this->m_preprocessor; delete this->m_lexer; delete this->m_parser; delete this->m_validator; } std::optional> PatternLanguage::executeString(prv::Provider *provider, const std::string &string) { this->m_currError.reset(); this->m_evaluator->getConsole().clear(); this->m_evaluator->setProvider(provider); this->m_evalDepth = 32; this->m_arrayLimit = 0x1000; for (auto &node : this->m_currAST) delete node; this->m_currAST.clear(); auto preprocessedCode = this->m_preprocessor->preprocess(string); if (!preprocessedCode.has_value()) { this->m_currError = this->m_preprocessor->getError(); return { }; } this->m_evaluator->setDefaultEndian(this->m_defaultEndian); this->m_evaluator->setEvaluationDepth(this->m_evalDepth); this->m_evaluator->setArrayLimit(this->m_arrayLimit); auto tokens = this->m_lexer->lex(preprocessedCode.value()); if (!tokens.has_value()) { this->m_currError = this->m_lexer->getError(); return { }; } auto ast = this->m_parser->parse(tokens.value()); if (!ast.has_value()) { this->m_currError = this->m_parser->getError(); return { }; } this->m_currAST = ast.value(); auto patterns = this->m_evaluator->evaluate(ast.value()); if (!patterns.has_value()) { this->m_currError = this->m_evaluator->getConsole().getLastHardError(); return { }; } return patterns; } std::optional> PatternLanguage::executeFile(prv::Provider *provider, const std::string &path) { File file(path, File::Mode::Read); return this->executeString(provider, file.readString()); } const std::vector>& PatternLanguage::getConsoleLog() { return this->m_evaluator->getConsole().getLog(); } const std::optional>& PatternLanguage::getError() { return this->m_currError; } }