2020-11-10 15:26:38 +01:00
# pragma once
2021-09-21 21:29:18 +02:00
# include <hex/pattern_language/token.hpp>
# include <hex/pattern_language/evaluator.hpp>
# include <hex/pattern_language/pattern_data.hpp>
2022-01-31 14:37:12 +01:00
# include <hex/helpers/concepts.hpp>
2020-11-10 15:26:38 +01:00
2021-10-09 23:07:58 +02:00
# include <algorithm>
2020-12-06 21:40:57 +01:00
# include <bit>
2021-12-19 12:32:15 +01:00
# include <chrono>
2020-11-10 15:26:38 +01:00
# include <optional>
2021-09-11 23:14:22 +02:00
# include <map>
2021-12-19 12:32:15 +01:00
# include <thread>
2021-04-21 10:17:42 +02:00
# include <variant>
2020-11-10 15:26:38 +01:00
# include <vector>
2021-09-08 15:18:24 +02:00
namespace hex : : pl {
2020-11-10 15:26:38 +01:00
2021-09-21 21:29:18 +02:00
class PatternData ;
2021-10-31 15:06:48 +01:00
class Evaluator ;
2022-01-31 14:37:12 +01:00
class ASTNode : public Cloneable < ASTNode > {
2021-10-31 15:06:48 +01:00
public :
constexpr ASTNode ( ) = default ;
constexpr virtual ~ ASTNode ( ) = default ;
constexpr ASTNode ( const ASTNode & ) = default ;
[[nodiscard]] constexpr u32 getLineNumber ( ) const { return this - > m_lineNumber ; }
[[maybe_unused]] constexpr void setLineNumber ( u32 lineNumber ) { this - > m_lineNumber = lineNumber ; }
[[nodiscard]] virtual ASTNode * evaluate ( Evaluator * evaluator ) const { return this - > clone ( ) ; }
[[nodiscard]] virtual std : : vector < PatternData * > createPatterns ( Evaluator * evaluator ) const { return { } ; }
2021-12-30 14:44:46 +01:00
using FunctionResult = std : : optional < Token : : Literal > ;
2022-01-31 14:37:12 +01:00
virtual FunctionResult execute ( Evaluator * evaluator ) const { LogConsole : : abortEvaluation ( " cannot execute non-function statement " , this ) ; }
2021-10-31 15:06:48 +01:00
private :
u32 m_lineNumber = 1 ;
} ;
2021-01-21 17:49:30 +01:00
2021-09-21 21:29:18 +02:00
class ASTNodeAttribute : public ASTNode {
2021-01-21 17:49:30 +01:00
public :
2021-09-21 21:29:18 +02:00
explicit ASTNodeAttribute ( std : : string attribute , std : : optional < std : : string > value = std : : nullopt )
2022-01-24 20:53:17 +01:00
: ASTNode ( ) , m_attribute ( std : : move ( attribute ) ) , m_value ( std : : move ( value ) ) { }
2021-01-21 17:49:30 +01:00
2021-09-21 21:29:18 +02:00
~ ASTNodeAttribute ( ) override = default ;
2021-01-21 17:49:30 +01:00
2021-09-21 21:29:18 +02:00
ASTNodeAttribute ( const ASTNodeAttribute & other ) : ASTNode ( other ) {
this - > m_attribute = other . m_attribute ;
2022-02-01 22:09:44 +01:00
this - > m_value = other . m_value ;
2021-01-21 17:49:30 +01:00
}
2022-01-24 20:53:17 +01:00
[[nodiscard]] ASTNode * clone ( ) const override {
2021-09-21 21:29:18 +02:00
return new ASTNodeAttribute ( * this ) ;
}
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
2022-01-24 20:53:17 +01:00
[[nodiscard]] const std : : string & getAttribute ( ) const {
2021-09-21 21:29:18 +02:00
return this - > m_attribute ;
}
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
2022-01-24 20:53:17 +01:00
[[nodiscard]] const std : : optional < std : : string > & getValue ( ) const {
2021-09-21 21:29:18 +02:00
return this - > m_value ;
}
2020-11-10 15:26:38 +01:00
private :
2021-09-21 21:29:18 +02:00
std : : string m_attribute ;
std : : optional < std : : string > m_value ;
2020-11-10 15:26:38 +01:00
} ;
2022-01-12 22:02:47 +01:00
class Attributable {
protected :
Attributable ( ) = default ;
~ Attributable ( ) {
for ( auto & attribute : this - > m_attributes )
delete attribute ;
}
Attributable ( const Attributable & other ) {
2022-01-31 14:37:12 +01:00
for ( auto & attribute : other . m_attributes ) {
2022-02-01 22:09:44 +01:00
if ( auto node = dynamic_cast < ASTNodeAttribute * > ( attribute - > clone ( ) ) )
2022-01-31 14:37:12 +01:00
this - > m_attributes . push_back ( node ) ;
else
delete node ;
}
2022-01-12 22:02:47 +01:00
}
public :
2022-02-07 20:08:19 +01:00
virtual void addAttribute ( ASTNodeAttribute * attribute ) {
2022-01-12 22:02:47 +01:00
this - > m_attributes . push_back ( attribute ) ;
}
[[nodiscard]] const auto & getAttributes ( ) const {
return this - > m_attributes ;
}
2022-02-08 17:50:42 +01:00
bool hasAttribute ( const std : : string & key , bool needsParameter ) const {
return std : : any_of ( this - > m_attributes . begin ( ) , this - > m_attributes . end ( ) , [ & ] ( ASTNodeAttribute * attribute ) {
if ( attribute - > getAttribute ( ) = = key ) {
if ( needsParameter & & ! attribute - > getValue ( ) . has_value ( ) )
LogConsole : : abortEvaluation ( hex : : format ( " attribute '{}' expected a parameter " ) , attribute ) ;
else if ( ! needsParameter & & attribute - > getValue ( ) . has_value ( ) )
LogConsole : : abortEvaluation ( hex : : format ( " attribute '{}' did not expect a parameter " ) , attribute ) ;
else
return true ;
}
return false ;
} ) ;
}
[[nodiscard]] std : : optional < std : : string > getAttributeValue ( const std : : string & key ) const {
auto attribute = std : : find_if ( this - > m_attributes . begin ( ) , this - > m_attributes . end ( ) , [ & ] ( ASTNodeAttribute * attribute ) {
return attribute - > getAttribute ( ) = = key ;
} ) ;
if ( attribute ! = this - > m_attributes . end ( ) )
return ( * attribute ) - > getValue ( ) ;
else
return std : : nullopt ;
}
2022-01-12 22:02:47 +01:00
private :
std : : vector < ASTNodeAttribute * > m_attributes ;
} ;
2022-02-19 18:03:07 +01:00
inline void applyTypeAttributes ( Evaluator * evaluator , const ASTNode * node , PatternData * pattern ) {
auto attributable = dynamic_cast < const Attributable * > ( node ) ;
if ( attributable = = nullptr )
LogConsole : : abortEvaluation ( " attribute cannot be applied here " , node ) ;
if ( attributable - > hasAttribute ( " inline " , false ) ) {
auto inlinable = dynamic_cast < Inlinable * > ( pattern ) ;
if ( inlinable = = nullptr )
LogConsole : : abortEvaluation ( " inline attribute can only be applied to nested types " , node ) ;
else
inlinable - > setInlined ( true ) ;
}
if ( auto value = attributable - > getAttributeValue ( " format " ) ; value ) {
auto functions = evaluator - > getCustomFunctions ( ) ;
if ( ! functions . contains ( * value ) )
LogConsole : : abortEvaluation ( hex : : format ( " cannot find formatter function '{}' " , * value ) , node ) ;
const auto & function = functions [ * value ] ;
if ( function . parameterCount ! = 1 )
LogConsole : : abortEvaluation ( " formatter function needs exactly one parameter " , node ) ;
pattern - > setFormatterFunction ( function ) ;
}
if ( auto value = attributable - > getAttributeValue ( " format_entries " ) ; value ) {
auto functions = evaluator - > getCustomFunctions ( ) ;
if ( ! functions . contains ( * value ) )
LogConsole : : abortEvaluation ( hex : : format ( " cannot find formatter function '{}' " , * value ) , node ) ;
const auto & function = functions [ * value ] ;
if ( function . parameterCount ! = 1 )
LogConsole : : abortEvaluation ( " formatter function needs exactly one parameter " , node ) ;
auto array = dynamic_cast < PatternDataDynamicArray * > ( pattern ) ;
if ( array = = nullptr )
LogConsole : : abortEvaluation ( " inline_array attribute can only be applied to array types " , node ) ;
for ( const auto & entry : array - > getEntries ( ) ) {
entry - > setFormatterFunction ( function ) ;
}
}
if ( auto value = attributable - > getAttributeValue ( " transform " ) ; value ) {
auto functions = evaluator - > getCustomFunctions ( ) ;
if ( ! functions . contains ( * value ) )
LogConsole : : abortEvaluation ( hex : : format ( " cannot find transform function '{}' " , * value ) , node ) ;
const auto & function = functions [ * value ] ;
if ( function . parameterCount ! = 1 )
LogConsole : : abortEvaluation ( " transform function needs exactly one parameter " , node ) ;
pattern - > setTransformFunction ( function ) ;
}
if ( auto value = attributable - > getAttributeValue ( " pointer_base " ) ; value ) {
auto functions = evaluator - > getCustomFunctions ( ) ;
if ( ! functions . contains ( * value ) )
LogConsole : : abortEvaluation ( hex : : format ( " cannot find pointer base function '{}' " , * value ) , node ) ;
const auto & function = functions [ * value ] ;
if ( function . parameterCount ! = 1 )
LogConsole : : abortEvaluation ( " pointer base function needs exactly one parameter " , node ) ;
if ( auto pointerPattern = dynamic_cast < PatternDataPointer * > ( pattern ) ) {
u128 pointerValue = pointerPattern - > getPointedAtAddress ( ) ;
auto result = function . func ( evaluator , { pointerValue } ) ;
if ( ! result . has_value ( ) )
LogConsole : : abortEvaluation ( " pointer base function did not return a value " , node ) ;
pointerPattern - > setPointedAtAddress ( Token : : literalToUnsigned ( result . value ( ) ) + pointerValue ) ;
} else {
LogConsole : : abortEvaluation ( " pointer_base attribute may only be applied to a pointer " ) ;
}
}
if ( attributable - > hasAttribute ( " hidden " , false ) ) {
pattern - > setHidden ( true ) ;
}
if ( ! pattern - > hasOverriddenColor ( ) ) {
if ( auto value = attributable - > getAttributeValue ( " color " ) ; value ) {
u32 color = strtoul ( value - > c_str ( ) , nullptr , 16 ) ;
pattern - > setColor ( hex : : changeEndianess ( color , std : : endian : : big ) > > 8 ) ;
} else if ( auto value = attributable - > hasAttribute ( " single_color " , false ) ; value ) {
pattern - > setColor ( ContentRegistry : : PatternLanguage : : getNextColor ( ) ) ;
}
}
}
inline void applyVariableAttributes ( Evaluator * evaluator , const ASTNode * node , PatternData * pattern ) {
auto attributable = dynamic_cast < const Attributable * > ( node ) ;
if ( attributable = = nullptr )
LogConsole : : abortEvaluation ( " attribute cannot be applied here " , node ) ;
auto endOffset = evaluator - > dataOffset ( ) ;
evaluator - > dataOffset ( ) = pattern - > getOffset ( ) ;
ON_SCOPE_EXIT { evaluator - > dataOffset ( ) = endOffset ; } ;
applyTypeAttributes ( evaluator , node , pattern ) ;
if ( auto value = attributable - > getAttributeValue ( " color " ) ; value ) {
u32 color = strtoul ( value - > c_str ( ) , nullptr , 16 ) ;
pattern - > setColor ( hex : : changeEndianess ( color , std : : endian : : big ) > > 8 ) ;
} else if ( auto value = attributable - > hasAttribute ( " single_color " , false ) ; value ) {
pattern - > setColor ( ContentRegistry : : PatternLanguage : : getNextColor ( ) ) ;
}
if ( auto value = attributable - > getAttributeValue ( " name " ) ; value ) {
pattern - > setDisplayName ( * value ) ;
}
if ( auto value = attributable - > getAttributeValue ( " comment " ) ; value ) {
pattern - > setComment ( * value ) ;
}
if ( attributable - > hasAttribute ( " no_unique_address " , false ) ) {
endOffset - = pattern - > getSize ( ) ;
}
}
2021-09-21 21:29:18 +02:00
class ASTNodeLiteral : public ASTNode {
2020-11-10 15:26:38 +01:00
public :
2022-01-16 14:20:52 +01:00
explicit ASTNodeLiteral ( Token : : Literal literal ) : ASTNode ( ) , m_literal ( std : : move ( literal ) ) { }
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
2022-01-24 20:53:17 +01:00
ASTNodeLiteral ( const ASTNodeLiteral & ) = default ;
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
2022-01-24 20:53:17 +01:00
[[nodiscard]] ASTNode * clone ( ) const override {
2021-09-21 21:29:18 +02:00
return new ASTNodeLiteral ( * this ) ;
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
}
2022-01-24 20:53:17 +01:00
[[nodiscard]] const auto & getValue ( ) const {
2021-06-20 21:22:31 +02:00
return this - > m_literal ;
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
}
2020-11-10 15:26:38 +01:00
private :
2021-09-21 21:29:18 +02:00
Token : : Literal m_literal ;
2020-11-10 15:26:38 +01:00
} ;
2021-09-21 21:29:18 +02:00
class ASTNodeMathematicalExpression : public ASTNode {
2022-01-24 20:53:17 +01:00
# define FLOAT_BIT_OPERATION(name) \
auto name ( hex : : floating_point auto left , auto right ) const { \
LogConsole : : abortEvaluation ( " invalid floating point operation " , this ) ; \
return 0 ; \
} \
auto name ( auto left , hex : : floating_point auto right ) const { \
LogConsole : : abortEvaluation ( " invalid floating point operation " , this ) ; \
return 0 ; \
} \
auto name ( hex : : floating_point auto left , hex : : floating_point auto right ) const { \
LogConsole : : abortEvaluation ( " invalid floating point operation " , this ) ; \
return 0 ; \
} \
auto name ( hex : : integral auto left , hex : : integral auto right ) const
2021-09-21 21:29:18 +02:00
FLOAT_BIT_OPERATION ( shiftLeft ) {
return left < < right ;
}
FLOAT_BIT_OPERATION ( shiftRight ) {
return left > > right ;
}
FLOAT_BIT_OPERATION ( bitAnd ) {
return left & right ;
}
FLOAT_BIT_OPERATION ( bitOr ) {
return left | right ;
}
FLOAT_BIT_OPERATION ( bitXor ) {
return left ^ right ;
}
FLOAT_BIT_OPERATION ( bitNot ) {
return ~ right ;
}
FLOAT_BIT_OPERATION ( modulus ) {
return left % right ;
}
2022-01-22 13:44:00 +01:00
# undef FLOAT_BIT_OPERATION
2020-11-10 15:26:38 +01:00
public :
2021-09-21 21:29:18 +02:00
ASTNodeMathematicalExpression ( ASTNode * left , ASTNode * right , Token : : Operator op )
2022-01-24 20:53:17 +01:00
: ASTNode ( ) , m_left ( left ) , m_right ( right ) , m_operator ( op ) { }
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
2021-09-21 21:29:18 +02:00
~ ASTNodeMathematicalExpression ( ) override {
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
delete this - > m_left ;
delete this - > m_right ;
}
2021-09-21 21:29:18 +02:00
ASTNodeMathematicalExpression ( const ASTNodeMathematicalExpression & other ) : ASTNode ( other ) {
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
this - > m_operator = other . m_operator ;
2022-02-01 22:09:44 +01:00
this - > m_left = other . m_left - > clone ( ) ;
this - > m_right = other . m_right - > clone ( ) ;
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
}
2022-01-24 20:53:17 +01:00
[[nodiscard]] ASTNode * clone ( ) const override {
2021-09-21 21:29:18 +02:00
return new ASTNodeMathematicalExpression ( * this ) ;
}
2022-01-24 20:53:17 +01:00
[[nodiscard]] ASTNode * evaluate ( Evaluator * evaluator ) const override {
2021-09-21 21:29:18 +02:00
if ( this - > getLeftOperand ( ) = = nullptr | | this - > getRightOperand ( ) = = nullptr )
LogConsole : : abortEvaluation ( " attempted to use void expression in mathematical expression " , this ) ;
2022-02-01 22:09:44 +01:00
auto * left = dynamic_cast < ASTNodeLiteral * > ( this - > getLeftOperand ( ) - > evaluate ( evaluator ) ) ;
2022-01-24 20:53:17 +01:00
auto * right = dynamic_cast < ASTNodeLiteral * > ( this - > getRightOperand ( ) - > evaluate ( evaluator ) ) ;
ON_SCOPE_EXIT {
delete left ;
delete right ;
} ;
2021-09-21 21:29:18 +02:00
return std : : visit ( overloaded {
2022-01-24 20:53:17 +01:00
// TODO: :notlikethis:
[ this ] ( u128 left , PatternData * const & right ) - > ASTNode * { LogConsole : : abortEvaluation ( " invalid operand used in mathematical expression " , this ) ; } ,
[ this ] ( i128 left , PatternData * const & right ) - > ASTNode * { LogConsole : : abortEvaluation ( " invalid operand used in mathematical expression " , this ) ; } ,
[ this ] ( double left , PatternData * const & right ) - > ASTNode * { LogConsole : : abortEvaluation ( " invalid operand used in mathematical expression " , this ) ; } ,
[ this ] ( char left , PatternData * const & right ) - > ASTNode * { LogConsole : : abortEvaluation ( " invalid operand used in mathematical expression " , this ) ; } ,
[ this ] ( bool left , PatternData * const & right ) - > ASTNode * { LogConsole : : abortEvaluation ( " invalid operand used in mathematical expression " , this ) ; } ,
2022-01-31 14:37:12 +01:00
[ this ] ( const std : : string & left , PatternData * const & right ) - > ASTNode * { LogConsole : : abortEvaluation ( " invalid operand used in mathematical expression " , this ) ; } ,
2022-01-24 20:53:17 +01:00
[ this ] ( PatternData * const & left , u128 right ) - > ASTNode * { LogConsole : : abortEvaluation ( " invalid operand used in mathematical expression " , this ) ; } ,
[ this ] ( PatternData * const & left , i128 right ) - > ASTNode * { LogConsole : : abortEvaluation ( " invalid operand used in mathematical expression " , this ) ; } ,
[ this ] ( PatternData * const & left , double right ) - > ASTNode * { LogConsole : : abortEvaluation ( " invalid operand used in mathematical expression " , this ) ; } ,
[ this ] ( PatternData * const & left , char right ) - > ASTNode * { LogConsole : : abortEvaluation ( " invalid operand used in mathematical expression " , this ) ; } ,
[ this ] ( PatternData * const & left , bool right ) - > ASTNode * { LogConsole : : abortEvaluation ( " invalid operand used in mathematical expression " , this ) ; } ,
2022-01-31 14:37:12 +01:00
[ this ] ( PatternData * const & left , const std : : string & right ) - > ASTNode * { LogConsole : : abortEvaluation ( " invalid operand used in mathematical expression " , this ) ; } ,
[ this ] ( PatternData * const & left , PatternData * const & right ) - > ASTNode * { LogConsole : : abortEvaluation ( " invalid operand used in mathematical expression " , this ) ; } ,
2022-01-24 20:53:17 +01:00
2022-01-31 14:37:12 +01:00
[ this ] ( auto & & left , const std : : string & right ) - > ASTNode * { LogConsole : : abortEvaluation ( " invalid operand used in mathematical expression " , this ) ; } ,
[ this ] ( const std : : string & left , auto & & right ) - > ASTNode * {
2022-01-24 20:53:17 +01:00
switch ( this - > getOperator ( ) ) {
2022-02-01 22:09:44 +01:00
case Token : : Operator : : Star :
{
std : : string result ;
for ( auto i = 0 ; i < right ; i + + )
result + = left ;
return new ASTNodeLiteral ( result ) ;
}
default :
LogConsole : : abortEvaluation ( " invalid operand used in mathematical expression " , this ) ;
2022-01-24 20:53:17 +01:00
}
} ,
2022-01-31 14:37:12 +01:00
[ this ] ( const std : : string & left , const std : : string & right ) - > ASTNode * {
2022-01-24 20:53:17 +01:00
switch ( this - > getOperator ( ) ) {
2022-02-01 22:09:44 +01:00
case Token : : Operator : : Plus :
return new ASTNodeLiteral ( left + right ) ;
case Token : : Operator : : BoolEquals :
return new ASTNodeLiteral ( left = = right ) ;
case Token : : Operator : : BoolNotEquals :
return new ASTNodeLiteral ( left ! = right ) ;
case Token : : Operator : : BoolGreaterThan :
return new ASTNodeLiteral ( left > right ) ;
case Token : : Operator : : BoolLessThan :
return new ASTNodeLiteral ( left < right ) ;
case Token : : Operator : : BoolGreaterThanOrEquals :
return new ASTNodeLiteral ( left > = right ) ;
case Token : : Operator : : BoolLessThanOrEquals :
return new ASTNodeLiteral ( left < = right ) ;
default :
LogConsole : : abortEvaluation ( " invalid operand used in mathematical expression " , this ) ;
2022-01-24 20:53:17 +01:00
}
} ,
2022-01-31 14:37:12 +01:00
[ this ] ( const std : : string & left , char right ) - > ASTNode * {
2022-01-24 20:53:17 +01:00
switch ( this - > getOperator ( ) ) {
2022-02-01 22:09:44 +01:00
case Token : : Operator : : Plus :
return new ASTNodeLiteral ( left + right ) ;
default :
LogConsole : : abortEvaluation ( " invalid operand used in mathematical expression " , this ) ;
2022-01-24 20:53:17 +01:00
}
} ,
2022-01-31 14:37:12 +01:00
[ this ] ( char left , const std : : string & right ) - > ASTNode * {
2022-01-24 20:53:17 +01:00
switch ( this - > getOperator ( ) ) {
2022-02-01 22:09:44 +01:00
case Token : : Operator : : Plus :
return new ASTNodeLiteral ( left + right ) ;
default :
LogConsole : : abortEvaluation ( " invalid operand used in mathematical expression " , this ) ;
2022-01-24 20:53:17 +01:00
}
} ,
[ this ] ( auto & & left , auto & & right ) - > ASTNode * {
switch ( this - > getOperator ( ) ) {
2022-02-01 22:09:44 +01:00
case Token : : Operator : : Plus :
return new ASTNodeLiteral ( left + right ) ;
case Token : : Operator : : Minus :
return new ASTNodeLiteral ( left - right ) ;
case Token : : Operator : : Star :
return new ASTNodeLiteral ( left * right ) ;
case Token : : Operator : : Slash :
if ( right = = 0 ) LogConsole : : abortEvaluation ( " division by zero! " , this ) ;
return new ASTNodeLiteral ( left / right ) ;
case Token : : Operator : : Percent :
if ( right = = 0 ) LogConsole : : abortEvaluation ( " division by zero! " , this ) ;
return new ASTNodeLiteral ( modulus ( left , right ) ) ;
case Token : : Operator : : ShiftLeft :
return new ASTNodeLiteral ( shiftLeft ( left , right ) ) ;
case Token : : Operator : : ShiftRight :
return new ASTNodeLiteral ( shiftRight ( left , right ) ) ;
case Token : : Operator : : BitAnd :
return new ASTNodeLiteral ( bitAnd ( left , right ) ) ;
case Token : : Operator : : BitXor :
return new ASTNodeLiteral ( bitXor ( left , right ) ) ;
case Token : : Operator : : BitOr :
return new ASTNodeLiteral ( bitOr ( left , right ) ) ;
case Token : : Operator : : BitNot :
return new ASTNodeLiteral ( bitNot ( left , right ) ) ;
case Token : : Operator : : BoolEquals :
return new ASTNodeLiteral ( bool ( left = = right ) ) ;
case Token : : Operator : : BoolNotEquals :
return new ASTNodeLiteral ( bool ( left ! = right ) ) ;
case Token : : Operator : : BoolGreaterThan :
return new ASTNodeLiteral ( bool ( left > right ) ) ;
case Token : : Operator : : BoolLessThan :
return new ASTNodeLiteral ( bool ( left < right ) ) ;
case Token : : Operator : : BoolGreaterThanOrEquals :
return new ASTNodeLiteral ( bool ( left > = right ) ) ;
case Token : : Operator : : BoolLessThanOrEquals :
return new ASTNodeLiteral ( bool ( left < = right ) ) ;
case Token : : Operator : : BoolAnd :
return new ASTNodeLiteral ( bool ( left & & right ) ) ;
case Token : : Operator : : BoolXor :
return new ASTNodeLiteral ( bool ( left & & ! right | | ! left & & right ) ) ;
case Token : : Operator : : BoolOr :
return new ASTNodeLiteral ( bool ( left | | right ) ) ;
case Token : : Operator : : BoolNot :
return new ASTNodeLiteral ( bool ( ! right ) ) ;
default :
LogConsole : : abortEvaluation ( " invalid operand used in mathematical expression " , this ) ;
2022-01-24 20:53:17 +01:00
}
} } ,
2022-02-01 22:09:44 +01:00
left - > getValue ( ) ,
right - > getValue ( ) ) ;
2021-09-21 21:29:18 +02:00
}
[[nodiscard]] ASTNode * getLeftOperand ( ) const { return this - > m_left ; }
[[nodiscard]] ASTNode * getRightOperand ( ) const { return this - > m_right ; }
[[nodiscard]] Token : : Operator getOperator ( ) const { return this - > m_operator ; }
2020-11-10 15:26:38 +01:00
private :
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
ASTNode * m_left , * m_right ;
Token : : Operator m_operator ;
2020-11-10 15:26:38 +01:00
} ;
2021-01-07 01:19:54 +01:00
class ASTNodeTernaryExpression : public ASTNode {
public :
ASTNodeTernaryExpression ( ASTNode * first , ASTNode * second , ASTNode * third , Token : : Operator op )
2022-01-24 20:53:17 +01:00
: ASTNode ( ) , m_first ( first ) , m_second ( second ) , m_third ( third ) , m_operator ( op ) { }
2021-01-07 01:19:54 +01:00
~ ASTNodeTernaryExpression ( ) override {
delete this - > m_first ;
delete this - > m_second ;
delete this - > m_third ;
}
ASTNodeTernaryExpression ( const ASTNodeTernaryExpression & other ) : ASTNode ( other ) {
this - > m_operator = other . m_operator ;
2022-02-01 22:09:44 +01:00
this - > m_first = other . m_first - > clone ( ) ;
this - > m_second = other . m_second - > clone ( ) ;
this - > m_third = other . m_third - > clone ( ) ;
2021-01-07 01:19:54 +01:00
}
2022-01-24 20:53:17 +01:00
[[nodiscard]] ASTNode * clone ( ) const override {
2021-01-07 01:19:54 +01:00
return new ASTNodeTernaryExpression ( * this ) ;
}
2022-01-24 20:53:17 +01:00
[[nodiscard]] ASTNode * evaluate ( Evaluator * evaluator ) const override {
2021-09-21 21:29:18 +02:00
if ( this - > getFirstOperand ( ) = = nullptr | | this - > getSecondOperand ( ) = = nullptr | | this - > getThirdOperand ( ) = = nullptr )
LogConsole : : abortEvaluation ( " attempted to use void expression in mathematical expression " , this ) ;
2022-02-01 22:09:44 +01:00
auto * first = dynamic_cast < ASTNodeLiteral * > ( this - > getFirstOperand ( ) - > evaluate ( evaluator ) ) ;
2022-01-24 20:53:17 +01:00
auto * second = dynamic_cast < ASTNodeLiteral * > ( this - > getSecondOperand ( ) - > evaluate ( evaluator ) ) ;
2022-02-01 22:09:44 +01:00
auto * third = dynamic_cast < ASTNodeLiteral * > ( this - > getThirdOperand ( ) - > evaluate ( evaluator ) ) ;
2022-01-24 20:53:17 +01:00
ON_SCOPE_EXIT {
delete first ;
delete second ;
delete third ;
} ;
2021-09-21 21:29:18 +02:00
auto condition = std : : visit ( overloaded {
2022-01-31 14:37:12 +01:00
[ ] ( const std : : string & value ) - > bool { return ! value . empty ( ) ; } ,
2022-01-24 20:53:17 +01:00
[ this ] ( PatternData * const & ) - > bool { LogConsole : : abortEvaluation ( " cannot cast custom type to bool " , this ) ; } ,
[ ] ( auto & & value ) - > bool { return bool ( value ) ; } } ,
2022-02-01 22:09:44 +01:00
first - > getValue ( ) ) ;
2021-09-21 21:29:18 +02:00
return std : : visit ( overloaded {
2022-01-24 20:53:17 +01:00
[ condition ] < typename T > ( const T & second , const T & third ) - > ASTNode * { return new ASTNodeLiteral ( condition ? second : third ) ; } ,
[ this ] ( auto & & second , auto & & third ) - > ASTNode * { LogConsole : : abortEvaluation ( " operands to ternary expression have different types " , this ) ; } } ,
2022-02-01 22:09:44 +01:00
second - > getValue ( ) ,
third - > getValue ( ) ) ;
2021-09-21 21:29:18 +02:00
}
[[nodiscard]] ASTNode * getFirstOperand ( ) const { return this - > m_first ; }
[[nodiscard]] ASTNode * getSecondOperand ( ) const { return this - > m_second ; }
[[nodiscard]] ASTNode * getThirdOperand ( ) const { return this - > m_third ; }
[[nodiscard]] Token : : Operator getOperator ( ) const { return this - > m_operator ; }
2021-01-07 01:19:54 +01:00
private :
ASTNode * m_first , * m_second , * m_third ;
Token : : Operator m_operator ;
} ;
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
class ASTNodeBuiltinType : public ASTNode {
public :
constexpr explicit ASTNodeBuiltinType ( Token : : ValueType type )
2022-01-24 20:53:17 +01:00
: ASTNode ( ) , m_type ( type ) { }
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
2022-01-24 20:53:17 +01:00
[[nodiscard]] constexpr const auto & getType ( ) const { return this - > m_type ; }
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
2022-01-24 20:53:17 +01:00
[[nodiscard]] ASTNode * clone ( ) const override {
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
return new ASTNodeBuiltinType ( * this ) ;
}
2022-01-24 20:53:17 +01:00
[[nodiscard]] std : : vector < PatternData * > createPatterns ( Evaluator * evaluator ) const override {
2021-09-21 21:29:18 +02:00
auto offset = evaluator - > dataOffset ( ) ;
2022-02-01 22:09:44 +01:00
auto size = Token : : getTypeSize ( this - > m_type ) ;
2021-09-21 21:29:18 +02:00
evaluator - > dataOffset ( ) + = size ;
PatternData * pattern ;
if ( Token : : isUnsigned ( this - > m_type ) )
2022-02-01 18:09:40 +01:00
pattern = new PatternDataUnsigned ( evaluator , offset , size ) ;
2021-09-21 21:29:18 +02:00
else if ( Token : : isSigned ( this - > m_type ) )
2022-02-01 18:09:40 +01:00
pattern = new PatternDataSigned ( evaluator , offset , size ) ;
2021-09-21 21:29:18 +02:00
else if ( Token : : isFloatingPoint ( this - > m_type ) )
2022-02-01 18:09:40 +01:00
pattern = new PatternDataFloat ( evaluator , offset , size ) ;
2021-09-21 21:29:18 +02:00
else if ( this - > m_type = = Token : : ValueType : : Boolean )
2022-02-01 18:09:40 +01:00
pattern = new PatternDataBoolean ( evaluator , offset ) ;
2021-09-21 21:29:18 +02:00
else if ( this - > m_type = = Token : : ValueType : : Character )
2022-02-01 18:09:40 +01:00
pattern = new PatternDataCharacter ( evaluator , offset ) ;
2021-09-21 21:29:18 +02:00
else if ( this - > m_type = = Token : : ValueType : : Character16 )
2022-02-01 18:09:40 +01:00
pattern = new PatternDataCharacter16 ( evaluator , offset ) ;
2021-09-21 21:29:18 +02:00
else if ( this - > m_type = = Token : : ValueType : : Padding )
2022-02-01 18:09:40 +01:00
pattern = new PatternDataPadding ( evaluator , offset , 1 ) ;
2021-09-21 21:29:18 +02:00
else if ( this - > m_type = = Token : : ValueType : : String )
2022-02-01 18:09:40 +01:00
pattern = new PatternDataString ( evaluator , offset , 1 ) ;
2021-09-23 23:43:16 +02:00
else if ( this - > m_type = = Token : : ValueType : : Auto )
return { nullptr } ;
2021-09-21 21:29:18 +02:00
else
LogConsole : : abortEvaluation ( " invalid built-in type " , this ) ;
pattern - > setTypeName ( Token : : getTypeName ( this - > m_type ) ) ;
return { pattern } ;
}
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
private :
const Token : : ValueType m_type ;
} ;
2022-01-24 20:53:17 +01:00
class ASTNodeTypeDecl : public ASTNode ,
public Attributable {
2020-11-10 15:26:38 +01:00
public :
2021-09-08 15:18:24 +02:00
ASTNodeTypeDecl ( std : : string name , ASTNode * type , std : : optional < std : : endian > endian = std : : nullopt )
2022-01-24 20:53:17 +01:00
: ASTNode ( ) , m_name ( std : : move ( name ) ) , m_type ( type ) , m_endian ( endian ) { }
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
2022-01-24 20:53:17 +01:00
ASTNodeTypeDecl ( const ASTNodeTypeDecl & other ) : ASTNode ( other ) , Attributable ( other ) {
2022-02-01 22:09:44 +01:00
this - > m_name = other . m_name ;
this - > m_type = other . m_type ;
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
this - > m_endian = other . m_endian ;
}
2022-01-24 20:53:17 +01:00
[[nodiscard]] ASTNode * clone ( ) const override {
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
return new ASTNodeTypeDecl ( * this ) ;
}
2021-08-25 17:07:01 +02:00
void setName ( const std : : string & name ) { this - > m_name = name ; }
2022-01-24 20:53:17 +01:00
[[nodiscard]] const std : : string & getName ( ) const { return this - > m_name ; }
[[nodiscard]] ASTNode * getType ( ) { return this - > m_type ; }
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
[[nodiscard]] std : : optional < std : : endian > getEndian ( ) const { return this - > m_endian ; }
2020-11-10 15:26:38 +01:00
2021-09-21 21:29:18 +02:00
[[nodiscard]] ASTNode * evaluate ( Evaluator * evaluator ) const override {
2022-01-12 22:02:47 +01:00
auto type = this - > m_type - > evaluate ( evaluator ) ;
2022-01-24 20:53:17 +01:00
if ( auto attributable = dynamic_cast < Attributable * > ( type ) ) {
2022-01-31 14:37:12 +01:00
for ( auto & attribute : this - > getAttributes ( ) ) {
2022-02-01 22:09:44 +01:00
if ( auto node = dynamic_cast < ASTNodeAttribute * > ( attribute - > clone ( ) ) )
2022-01-31 14:37:12 +01:00
attributable - > addAttribute ( node ) ;
else
delete node ;
}
2022-01-12 22:02:47 +01:00
}
return type ;
2021-09-21 21:29:18 +02:00
}
2022-01-24 20:53:17 +01:00
[[nodiscard]] std : : vector < PatternData * > createPatterns ( Evaluator * evaluator ) const override {
2021-09-21 21:29:18 +02:00
auto patterns = this - > m_type - > createPatterns ( evaluator ) ;
for ( auto & pattern : patterns ) {
2021-09-23 23:43:16 +02:00
if ( pattern = = nullptr )
continue ;
2021-09-21 21:29:18 +02:00
if ( ! this - > m_name . empty ( ) )
pattern - > setTypeName ( this - > m_name ) ;
2022-01-29 02:51:46 +01:00
if ( this - > m_endian . has_value ( ) )
pattern - > setEndian ( this - > m_endian . value ( ) ) ;
2022-02-19 18:03:07 +01:00
applyTypeAttributes ( evaluator , this , pattern ) ;
2021-09-21 21:29:18 +02:00
}
return patterns ;
}
2022-02-07 20:08:19 +01:00
void addAttribute ( ASTNodeAttribute * attribute ) override {
Attributable : : addAttribute ( attribute ) ;
if ( auto attributable = dynamic_cast < Attributable * > ( this - > m_type ) ; attributable ! = nullptr )
attributable - > addAttribute ( static_cast < ASTNodeAttribute * > ( attribute - > clone ( ) ) ) ;
}
2020-11-10 15:26:38 +01:00
private :
std : : string m_name ;
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
ASTNode * m_type ;
std : : optional < std : : endian > m_endian ;
2020-11-10 15:26:38 +01:00
} ;
2021-09-21 21:29:18 +02:00
class ASTNodeCast : public ASTNode {
public :
ASTNodeCast ( ASTNode * value , ASTNode * type ) : m_value ( value ) , m_type ( type ) { }
2022-01-31 14:37:12 +01:00
ASTNodeCast ( const ASTNodeCast & other ) : ASTNode ( other ) {
2021-09-21 21:29:18 +02:00
this - > m_value = other . m_value - > clone ( ) ;
2022-02-01 22:09:44 +01:00
this - > m_type = other . m_type - > clone ( ) ;
2021-09-21 21:29:18 +02:00
}
~ ASTNodeCast ( ) override {
delete this - > m_value ;
delete this - > m_type ;
}
2022-01-24 20:53:17 +01:00
[[nodiscard]] ASTNode * clone ( ) const override {
2021-09-21 21:29:18 +02:00
return new ASTNodeCast ( * this ) ;
}
2022-01-24 20:53:17 +01:00
[[nodiscard]] ASTNode * evaluate ( Evaluator * evaluator ) const override {
auto literal = dynamic_cast < ASTNodeLiteral * > ( this - > m_value - > evaluate ( evaluator ) ) ;
2022-02-01 22:09:44 +01:00
auto type = dynamic_cast < ASTNodeBuiltinType * > ( this - > m_type - > evaluate ( evaluator ) ) - > getType ( ) ;
2021-09-21 21:29:18 +02:00
2022-01-24 20:53:17 +01:00
auto startOffset = evaluator - > dataOffset ( ) ;
2021-12-03 16:06:40 +01:00
2021-09-21 21:29:18 +02:00
auto typePattern = this - > m_type - > createPatterns ( evaluator ) . front ( ) ;
2021-12-03 16:06:40 +01:00
ON_SCOPE_EXIT {
2022-01-24 20:53:17 +01:00
evaluator - > dataOffset ( ) = startOffset ;
delete typePattern ;
} ;
2021-09-21 21:29:18 +02:00
return std : : visit ( overloaded {
2022-01-24 20:53:17 +01:00
[ & , this ] ( PatternData * value ) - > ASTNode * { LogConsole : : abortEvaluation ( hex : : format ( " cannot cast custom type '{}' to '{}' " , value - > getTypeName ( ) , Token : : getTypeName ( type ) ) , this ) ; } ,
[ & , this ] ( const std : : string & ) - > ASTNode * { LogConsole : : abortEvaluation ( hex : : format ( " cannot cast string to '{}' " , Token : : getTypeName ( type ) ) , this ) ; } ,
[ & , this ] ( auto & & value ) - > ASTNode * {
auto endianAdjustedValue = hex : : changeEndianess ( value , typePattern - > getSize ( ) , typePattern - > getEndian ( ) ) ;
switch ( type ) {
2022-02-01 22:09:44 +01:00
case Token : : ValueType : : Unsigned8Bit :
return new ASTNodeLiteral ( u128 ( u8 ( endianAdjustedValue ) ) ) ;
case Token : : ValueType : : Unsigned16Bit :
return new ASTNodeLiteral ( u128 ( u16 ( endianAdjustedValue ) ) ) ;
case Token : : ValueType : : Unsigned32Bit :
return new ASTNodeLiteral ( u128 ( u32 ( endianAdjustedValue ) ) ) ;
case Token : : ValueType : : Unsigned64Bit :
return new ASTNodeLiteral ( u128 ( u64 ( endianAdjustedValue ) ) ) ;
case Token : : ValueType : : Unsigned128Bit :
return new ASTNodeLiteral ( u128 ( endianAdjustedValue ) ) ;
case Token : : ValueType : : Signed8Bit :
return new ASTNodeLiteral ( i128 ( i8 ( endianAdjustedValue ) ) ) ;
case Token : : ValueType : : Signed16Bit :
return new ASTNodeLiteral ( i128 ( i16 ( endianAdjustedValue ) ) ) ;
case Token : : ValueType : : Signed32Bit :
return new ASTNodeLiteral ( i128 ( i32 ( endianAdjustedValue ) ) ) ;
case Token : : ValueType : : Signed64Bit :
return new ASTNodeLiteral ( i128 ( i64 ( endianAdjustedValue ) ) ) ;
case Token : : ValueType : : Signed128Bit :
return new ASTNodeLiteral ( i128 ( endianAdjustedValue ) ) ;
case Token : : ValueType : : Float :
return new ASTNodeLiteral ( double ( float ( endianAdjustedValue ) ) ) ;
case Token : : ValueType : : Double :
return new ASTNodeLiteral ( double ( endianAdjustedValue ) ) ;
case Token : : ValueType : : Character :
return new ASTNodeLiteral ( char ( endianAdjustedValue ) ) ;
case Token : : ValueType : : Character16 :
return new ASTNodeLiteral ( u128 ( char16_t ( endianAdjustedValue ) ) ) ;
case Token : : ValueType : : Boolean :
return new ASTNodeLiteral ( bool ( endianAdjustedValue ) ) ;
case Token : : ValueType : : String :
{
std : : string string ( sizeof ( value ) , ' \x00 ' ) ;
std : : memcpy ( string . data ( ) , & value , string . size ( ) ) ;
hex : : trim ( string ) ;
if ( typePattern - > getEndian ( ) ! = std : : endian : : native )
std : : reverse ( string . begin ( ) , string . end ( ) ) ;
return new ASTNodeLiteral ( string ) ;
}
default :
LogConsole : : abortEvaluation ( hex : : format ( " cannot cast value to '{}' " , Token : : getTypeName ( type ) ) , this ) ;
2022-01-24 20:53:17 +01:00
}
} ,
} ,
2022-02-01 22:09:44 +01:00
literal - > getValue ( ) ) ;
2021-09-21 21:29:18 +02:00
}
private :
ASTNode * m_value ;
ASTNode * m_type ;
} ;
class ASTNodeWhileStatement : public ASTNode {
public :
2022-01-24 20:53:17 +01:00
explicit ASTNodeWhileStatement ( ASTNode * condition , std : : vector < ASTNode * > body , ASTNode * postExpression = nullptr )
: ASTNode ( ) , m_condition ( condition ) , m_body ( std : : move ( body ) ) , m_postExpression ( postExpression ) { }
2021-09-21 21:29:18 +02:00
~ ASTNodeWhileStatement ( ) override {
delete this - > m_condition ;
for ( auto & statement : this - > m_body )
delete statement ;
2021-12-30 14:44:46 +01:00
delete this - > m_postExpression ;
2021-09-21 21:29:18 +02:00
}
ASTNodeWhileStatement ( const ASTNodeWhileStatement & other ) : ASTNode ( other ) {
this - > m_condition = other . m_condition - > clone ( ) ;
2021-09-26 18:27:18 +02:00
for ( auto & statement : other . m_body )
this - > m_body . push_back ( statement - > clone ( ) ) ;
2021-12-30 14:44:46 +01:00
2021-12-30 18:31:24 +01:00
if ( other . m_postExpression ! = nullptr )
this - > m_postExpression = other . m_postExpression - > clone ( ) ;
else
this - > m_postExpression = nullptr ;
2021-09-21 21:29:18 +02:00
}
2022-01-24 20:53:17 +01:00
[[nodiscard]] ASTNode * clone ( ) const override {
2021-09-21 21:29:18 +02:00
return new ASTNodeWhileStatement ( * this ) ;
}
2022-01-24 20:53:17 +01:00
[[nodiscard]] ASTNode * getCondition ( ) {
2021-09-21 21:29:18 +02:00
return this - > m_condition ;
}
2022-01-24 20:53:17 +01:00
[[nodiscard]] const std : : vector < ASTNode * > & getBody ( ) {
2021-09-21 21:29:18 +02:00
return this - > m_body ;
}
2021-11-27 12:57:59 +01:00
FunctionResult execute ( Evaluator * evaluator ) const override {
2021-09-21 21:29:18 +02:00
2021-10-07 11:34:46 +02:00
u64 loopIterations = 0 ;
2021-09-21 21:29:18 +02:00
while ( evaluateCondition ( evaluator ) ) {
2021-10-07 11:34:46 +02:00
evaluator - > handleAbort ( ) ;
2022-02-01 22:09:44 +01:00
auto variables = * evaluator - > getScope ( 0 ) . scope ;
auto parameterPack = evaluator - > getScope ( 0 ) . parameterPack ;
2021-09-21 21:29:18 +02:00
u32 startVariableCount = variables . size ( ) ;
ON_SCOPE_EXIT {
2022-01-31 14:37:12 +01:00
ssize_t stackSize = evaluator - > getStack ( ) . size ( ) ;
2022-01-24 20:53:17 +01:00
for ( u32 i = startVariableCount ; i < variables . size ( ) ; i + + ) {
stackSize - - ;
delete variables [ i ] ;
}
if ( stackSize < 0 ) LogConsole : : abortEvaluation ( " stack pointer underflow! " , this ) ;
evaluator - > getStack ( ) . resize ( stackSize ) ;
} ;
2021-09-21 21:29:18 +02:00
evaluator - > pushScope ( nullptr , variables ) ;
2022-01-30 17:48:51 +01:00
evaluator - > getScope ( 0 ) . parameterPack = parameterPack ;
2022-02-14 22:44:43 +03:00
ON_SCOPE_EXIT {
evaluator - > popScope ( ) ;
} ;
2021-09-21 21:29:18 +02:00
2021-12-30 14:44:46 +01:00
auto ctrlFlow = ControlFlowStatement : : None ;
2021-09-21 21:29:18 +02:00
for ( auto & statement : this - > m_body ) {
2021-12-30 14:44:46 +01:00
auto result = statement - > execute ( evaluator ) ;
ctrlFlow = evaluator - > getCurrentControlFlowStatement ( ) ;
evaluator - > setCurrentControlFlowStatement ( ControlFlowStatement : : None ) ;
if ( ctrlFlow = = ControlFlowStatement : : Return )
return result ;
else if ( ctrlFlow ! = ControlFlowStatement : : None )
break ;
2021-09-21 21:29:18 +02:00
}
2021-10-07 11:34:46 +02:00
2021-12-30 14:44:46 +01:00
if ( this - > m_postExpression ! = nullptr )
this - > m_postExpression - > execute ( evaluator ) ;
2021-10-07 11:34:46 +02:00
loopIterations + + ;
if ( loopIterations > = evaluator - > getLoopLimit ( ) )
LogConsole : : abortEvaluation ( hex : : format ( " loop iterations exceeded limit of {} " , evaluator - > getLoopLimit ( ) ) , this ) ;
evaluator - > handleAbort ( ) ;
2021-12-30 14:44:46 +01:00
if ( ctrlFlow = = ControlFlowStatement : : Break )
break ;
else if ( ctrlFlow = = ControlFlowStatement : : Continue )
continue ;
2021-09-21 21:29:18 +02:00
}
2022-01-30 12:43:43 +01:00
return std : : nullopt ;
2021-09-21 21:29:18 +02:00
}
2022-01-24 20:53:17 +01:00
[[nodiscard]] bool evaluateCondition ( Evaluator * evaluator ) const {
auto literal = dynamic_cast < ASTNodeLiteral * > ( this - > m_condition - > evaluate ( evaluator ) ) ;
2021-09-21 21:29:18 +02:00
ON_SCOPE_EXIT { delete literal ; } ;
return std : : visit ( overloaded {
2022-01-31 14:37:12 +01:00
[ ] ( const std : : string & value ) - > bool { return ! value . empty ( ) ; } ,
2022-01-24 20:53:17 +01:00
[ this ] ( PatternData * const & ) - > bool { LogConsole : : abortEvaluation ( " cannot cast custom type to bool " , this ) ; } ,
[ ] ( auto & & value ) - > bool { return value ! = 0 ; } } ,
2022-02-01 22:09:44 +01:00
literal - > getValue ( ) ) ;
2021-09-21 21:29:18 +02:00
}
private :
ASTNode * m_condition ;
2022-01-24 20:53:17 +01:00
std : : vector < ASTNode * > m_body ;
2021-12-30 14:44:46 +01:00
ASTNode * m_postExpression ;
2021-09-21 21:29:18 +02:00
} ;
2022-01-24 20:53:17 +01:00
class ASTNodeVariableDecl : public ASTNode ,
public Attributable {
2020-11-20 21:29:28 +01:00
public :
2021-12-18 22:56:36 +01:00
ASTNodeVariableDecl ( std : : string name , ASTNode * type , ASTNode * placementOffset = nullptr , bool inVariable = false , bool outVariable = false )
2022-01-24 20:53:17 +01:00
: ASTNode ( ) , m_name ( std : : move ( name ) ) , m_type ( type ) , m_placementOffset ( placementOffset ) , m_inVariable ( inVariable ) , m_outVariable ( outVariable ) { }
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
2021-01-21 17:49:30 +01:00
ASTNodeVariableDecl ( const ASTNodeVariableDecl & other ) : ASTNode ( other ) , Attributable ( other ) {
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
this - > m_name = other . m_name ;
this - > m_type = other . m_type - > clone ( ) ;
if ( other . m_placementOffset ! = nullptr )
this - > m_placementOffset = other . m_placementOffset - > clone ( ) ;
else
this - > m_placementOffset = nullptr ;
2022-01-31 14:37:12 +01:00
2022-02-01 22:09:44 +01:00
this - > m_inVariable = other . m_inVariable ;
2022-01-31 14:37:12 +01:00
this - > m_outVariable = other . m_outVariable ;
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
}
~ ASTNodeVariableDecl ( ) override {
delete this - > m_type ;
2021-02-19 10:51:30 +01:00
delete this - > m_placementOffset ;
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
}
2022-01-24 20:53:17 +01:00
[[nodiscard]] ASTNode * clone ( ) const override {
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
return new ASTNodeVariableDecl ( * this ) ;
}
2022-01-24 20:53:17 +01:00
[[nodiscard]] const std : : string & getName ( ) const { return this - > m_name ; }
[[nodiscard]] constexpr ASTNode * getType ( ) const { return this - > m_type ; }
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
[[nodiscard]] constexpr auto getPlacementOffset ( ) const { return this - > m_placementOffset ; }
2020-11-20 21:29:28 +01:00
2021-12-18 22:56:36 +01:00
[[nodiscard]] constexpr bool isInVariable ( ) const { return this - > m_inVariable ; }
[[nodiscard]] constexpr bool isOutVariable ( ) const { return this - > m_outVariable ; }
2022-01-24 20:53:17 +01:00
[[nodiscard]] std : : vector < PatternData * > createPatterns ( Evaluator * evaluator ) const override {
2022-02-06 19:46:39 +01:00
u64 startOffset = evaluator - > dataOffset ( ) ;
2021-09-21 21:29:18 +02:00
if ( this - > m_placementOffset ! = nullptr ) {
auto offset = dynamic_cast < ASTNodeLiteral * > ( this - > m_placementOffset - > evaluate ( evaluator ) ) ;
ON_SCOPE_EXIT { delete offset ; } ;
evaluator - > dataOffset ( ) = std : : visit ( overloaded {
2022-01-31 14:37:12 +01:00
[ this ] ( const std : : string & ) - > u64 { LogConsole : : abortEvaluation ( " placement offset cannot be a string " , this ) ; } ,
2022-01-24 20:53:17 +01:00
[ this ] ( PatternData * const & ) - > u64 { LogConsole : : abortEvaluation ( " placement offset cannot be a custom type " , this ) ; } ,
[ ] ( auto & & offset ) - > u64 { return offset ; } } ,
2022-02-01 22:09:44 +01:00
offset - > getValue ( ) ) ;
2021-09-21 21:29:18 +02:00
}
auto pattern = this - > m_type - > createPatterns ( evaluator ) . front ( ) ;
pattern - > setVariableName ( this - > m_name ) ;
applyVariableAttributes ( evaluator , this , pattern ) ;
2022-02-09 08:55:51 +01:00
if ( this - > m_placementOffset ! = nullptr & & ! evaluator - > isGlobalScope ( ) ) {
2022-02-06 19:46:39 +01:00
evaluator - > dataOffset ( ) = startOffset ;
}
2021-09-21 21:29:18 +02:00
return { pattern } ;
}
2021-11-27 12:57:59 +01:00
FunctionResult execute ( Evaluator * evaluator ) const override {
2021-09-21 21:29:18 +02:00
evaluator - > createVariable ( this - > getName ( ) , this - > getType ( ) ) ;
2022-01-30 12:43:43 +01:00
return std : : nullopt ;
2021-09-21 21:29:18 +02:00
}
2020-11-20 21:29:28 +01:00
private :
std : : string m_name ;
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
ASTNode * m_type ;
ASTNode * m_placementOffset ;
2021-12-18 22:56:36 +01:00
2022-01-31 14:37:12 +01:00
bool m_inVariable = false , m_outVariable = false ;
2020-11-20 21:29:28 +01:00
} ;
2022-01-24 20:53:17 +01:00
class ASTNodeArrayVariableDecl : public ASTNode ,
public Attributable {
2020-11-20 20:26:19 +01:00
public :
2021-09-08 15:18:24 +02:00
ASTNodeArrayVariableDecl ( std : : string name , ASTNode * type , ASTNode * size , ASTNode * placementOffset = nullptr )
2022-01-24 20:53:17 +01:00
: ASTNode ( ) , m_name ( std : : move ( name ) ) , m_type ( type ) , m_size ( size ) , m_placementOffset ( placementOffset ) { }
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
2021-01-21 17:49:30 +01:00
ASTNodeArrayVariableDecl ( const ASTNodeArrayVariableDecl & other ) : ASTNode ( other ) , Attributable ( other ) {
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
this - > m_name = other . m_name ;
this - > m_type = other . m_type - > clone ( ) ;
2021-01-07 21:16:34 +01:00
if ( other . m_size ! = nullptr )
this - > m_size = other . m_size - > clone ( ) ;
else
this - > m_size = nullptr ;
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
if ( other . m_placementOffset ! = nullptr )
this - > m_placementOffset = other . m_placementOffset - > clone ( ) ;
else
this - > m_placementOffset = nullptr ;
}
~ ASTNodeArrayVariableDecl ( ) override {
delete this - > m_type ;
delete this - > m_size ;
2021-02-19 10:51:30 +01:00
delete this - > m_placementOffset ;
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
}
2022-01-24 20:53:17 +01:00
[[nodiscard]] ASTNode * clone ( ) const override {
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
return new ASTNodeArrayVariableDecl ( * this ) ;
}
2022-01-24 20:53:17 +01:00
[[nodiscard]] std : : vector < PatternData * > createPatterns ( Evaluator * evaluator ) const override {
2022-02-06 19:46:39 +01:00
auto startOffset = evaluator - > dataOffset ( ) ;
2021-09-21 21:29:18 +02:00
if ( this - > m_placementOffset ! = nullptr ) {
2022-01-24 20:53:17 +01:00
auto offset = dynamic_cast < ASTNodeLiteral * > ( this - > m_placementOffset - > evaluate ( evaluator ) ) ;
2021-09-21 21:29:18 +02:00
ON_SCOPE_EXIT { delete offset ; } ;
evaluator - > dataOffset ( ) = std : : visit ( overloaded {
2022-01-31 14:37:12 +01:00
[ this ] ( const std : : string & ) - > u64 { LogConsole : : abortEvaluation ( " placement offset cannot be a string " , this ) ; } ,
2022-01-24 20:53:17 +01:00
[ this ] ( PatternData * const & ) - > u64 { LogConsole : : abortEvaluation ( " placement offset cannot be a custom type " , this ) ; } ,
[ ] ( auto & & offset ) - > u64 { return offset ; } } ,
2022-02-01 22:09:44 +01:00
offset - > getValue ( ) ) ;
2021-09-21 21:29:18 +02:00
}
auto type = this - > m_type - > evaluate ( evaluator ) ;
ON_SCOPE_EXIT { delete type ; } ;
PatternData * pattern ;
2022-01-24 20:53:17 +01:00
if ( dynamic_cast < ASTNodeBuiltinType * > ( type ) )
2021-09-21 21:29:18 +02:00
pattern = createStaticArray ( evaluator ) ;
2022-01-24 20:53:17 +01:00
else if ( auto attributable = dynamic_cast < Attributable * > ( type ) ) {
2022-02-08 17:50:42 +01:00
bool isStaticType = attributable - > hasAttribute ( " static " , false ) ;
2021-09-21 21:29:18 +02:00
if ( isStaticType )
pattern = createStaticArray ( evaluator ) ;
else
pattern = createDynamicArray ( evaluator ) ;
} else {
LogConsole : : abortEvaluation ( " invalid type used in array " , this ) ;
}
applyVariableAttributes ( evaluator , this , pattern ) ;
2022-02-06 19:46:39 +01:00
2022-02-09 08:55:51 +01:00
if ( this - > m_placementOffset ! = nullptr & & ! evaluator - > isGlobalScope ( ) ) {
2022-02-06 19:46:39 +01:00
evaluator - > dataOffset ( ) = startOffset ;
}
2021-09-21 21:29:18 +02:00
return { pattern } ;
}
2022-01-24 20:53:17 +01:00
[[nodiscard]] const std : : string & getName ( ) const { return this - > m_name ; }
[[nodiscard]] constexpr ASTNode * getType ( ) const { return this - > m_type ; }
[[nodiscard]] constexpr ASTNode * getSize ( ) const { return this - > m_size ; }
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
[[nodiscard]] constexpr auto getPlacementOffset ( ) const { return this - > m_placementOffset ; }
2020-11-20 20:26:19 +01:00
private :
std : : string m_name ;
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
ASTNode * m_type ;
ASTNode * m_size ;
ASTNode * m_placementOffset ;
2021-09-21 21:29:18 +02:00
2022-01-24 20:53:17 +01:00
PatternData * createStaticArray ( Evaluator * evaluator ) const {
2021-09-21 21:29:18 +02:00
u64 startOffset = evaluator - > dataOffset ( ) ;
PatternData * templatePattern = this - > m_type - > createPatterns ( evaluator ) . front ( ) ;
ON_SCOPE_EXIT { delete templatePattern ; } ;
evaluator - > dataOffset ( ) = startOffset ;
2022-02-11 16:52:51 +01:00
i128 entryCount = 0 ;
2021-09-21 21:29:18 +02:00
if ( this - > m_size ! = nullptr ) {
auto sizeNode = this - > m_size - > evaluate ( evaluator ) ;
ON_SCOPE_EXIT { delete sizeNode ; } ;
2022-01-24 20:53:17 +01:00
if ( auto literal = dynamic_cast < ASTNodeLiteral * > ( sizeNode ) ) {
2021-09-21 21:29:18 +02:00
entryCount = std : : visit ( overloaded {
2022-01-31 14:37:12 +01:00
[ this ] ( const std : : string & ) - > u128 { LogConsole : : abortEvaluation ( " cannot use string to index array " , this ) ; } ,
2022-01-24 20:53:17 +01:00
[ this ] ( PatternData * ) - > u128 { LogConsole : : abortEvaluation ( " cannot use custom type to index array " , this ) ; } ,
[ ] ( auto & & size ) - > u128 { return size ; } } ,
2022-02-01 22:09:44 +01:00
literal - > getValue ( ) ) ;
2022-01-24 20:53:17 +01:00
} else if ( auto whileStatement = dynamic_cast < ASTNodeWhileStatement * > ( sizeNode ) ) {
2021-09-21 21:29:18 +02:00
while ( whileStatement - > evaluateCondition ( evaluator ) ) {
entryCount + + ;
evaluator - > dataOffset ( ) + = templatePattern - > getSize ( ) ;
2021-10-07 11:34:46 +02:00
evaluator - > handleAbort ( ) ;
2021-09-21 21:29:18 +02:00
}
}
2022-02-11 16:52:51 +01:00
if ( entryCount < 0 )
LogConsole : : abortEvaluation ( " array cannot have a negative size " , this ) ;
2021-09-21 21:29:18 +02:00
} else {
std : : vector < u8 > buffer ( templatePattern - > getSize ( ) ) ;
while ( true ) {
if ( evaluator - > dataOffset ( ) > = evaluator - > getProvider ( ) - > getActualSize ( ) - buffer . size ( ) )
LogConsole : : abortEvaluation ( " reached end of file before finding end of unsized array " , this ) ;
evaluator - > getProvider ( ) - > read ( evaluator - > dataOffset ( ) , buffer . data ( ) , buffer . size ( ) ) ;
evaluator - > dataOffset ( ) + = buffer . size ( ) ;
entryCount + + ;
bool reachedEnd = true ;
for ( u8 & byte : buffer ) {
if ( byte ! = 0x00 ) {
reachedEnd = false ;
break ;
}
}
if ( reachedEnd ) break ;
2021-10-07 11:34:46 +02:00
evaluator - > handleAbort ( ) ;
2021-09-21 21:29:18 +02:00
}
}
PatternData * outputPattern ;
2022-01-24 20:53:17 +01:00
if ( dynamic_cast < PatternDataPadding * > ( templatePattern ) ) {
2022-02-01 18:09:40 +01:00
outputPattern = new PatternDataPadding ( evaluator , startOffset , 0 ) ;
2022-01-24 20:53:17 +01:00
} else if ( dynamic_cast < PatternDataCharacter * > ( templatePattern ) ) {
2022-02-01 18:09:40 +01:00
outputPattern = new PatternDataString ( evaluator , startOffset , 0 ) ;
2022-01-24 20:53:17 +01:00
} else if ( dynamic_cast < PatternDataCharacter16 * > ( templatePattern ) ) {
2022-02-01 18:09:40 +01:00
outputPattern = new PatternDataString16 ( evaluator , startOffset , 0 ) ;
2021-09-21 21:29:18 +02:00
} else {
2022-02-01 18:09:40 +01:00
auto arrayPattern = new PatternDataStaticArray ( evaluator , startOffset , 0 ) ;
2021-09-21 21:29:18 +02:00
arrayPattern - > setEntries ( templatePattern - > clone ( ) , entryCount ) ;
outputPattern = arrayPattern ;
}
outputPattern - > setVariableName ( this - > m_name ) ;
outputPattern - > setEndian ( templatePattern - > getEndian ( ) ) ;
outputPattern - > setTypeName ( templatePattern - > getTypeName ( ) ) ;
outputPattern - > setSize ( templatePattern - > getSize ( ) * entryCount ) ;
evaluator - > dataOffset ( ) = startOffset + outputPattern - > getSize ( ) ;
return outputPattern ;
}
2022-01-24 20:53:17 +01:00
PatternData * createDynamicArray ( Evaluator * evaluator ) const {
2022-02-01 18:09:40 +01:00
auto arrayPattern = new PatternDataDynamicArray ( evaluator , evaluator - > dataOffset ( ) , 0 ) ;
2021-09-21 21:29:18 +02:00
arrayPattern - > setVariableName ( this - > m_name ) ;
std : : vector < PatternData * > entries ;
2021-10-02 15:22:38 +02:00
auto arrayCleanup = SCOPE_GUARD {
for ( auto entry : entries )
delete entry ;
} ;
2022-02-01 22:09:44 +01:00
size_t size = 0 ;
2021-12-30 14:44:46 +01:00
u64 entryIndex = 0 ;
2022-01-24 20:53:17 +01:00
auto addEntries = [ & ] ( const std : : vector < PatternData * > & patterns ) {
2022-01-22 13:44:00 +01:00
for ( auto pattern : patterns ) {
pattern - > setVariableName ( hex : : format ( " [{}] " , entryIndex ) ) ;
pattern - > setEndian ( arrayPattern - > getEndian ( ) ) ;
entries . push_back ( pattern ) ;
2021-12-30 14:44:46 +01:00
2022-01-22 13:44:00 +01:00
size + = pattern - > getSize ( ) ;
entryIndex + + ;
2021-12-30 14:44:46 +01:00
2022-01-22 13:44:00 +01:00
evaluator - > handleAbort ( ) ;
}
2021-12-30 14:44:46 +01:00
} ;
2021-09-21 21:29:18 +02:00
2022-01-22 13:44:00 +01:00
auto discardEntries = [ & ] ( u32 count ) {
for ( u32 i = 0 ; i < count ; i + + ) {
delete entries . back ( ) ;
entries . pop_back ( ) ;
entryIndex - - ;
}
2021-12-30 18:31:24 +01:00
} ;
2021-09-21 21:29:18 +02:00
if ( this - > m_size ! = nullptr ) {
auto sizeNode = this - > m_size - > evaluate ( evaluator ) ;
ON_SCOPE_EXIT { delete sizeNode ; } ;
2022-01-24 20:53:17 +01:00
if ( auto literal = dynamic_cast < ASTNodeLiteral * > ( sizeNode ) ) {
auto entryCount = std : : visit ( overloaded {
2022-01-31 14:37:12 +01:00
[ this ] ( const std : : string & ) - > u128 { LogConsole : : abortEvaluation ( " cannot use string to index array " , this ) ; } ,
2022-01-24 20:53:17 +01:00
[ this ] ( PatternData * ) - > u128 { LogConsole : : abortEvaluation ( " cannot use custom type to index array " , this ) ; } ,
[ ] ( auto & & size ) - > u128 { return size ; } } ,
2022-02-01 22:09:44 +01:00
literal - > getValue ( ) ) ;
2021-09-21 21:29:18 +02:00
2021-09-22 00:45:04 +02:00
auto limit = evaluator - > getArrayLimit ( ) ;
if ( entryCount > limit )
LogConsole : : abortEvaluation ( hex : : format ( " array grew past set limit of {} " , limit ) , this ) ;
2021-09-21 21:29:18 +02:00
for ( u64 i = 0 ; i < entryCount ; i + + ) {
2022-01-21 19:06:19 +01:00
evaluator - > setCurrentControlFlowStatement ( ControlFlowStatement : : None ) ;
2021-12-30 18:31:24 +01:00
auto patterns = this - > m_type - > createPatterns ( evaluator ) ;
2021-09-21 21:29:18 +02:00
2021-12-30 18:31:24 +01:00
if ( ! patterns . empty ( ) )
2022-01-22 13:44:00 +01:00
addEntries ( patterns ) ;
2021-12-30 18:31:24 +01:00
auto ctrlFlow = evaluator - > getCurrentControlFlowStatement ( ) ;
if ( ctrlFlow = = ControlFlowStatement : : Break )
break ;
else if ( ctrlFlow = = ControlFlowStatement : : Continue ) {
2022-01-22 13:44:00 +01:00
discardEntries ( patterns . size ( ) ) ;
2021-12-30 18:31:24 +01:00
continue ;
}
2021-09-21 21:29:18 +02:00
}
2022-01-24 20:53:17 +01:00
} else if ( auto whileStatement = dynamic_cast < ASTNodeWhileStatement * > ( sizeNode ) ) {
2021-09-21 21:29:18 +02:00
while ( whileStatement - > evaluateCondition ( evaluator ) ) {
2021-09-22 00:45:04 +02:00
auto limit = evaluator - > getArrayLimit ( ) ;
2021-12-30 14:44:46 +01:00
if ( entryIndex > limit )
2021-09-22 00:45:04 +02:00
LogConsole : : abortEvaluation ( hex : : format ( " array grew past set limit of {} " , limit ) , this ) ;
2022-01-21 19:06:19 +01:00
evaluator - > setCurrentControlFlowStatement ( ControlFlowStatement : : None ) ;
2021-12-30 18:31:24 +01:00
auto patterns = this - > m_type - > createPatterns ( evaluator ) ;
2021-09-21 21:29:18 +02:00
2021-12-30 18:31:24 +01:00
if ( ! patterns . empty ( ) )
2022-01-22 13:44:00 +01:00
addEntries ( patterns ) ;
2021-12-30 18:31:24 +01:00
auto ctrlFlow = evaluator - > getCurrentControlFlowStatement ( ) ;
if ( ctrlFlow = = ControlFlowStatement : : Break )
break ;
else if ( ctrlFlow = = ControlFlowStatement : : Continue ) {
2022-01-22 13:44:00 +01:00
discardEntries ( patterns . size ( ) ) ;
2021-12-30 18:31:24 +01:00
continue ;
}
2021-09-21 21:29:18 +02:00
}
}
} else {
while ( true ) {
2022-01-22 13:44:00 +01:00
bool reachedEnd = true ;
2022-02-01 22:09:44 +01:00
auto limit = evaluator - > getArrayLimit ( ) ;
2021-12-30 14:44:46 +01:00
if ( entryIndex > limit )
2021-09-22 00:45:04 +02:00
LogConsole : : abortEvaluation ( hex : : format ( " array grew past set limit of {} " , limit ) , this ) ;
2022-01-21 19:06:19 +01:00
evaluator - > setCurrentControlFlowStatement ( ControlFlowStatement : : None ) ;
2021-12-30 18:31:24 +01:00
auto patterns = this - > m_type - > createPatterns ( evaluator ) ;
2021-09-21 21:29:18 +02:00
2022-01-22 13:44:00 +01:00
for ( auto pattern : patterns ) {
2021-12-30 18:31:24 +01:00
std : : vector < u8 > buffer ( pattern - > getSize ( ) ) ;
2021-09-21 21:29:18 +02:00
2021-12-30 18:31:24 +01:00
if ( evaluator - > dataOffset ( ) > = evaluator - > getProvider ( ) - > getActualSize ( ) - buffer . size ( ) ) {
delete pattern ;
LogConsole : : abortEvaluation ( " reached end of file before finding end of unsized array " , this ) ;
}
2022-01-22 13:44:00 +01:00
addEntries ( { pattern } ) ;
2021-12-30 18:31:24 +01:00
auto ctrlFlow = evaluator - > getCurrentControlFlowStatement ( ) ;
2022-01-22 13:44:00 +01:00
if ( ctrlFlow = = ControlFlowStatement : : None )
2021-09-21 21:29:18 +02:00
break ;
2021-12-30 18:31:24 +01:00
evaluator - > getProvider ( ) - > read ( evaluator - > dataOffset ( ) - pattern - > getSize ( ) , buffer . data ( ) , buffer . size ( ) ) ;
2022-01-22 13:44:00 +01:00
reachedEnd = true ;
2021-12-30 18:31:24 +01:00
for ( u8 & byte : buffer ) {
if ( byte ! = 0x00 ) {
reachedEnd = false ;
break ;
}
}
if ( reachedEnd ) break ;
}
2022-01-22 13:44:00 +01:00
auto ctrlFlow = evaluator - > getCurrentControlFlowStatement ( ) ;
if ( ctrlFlow = = ControlFlowStatement : : Break )
break ;
else if ( ctrlFlow = = ControlFlowStatement : : Continue ) {
discardEntries ( 1 ) ;
continue ;
}
if ( reachedEnd ) break ;
2021-09-21 21:29:18 +02:00
}
}
arrayPattern - > setEntries ( entries ) ;
arrayPattern - > setSize ( size ) ;
2021-12-30 14:44:46 +01:00
if ( auto & entries = arrayPattern - > getEntries ( ) ; ! entries . empty ( ) )
arrayPattern - > setTypeName ( entries . front ( ) - > getTypeName ( ) ) ;
2021-10-02 15:22:38 +02:00
arrayCleanup . release ( ) ;
2021-09-21 21:29:18 +02:00
return arrayPattern ;
}
2020-11-20 20:26:19 +01:00
} ;
2022-01-24 20:53:17 +01:00
class ASTNodePointerVariableDecl : public ASTNode ,
public Attributable {
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
public :
2021-09-08 15:18:24 +02:00
ASTNodePointerVariableDecl ( std : : string name , ASTNode * type , ASTNode * sizeType , ASTNode * placementOffset = nullptr )
2022-01-24 20:53:17 +01:00
: ASTNode ( ) , m_name ( std : : move ( name ) ) , m_type ( type ) , m_sizeType ( sizeType ) , m_placementOffset ( placementOffset ) { }
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
2021-01-21 17:49:30 +01:00
ASTNodePointerVariableDecl ( const ASTNodePointerVariableDecl & other ) : ASTNode ( other ) , Attributable ( other ) {
2022-02-01 22:09:44 +01:00
this - > m_name = other . m_name ;
this - > m_type = other . m_type - > clone ( ) ;
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
this - > m_sizeType = other . m_sizeType - > clone ( ) ;
if ( other . m_placementOffset ! = nullptr )
this - > m_placementOffset = other . m_placementOffset - > clone ( ) ;
else
this - > m_placementOffset = nullptr ;
}
~ ASTNodePointerVariableDecl ( ) override {
delete this - > m_type ;
2021-02-19 10:51:30 +01:00
delete this - > m_sizeType ;
delete this - > m_placementOffset ;
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
}
2022-01-24 20:53:17 +01:00
[[nodiscard]] ASTNode * clone ( ) const override {
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
return new ASTNodePointerVariableDecl ( * this ) ;
}
2022-01-24 20:53:17 +01:00
[[nodiscard]] const std : : string & getName ( ) const { return this - > m_name ; }
[[nodiscard]] constexpr ASTNode * getType ( ) const { return this - > m_type ; }
[[nodiscard]] constexpr ASTNode * getSizeType ( ) const { return this - > m_sizeType ; }
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
[[nodiscard]] constexpr auto getPlacementOffset ( ) const { return this - > m_placementOffset ; }
2022-01-24 20:53:17 +01:00
[[nodiscard]] std : : vector < PatternData * > createPatterns ( Evaluator * evaluator ) const override {
2022-02-06 19:46:39 +01:00
auto startOffset = evaluator - > dataOffset ( ) ;
2021-09-21 21:29:18 +02:00
if ( this - > m_placementOffset ! = nullptr ) {
auto offset = dynamic_cast < ASTNodeLiteral * > ( this - > m_placementOffset - > evaluate ( evaluator ) ) ;
ON_SCOPE_EXIT { delete offset ; } ;
evaluator - > dataOffset ( ) = std : : visit ( overloaded {
2022-01-31 14:37:12 +01:00
[ this ] ( const std : : string & ) - > u64 { LogConsole : : abortEvaluation ( " placement offset cannot be a string " , this ) ; } ,
2022-01-24 20:53:17 +01:00
[ this ] ( PatternData * ) - > u64 { LogConsole : : abortEvaluation ( " placement offset cannot be a custom type " , this ) ; } ,
[ ] ( auto & & offset ) - > u64 { return u64 ( offset ) ; } } ,
2022-02-01 22:09:44 +01:00
offset - > getValue ( ) ) ;
2021-09-21 21:29:18 +02:00
}
2022-02-06 19:46:39 +01:00
auto pointerStartOffset = evaluator - > dataOffset ( ) ;
2021-09-25 18:45:23 +02:00
2021-09-21 21:29:18 +02:00
auto sizePattern = this - > m_sizeType - > createPatterns ( evaluator ) . front ( ) ;
ON_SCOPE_EXIT { delete sizePattern ; } ;
2022-02-06 19:46:39 +01:00
auto pattern = new PatternDataPointer ( evaluator , pointerStartOffset , sizePattern - > getSize ( ) ) ;
2021-09-27 20:16:23 +02:00
pattern - > setVariableName ( this - > m_name ) ;
2022-02-06 19:46:39 +01:00
auto pointerEndOffset = evaluator - > dataOffset ( ) ;
2021-09-25 18:45:23 +02:00
{
u128 pointerAddress = 0 ;
evaluator - > getProvider ( ) - > read ( pattern - > getOffset ( ) , & pointerAddress , pattern - > getSize ( ) ) ;
2021-10-31 15:20:19 +01:00
pointerAddress = hex : : changeEndianess ( pointerAddress , sizePattern - > getSize ( ) , sizePattern - > getEndian ( ) ) ;
2022-02-06 19:46:39 +01:00
evaluator - > dataOffset ( ) = pointerStartOffset ;
2022-01-10 00:40:34 +01:00
pattern - > setPointedAtAddress ( pointerAddress ) ;
applyVariableAttributes ( evaluator , this , pattern ) ;
evaluator - > dataOffset ( ) = pattern - > getPointedAtAddress ( ) ;
2021-12-05 21:54:09 +01:00
auto pointedAtPattern = this - > m_type - > createPatterns ( evaluator ) . front ( ) ;
2021-09-25 18:45:23 +02:00
pattern - > setPointedAtPattern ( pointedAtPattern ) ;
2021-10-31 15:20:19 +01:00
pattern - > setEndian ( sizePattern - > getEndian ( ) ) ;
2021-09-25 18:45:23 +02:00
}
2022-02-09 08:55:51 +01:00
if ( this - > m_placementOffset ! = nullptr & & ! evaluator - > isGlobalScope ( ) ) {
2022-02-06 19:46:39 +01:00
evaluator - > dataOffset ( ) = startOffset ;
} else {
evaluator - > dataOffset ( ) = pointerEndOffset ;
}
2021-09-27 18:32:48 +02:00
2021-09-21 21:29:18 +02:00
return { pattern } ;
}
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
private :
std : : string m_name ;
ASTNode * m_type ;
ASTNode * m_sizeType ;
ASTNode * m_placementOffset ;
} ;
2021-08-25 17:54:47 +02:00
class ASTNodeMultiVariableDecl : public ASTNode {
public :
2022-01-24 20:53:17 +01:00
explicit ASTNodeMultiVariableDecl ( std : : vector < ASTNode * > variables ) : m_variables ( std : : move ( variables ) ) { }
2021-08-25 17:54:47 +02:00
ASTNodeMultiVariableDecl ( const ASTNodeMultiVariableDecl & other ) : ASTNode ( other ) {
for ( auto & variable : other . m_variables )
this - > m_variables . push_back ( variable - > clone ( ) ) ;
}
~ ASTNodeMultiVariableDecl ( ) override {
for ( auto & variable : this - > m_variables )
delete variable ;
}
2022-01-24 20:53:17 +01:00
[[nodiscard]] ASTNode * clone ( ) const override {
2021-08-25 17:54:47 +02:00
return new ASTNodeMultiVariableDecl ( * this ) ;
}
2022-01-24 20:53:17 +01:00
[[nodiscard]] std : : vector < ASTNode * > getVariables ( ) {
2021-08-25 17:54:47 +02:00
return this - > m_variables ;
}
2022-01-24 20:53:17 +01:00
[[nodiscard]] std : : vector < PatternData * > createPatterns ( Evaluator * evaluator ) const override {
std : : vector < PatternData * > patterns ;
2021-09-21 21:29:18 +02:00
for ( auto & node : this - > m_variables ) {
auto newPatterns = node - > createPatterns ( evaluator ) ;
patterns . insert ( patterns . end ( ) , newPatterns . begin ( ) , newPatterns . end ( ) ) ;
}
return patterns ;
}
2021-11-27 12:57:59 +01:00
FunctionResult execute ( Evaluator * evaluator ) const override {
2021-09-21 21:29:18 +02:00
for ( auto & variable : this - > m_variables ) {
2022-01-24 20:53:17 +01:00
auto variableDecl = dynamic_cast < ASTNodeVariableDecl * > ( variable ) ;
2021-09-21 21:29:18 +02:00
evaluator - > createVariable ( variableDecl - > getName ( ) , variableDecl - > getType ( ) - > evaluate ( evaluator ) ) ;
}
2022-01-30 12:43:43 +01:00
return std : : nullopt ;
2021-09-21 21:29:18 +02:00
}
2021-08-25 17:54:47 +02:00
private :
2022-01-24 20:53:17 +01:00
std : : vector < ASTNode * > m_variables ;
2021-08-25 17:54:47 +02:00
} ;
2022-01-24 20:53:17 +01:00
class ASTNodeStruct : public ASTNode ,
public Attributable {
2020-11-10 15:26:38 +01:00
public :
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
ASTNodeStruct ( ) : ASTNode ( ) { }
2021-01-21 17:49:30 +01:00
ASTNodeStruct ( const ASTNodeStruct & other ) : ASTNode ( other ) , Attributable ( other ) {
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
for ( const auto & otherMember : other . getMembers ( ) )
this - > m_members . push_back ( otherMember - > clone ( ) ) ;
2021-09-24 00:47:34 +02:00
for ( const auto & otherInheritance : other . getInheritance ( ) )
this - > m_inheritance . push_back ( otherInheritance - > clone ( ) ) ;
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
}
~ ASTNodeStruct ( ) override {
for ( auto & member : this - > m_members )
delete member ;
2021-09-24 00:47:34 +02:00
for ( auto & inheritance : this - > m_inheritance )
delete inheritance ;
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
}
2020-11-10 15:26:38 +01:00
2022-01-24 20:53:17 +01:00
[[nodiscard]] ASTNode * clone ( ) const override {
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
return new ASTNodeStruct ( * this ) ;
}
2022-01-24 20:53:17 +01:00
[[nodiscard]] std : : vector < PatternData * > createPatterns ( Evaluator * evaluator ) const override {
2022-02-01 18:09:40 +01:00
auto pattern = new PatternDataStruct ( evaluator , evaluator - > dataOffset ( ) , 0 ) ;
2021-09-21 21:29:18 +02:00
u64 startOffset = evaluator - > dataOffset ( ) ;
2022-01-24 20:53:17 +01:00
std : : vector < PatternData * > memberPatterns ;
2021-10-02 15:22:38 +02:00
auto structCleanup = SCOPE_GUARD {
delete pattern ;
for ( auto member : memberPatterns )
delete member ;
} ;
2021-09-21 21:29:18 +02:00
evaluator - > pushScope ( pattern , memberPatterns ) ;
2022-02-14 22:44:43 +03:00
ON_SCOPE_EXIT {
evaluator - > popScope ( ) ;
} ;
2021-09-24 00:47:34 +02:00
for ( auto inheritance : this - > m_inheritance ) {
auto inheritancePatterns = inheritance - > createPatterns ( evaluator ) . front ( ) ;
ON_SCOPE_EXIT {
2022-01-24 20:53:17 +01:00
delete inheritancePatterns ;
} ;
2021-09-24 00:47:34 +02:00
2022-01-24 20:53:17 +01:00
if ( auto structPattern = dynamic_cast < PatternDataStruct * > ( inheritancePatterns ) ) {
2021-09-24 00:47:34 +02:00
for ( auto member : structPattern - > getMembers ( ) ) {
memberPatterns . push_back ( member - > clone ( ) ) ;
}
}
}
2021-09-21 21:29:18 +02:00
for ( auto member : this - > m_members ) {
for ( auto & memberPattern : member - > createPatterns ( evaluator ) ) {
memberPatterns . push_back ( memberPattern ) ;
}
}
2021-09-24 00:47:34 +02:00
2021-09-21 21:29:18 +02:00
pattern - > setMembers ( memberPatterns ) ;
pattern - > setSize ( evaluator - > dataOffset ( ) - startOffset ) ;
2021-10-02 15:22:38 +02:00
structCleanup . release ( ) ;
2022-02-19 18:03:07 +01:00
applyTypeAttributes ( evaluator , this , pattern ) ;
2022-02-18 13:31:44 +01:00
2021-09-21 21:29:18 +02:00
return { pattern } ;
}
2022-01-24 20:53:17 +01:00
[[nodiscard]] const std : : vector < ASTNode * > & getMembers ( ) const { return this - > m_members ; }
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
void addMember ( ASTNode * node ) { this - > m_members . push_back ( node ) ; }
2020-11-10 15:26:38 +01:00
2022-01-24 20:53:17 +01:00
[[nodiscard]] const std : : vector < ASTNode * > & getInheritance ( ) const { return this - > m_inheritance ; }
2021-09-24 00:47:34 +02:00
void addInheritance ( ASTNode * node ) { this - > m_inheritance . push_back ( node ) ; }
2020-11-10 15:26:38 +01:00
private :
2022-01-24 20:53:17 +01:00
std : : vector < ASTNode * > m_members ;
std : : vector < ASTNode * > m_inheritance ;
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
} ;
2022-01-24 20:53:17 +01:00
class ASTNodeUnion : public ASTNode ,
public Attributable {
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
public :
ASTNodeUnion ( ) : ASTNode ( ) { }
2021-01-21 17:49:30 +01:00
ASTNodeUnion ( const ASTNodeUnion & other ) : ASTNode ( other ) , Attributable ( other ) {
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
for ( const auto & otherMember : other . getMembers ( ) )
this - > m_members . push_back ( otherMember - > clone ( ) ) ;
}
~ ASTNodeUnion ( ) override {
for ( auto & member : this - > m_members )
delete member ;
}
2022-01-24 20:53:17 +01:00
[[nodiscard]] ASTNode * clone ( ) const override {
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
return new ASTNodeUnion ( * this ) ;
}
2022-01-24 20:53:17 +01:00
[[nodiscard]] std : : vector < PatternData * > createPatterns ( Evaluator * evaluator ) const override {
2022-02-01 18:09:40 +01:00
auto pattern = new PatternDataUnion ( evaluator , evaluator - > dataOffset ( ) , 0 ) ;
2021-09-21 21:29:18 +02:00
size_t size = 0 ;
2022-01-24 20:53:17 +01:00
std : : vector < PatternData * > memberPatterns ;
2021-09-21 21:29:18 +02:00
u64 startOffset = evaluator - > dataOffset ( ) ;
2021-10-02 15:22:38 +02:00
auto unionCleanup = SCOPE_GUARD {
delete pattern ;
for ( auto member : memberPatterns )
delete member ;
} ;
2021-09-21 21:29:18 +02:00
evaluator - > pushScope ( pattern , memberPatterns ) ;
2022-02-14 22:44:43 +03:00
ON_SCOPE_EXIT {
evaluator - > popScope ( ) ;
} ;
2021-09-21 21:29:18 +02:00
for ( auto member : this - > m_members ) {
for ( auto & memberPattern : member - > createPatterns ( evaluator ) ) {
memberPattern - > setOffset ( startOffset ) ;
memberPatterns . push_back ( memberPattern ) ;
size = std : : max ( memberPattern - > getSize ( ) , size ) ;
}
}
evaluator - > dataOffset ( ) = startOffset + size ;
pattern - > setMembers ( memberPatterns ) ;
pattern - > setSize ( size ) ;
2022-02-19 18:03:07 +01:00
applyTypeAttributes ( evaluator , this , pattern ) ;
2021-10-02 15:22:38 +02:00
unionCleanup . release ( ) ;
2021-09-21 21:29:18 +02:00
return { pattern } ;
}
2022-01-24 20:53:17 +01:00
[[nodiscard]] const std : : vector < ASTNode * > & getMembers ( ) const { return this - > m_members ; }
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
void addMember ( ASTNode * node ) { this - > m_members . push_back ( node ) ; }
private :
2022-01-24 20:53:17 +01:00
std : : vector < ASTNode * > m_members ;
2020-11-10 15:26:38 +01:00
} ;
2022-01-24 20:53:17 +01:00
class ASTNodeEnum : public ASTNode ,
public Attributable {
2020-11-14 14:40:21 +01:00
public :
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
explicit ASTNodeEnum ( ASTNode * underlyingType ) : ASTNode ( ) , m_underlyingType ( underlyingType ) { }
2021-01-21 17:49:30 +01:00
ASTNodeEnum ( const ASTNodeEnum & other ) : ASTNode ( other ) , Attributable ( other ) {
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
for ( const auto & [ name , entry ] : other . getEntries ( ) )
2021-09-21 21:29:18 +02:00
this - > m_entries . emplace ( name , entry - > clone ( ) ) ;
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
this - > m_underlyingType = other . m_underlyingType - > clone ( ) ;
}
~ ASTNodeEnum ( ) override {
for ( auto & [ name , expr ] : this - > m_entries )
delete expr ;
delete this - > m_underlyingType ;
}
2022-01-24 20:53:17 +01:00
[[nodiscard]] ASTNode * clone ( ) const override {
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
return new ASTNodeEnum ( * this ) ;
}
2022-01-24 20:53:17 +01:00
[[nodiscard]] std : : vector < PatternData * > createPatterns ( Evaluator * evaluator ) const override {
2022-02-01 22:09:44 +01:00
auto pattern = new PatternDataEnum ( evaluator , evaluator - > dataOffset ( ) , 0 ) ;
2021-10-02 15:22:38 +02:00
auto enumCleanup = SCOPE_GUARD { delete pattern ; } ;
2021-09-21 21:29:18 +02:00
std : : vector < std : : pair < Token : : Literal , std : : string > > enumEntries ;
for ( const auto & [ name , value ] : this - > m_entries ) {
2022-01-24 20:53:17 +01:00
auto literal = dynamic_cast < ASTNodeLiteral * > ( value - > evaluate ( evaluator ) ) ;
2021-09-21 21:29:18 +02:00
ON_SCOPE_EXIT { delete literal ; } ;
enumEntries . emplace_back ( literal - > getValue ( ) , name ) ;
}
pattern - > setEnumValues ( enumEntries ) ;
auto underlying = this - > m_underlyingType - > createPatterns ( evaluator ) . front ( ) ;
ON_SCOPE_EXIT { delete underlying ; } ;
pattern - > setSize ( underlying - > getSize ( ) ) ;
pattern - > setEndian ( underlying - > getEndian ( ) ) ;
2022-02-19 18:03:07 +01:00
applyTypeAttributes ( evaluator , this , pattern ) ;
2021-10-02 15:22:38 +02:00
enumCleanup . release ( ) ;
2021-09-21 21:29:18 +02:00
return { pattern } ;
}
2022-01-24 20:53:17 +01:00
[[nodiscard]] const std : : map < std : : string , ASTNode * > & getEntries ( ) const { return this - > m_entries ; }
void addEntry ( const std : : string & name , ASTNode * expression ) { this - > m_entries . insert ( { name , expression } ) ; }
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
2021-01-10 17:14:38 +01:00
[[nodiscard]] ASTNode * getUnderlyingType ( ) { return this - > m_underlyingType ; }
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
private :
2022-01-24 20:53:17 +01:00
std : : map < std : : string , ASTNode * > m_entries ;
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
ASTNode * m_underlyingType ;
} ;
2022-01-24 20:53:17 +01:00
class ASTNodeBitfield : public ASTNode ,
public Attributable {
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
public :
ASTNodeBitfield ( ) : ASTNode ( ) { }
2021-01-21 17:49:30 +01:00
ASTNodeBitfield ( const ASTNodeBitfield & other ) : ASTNode ( other ) , Attributable ( other ) {
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
for ( const auto & [ name , entry ] : other . getEntries ( ) )
this - > m_entries . emplace_back ( name , entry - > clone ( ) ) ;
}
~ ASTNodeBitfield ( ) override {
for ( auto & [ name , expr ] : this - > m_entries )
delete expr ;
}
2020-11-14 14:40:21 +01:00
2022-01-24 20:53:17 +01:00
[[nodiscard]] ASTNode * clone ( ) const override {
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
return new ASTNodeBitfield ( * this ) ;
}
2022-01-24 20:53:17 +01:00
[[nodiscard]] const std : : vector < std : : pair < std : : string , ASTNode * > > & getEntries ( ) const { return this - > m_entries ; }
void addEntry ( const std : : string & name , ASTNode * size ) { this - > m_entries . emplace_back ( name , size ) ; }
2020-11-14 14:40:21 +01:00
2022-01-24 20:53:17 +01:00
[[nodiscard]] std : : vector < PatternData * > createPatterns ( Evaluator * evaluator ) const override {
2022-02-01 18:09:40 +01:00
auto pattern = new PatternDataBitfield ( evaluator , evaluator - > dataOffset ( ) , 0 ) ;
2021-09-21 21:29:18 +02:00
size_t bitOffset = 0 ;
2022-01-24 20:53:17 +01:00
std : : vector < PatternData * > fields ;
2021-10-02 15:22:38 +02:00
auto bitfieldCleanup = SCOPE_GUARD {
delete pattern ;
for ( auto field : fields )
delete field ;
} ;
2022-02-07 20:08:19 +01:00
bool isLeftToRight = false ;
2022-02-08 17:50:42 +01:00
if ( this - > hasAttribute ( " left_to_right " , false ) )
2022-02-07 20:08:19 +01:00
isLeftToRight = true ;
2022-02-08 17:50:42 +01:00
else if ( this - > hasAttribute ( " right_to_left " , false ) )
2022-02-07 20:08:19 +01:00
isLeftToRight = false ;
2022-02-08 17:50:42 +01:00
auto entries = this - > m_entries ;
if ( isLeftToRight )
std : : reverse ( entries . begin ( ) , entries . end ( ) ) ;
2021-09-21 21:29:18 +02:00
evaluator - > pushScope ( pattern , fields ) ;
2022-02-14 22:44:43 +03:00
ON_SCOPE_EXIT {
evaluator - > popScope ( ) ;
} ;
2022-02-08 17:50:42 +01:00
for ( auto [ name , bitSizeNode ] : entries ) {
2021-09-21 21:29:18 +02:00
auto literal = bitSizeNode - > evaluate ( evaluator ) ;
ON_SCOPE_EXIT { delete literal ; } ;
u8 bitSize = std : : visit ( overloaded {
2022-01-31 14:37:12 +01:00
[ this ] ( const std : : string & ) - > u8 { LogConsole : : abortEvaluation ( " bitfield field size cannot be a string " , this ) ; } ,
2022-01-24 20:53:17 +01:00
[ this ] ( PatternData * ) - > u8 { LogConsole : : abortEvaluation ( " bitfield field size cannot be a custom type " , this ) ; } ,
[ ] ( auto & & offset ) - > u8 { return static_cast < u8 > ( offset ) ; } } ,
2022-02-01 22:09:44 +01:00
dynamic_cast < ASTNodeLiteral * > ( literal ) - > getValue ( ) ) ;
2021-09-21 21:29:18 +02:00
2021-10-14 20:28:13 +02:00
// If a field is named padding, it was created through a padding expression and only advances the bit position
if ( name ! = " padding " ) {
2022-02-01 18:09:40 +01:00
auto field = new PatternDataBitfieldField ( evaluator , evaluator - > dataOffset ( ) , bitOffset , bitSize , pattern ) ;
2021-10-14 20:28:13 +02:00
field - > setVariableName ( name ) ;
2022-02-07 20:08:19 +01:00
2022-02-08 17:50:42 +01:00
fields . push_back ( field ) ;
2021-10-14 20:28:13 +02:00
}
2021-09-21 21:29:18 +02:00
bitOffset + = bitSize ;
}
pattern - > setSize ( ( bitOffset + 7 ) / 8 ) ;
pattern - > setFields ( fields ) ;
evaluator - > dataOffset ( ) + = pattern - > getSize ( ) ;
2022-02-19 18:03:07 +01:00
applyTypeAttributes ( evaluator , this , pattern ) ;
2021-10-02 15:22:38 +02:00
bitfieldCleanup . release ( ) ;
2021-09-21 21:29:18 +02:00
return { pattern } ;
}
2020-11-14 14:40:21 +01:00
private :
2022-01-24 20:53:17 +01:00
std : : vector < std : : pair < std : : string , ASTNode * > > m_entries ;
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
} ;
2022-01-30 15:18:45 +01:00
class ASTNodeParameterPack : public ASTNode {
public :
2022-02-01 22:09:44 +01:00
explicit ASTNodeParameterPack ( std : : vector < Token : : Literal > values ) : m_values ( std : : move ( values ) ) { }
2022-01-30 15:18:45 +01:00
[[nodiscard]] ASTNode * clone ( ) const override {
return new ASTNodeParameterPack ( * this ) ;
}
2022-01-31 14:37:12 +01:00
[[nodiscard]] const std : : vector < Token : : Literal > & getValues ( ) const {
2022-01-30 15:18:45 +01:00
return this - > m_values ;
}
private :
std : : vector < Token : : Literal > m_values ;
} ;
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
class ASTNodeRValue : public ASTNode {
public :
2022-01-24 20:53:17 +01:00
using Path = std : : vector < std : : variant < std : : string , ASTNode * > > ;
2021-04-21 10:17:42 +02:00
explicit ASTNodeRValue ( Path path ) : ASTNode ( ) , m_path ( std : : move ( path ) ) { }
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
2022-02-06 18:04:33 +01:00
ASTNodeRValue ( const ASTNodeRValue & other ) {
for ( auto & part : other . m_path ) {
if ( auto stringPart = std : : get_if < std : : string > ( & part ) ; stringPart ! = nullptr )
this - > m_path . push_back ( * stringPart ) ;
else if ( auto nodePart = std : : get_if < ASTNode * > ( & part ) ; nodePart ! = nullptr )
this - > m_path . push_back ( ( * nodePart ) - > clone ( ) ) ;
}
}
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
2021-04-21 10:17:42 +02:00
~ ASTNodeRValue ( ) override {
for ( auto & part : this - > m_path ) {
2022-01-24 20:53:17 +01:00
if ( auto node = std : : get_if < ASTNode * > ( & part ) ; node ! = nullptr )
2021-04-21 10:17:42 +02:00
delete * node ;
}
}
2022-01-24 20:53:17 +01:00
[[nodiscard]] ASTNode * clone ( ) const override {
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
return new ASTNodeRValue ( * this ) ;
}
2022-01-24 20:53:17 +01:00
[[nodiscard]] const Path & getPath ( ) const {
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
return this - > m_path ;
}
2022-01-24 20:53:17 +01:00
[[nodiscard]] ASTNode * evaluate ( Evaluator * evaluator ) const override {
2021-09-21 21:29:18 +02:00
if ( this - > getPath ( ) . size ( ) = = 1 ) {
if ( auto name = std : : get_if < std : : string > ( & this - > getPath ( ) . front ( ) ) ; name ! = nullptr ) {
if ( * name = = " $ " ) return new ASTNodeLiteral ( u128 ( evaluator - > dataOffset ( ) ) ) ;
2022-01-30 15:18:45 +01:00
auto & parameterPack = evaluator - > getScope ( 0 ) . parameterPack ;
if ( parameterPack & & * name = = parameterPack - > name )
return new ASTNodeParameterPack ( parameterPack - > values ) ;
2021-09-21 21:29:18 +02:00
}
}
auto pattern = this - > createPatterns ( evaluator ) . front ( ) ;
ON_SCOPE_EXIT { delete pattern ; } ;
Token : : Literal literal ;
2022-01-24 20:53:17 +01:00
if ( dynamic_cast < PatternDataUnsigned * > ( pattern ) | | dynamic_cast < PatternDataEnum * > ( pattern ) ) {
2021-09-21 21:29:18 +02:00
u128 value = 0 ;
2022-01-11 19:35:28 +01:00
readVariable ( evaluator , value , pattern ) ;
2021-09-21 21:29:18 +02:00
literal = value ;
2022-01-24 20:53:17 +01:00
} else if ( dynamic_cast < PatternDataSigned * > ( pattern ) ) {
2022-01-22 22:37:52 +01:00
i128 value = 0 ;
2022-01-11 19:35:28 +01:00
readVariable ( evaluator , value , pattern ) ;
2022-02-01 22:09:44 +01:00
value = hex : : signExtend ( pattern - > getSize ( ) * 8 , value ) ;
2021-09-21 21:29:18 +02:00
literal = value ;
2022-01-24 20:53:17 +01:00
} else if ( dynamic_cast < PatternDataFloat * > ( pattern ) ) {
2021-09-21 21:29:18 +02:00
if ( pattern - > getSize ( ) = = sizeof ( u16 ) ) {
u16 value = 0 ;
2022-01-11 19:35:28 +01:00
readVariable ( evaluator , value , pattern ) ;
2021-09-21 21:29:18 +02:00
literal = double ( float16ToFloat32 ( value ) ) ;
} else if ( pattern - > getSize ( ) = = sizeof ( float ) ) {
float value = 0 ;
2022-01-11 19:35:28 +01:00
readVariable ( evaluator , value , pattern ) ;
2021-09-21 21:29:18 +02:00
literal = double ( value ) ;
} else if ( pattern - > getSize ( ) = = sizeof ( double ) ) {
double value = 0 ;
2022-01-11 19:35:28 +01:00
readVariable ( evaluator , value , pattern ) ;
2021-09-21 21:29:18 +02:00
literal = value ;
} else LogConsole : : abortEvaluation ( " invalid floating point type access " , this ) ;
2022-01-24 20:53:17 +01:00
} else if ( dynamic_cast < PatternDataCharacter * > ( pattern ) ) {
2021-09-21 21:29:18 +02:00
char value = 0 ;
2022-01-11 19:35:28 +01:00
readVariable ( evaluator , value , pattern ) ;
2021-09-21 21:29:18 +02:00
literal = value ;
2022-01-24 20:53:17 +01:00
} else if ( dynamic_cast < PatternDataBoolean * > ( pattern ) ) {
2021-09-21 21:29:18 +02:00
bool value = false ;
2022-01-11 19:35:28 +01:00
readVariable ( evaluator , value , pattern ) ;
2021-09-21 21:29:18 +02:00
literal = value ;
2022-01-24 20:53:17 +01:00
} else if ( dynamic_cast < PatternDataString * > ( pattern ) ) {
2021-09-21 21:29:18 +02:00
std : : string value ;
if ( pattern - > isLocal ( ) ) {
auto & literal = evaluator - > getStack ( ) [ pattern - > getOffset ( ) ] ;
std : : visit ( overloaded {
2022-01-24 20:53:17 +01:00
[ & ] ( char assignmentValue ) { if ( assignmentValue ! = 0x00 ) value = std : : string ( { assignmentValue } ) ; } ,
2022-02-01 18:09:40 +01:00
[ & ] ( std : : string assignmentValue ) { value = assignmentValue ; } ,
2022-01-24 20:53:17 +01:00
[ & , this ] ( PatternData * const & assignmentValue ) {
if ( ! dynamic_cast < PatternDataString * > ( assignmentValue ) & & ! dynamic_cast < PatternDataCharacter * > ( assignmentValue ) )
LogConsole : : abortEvaluation ( hex : : format ( " cannot assign '{}' to string " , pattern - > getTypeName ( ) ) , this ) ;
readVariable ( evaluator , value , assignmentValue ) ;
} ,
[ & , this ] ( auto & & assignmentValue ) { LogConsole : : abortEvaluation ( hex : : format ( " cannot assign '{}' to string " , pattern - > getTypeName ( ) ) , this ) ; } } ,
2022-02-01 22:09:44 +01:00
literal ) ;
2022-01-24 20:53:17 +01:00
} else {
2021-09-25 14:52:34 +02:00
value . resize ( pattern - > getSize ( ) ) ;
evaluator - > getProvider ( ) - > read ( pattern - > getOffset ( ) , value . data ( ) , value . size ( ) ) ;
2021-10-07 21:47:39 +02:00
value . erase ( std : : find ( value . begin ( ) , value . end ( ) , ' \0 ' ) , value . end ( ) ) ;
2021-09-25 14:52:34 +02:00
}
2021-09-21 21:29:18 +02:00
literal = value ;
2022-01-24 20:53:17 +01:00
} else if ( auto bitfieldFieldPattern = dynamic_cast < PatternDataBitfieldField * > ( pattern ) ) {
2021-09-21 21:29:18 +02:00
u64 value = 0 ;
2022-01-11 19:35:28 +01:00
readVariable ( evaluator , value , pattern ) ;
2021-09-21 21:29:18 +02:00
literal = u128 ( hex : : extract ( bitfieldFieldPattern - > getBitOffset ( ) + ( bitfieldFieldPattern - > getBitSize ( ) - 1 ) , bitfieldFieldPattern - > getBitOffset ( ) , value ) ) ;
} else {
literal = pattern - > clone ( ) ;
}
2022-01-31 14:37:12 +01:00
if ( auto transformFunc = pattern - > getTransformFunction ( ) ; transformFunc . has_value ( ) ) {
2021-09-27 18:32:48 +02:00
auto result = transformFunc - > func ( evaluator , { literal } ) ;
if ( ! result . has_value ( ) )
LogConsole : : abortEvaluation ( " transform function did not return a value " , this ) ;
literal = result . value ( ) ;
}
2021-09-21 21:29:18 +02:00
return new ASTNodeLiteral ( literal ) ;
}
2022-01-24 20:53:17 +01:00
[[nodiscard]] std : : vector < PatternData * > createPatterns ( Evaluator * evaluator ) const override {
std : : vector < PatternData * > searchScope ;
2021-10-11 22:01:15 +02:00
PatternData * currPattern = nullptr ;
2022-02-01 22:09:44 +01:00
i32 scopeIndex = 0 ;
2021-09-21 21:29:18 +02:00
2022-01-30 15:18:45 +01:00
2022-01-24 20:53:17 +01:00
if ( ! evaluator - > isGlobalScope ( ) ) {
2021-10-11 22:01:15 +02:00
auto globalScope = evaluator - > getGlobalScope ( ) . scope ;
std : : copy ( globalScope - > begin ( ) , globalScope - > end ( ) , std : : back_inserter ( searchScope ) ) ;
}
{
auto currScope = evaluator - > getScope ( scopeIndex ) . scope ;
std : : copy ( currScope - > begin ( ) , currScope - > end ( ) , std : : back_inserter ( searchScope ) ) ;
}
2021-09-21 21:29:18 +02:00
for ( const auto & part : this - > getPath ( ) ) {
if ( part . index ( ) = = 0 ) {
// Variable access
auto name = std : : get < std : : string > ( part ) ;
if ( name = = " parent " ) {
scopeIndex - - ;
2021-10-17 21:49:33 +02:00
if ( - scopeIndex > = evaluator - > getScopeCount ( ) )
LogConsole : : abortEvaluation ( " cannot access parent of global scope " , this ) ;
2022-02-01 22:09:44 +01:00
searchScope = * evaluator - > getScope ( scopeIndex ) . scope ;
2021-09-27 18:32:48 +02:00
auto currParent = evaluator - > getScope ( scopeIndex ) . parent ;
2021-10-17 21:49:33 +02:00
if ( currParent = = nullptr ) {
currPattern = nullptr ;
} else {
currPattern = currParent - > clone ( ) ;
}
2021-09-27 18:32:48 +02:00
continue ;
2021-09-21 21:29:18 +02:00
} else if ( name = = " this " ) {
searchScope = * evaluator - > getScope ( scopeIndex ) . scope ;
auto currParent = evaluator - > getScope ( 0 ) . parent ;
if ( currParent = = nullptr )
LogConsole : : abortEvaluation ( " invalid use of 'this' outside of struct-like type " , this ) ;
currPattern = currParent - > clone ( ) ;
continue ;
} else {
bool found = false ;
2021-10-09 23:07:58 +02:00
for ( auto iter = searchScope . crbegin ( ) ; iter ! = searchScope . crend ( ) ; + + iter ) {
if ( ( * iter ) - > getVariableName ( ) = = name ) {
auto newPattern = ( * iter ) - > clone ( ) ;
2021-09-21 21:29:18 +02:00
delete currPattern ;
currPattern = newPattern ;
2022-02-01 22:09:44 +01:00
found = true ;
2021-09-21 21:29:18 +02:00
break ;
}
}
if ( name = = " $ " )
LogConsole : : abortEvaluation ( " invalid use of placeholder operator in rvalue " ) ;
2022-01-30 15:18:45 +01:00
if ( ! found ) {
2021-09-21 21:29:18 +02:00
LogConsole : : abortEvaluation ( hex : : format ( " no variable named '{}' found " , name ) , this ) ;
2022-01-30 15:18:45 +01:00
}
2021-09-21 21:29:18 +02:00
}
} else {
// Array indexing
2022-01-24 20:53:17 +01:00
auto index = dynamic_cast < ASTNodeLiteral * > ( std : : get < ASTNode * > ( part ) - > evaluate ( evaluator ) ) ;
2021-09-21 21:29:18 +02:00
ON_SCOPE_EXIT { delete index ; } ;
std : : visit ( overloaded {
2022-01-31 14:37:12 +01:00
[ this ] ( const std : : string & ) { LogConsole : : abortEvaluation ( " cannot use string to index array " , this ) ; } ,
[ this ] ( PatternData * const & ) { LogConsole : : abortEvaluation ( " cannot use custom type to index array " , this ) ; } ,
2022-01-24 20:53:17 +01:00
[ & , this ] ( auto & & index ) {
if ( auto dynamicArrayPattern = dynamic_cast < PatternDataDynamicArray * > ( currPattern ) ) {
if ( index > = searchScope . size ( ) | | index < 0 )
LogConsole : : abortEvaluation ( " array index out of bounds " , this ) ;
auto newPattern = searchScope [ index ] - > clone ( ) ;
delete currPattern ;
currPattern = newPattern ;
} else if ( auto staticArrayPattern = dynamic_cast < PatternDataStaticArray * > ( currPattern ) ) {
if ( index > = staticArrayPattern - > getEntryCount ( ) | | index < 0 )
LogConsole : : abortEvaluation ( " array index out of bounds " , this ) ;
auto newPattern = searchScope . front ( ) - > clone ( ) ;
newPattern - > setOffset ( staticArrayPattern - > getOffset ( ) + index * staticArrayPattern - > getTemplate ( ) - > getSize ( ) ) ;
delete currPattern ;
currPattern = newPattern ;
}
} } ,
2022-02-01 22:09:44 +01:00
index - > getValue ( ) ) ;
2021-09-21 21:29:18 +02:00
}
2021-10-17 21:49:33 +02:00
if ( currPattern = = nullptr )
break ;
2022-01-24 20:53:17 +01:00
if ( auto pointerPattern = dynamic_cast < PatternDataPointer * > ( currPattern ) ) {
2021-09-21 21:29:18 +02:00
auto newPattern = pointerPattern - > getPointedAtPattern ( ) - > clone ( ) ;
delete currPattern ;
currPattern = newPattern ;
}
2022-01-10 20:02:18 +01:00
PatternData * indexPattern ;
2022-01-11 19:35:28 +01:00
if ( currPattern - > isLocal ( ) ) {
auto stackLiteral = evaluator - > getStack ( ) [ currPattern - > getOffset ( ) ] ;
2022-01-24 20:53:17 +01:00
if ( auto stackPattern = std : : get_if < PatternData * > ( & stackLiteral ) ; stackPattern ! = nullptr )
2022-01-11 19:35:28 +01:00
indexPattern = * stackPattern ;
else
return { currPattern } ;
2022-01-24 20:53:17 +01:00
} else
2022-01-10 20:02:18 +01:00
indexPattern = currPattern ;
2022-01-24 20:53:17 +01:00
if ( auto structPattern = dynamic_cast < PatternDataStruct * > ( indexPattern ) )
2021-09-21 21:29:18 +02:00
searchScope = structPattern - > getMembers ( ) ;
2022-01-24 20:53:17 +01:00
else if ( auto unionPattern = dynamic_cast < PatternDataUnion * > ( indexPattern ) )
2021-09-21 21:29:18 +02:00
searchScope = unionPattern - > getMembers ( ) ;
2022-01-24 20:53:17 +01:00
else if ( auto bitfieldPattern = dynamic_cast < PatternDataBitfield * > ( indexPattern ) )
2021-09-21 21:29:18 +02:00
searchScope = bitfieldPattern - > getFields ( ) ;
2022-01-24 20:53:17 +01:00
else if ( auto dynamicArrayPattern = dynamic_cast < PatternDataDynamicArray * > ( indexPattern ) )
2021-09-21 21:29:18 +02:00
searchScope = dynamicArrayPattern - > getEntries ( ) ;
2022-01-24 20:53:17 +01:00
else if ( auto staticArrayPattern = dynamic_cast < PatternDataStaticArray * > ( indexPattern ) )
2021-09-21 21:29:18 +02:00
searchScope = { staticArrayPattern - > getTemplate ( ) } ;
}
2021-10-17 21:49:33 +02:00
if ( currPattern = = nullptr )
LogConsole : : abortEvaluation ( " cannot reference global scope " , this ) ;
2021-09-21 21:29:18 +02:00
return { currPattern } ;
}
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
private :
2021-04-21 10:17:42 +02:00
Path m_path ;
2022-01-11 19:35:28 +01:00
void readVariable ( Evaluator * evaluator , auto & value , PatternData * variablePattern ) const {
constexpr bool isString = std : : same_as < std : : remove_cvref_t < decltype ( value ) > , std : : string > ;
if ( variablePattern - > isLocal ( ) ) {
auto & literal = evaluator - > getStack ( ) [ variablePattern - > getOffset ( ) ] ;
std : : visit ( overloaded {
2022-01-24 20:53:17 +01:00
[ & ] ( std : : string & assignmentValue ) {
if constexpr ( isString ) value = assignmentValue ;
} ,
[ & ] ( PatternData * assignmentValue ) { readVariable ( evaluator , value , assignmentValue ) ; } ,
[ & ] ( auto & & assignmentValue ) { value = assignmentValue ; } } ,
2022-02-01 22:09:44 +01:00
literal ) ;
2022-01-24 20:53:17 +01:00
} else {
2022-01-11 19:35:28 +01:00
if constexpr ( isString ) {
value . resize ( variablePattern - > getSize ( ) ) ;
evaluator - > getProvider ( ) - > read ( variablePattern - > getOffset ( ) , value . data ( ) , value . size ( ) ) ;
value . erase ( std : : find ( value . begin ( ) , value . end ( ) , ' \0 ' ) , value . end ( ) ) ;
} else {
evaluator - > getProvider ( ) - > read ( variablePattern - > getOffset ( ) , & value , variablePattern - > getSize ( ) ) ;
}
}
if constexpr ( ! isString )
value = hex : : changeEndianess ( value , variablePattern - > getSize ( ) , variablePattern - > getEndian ( ) ) ;
}
2020-11-14 14:40:21 +01:00
} ;
2021-01-05 14:42:08 +01:00
class ASTNodeScopeResolution : public ASTNode {
public :
2021-09-25 14:52:17 +02:00
explicit ASTNodeScopeResolution ( ASTNode * type , std : : string name ) : ASTNode ( ) , m_type ( type ) , m_name ( std : : move ( name ) ) { }
2021-01-05 14:42:08 +01:00
2022-01-31 14:37:12 +01:00
ASTNodeScopeResolution ( const ASTNodeScopeResolution & other ) : ASTNode ( other ) {
2021-09-25 14:52:17 +02:00
this - > m_type = other . m_type - > clone ( ) ;
this - > m_name = other . m_name ;
}
~ ASTNodeScopeResolution ( ) override {
delete this - > m_type ;
}
2021-01-05 14:42:08 +01:00
2022-01-24 20:53:17 +01:00
[[nodiscard]] ASTNode * clone ( ) const override {
2021-01-05 14:42:08 +01:00
return new ASTNodeScopeResolution ( * this ) ;
}
2022-01-24 20:53:17 +01:00
[[nodiscard]] ASTNode * evaluate ( Evaluator * evaluator ) const override {
2021-09-25 14:52:17 +02:00
auto type = this - > m_type - > evaluate ( evaluator ) ;
ON_SCOPE_EXIT { delete type ; } ;
2022-01-24 20:53:17 +01:00
if ( auto enumType = dynamic_cast < ASTNodeEnum * > ( type ) ) {
2021-09-25 14:52:17 +02:00
for ( auto & [ name , value ] : enumType - > getEntries ( ) ) {
if ( name = = this - > m_name )
return value - > evaluate ( evaluator ) ;
}
} else {
LogConsole : : abortEvaluation ( " invalid scope resolution. Cannot access this type " ) ;
}
LogConsole : : abortEvaluation ( hex : : format ( " could not find constant '{}' " , this - > m_name ) , this ) ;
2021-01-05 14:42:08 +01:00
}
private :
2021-09-25 14:52:17 +02:00
ASTNode * m_type ;
std : : string m_name ;
2021-01-05 14:42:08 +01:00
} ;
2021-01-07 00:02:51 +01:00
class ASTNodeConditionalStatement : public ASTNode {
public :
2022-01-24 20:53:17 +01:00
explicit ASTNodeConditionalStatement ( ASTNode * condition , std : : vector < ASTNode * > trueBody , std : : vector < ASTNode * > falseBody )
: ASTNode ( ) , m_condition ( condition ) , m_trueBody ( std : : move ( trueBody ) ) , m_falseBody ( std : : move ( falseBody ) ) { }
2021-01-07 00:02:51 +01:00
~ ASTNodeConditionalStatement ( ) override {
delete this - > m_condition ;
for ( auto & statement : this - > m_trueBody )
delete statement ;
for ( auto & statement : this - > m_falseBody )
delete statement ;
}
ASTNodeConditionalStatement ( const ASTNodeConditionalStatement & other ) : ASTNode ( other ) {
this - > m_condition = other . m_condition - > clone ( ) ;
for ( auto & statement : other . m_trueBody )
this - > m_trueBody . push_back ( statement - > clone ( ) ) ;
for ( auto & statement : other . m_falseBody )
this - > m_falseBody . push_back ( statement - > clone ( ) ) ;
}
2022-01-24 20:53:17 +01:00
[[nodiscard]] ASTNode * clone ( ) const override {
2021-01-07 00:02:51 +01:00
return new ASTNodeConditionalStatement ( * this ) ;
}
2022-01-24 20:53:17 +01:00
[[nodiscard]] std : : vector < PatternData * > createPatterns ( Evaluator * evaluator ) const override {
2021-09-26 02:22:50 +02:00
auto & scope = * evaluator - > getScope ( 0 ) . scope ;
2022-02-01 22:09:44 +01:00
auto & body = evaluateCondition ( evaluator ) ? this - > m_trueBody : this - > m_falseBody ;
2021-09-21 21:29:18 +02:00
for ( auto & node : body ) {
auto newPatterns = node - > createPatterns ( evaluator ) ;
2022-01-22 13:44:00 +01:00
for ( auto & pattern : newPatterns ) {
scope . push_back ( pattern ) ;
}
2021-09-21 21:29:18 +02:00
}
2022-01-24 20:53:17 +01:00
return { } ;
2021-09-21 21:29:18 +02:00
}
2022-01-24 20:53:17 +01:00
[[nodiscard]] ASTNode * getCondition ( ) {
2021-01-07 00:02:51 +01:00
return this - > m_condition ;
}
2021-11-27 12:57:59 +01:00
FunctionResult execute ( Evaluator * evaluator ) const override {
2021-09-21 21:29:18 +02:00
auto & body = evaluateCondition ( evaluator ) ? this - > m_trueBody : this - > m_falseBody ;
2022-02-01 22:09:44 +01:00
auto variables = * evaluator - > getScope ( 0 ) . scope ;
2022-01-30 17:48:51 +01:00
auto parameterPack = evaluator - > getScope ( 0 ) . parameterPack ;
2021-09-21 21:29:18 +02:00
u32 startVariableCount = variables . size ( ) ;
ON_SCOPE_EXIT {
2022-01-24 20:53:17 +01:00
i64 stackSize = evaluator - > getStack ( ) . size ( ) ;
for ( u32 i = startVariableCount ; i < variables . size ( ) ; i + + ) {
stackSize - - ;
delete variables [ i ] ;
}
if ( stackSize < 0 ) LogConsole : : abortEvaluation ( " stack pointer underflow! " , this ) ;
evaluator - > getStack ( ) . resize ( stackSize ) ;
} ;
2021-09-21 21:29:18 +02:00
evaluator - > pushScope ( nullptr , variables ) ;
2022-01-30 17:48:51 +01:00
evaluator - > getScope ( 0 ) . parameterPack = parameterPack ;
2022-02-14 22:44:43 +03:00
ON_SCOPE_EXIT {
evaluator - > popScope ( ) ;
} ;
2021-09-21 21:29:18 +02:00
for ( auto & statement : body ) {
2021-12-30 14:44:46 +01:00
auto result = statement - > execute ( evaluator ) ;
if ( auto ctrlStatement = evaluator - > getCurrentControlFlowStatement ( ) ; ctrlStatement ! = ControlFlowStatement : : None ) {
return result ;
2021-09-21 21:29:18 +02:00
}
}
2021-06-17 23:13:58 +02:00
2022-01-30 12:43:43 +01:00
return std : : nullopt ;
2021-06-17 23:13:58 +02:00
}
2021-09-21 21:29:18 +02:00
private :
2022-01-24 20:53:17 +01:00
[[nodiscard]] bool evaluateCondition ( Evaluator * evaluator ) const {
auto literal = dynamic_cast < ASTNodeLiteral * > ( this - > m_condition - > evaluate ( evaluator ) ) ;
2021-09-21 21:29:18 +02:00
ON_SCOPE_EXIT { delete literal ; } ;
2021-06-17 23:13:58 +02:00
2021-09-21 21:29:18 +02:00
return std : : visit ( overloaded {
2022-01-31 14:37:12 +01:00
[ ] ( const std : : string & value ) - > bool { return ! value . empty ( ) ; } ,
2022-01-24 20:53:17 +01:00
[ this ] ( PatternData * const & ) - > bool { LogConsole : : abortEvaluation ( " cannot cast custom type to bool " , this ) ; } ,
[ ] ( auto & & value ) - > bool { return value ! = 0 ; } } ,
2022-02-01 22:09:44 +01:00
literal - > getValue ( ) ) ;
2021-06-17 23:13:58 +02:00
}
ASTNode * m_condition ;
2022-01-24 20:53:17 +01:00
std : : vector < ASTNode * > m_trueBody , m_falseBody ;
2021-06-17 23:13:58 +02:00
} ;
2021-01-07 15:37:37 +01:00
class ASTNodeFunctionCall : public ASTNode {
public :
2022-01-24 20:53:17 +01:00
explicit ASTNodeFunctionCall ( std : : string functionName , std : : vector < ASTNode * > params )
: ASTNode ( ) , m_functionName ( std : : move ( functionName ) ) , m_params ( std : : move ( params ) ) { }
2021-01-07 15:37:37 +01:00
~ ASTNodeFunctionCall ( ) override {
for ( auto & param : this - > m_params )
delete param ;
}
ASTNodeFunctionCall ( const ASTNodeFunctionCall & other ) : ASTNode ( other ) {
this - > m_functionName = other . m_functionName ;
for ( auto & param : other . m_params )
this - > m_params . push_back ( param - > clone ( ) ) ;
}
2022-01-24 20:53:17 +01:00
[[nodiscard]] ASTNode * clone ( ) const override {
2021-01-07 15:37:37 +01:00
return new ASTNodeFunctionCall ( * this ) ;
}
2022-01-24 20:53:17 +01:00
[[nodiscard]] const std : : string & getFunctionName ( ) {
2021-01-07 15:37:37 +01:00
return this - > m_functionName ;
}
2022-01-24 20:53:17 +01:00
[[nodiscard]] const std : : vector < ASTNode * > & getParams ( ) const {
2021-01-07 15:37:37 +01:00
return this - > m_params ;
}
2022-01-24 20:53:17 +01:00
[[nodiscard]] std : : vector < PatternData * > createPatterns ( Evaluator * evaluator ) const override {
2021-11-27 12:57:59 +01:00
this - > execute ( evaluator ) ;
2022-01-24 20:53:17 +01:00
return { } ;
2021-11-27 12:57:59 +01:00
}
2022-01-24 20:53:17 +01:00
[[nodiscard]] ASTNode * evaluate ( Evaluator * evaluator ) const override {
2021-09-21 21:29:18 +02:00
std : : vector < Token : : Literal > evaluatedParams ;
for ( auto param : this - > m_params ) {
auto expression = param - > evaluate ( evaluator ) ;
ON_SCOPE_EXIT { delete expression ; } ;
2021-01-07 15:37:37 +01:00
2022-01-30 15:18:45 +01:00
if ( auto literal = dynamic_cast < ASTNodeLiteral * > ( expression - > evaluate ( evaluator ) ) ) {
evaluatedParams . push_back ( literal - > getValue ( ) ) ;
delete literal ;
} else if ( auto parameterPack = dynamic_cast < ASTNodeParameterPack * > ( expression - > evaluate ( evaluator ) ) ) {
for ( auto & value : parameterPack - > getValues ( ) ) {
evaluatedParams . push_back ( value ) ;
}
2021-01-09 21:47:11 +01:00
2022-01-30 15:18:45 +01:00
delete parameterPack ;
}
2021-09-21 21:29:18 +02:00
}
2021-01-09 21:47:11 +01:00
2021-09-21 21:29:18 +02:00
auto & customFunctions = evaluator - > getCustomFunctions ( ) ;
2022-02-01 22:09:44 +01:00
auto functions = ContentRegistry : : PatternLanguage : : getFunctions ( ) ;
2021-01-09 21:47:11 +01:00
2021-09-21 21:29:18 +02:00
for ( auto & func : customFunctions )
functions . insert ( func ) ;
2021-01-09 21:47:11 +01:00
2022-02-05 23:02:38 +01:00
if ( ! functions . contains ( this - > m_functionName ) ) {
if ( this - > m_functionName . starts_with ( " std:: " ) ) {
2022-02-06 19:46:39 +01:00
evaluator - > getConsole ( ) . log ( LogConsole : : Level : : Warning , " This function might be part of the standard library. \n You can install the standard library though \n the Content Store found under Help -> Content Store and then \n include the correct file. " ) ;
2022-02-05 23:02:38 +01:00
}
2021-09-21 21:29:18 +02:00
LogConsole : : abortEvaluation ( hex : : format ( " call to unknown function '{}' " , this - > m_functionName ) , this ) ;
2022-02-05 23:02:38 +01:00
}
2021-01-09 21:47:11 +01:00
2021-09-21 21:29:18 +02:00
auto function = functions [ this - > m_functionName ] ;
2021-12-20 20:40:28 +01:00
if ( function . parameterCount = = ContentRegistry : : PatternLanguage : : UnlimitedParameters ) {
2022-01-24 20:53:17 +01:00
; // Don't check parameter count
} else if ( function . parameterCount & ContentRegistry : : PatternLanguage : : LessParametersThan ) {
2021-12-20 20:40:28 +01:00
if ( evaluatedParams . size ( ) > = ( function . parameterCount & ~ ContentRegistry : : PatternLanguage : : LessParametersThan ) )
2022-01-30 15:18:45 +01:00
LogConsole : : abortEvaluation ( hex : : format ( " too many parameters for function '{0}'. Expected less than {1} " , this - > m_functionName , function . parameterCount & ~ ContentRegistry : : PatternLanguage : : LessParametersThan ) , this ) ;
2021-12-20 20:40:28 +01:00
} else if ( function . parameterCount & ContentRegistry : : PatternLanguage : : MoreParametersThan ) {
if ( evaluatedParams . size ( ) < = ( function . parameterCount & ~ ContentRegistry : : PatternLanguage : : MoreParametersThan ) )
2022-01-30 15:18:45 +01:00
LogConsole : : abortEvaluation ( hex : : format ( " too few parameters for function '{0}'. Expected more than {1} " , this - > m_functionName , function . parameterCount & ~ ContentRegistry : : PatternLanguage : : MoreParametersThan ) , this ) ;
} else if ( function . parameterCount & ContentRegistry : : PatternLanguage : : ExactlyOrMoreParametersThan ) {
if ( evaluatedParams . size ( ) < ( function . parameterCount & ~ ContentRegistry : : PatternLanguage : : ExactlyOrMoreParametersThan ) )
LogConsole : : abortEvaluation ( hex : : format ( " too few parameters for function '{0}'. Expected more than {1} " , this - > m_functionName , ( function . parameterCount - 1 ) & ~ ContentRegistry : : PatternLanguage : : ExactlyOrMoreParametersThan ) , this ) ;
2021-09-21 21:29:18 +02:00
} else if ( function . parameterCount ! = evaluatedParams . size ( ) ) {
LogConsole : : abortEvaluation ( hex : : format ( " invalid number of parameters for function '{0}'. Expected {1} " , this - > m_functionName , function . parameterCount ) , this ) ;
}
2021-01-09 21:47:11 +01:00
2021-09-21 21:29:18 +02:00
try {
2021-12-19 12:32:15 +01:00
auto & function = functions [ this - > m_functionName ] ;
if ( function . dangerous & & evaluator - > getDangerousFunctionPermission ( ) ! = DangerousFunctionPermission : : Allow ) {
evaluator - > dangerousFunctionCalled ( ) ;
while ( evaluator - > getDangerousFunctionPermission ( ) = = DangerousFunctionPermission : : Ask ) {
using namespace std : : literals : : chrono_literals ;
std : : this_thread : : sleep_for ( 100 ms ) ;
}
if ( evaluator - > getDangerousFunctionPermission ( ) = = DangerousFunctionPermission : : Deny ) {
LogConsole : : abortEvaluation ( hex : : format ( " calling of dangerous function '{}' is not allowed " , this - > m_functionName ) , this ) ;
}
}
auto result = function . func ( evaluator , evaluatedParams ) ;
2021-01-21 17:49:30 +01:00
2021-09-21 21:29:18 +02:00
if ( result . has_value ( ) )
return new ASTNodeLiteral ( result . value ( ) ) ;
else
return new ASTNodeMathematicalExpression ( nullptr , nullptr , Token : : Operator : : Plus ) ;
} catch ( std : : string & error ) {
LogConsole : : abortEvaluation ( error , this ) ;
}
2021-01-21 17:49:30 +01:00
2021-09-21 21:29:18 +02:00
return nullptr ;
2021-01-21 17:49:30 +01:00
}
2021-11-27 12:57:59 +01:00
FunctionResult execute ( Evaluator * evaluator ) const override {
2021-09-21 21:29:18 +02:00
delete this - > evaluate ( evaluator ) ;
2021-01-21 17:49:30 +01:00
2022-01-24 20:53:17 +01:00
return { } ;
2021-01-21 17:49:30 +01:00
}
private :
2021-09-21 21:29:18 +02:00
std : : string m_functionName ;
2022-01-24 20:53:17 +01:00
std : : vector < ASTNode * > m_params ;
2021-01-21 17:49:30 +01:00
} ;
2021-01-07 15:37:37 +01:00
2021-04-21 10:17:42 +02:00
class ASTNodeTypeOperator : public ASTNode {
public :
ASTNodeTypeOperator ( Token : : Operator op , ASTNode * expression ) : m_op ( op ) , m_expression ( expression ) {
}
ASTNodeTypeOperator ( const ASTNodeTypeOperator & other ) : ASTNode ( other ) {
2022-02-01 22:09:44 +01:00
this - > m_op = other . m_op ;
2021-04-21 10:17:42 +02:00
this - > m_expression = other . m_expression - > clone ( ) ;
}
2022-01-24 20:53:17 +01:00
[[nodiscard]] ASTNode * clone ( ) const override {
2021-04-21 10:17:42 +02:00
return new ASTNodeTypeOperator ( * this ) ;
}
~ ASTNodeTypeOperator ( ) override {
delete this - > m_expression ;
}
2022-01-24 20:53:17 +01:00
[[nodiscard]] Token : : Operator getOperator ( ) const {
2021-04-21 10:17:42 +02:00
return this - > m_op ;
}
2022-01-24 20:53:17 +01:00
[[nodiscard]] ASTNode * getExpression ( ) const {
2021-04-21 10:17:42 +02:00
return this - > m_expression ;
}
2022-01-24 20:53:17 +01:00
[[nodiscard]] ASTNode * evaluate ( Evaluator * evaluator ) const override {
2021-09-21 21:29:18 +02:00
auto pattern = this - > m_expression - > createPatterns ( evaluator ) . front ( ) ;
ON_SCOPE_EXIT { delete pattern ; } ;
2021-06-20 21:22:31 +02:00
2021-09-21 21:29:18 +02:00
switch ( this - > getOperator ( ) ) {
2022-02-01 22:09:44 +01:00
case Token : : Operator : : AddressOf :
return new ASTNodeLiteral ( u128 ( pattern - > getOffset ( ) ) ) ;
case Token : : Operator : : SizeOf :
return new ASTNodeLiteral ( u128 ( pattern - > getSize ( ) ) ) ;
default :
LogConsole : : abortEvaluation ( " invalid type operator " , this ) ;
2021-06-20 21:22:31 +02:00
}
}
private :
2021-09-21 21:29:18 +02:00
Token : : Operator m_op ;
ASTNode * m_expression ;
2021-06-20 21:22:31 +02:00
} ;
2021-09-21 21:29:18 +02:00
2021-06-20 21:22:31 +02:00
class ASTNodeAssignment : public ASTNode {
public :
ASTNodeAssignment ( std : : string lvalueName , ASTNode * rvalue ) : m_lvalueName ( std : : move ( lvalueName ) ) , m_rvalue ( rvalue ) {
}
ASTNodeAssignment ( const ASTNodeAssignment & other ) : ASTNode ( other ) {
this - > m_lvalueName = other . m_lvalueName ;
2022-02-01 22:09:44 +01:00
this - > m_rvalue = other . m_rvalue - > clone ( ) ;
2021-06-20 21:22:31 +02:00
}
2022-01-24 20:53:17 +01:00
[[nodiscard]] ASTNode * clone ( ) const override {
2021-06-20 21:22:31 +02:00
return new ASTNodeAssignment ( * this ) ;
}
~ ASTNodeAssignment ( ) override {
delete this - > m_rvalue ;
}
2022-01-24 20:53:17 +01:00
[[nodiscard]] const std : : string & getLValueName ( ) const {
2021-06-20 21:22:31 +02:00
return this - > m_lvalueName ;
}
2022-01-24 20:53:17 +01:00
[[nodiscard]] ASTNode * getRValue ( ) const {
2021-06-20 21:22:31 +02:00
return this - > m_rvalue ;
}
2022-01-29 03:06:34 +01:00
[[nodiscard]] std : : vector < PatternData * > createPatterns ( Evaluator * evaluator ) const override {
this - > execute ( evaluator ) ;
2022-01-30 12:43:43 +01:00
return { } ;
2022-01-29 03:06:34 +01:00
}
2021-11-27 12:57:59 +01:00
FunctionResult execute ( Evaluator * evaluator ) const override {
2022-01-24 20:53:17 +01:00
auto literal = dynamic_cast < ASTNodeLiteral * > ( this - > getRValue ( ) - > evaluate ( evaluator ) ) ;
2021-09-21 21:29:18 +02:00
ON_SCOPE_EXIT { delete literal ; } ;
2022-01-29 03:06:34 +01:00
if ( this - > getLValueName ( ) = = " $ " )
evaluator - > dataOffset ( ) = Token : : literalToUnsigned ( literal - > getValue ( ) ) ;
else
evaluator - > setVariable ( this - > getLValueName ( ) , literal - > getValue ( ) ) ;
2021-09-21 21:29:18 +02:00
2022-01-24 20:53:17 +01:00
return { } ;
2021-09-21 21:29:18 +02:00
}
2021-06-20 21:22:31 +02:00
private :
std : : string m_lvalueName ;
ASTNode * m_rvalue ;
} ;
2021-12-30 14:44:46 +01:00
class ASTNodeControlFlowStatement : public ASTNode {
2021-06-20 21:22:31 +02:00
public :
2021-12-30 14:44:46 +01:00
explicit ASTNodeControlFlowStatement ( ControlFlowStatement type , ASTNode * rvalue ) : m_type ( type ) , m_rvalue ( rvalue ) {
2021-06-20 21:22:31 +02:00
}
2021-12-30 14:44:46 +01:00
ASTNodeControlFlowStatement ( const ASTNodeControlFlowStatement & other ) : ASTNode ( other ) {
this - > m_type = other . m_type ;
2021-12-30 18:31:24 +01:00
if ( other . m_rvalue ! = nullptr )
this - > m_rvalue = other . m_rvalue - > clone ( ) ;
else
this - > m_rvalue = nullptr ;
2021-06-20 21:22:31 +02:00
}
2022-01-24 20:53:17 +01:00
[[nodiscard]] ASTNode * clone ( ) const override {
2021-12-30 14:44:46 +01:00
return new ASTNodeControlFlowStatement ( * this ) ;
2021-06-20 21:22:31 +02:00
}
2021-12-30 14:44:46 +01:00
~ ASTNodeControlFlowStatement ( ) override {
2021-06-20 21:22:31 +02:00
delete this - > m_rvalue ;
}
2022-01-24 20:53:17 +01:00
[[nodiscard]] ASTNode * getReturnValue ( ) const {
2021-06-20 21:22:31 +02:00
return this - > m_rvalue ;
}
2022-01-24 20:53:17 +01:00
[[nodiscard]] std : : vector < PatternData * > createPatterns ( Evaluator * evaluator ) const override {
2021-12-30 18:31:24 +01:00
this - > execute ( evaluator ) ;
2022-01-24 20:53:17 +01:00
return { } ;
2021-12-30 18:31:24 +01:00
}
2021-11-27 12:57:59 +01:00
FunctionResult execute ( Evaluator * evaluator ) const override {
2021-09-21 21:29:18 +02:00
auto returnValue = this - > getReturnValue ( ) ;
2022-02-11 21:02:18 +01:00
auto literal = dynamic_cast < ASTNodeLiteral * > ( returnValue - > evaluate ( evaluator ) ) ;
ON_SCOPE_EXIT { delete literal ; } ;
2021-12-30 14:44:46 +01:00
evaluator - > setCurrentControlFlowStatement ( this - > m_type ) ;
2022-02-11 21:02:18 +01:00
if ( literal = = nullptr )
2021-12-30 14:44:46 +01:00
return std : : nullopt ;
2022-02-11 21:02:18 +01:00
else
2021-12-30 14:44:46 +01:00
return literal - > getValue ( ) ;
2021-09-21 21:29:18 +02:00
}
2021-06-20 21:22:31 +02:00
private :
2021-12-30 14:44:46 +01:00
ControlFlowStatement m_type ;
2021-06-20 21:22:31 +02:00
ASTNode * m_rvalue ;
} ;
2021-09-21 21:29:18 +02:00
class ASTNodeFunctionDefinition : public ASTNode {
public :
2022-01-30 15:18:45 +01:00
ASTNodeFunctionDefinition ( std : : string name , std : : vector < std : : pair < std : : string , ASTNode * > > params , std : : vector < ASTNode * > body , std : : optional < std : : string > parameterPack )
: m_name ( std : : move ( name ) ) , m_params ( std : : move ( params ) ) , m_body ( std : : move ( body ) ) , m_parameterPack ( std : : move ( parameterPack ) ) {
2021-09-21 21:29:18 +02:00
}
ASTNodeFunctionDefinition ( const ASTNodeFunctionDefinition & other ) : ASTNode ( other ) {
2022-02-01 22:09:44 +01:00
this - > m_name = other . m_name ;
2021-09-21 21:29:18 +02:00
this - > m_params = other . m_params ;
for ( const auto & [ name , type ] : other . m_params ) {
2021-09-24 11:34:06 +02:00
this - > m_params . emplace_back ( name , type - > clone ( ) ) ;
2021-09-21 21:29:18 +02:00
}
for ( auto statement : other . m_body ) {
this - > m_body . push_back ( statement - > clone ( ) ) ;
}
}
2022-01-24 20:53:17 +01:00
[[nodiscard]] ASTNode * clone ( ) const override {
2021-09-21 21:29:18 +02:00
return new ASTNodeFunctionDefinition ( * this ) ;
}
~ ASTNodeFunctionDefinition ( ) override {
for ( auto & [ name , type ] : this - > m_params )
delete type ;
for ( auto statement : this - > m_body )
delete statement ;
}
2022-01-24 20:53:17 +01:00
[[nodiscard]] const std : : string & getName ( ) const {
2021-09-21 21:29:18 +02:00
return this - > m_name ;
}
2022-01-24 20:53:17 +01:00
[[nodiscard]] const auto & getParams ( ) const {
2021-09-21 21:29:18 +02:00
return this - > m_params ;
}
2022-01-24 20:53:17 +01:00
[[nodiscard]] const auto & getBody ( ) const {
2021-09-21 21:29:18 +02:00
return this - > m_body ;
}
2022-01-24 20:53:17 +01:00
[[nodiscard]] ASTNode * evaluate ( Evaluator * evaluator ) const override {
2021-09-21 21:29:18 +02:00
2022-01-30 15:18:45 +01:00
size_t paramCount = this - > m_params . size ( ) ;
if ( this - > m_parameterPack . has_value ( ) )
paramCount | = ContentRegistry : : PatternLanguage : : ExactlyOrMoreParametersThan ;
evaluator - > addCustomFunction ( this - > m_name , paramCount , [ this ] ( Evaluator * ctx , const std : : vector < Token : : Literal > & params ) - > std : : optional < Token : : Literal > {
2022-01-24 20:53:17 +01:00
std : : vector < PatternData * > variables ;
2021-09-21 21:29:18 +02:00
ctx - > pushScope ( nullptr , variables ) ;
2021-10-02 15:22:38 +02:00
ON_SCOPE_EXIT {
2022-01-24 20:53:17 +01:00
for ( auto variable : variables )
delete variable ;
2021-10-02 15:22:38 +02:00
2022-01-24 20:53:17 +01:00
ctx - > popScope ( ) ;
} ;
2021-09-21 21:29:18 +02:00
2022-01-30 15:18:45 +01:00
if ( this - > m_parameterPack . has_value ( ) ) {
std : : vector < Token : : Literal > parameterPackContent ;
for ( u32 paramIndex = this - > m_params . size ( ) ; paramIndex < params . size ( ) ; paramIndex + + )
parameterPackContent . push_back ( params [ paramIndex ] ) ;
ctx - > createParameterPack ( this - > m_parameterPack . value ( ) , parameterPackContent ) ;
}
for ( u32 paramIndex = 0 ; paramIndex < this - > m_params . size ( ) ; paramIndex + + ) {
const auto & [ name , type ] = this - > m_params [ paramIndex ] ;
2021-09-23 23:43:16 +02:00
ctx - > createVariable ( name , type , params [ paramIndex ] ) ;
2021-09-21 21:29:18 +02:00
ctx - > setVariable ( name , params [ paramIndex ] ) ;
}
for ( auto statement : this - > m_body ) {
2021-12-30 14:44:46 +01:00
auto result = statement - > execute ( ctx ) ;
if ( ctx - > getCurrentControlFlowStatement ( ) ! = ControlFlowStatement : : None ) {
switch ( ctx - > getCurrentControlFlowStatement ( ) ) {
2022-02-01 22:09:44 +01:00
case ControlFlowStatement : : Break :
LogConsole : : abortEvaluation ( " break statement not within a loop " , statement ) ;
case ControlFlowStatement : : Continue :
LogConsole : : abortEvaluation ( " continue statement not within a loop " , statement ) ;
default :
break ;
2021-12-30 14:44:46 +01:00
}
2021-09-21 21:29:18 +02:00
2021-12-30 14:44:46 +01:00
ctx - > setCurrentControlFlowStatement ( ControlFlowStatement : : None ) ;
2021-09-21 21:29:18 +02:00
return result ;
}
}
2022-01-24 20:53:17 +01:00
return { } ;
2021-09-21 21:29:18 +02:00
} ) ;
return nullptr ;
}
private :
std : : string m_name ;
2022-01-24 20:53:17 +01:00
std : : vector < std : : pair < std : : string , ASTNode * > > m_params ;
std : : vector < ASTNode * > m_body ;
2022-01-30 15:18:45 +01:00
std : : optional < std : : string > m_parameterPack ;
2021-09-21 21:29:18 +02:00
} ;
2021-10-10 13:05:32 +02:00
class ASTNodeCompoundStatement : public ASTNode {
public :
2022-01-31 14:37:12 +01:00
explicit ASTNodeCompoundStatement ( std : : vector < ASTNode * > statements , bool newScope = false ) : m_statements ( std : : move ( statements ) ) , m_newScope ( newScope ) {
2021-10-10 13:05:32 +02:00
}
ASTNodeCompoundStatement ( const ASTNodeCompoundStatement & other ) : ASTNode ( other ) {
for ( const auto & statement : other . m_statements ) {
this - > m_statements . push_back ( statement - > clone ( ) ) ;
}
2022-01-31 14:37:12 +01:00
this - > m_newScope = other . m_newScope ;
2021-10-10 13:05:32 +02:00
}
2022-01-24 20:53:17 +01:00
[[nodiscard]] ASTNode * clone ( ) const override {
2021-10-10 13:05:32 +02:00
return new ASTNodeCompoundStatement ( * this ) ;
}
~ ASTNodeCompoundStatement ( ) override {
for ( const auto & statement : this - > m_statements ) {
delete statement ;
}
}
2022-01-24 20:53:17 +01:00
[[nodiscard]] ASTNode * evaluate ( Evaluator * evaluator ) const override {
2021-10-10 13:05:32 +02:00
ASTNode * result = nullptr ;
for ( const auto & statement : this - > m_statements ) {
delete result ;
result = statement - > evaluate ( evaluator ) ;
}
return result ;
}
2022-01-24 20:53:17 +01:00
[[nodiscard]] std : : vector < PatternData * > createPatterns ( Evaluator * evaluator ) const override {
std : : vector < PatternData * > result ;
2021-10-10 13:05:32 +02:00
for ( const auto & statement : this - > m_statements ) {
auto patterns = statement - > createPatterns ( evaluator ) ;
std : : copy ( patterns . begin ( ) , patterns . end ( ) , std : : back_inserter ( result ) ) ;
}
return result ;
}
2021-11-27 12:57:59 +01:00
FunctionResult execute ( Evaluator * evaluator ) const override {
2021-10-10 13:05:32 +02:00
FunctionResult result ;
2022-02-01 22:09:44 +01:00
auto variables = * evaluator - > getScope ( 0 ) . scope ;
2021-10-10 13:47:48 +02:00
u32 startVariableCount = variables . size ( ) ;
if ( this - > m_newScope ) {
evaluator - > pushScope ( nullptr , variables ) ;
}
2021-10-10 13:05:32 +02:00
for ( const auto & statement : this - > m_statements ) {
result = statement - > execute ( evaluator ) ;
2021-12-30 14:44:46 +01:00
if ( evaluator - > getCurrentControlFlowStatement ( ) ! = ControlFlowStatement : : None )
2021-10-10 13:05:32 +02:00
return result ;
}
2021-10-10 13:47:48 +02:00
if ( this - > m_newScope ) {
2022-01-22 22:37:52 +01:00
i64 stackSize = evaluator - > getStack ( ) . size ( ) ;
2021-10-10 13:47:48 +02:00
for ( u32 i = startVariableCount ; i < variables . size ( ) ; i + + ) {
stackSize - - ;
delete variables [ i ] ;
}
if ( stackSize < 0 ) LogConsole : : abortEvaluation ( " stack pointer underflow! " , this ) ;
evaluator - > getStack ( ) . resize ( stackSize ) ;
evaluator - > popScope ( ) ;
}
2021-10-10 13:05:32 +02:00
return result ;
}
public :
2022-01-24 20:53:17 +01:00
std : : vector < ASTNode * > m_statements ;
2022-01-31 14:37:12 +01:00
bool m_newScope = false ;
2021-10-10 13:05:32 +02:00
} ;
2022-02-18 13:31:44 +01:00
} ;