2021-09-08 15:18:24 +02:00
|
|
|
#include <hex/pattern_language/evaluator.hpp>
|
2022-02-27 23:25:39 +01:00
|
|
|
#include <hex/pattern_language/patterns/pattern.hpp>
|
|
|
|
|
|
|
|
#include <hex/pattern_language/ast/ast_node.hpp>
|
|
|
|
#include <hex/pattern_language/ast/ast_node_type_decl.hpp>
|
|
|
|
#include <hex/pattern_language/ast/ast_node_variable_decl.hpp>
|
|
|
|
#include <hex/pattern_language/ast/ast_node_function_call.hpp>
|
|
|
|
#include <hex/pattern_language/ast/ast_node_function_definition.hpp>
|
|
|
|
|
|
|
|
#include <hex/pattern_language/patterns/pattern_unsigned.hpp>
|
|
|
|
#include <hex/pattern_language/patterns/pattern_signed.hpp>
|
|
|
|
#include <hex/pattern_language/patterns/pattern_float.hpp>
|
|
|
|
#include <hex/pattern_language/patterns/pattern_boolean.hpp>
|
|
|
|
#include <hex/pattern_language/patterns/pattern_character.hpp>
|
|
|
|
#include <hex/pattern_language/patterns/pattern_string.hpp>
|
|
|
|
#include <hex/pattern_language/patterns/pattern_enum.hpp>
|
2020-11-19 11:36:52 +01:00
|
|
|
|
2022-03-03 12:11:47 +01:00
|
|
|
#include <hex/helpers/logger.hpp>
|
|
|
|
|
2021-09-08 15:18:24 +02:00
|
|
|
namespace hex::pl {
|
2020-12-06 21:40:57 +01:00
|
|
|
|
2022-01-30 15:18:45 +01:00
|
|
|
void Evaluator::createParameterPack(const std::string &name, const std::vector<Token::Literal> &values) {
|
|
|
|
this->getScope(0).parameterPack = ParameterPack {
|
|
|
|
name,
|
|
|
|
values
|
|
|
|
};
|
|
|
|
}
|
|
|
|
|
2021-12-18 22:56:36 +01:00
|
|
|
void Evaluator::createVariable(const std::string &name, ASTNode *type, const std::optional<Token::Literal> &value, bool outVariable) {
|
2021-09-21 21:29:18 +02:00
|
|
|
auto &variables = *this->getScope(0).scope;
|
|
|
|
for (auto &variable : variables) {
|
|
|
|
if (variable->getVariableName() == name) {
|
|
|
|
LogConsole::abortEvaluation(hex::format("variable with name '{}' already exists", name));
|
2021-04-21 10:17:42 +02:00
|
|
|
}
|
2021-01-05 14:42:08 +01:00
|
|
|
}
|
|
|
|
|
2022-02-27 23:25:39 +01:00
|
|
|
auto startOffset = this->dataOffset();
|
|
|
|
|
|
|
|
std::unique_ptr<Pattern> pattern;
|
2021-09-27 18:32:48 +02:00
|
|
|
this->dataOffset() = startOffset;
|
2021-01-05 14:42:08 +01:00
|
|
|
|
2022-02-21 20:00:54 +01:00
|
|
|
bool referenceType = false;
|
|
|
|
|
2022-03-03 13:34:05 +01:00
|
|
|
auto typePattern = type->createPatterns(this);
|
|
|
|
|
|
|
|
if (typePattern.empty()) {
|
2021-09-23 23:43:16 +02:00
|
|
|
// Handle auto variables
|
|
|
|
if (!value.has_value())
|
|
|
|
LogConsole::abortEvaluation("cannot determine type of auto variable", type);
|
|
|
|
|
2021-10-04 20:26:34 +02:00
|
|
|
if (std::get_if<u128>(&value.value()) != nullptr)
|
2022-02-27 23:25:39 +01:00
|
|
|
pattern = std::unique_ptr<Pattern>(new PatternUnsigned(this, 0, sizeof(u128)));
|
2022-01-22 22:37:52 +01:00
|
|
|
else if (std::get_if<i128>(&value.value()) != nullptr)
|
2022-02-27 23:25:39 +01:00
|
|
|
pattern = std::unique_ptr<Pattern>(new PatternSigned(this, 0, sizeof(i128)));
|
2021-10-04 20:26:34 +02:00
|
|
|
else if (std::get_if<double>(&value.value()) != nullptr)
|
2022-02-27 23:25:39 +01:00
|
|
|
pattern = std::unique_ptr<Pattern>(new PatternFloat(this, 0, sizeof(double)));
|
2021-10-04 20:26:34 +02:00
|
|
|
else if (std::get_if<bool>(&value.value()) != nullptr)
|
2022-02-27 23:25:39 +01:00
|
|
|
pattern = std::unique_ptr<Pattern>(new PatternBoolean(this, 0));
|
2021-10-04 20:26:34 +02:00
|
|
|
else if (std::get_if<char>(&value.value()) != nullptr)
|
2022-02-27 23:25:39 +01:00
|
|
|
pattern = std::unique_ptr<Pattern>(new PatternCharacter(this, 0));
|
2021-10-04 20:26:34 +02:00
|
|
|
else if (std::get_if<std::string>(&value.value()) != nullptr)
|
2022-02-27 23:25:39 +01:00
|
|
|
pattern = std::unique_ptr<Pattern>(new PatternString(this, 0, 1));
|
2022-03-03 12:11:47 +01:00
|
|
|
else if (auto patternValue = std::get_if<Pattern *>(&value.value()); patternValue != nullptr) {
|
2022-02-27 23:25:39 +01:00
|
|
|
pattern = (*patternValue)->clone();
|
2022-02-21 20:00:54 +01:00
|
|
|
referenceType = true;
|
|
|
|
} else
|
2022-01-30 15:18:45 +01:00
|
|
|
LogConsole::abortEvaluation("cannot determine type of auto variable", type);
|
2022-02-27 23:25:39 +01:00
|
|
|
} else {
|
2022-03-03 13:34:05 +01:00
|
|
|
pattern = std::move(typePattern.front());
|
2021-09-23 23:43:16 +02:00
|
|
|
}
|
|
|
|
|
2021-09-21 21:29:18 +02:00
|
|
|
pattern->setVariableName(name);
|
2021-01-09 21:47:11 +01:00
|
|
|
|
2022-02-21 20:00:54 +01:00
|
|
|
if (!referenceType) {
|
|
|
|
pattern->setOffset(this->getStack().size());
|
|
|
|
pattern->setLocal(true);
|
|
|
|
this->getStack().emplace_back();
|
|
|
|
}
|
|
|
|
|
2021-12-18 22:56:36 +01:00
|
|
|
if (outVariable)
|
|
|
|
this->m_outVariables[name] = pattern->getOffset();
|
2022-02-27 23:25:39 +01:00
|
|
|
|
|
|
|
variables.push_back(std::move(pattern));
|
2021-01-07 01:19:54 +01:00
|
|
|
}
|
|
|
|
|
2022-01-24 20:53:17 +01:00
|
|
|
void Evaluator::setVariable(const std::string &name, const Token::Literal &value) {
|
2022-02-27 23:25:39 +01:00
|
|
|
std::unique_ptr<Pattern> pattern = nullptr;
|
2021-01-07 01:19:54 +01:00
|
|
|
|
2021-10-18 22:04:54 +02:00
|
|
|
{
|
|
|
|
auto &variables = *this->getScope(0).scope;
|
|
|
|
for (auto &variable : variables) {
|
|
|
|
if (variable->getVariableName() == name) {
|
2022-02-27 23:25:39 +01:00
|
|
|
pattern = variable->clone();
|
2021-10-18 22:04:54 +02:00
|
|
|
break;
|
|
|
|
}
|
|
|
|
}
|
|
|
|
}
|
|
|
|
|
|
|
|
if (pattern == nullptr) {
|
|
|
|
auto &variables = *this->getGlobalScope().scope;
|
|
|
|
for (auto &variable : variables) {
|
|
|
|
if (variable->getVariableName() == name) {
|
|
|
|
if (!variable->isLocal())
|
|
|
|
LogConsole::abortEvaluation(hex::format("cannot modify global variable '{}' which has been placed in memory", name));
|
|
|
|
|
2022-02-27 23:25:39 +01:00
|
|
|
pattern = variable->clone();
|
2021-10-18 22:04:54 +02:00
|
|
|
break;
|
|
|
|
}
|
2021-01-07 01:19:54 +01:00
|
|
|
}
|
2021-06-20 21:22:31 +02:00
|
|
|
}
|
|
|
|
|
2021-09-21 21:29:18 +02:00
|
|
|
if (pattern == nullptr)
|
|
|
|
LogConsole::abortEvaluation(hex::format("no variable with name '{}' found", name));
|
2021-06-20 21:22:31 +02:00
|
|
|
|
2022-02-21 20:00:54 +01:00
|
|
|
if (!pattern->isLocal()) return;
|
|
|
|
|
2021-09-21 21:29:18 +02:00
|
|
|
Token::Literal castedLiteral = std::visit(overloaded {
|
2022-01-24 20:53:17 +01:00
|
|
|
[&](double &value) -> Token::Literal {
|
2022-02-27 23:25:39 +01:00
|
|
|
if (dynamic_cast<PatternUnsigned *>(pattern.get()))
|
2022-01-24 20:53:17 +01:00
|
|
|
return u128(value) & bitmask(pattern->getSize() * 8);
|
2022-02-27 23:25:39 +01:00
|
|
|
else if (dynamic_cast<PatternSigned *>(pattern.get()))
|
2022-01-24 20:53:17 +01:00
|
|
|
return i128(value) & bitmask(pattern->getSize() * 8);
|
2022-02-27 23:25:39 +01:00
|
|
|
else if (dynamic_cast<PatternFloat *>(pattern.get()))
|
2022-01-24 20:53:17 +01:00
|
|
|
return pattern->getSize() == sizeof(float) ? double(float(value)) : value;
|
|
|
|
else
|
|
|
|
LogConsole::abortEvaluation(hex::format("cannot cast type 'double' to type '{}'", pattern->getTypeName()));
|
|
|
|
},
|
|
|
|
[&](const std::string &value) -> Token::Literal {
|
2022-02-27 23:25:39 +01:00
|
|
|
if (dynamic_cast<PatternString *>(pattern.get()))
|
2022-01-24 20:53:17 +01:00
|
|
|
return value;
|
|
|
|
else
|
|
|
|
LogConsole::abortEvaluation(hex::format("cannot cast type 'string' to type '{}'", pattern->getTypeName()));
|
|
|
|
},
|
2022-03-03 12:11:47 +01:00
|
|
|
[&](Pattern *value) -> Token::Literal {
|
2022-01-24 20:53:17 +01:00
|
|
|
if (value->getTypeName() == pattern->getTypeName())
|
|
|
|
return value;
|
|
|
|
else
|
|
|
|
LogConsole::abortEvaluation(hex::format("cannot cast type '{}' to type '{}'", value->getTypeName(), pattern->getTypeName()));
|
|
|
|
},
|
|
|
|
[&](auto &&value) -> Token::Literal {
|
2022-02-27 23:25:39 +01:00
|
|
|
if (dynamic_cast<PatternUnsigned *>(pattern.get()) || dynamic_cast<PatternEnum *>(pattern.get()))
|
2022-01-24 20:53:17 +01:00
|
|
|
return u128(value) & bitmask(pattern->getSize() * 8);
|
2022-02-27 23:25:39 +01:00
|
|
|
else if (dynamic_cast<PatternSigned *>(pattern.get()))
|
2022-01-24 20:53:17 +01:00
|
|
|
return i128(value) & bitmask(pattern->getSize() * 8);
|
2022-02-27 23:25:39 +01:00
|
|
|
else if (dynamic_cast<PatternCharacter *>(pattern.get()))
|
2022-01-24 20:53:17 +01:00
|
|
|
return char(value);
|
2022-02-27 23:25:39 +01:00
|
|
|
else if (dynamic_cast<PatternBoolean *>(pattern.get()))
|
2022-01-24 20:53:17 +01:00
|
|
|
return bool(value);
|
2022-02-27 23:25:39 +01:00
|
|
|
else if (dynamic_cast<PatternFloat *>(pattern.get()))
|
2022-01-24 20:53:17 +01:00
|
|
|
return pattern->getSize() == sizeof(float) ? double(float(value)) : value;
|
|
|
|
else
|
|
|
|
LogConsole::abortEvaluation(hex::format("cannot cast integer literal to type '{}'", pattern->getTypeName()));
|
|
|
|
} },
|
2022-02-01 22:09:44 +01:00
|
|
|
value);
|
2021-06-21 00:21:38 +02:00
|
|
|
|
2021-09-25 00:03:32 +02:00
|
|
|
this->getStack()[pattern->getOffset()] = castedLiteral;
|
2021-06-20 23:46:13 +02:00
|
|
|
}
|
|
|
|
|
2022-02-27 23:25:39 +01:00
|
|
|
std::optional<std::vector<std::shared_ptr<Pattern>>> Evaluator::evaluate(const std::vector<std::shared_ptr<ASTNode>> &ast) {
|
2021-09-21 21:29:18 +02:00
|
|
|
this->m_stack.clear();
|
|
|
|
this->m_customFunctions.clear();
|
|
|
|
this->m_scopes.clear();
|
2022-02-06 00:18:04 +01:00
|
|
|
this->m_mainResult.reset();
|
2021-10-07 11:34:46 +02:00
|
|
|
this->m_aborted = false;
|
2021-01-21 17:49:30 +01:00
|
|
|
|
2021-12-19 12:32:15 +01:00
|
|
|
if (this->m_allowDangerousFunctions == DangerousFunctionPermission::Deny)
|
|
|
|
this->m_allowDangerousFunctions = DangerousFunctionPermission::Ask;
|
|
|
|
|
|
|
|
this->m_dangerousFunctionCalled = false;
|
|
|
|
|
2021-12-10 11:55:27 +01:00
|
|
|
ON_SCOPE_EXIT {
|
|
|
|
this->m_envVariables.clear();
|
|
|
|
};
|
|
|
|
|
2022-02-01 22:09:44 +01:00
|
|
|
this->dataOffset() = 0x00;
|
2021-10-02 15:22:38 +02:00
|
|
|
this->m_currPatternCount = 0;
|
2021-10-02 13:41:56 +02:00
|
|
|
|
2021-09-21 21:29:18 +02:00
|
|
|
this->m_customFunctionDefinitions.clear();
|
2021-01-21 17:49:30 +01:00
|
|
|
|
2022-02-27 23:25:39 +01:00
|
|
|
std::vector<std::shared_ptr<Pattern>> patterns;
|
2021-01-23 14:00:09 +01:00
|
|
|
|
Pattern Language rewrite (#111)
* Initial parser rewrite effort
Lexer and Token cleanup, Parser started over
* Greatly improved parser syntax
* Reimplemented using declarations and variable placement parsing
* Added back unions and structs
* Added enums as well as mathematical expressions (+, -, *, /, <<, >>, &, |, ^)
* Code style improvement
* Implemented arrays and fixed memory issues
* Fixed more memory issues in parser, reimplemented validator, evaluator and patterns
* Fixed builtin types, arrays and reimplemented strings
* Improved error messages
* Made character a distinct type, used for chars and strings
* Implemented padding, fixed arrays
* Added bitfields
* Added rvalue parsing, no evaluating yet
* Added .idea folder to gitignore
* Fixed build on MacOS
* Added custom implementation of integral concept if not available
* Rebased onto master
* Fixed array variable decl crash
* Added rvalues and dot syntax
* Lower case all pattern language error messages
* Fixed typo in variable name
* Fixed bug where preprocessor would not ignore commented out directives
* Reimplemented pointers
* Fixed rebase issues
2021-01-02 20:27:11 +01:00
|
|
|
try {
|
2021-12-30 14:44:46 +01:00
|
|
|
this->setCurrentControlFlowStatement(ControlFlowStatement::None);
|
2021-09-21 21:29:18 +02:00
|
|
|
pushScope(nullptr, patterns);
|
2022-02-14 20:44:43 +01:00
|
|
|
ON_SCOPE_EXIT {
|
|
|
|
popScope();
|
|
|
|
};
|
2021-10-08 18:47:05 +02:00
|
|
|
|
2022-02-27 23:25:39 +01:00
|
|
|
for (auto &node : ast) {
|
|
|
|
if (dynamic_cast<ASTNodeTypeDecl *>(node.get())) {
|
2022-01-24 20:53:17 +01:00
|
|
|
; // Don't create patterns from type declarations
|
2022-02-27 23:25:39 +01:00
|
|
|
} else if (dynamic_cast<ASTNodeFunctionCall *>(node.get())) {
|
|
|
|
(void)node->evaluate(this);
|
|
|
|
} else if (dynamic_cast<ASTNodeFunctionDefinition *>(node.get())) {
|
2021-09-21 21:29:18 +02:00
|
|
|
this->m_customFunctionDefinitions.push_back(node->evaluate(this));
|
2022-02-27 23:25:39 +01:00
|
|
|
} else if (auto varDeclNode = dynamic_cast<ASTNodeVariableDecl *>(node.get())) {
|
|
|
|
for (auto &pattern : node->createPatterns(this)) {
|
|
|
|
if (varDeclNode->getPlacementOffset() == nullptr) {
|
|
|
|
auto type = varDeclNode->getType()->evaluate(this);
|
|
|
|
|
|
|
|
auto &name = pattern->getVariableName();
|
|
|
|
this->createVariable(name, type.get(), std::nullopt, varDeclNode->isOutVariable());
|
|
|
|
|
|
|
|
if (varDeclNode->isInVariable() && this->m_inVariables.contains(name))
|
|
|
|
this->setVariable(name, this->m_inVariables[name]);
|
|
|
|
} else {
|
|
|
|
patterns.push_back(std::move(pattern));
|
|
|
|
}
|
2022-01-24 20:53:17 +01:00
|
|
|
}
|
2021-09-21 21:29:18 +02:00
|
|
|
} else {
|
|
|
|
auto newPatterns = node->createPatterns(this);
|
2022-02-27 23:25:39 +01:00
|
|
|
std::move(newPatterns.begin(), newPatterns.end(), std::back_inserter(patterns));
|
2021-04-13 20:40:21 +02:00
|
|
|
}
|
2021-10-08 18:47:05 +02:00
|
|
|
}
|
|
|
|
|
|
|
|
if (this->m_customFunctions.contains("main")) {
|
|
|
|
auto mainFunction = this->m_customFunctions["main"];
|
|
|
|
|
|
|
|
if (mainFunction.parameterCount > 0)
|
|
|
|
LogConsole::abortEvaluation("main function may not accept any arguments");
|
2021-04-13 20:40:21 +02:00
|
|
|
|
2022-02-06 00:18:04 +01:00
|
|
|
this->m_mainResult = mainFunction.func(this, {});
|
2021-04-13 20:40:21 +02:00
|
|
|
}
|
2022-01-31 14:37:12 +01:00
|
|
|
} catch (PatternLanguageError &error) {
|
|
|
|
this->m_console.log(LogConsole::Level::Error, error.what());
|
2021-04-13 20:40:21 +02:00
|
|
|
|
2022-01-31 14:37:12 +01:00
|
|
|
if (error.getLineNumber() != 0)
|
2021-09-21 21:29:18 +02:00
|
|
|
this->m_console.setHardError(error);
|
2020-11-19 11:36:52 +01:00
|
|
|
|
2021-09-21 21:29:18 +02:00
|
|
|
patterns.clear();
|
2021-01-08 15:03:53 +01:00
|
|
|
|
2021-10-02 15:22:38 +02:00
|
|
|
this->m_currPatternCount = 0;
|
|
|
|
|
2021-09-21 21:29:18 +02:00
|
|
|
return std::nullopt;
|
2020-11-19 11:36:52 +01:00
|
|
|
}
|
|
|
|
|
2021-10-18 22:04:54 +02:00
|
|
|
// Remove global local variables
|
2022-02-27 23:25:39 +01:00
|
|
|
std::erase_if(patterns, [](const std::shared_ptr<Pattern> &pattern) {
|
2021-10-18 22:04:54 +02:00
|
|
|
return pattern->isLocal();
|
|
|
|
});
|
|
|
|
|
2021-09-21 21:29:18 +02:00
|
|
|
return patterns;
|
2020-11-19 11:36:52 +01:00
|
|
|
}
|
|
|
|
|
2021-10-02 15:22:38 +02:00
|
|
|
void Evaluator::patternCreated() {
|
|
|
|
if (this->m_currPatternCount > this->m_patternLimit)
|
|
|
|
LogConsole::abortEvaluation(hex::format("exceeded maximum number of patterns: {}", this->m_patternLimit));
|
|
|
|
this->m_currPatternCount++;
|
|
|
|
}
|
|
|
|
|
|
|
|
void Evaluator::patternDestroyed() {
|
|
|
|
this->m_currPatternCount--;
|
|
|
|
}
|
|
|
|
|
2020-11-19 11:36:52 +01:00
|
|
|
}
|