diff --git a/base/components/interface.cpp b/base/components/interface.cpp index cfc67ee..6415e51 100644 --- a/base/components/interface.cpp +++ b/base/components/interface.cpp @@ -3,8 +3,8 @@ #include "code_serialization.cpp" #endif -internal void parser_init(); -internal void parser_deinit(); +internal void parser_init(Context* ctx); +internal void parser_deinit(Context* ctx); internal void* fallback_allocator_proc( void* allocator_data, AllocType type, ssize size, ssize alignment, void* old_memory, ssize old_size, u64 flags ) @@ -343,7 +343,7 @@ void init(Context* ctx) } define_constants(); - parser_init(); + parser_init(ctx); ++ context_counter; } @@ -392,7 +392,7 @@ void deinit(Context* ctx) while ( left--, left ); array_free( ctx->Fallback_AllocatorBuckets); } - parser_deinit(); + parser_deinit(ctx); if (_ctx == ctx) _ctx = nullptr; diff --git a/base/components/interface.parsing.cpp b/base/components/interface.parsing.cpp index 70166f3..84caf41 100644 --- a/base/components/interface.parsing.cpp +++ b/base/components/interface.parsing.cpp @@ -10,6 +10,9 @@ ParseInfo wip_parse_str(LexedInfo lexed, ParseOpts* opts) { + // TODO(Ed): Lift this. + Context* ctx = _ctx; + TokArray toks; if (lexed.tokens.Num == 0 && lexed.tokens.Ptr == nullptr) { check_parse_args(lexed.text); @@ -23,31 +26,44 @@ ParseInfo wip_parse_str(LexedInfo lexed, ParseOpts* opts) // TODO(Ed): ParseInfo should be set to the parser context. - _ctx->parser.Tokens = toks; - push_scope(); - CodeBody result = parse_global_nspace(_ctx, CT_Global_Body); - parser_pop(& _ctx->parser); + ctx->parser.Tokens = toks; + ParseStackNode scope = NullScope; + parser_push(& ctx->parser, & scope); + + CodeBody result = parse_global_nspace(ctx,CT_Global_Body); + + parser_pop(& ctx->parser); return info; } CodeClass parse_class( Str def ) { + // TODO(Ed): Lift this. + Context* ctx = _ctx; + check_parse_args( def ); TokArray toks = lex( def ); if ( toks.Arr == nullptr ) return InvalidCode; - _ctx->parser.Tokens = toks; - push_scope(); - CodeClass result = (CodeClass) parse_class_struct( _ctx, Tok_Decl_Class, parser_not_inplace_def ); - parser_pop(& _ctx->parser); + ctx->parser.Tokens = toks; + + ParseStackNode scope = NullScope; + parser_push(& ctx->parser, & scope); + + CodeClass result = (CodeClass) parse_class_struct( ctx, Tok_Decl_Class, parser_not_inplace_def ); + + parser_pop(& ctx->parser); return result; } -CodeConstructor parse_constructor( Str def ) +CodeConstructor parse_constructor(Str def ) { + // TODO(Ed): Lift this. + Context* ctx = _ctx; + check_parse_args( def ); TokArray toks = lex( def ); @@ -80,8 +96,8 @@ CodeConstructor parse_constructor( Str def ) break; default : - log_failure( "Invalid specifier %s for variable\n%S", spec_to_str( spec ), parser_to_strbuilder(_ctx->parser) ); - parser_pop(& _ctx->parser); + log_failure( "Invalid specifier %s for variable\n%S", spec_to_str( spec ), parser_to_strbuilder(ctx->parser, ctx->Allocator_Temp) ); + parser_pop(& ctx->parser); return InvalidCode; } @@ -100,28 +116,38 @@ CodeConstructor parse_constructor( Str def ) // ... } - _ctx->parser.Tokens = toks; - CodeConstructor result = parser_parse_constructor( specifiers ); + ctx->parser.Tokens = toks; + CodeConstructor result = parser_parse_constructor(ctx, specifiers); return result; } CodeDefine parse_define( Str def ) { + // TODO(Ed): Lift this. + Context* ctx = _ctx; + check_parse_args( def ); TokArray toks = lex( def ); if ( toks.Arr == nullptr ) return InvalidCode; - _ctx->parser.Tokens = toks; - push_scope(); - CodeDefine result = parser_parse_define(); - parser_pop(& _ctx->parser); + ctx->parser.Tokens = toks; + + ParseStackNode scope = NullScope; + parser_push(& ctx->parser, & scope); + + CodeDefine result = parser_parse_define(ctx); + + parser_pop(& ctx->parser); return result; } CodeDestructor parse_destructor( Str def ) { + // TODO(Ed): Lift this. + Context* ctx = _ctx; + check_parse_args( def ); TokArray toks = lex( def ); @@ -131,210 +157,269 @@ CodeDestructor parse_destructor( Str def ) // TODO(Ed): Destructors can have prefix attributes // TODO(Ed): Destructors can have virtual - _ctx->parser.Tokens = toks; - CodeDestructor result = parser_parse_destructor(NullCode); + ctx->parser.Tokens = toks; + CodeDestructor result = parser_parse_destructor(ctx, NullCode); return result; } CodeEnum parse_enum( Str def ) { + // TODO(Ed): Lift this. + Context* ctx = _ctx; + check_parse_args( def ); + ParseStackNode scope = NullScope; + parser_push(& ctx->parser, & scope); + TokArray toks = lex( def ); if ( toks.Arr == nullptr ) { - parser_pop(& _ctx->parser); + parser_pop(& ctx->parser); return InvalidCode; } - _ctx->parser.Tokens = toks; - return parser_parse_enum( parser_not_inplace_def); + ctx->parser.Tokens = toks; + return parser_parse_enum(ctx, parser_not_inplace_def); } CodeBody parse_export_body( Str def ) { + // TODO(Ed): Lift this. + Context* ctx = _ctx; + check_parse_args( def ); TokArray toks = lex( def ); if ( toks.Arr == nullptr ) return InvalidCode; - _ctx->parser.Tokens = toks; - return parser_parse_export_body(); + ctx->parser.Tokens = toks; + return parser_parse_export_body(ctx); } CodeExtern parse_extern_link( Str def ) { + // TODO(Ed): Lift this. + Context* ctx = _ctx; + check_parse_args( def ); TokArray toks = lex( def ); if ( toks.Arr == nullptr ) return InvalidCode; - _ctx->parser.Tokens = toks; - return parser_parse_extern_link(); + ctx->parser.Tokens = toks; + return parser_parse_extern_link(ctx); } CodeFriend parse_friend( Str def ) { + // TODO(Ed): Lift this. + Context* ctx = _ctx; + check_parse_args( def ); TokArray toks = lex( def ); if ( toks.Arr == nullptr ) return InvalidCode; - _ctx->parser.Tokens = toks; - return parser_parse_friend(); + ctx->parser.Tokens = toks; + return parser_parse_friend(ctx); } CodeFn parse_function( Str def ) { + // TODO(Ed): Lift this. + Context* ctx = _ctx; + check_parse_args( def ); TokArray toks = lex( def ); if ( toks.Arr == nullptr ) return InvalidCode; - _ctx->parser.Tokens = toks; - return (CodeFn) parser_parse_function(); + ctx->parser.Tokens = toks; + return (CodeFn) parser_parse_function(ctx); } CodeBody parse_global_body( Str def ) { + // TODO(Ed): Lift this. + Context* ctx = _ctx; + check_parse_args( def ); TokArray toks = lex( def ); if ( toks.Arr == nullptr ) return InvalidCode; - _ctx->parser.Tokens = toks; - push_scope(); - CodeBody result = parse_global_nspace(_ctx, CT_Global_Body ); - parser_pop(& _ctx->parser); + ctx->parser.Tokens = toks; + + ParseStackNode scope = NullScope; + parser_push(& ctx->parser, & scope); + + CodeBody result = parse_global_nspace(ctx, CT_Global_Body ); + + parser_pop(& ctx->parser); return result; } CodeNS parse_namespace( Str def ) { + // TODO(Ed): Lift this. + Context* ctx = _ctx; + check_parse_args( def ); TokArray toks = lex( def ); if ( toks.Arr == nullptr ) return InvalidCode; - _ctx->parser.Tokens = toks; - return parser_parse_namespace(); + ctx->parser.Tokens = toks; + return parser_parse_namespace(ctx); } CodeOperator parse_operator( Str def ) { + // TODO(Ed): Lift this. + Context* ctx = _ctx; + check_parse_args( def ); TokArray toks = lex( def ); if ( toks.Arr == nullptr ) return InvalidCode; - _ctx->parser.Tokens = toks; - return (CodeOperator) parser_parse_operator(); + ctx->parser.Tokens = toks; + return (CodeOperator) parser_parse_operator(ctx); } CodeOpCast parse_operator_cast( Str def ) { + // TODO(Ed): Lift this. + Context* ctx = _ctx; + check_parse_args( def ); TokArray toks = lex( def ); if ( toks.Arr == nullptr ) return InvalidCode; - _ctx->parser.Tokens = toks; - return parser_parse_operator_cast(NullCode); + ctx->parser.Tokens = toks; + return parser_parse_operator_cast(ctx, NullCode); } CodeStruct parse_struct( Str def ) { + // TODO(Ed): Lift this. + Context* ctx = _ctx; + check_parse_args( def ); TokArray toks = lex( def ); if ( toks.Arr == nullptr ) return InvalidCode; - _ctx->parser.Tokens = toks; - push_scope(); - CodeStruct result = (CodeStruct) parse_class_struct( _ctx, Tok_Decl_Struct, parser_not_inplace_def ); - parser_pop(& _ctx->parser); + ctx->parser.Tokens = toks; + + ParseStackNode scope = NullScope; + parser_push(& ctx->parser, & scope); + + CodeStruct result = (CodeStruct) parse_class_struct( ctx, Tok_Decl_Struct, parser_not_inplace_def ); + + parser_pop(& ctx->parser); return result; } CodeTemplate parse_template( Str def ) { + // TODO(Ed): Lift this. + Context* ctx = _ctx; + check_parse_args( def ); TokArray toks = lex( def ); if ( toks.Arr == nullptr ) return InvalidCode; - _ctx->parser.Tokens = toks; - return parser_parse_template(); + ctx->parser.Tokens = toks; + return parser_parse_template(ctx); } CodeTypename parse_type( Str def ) { + // TODO(Ed): Lift this. + Context* ctx = _ctx; + check_parse_args( def ); TokArray toks = lex( def ); if ( toks.Arr == nullptr ) return InvalidCode; - _ctx->parser.Tokens = toks; - return parser_parse_type( parser_not_from_template, nullptr); + ctx->parser.Tokens = toks; + return parser_parse_type( ctx, parser_not_from_template, nullptr); } CodeTypedef parse_typedef( Str def ) { + // TODO(Ed): Lift this. + Context* ctx = _ctx; + check_parse_args( def ); TokArray toks = lex( def ); if ( toks.Arr == nullptr ) return InvalidCode; - _ctx->parser.Tokens = toks; - return parser_parse_typedef(); + ctx->parser.Tokens = toks; + return parser_parse_typedef(ctx); } CodeUnion parse_union( Str def ) { + // TODO(Ed): Lift this. + Context* ctx = _ctx; + check_parse_args( def ); TokArray toks = lex( def ); if ( toks.Arr == nullptr ) return InvalidCode; - _ctx->parser.Tokens = toks; - return parser_parse_union( parser_not_inplace_def); + ctx->parser.Tokens = toks; + return parser_parse_union(ctx, parser_not_inplace_def); } CodeUsing parse_using( Str def ) { + // TODO(Ed): Lift this. + Context* ctx = _ctx; + check_parse_args( def ); TokArray toks = lex( def ); if ( toks.Arr == nullptr ) return InvalidCode; - _ctx->parser.Tokens = toks; - return parser_parse_using(); + ctx->parser.Tokens = toks; + return parser_parse_using(ctx); } CodeVar parse_variable( Str def ) { + // TODO(Ed): Lift this. + Context* ctx = _ctx; + check_parse_args( def ); TokArray toks = lex( def ); if ( toks.Arr == nullptr ) return InvalidCode; - _ctx->parser.Tokens = toks; - return parser_parse_variable(); + ctx->parser.Tokens = toks; + return parser_parse_variable(ctx); } // Undef helper macros diff --git a/base/components/lexer.cpp b/base/components/lexer.cpp index 0fc46b8..2d0e7ae 100644 --- a/base/components/lexer.cpp +++ b/base/components/lexer.cpp @@ -17,7 +17,7 @@ StrBuilder tok_to_strbuilder(Token tok) return result; } -bool lex__eat( TokArray* self, TokType type ); +bool lex__eat(Context* ctx, TokArray* self, TokType type ); Token* lex_current(TokArray* self, bool skip_formatting ) { diff --git a/base/components/parser.cpp b/base/components/parser.cpp index a2b3d68..3beba0c 100644 --- a/base/components/parser.cpp +++ b/base/components/parser.cpp @@ -17,21 +17,21 @@ void parser_push( ParseContext* ctx, ParseStackNode* node ) ctx->Scope = node; #if 0 && GEN_BUILD_DEBUG - log_fmt("\tEntering _ctx->parser: %.*s\n", Scope->ProcName.Len, Scope->ProcName.Ptr ); + log_fmt("\tEntering parser: %.*s\n", Scope->ProcName.Len, Scope->ProcName.Ptr ); #endif } void parser_pop(ParseContext* ctx) { #if 0 && GEN_BUILD_DEBUG - log_fmt("\tPopping _ctx->parser: %.*s\n", Scope->ProcName.Len, Scope->ProcName.Ptr ); + log_fmt("\tPopping parser: %.*s\n", Scope->ProcName.Len, Scope->ProcName.Ptr ); #endif ctx->Scope = ctx->Scope->Prev; } -StrBuilder parser_to_strbuilder(ParseContext ctx) +StrBuilder parser_to_strbuilder(ParseContext ctx, AllocatorInfo temp) { - StrBuilder result = strbuilder_make_reserve( _ctx->Allocator_Temp, kilobytes(4) ); + StrBuilder result = strbuilder_make_reserve( temp, kilobytes(4) ); Token scope_start = * ctx.Scope->Start; Token last_valid = ctx.Tokens.Idx >= array_num(ctx.Tokens.Arr) ? ctx.Tokens.Arr[array_num(ctx.Tokens.Arr) -1] : (* lex_current(& ctx.Tokens, true)); @@ -45,7 +45,7 @@ StrBuilder parser_to_strbuilder(ParseContext ctx) } Str scope_str = { scope_start.Text.Ptr, length }; - StrBuilder line = strbuilder_make_str( _ctx->Allocator_Temp, scope_str ); + StrBuilder line = strbuilder_make_str( temp, scope_str ); strbuilder_append_fmt( & result, "\tScope : %s\n", line ); strbuilder_free(& line); @@ -53,7 +53,7 @@ StrBuilder parser_to_strbuilder(ParseContext ctx) sptr length_from_err = dist; Str err_str = { last_valid.Text.Ptr, length_from_err }; - StrBuilder line_from_err = strbuilder_make_str( _ctx->Allocator_Temp, err_str ); + StrBuilder line_from_err = strbuilder_make_str( temp, err_str ); if ( length_from_err < 100 ) strbuilder_append_fmt(& result, "\t(%d, %d):%*c\n", last_valid.Line, last_valid.Column, length_from_err, '^' ); @@ -78,10 +78,10 @@ StrBuilder parser_to_strbuilder(ParseContext ctx) return result; } -bool lex__eat(TokArray* self, TokType type ) +bool lex__eat(Context* ctx, TokArray* self, TokType type) { if ( array_num(self->Arr) - self->Idx <= 0 ) { - log_failure( "No tokens left.\n%s", parser_to_strbuilder(_ctx->parser) ); + log_failure( "No tokens left.\n%s", parser_to_strbuilder(ctx->parser, ctx->Allocator_Temp) ); return false; } @@ -109,7 +109,7 @@ bool lex__eat(TokArray* self, TokType type ) , at_idx.Text.Len, at_idx.Text.Ptr , tok.Line , tok.Column - , parser_to_strbuilder(_ctx->parser) + , parser_to_strbuilder(ctx->parser, ctx->Allocator_Temp) ); GEN_DEBUG_TRAP(); return false; @@ -124,46 +124,46 @@ bool lex__eat(TokArray* self, TokType type ) } internal -void parser_init() +void parser_init(Context* ctx) { - _ctx->Lexer_Tokens = array_init_reserve(Token, _ctx->Allocator_DyanmicContainers, _ctx->InitSize_LexerTokens ); + ctx->Lexer_Tokens = array_init_reserve(Token, ctx->Allocator_DyanmicContainers, ctx->InitSize_LexerTokens ); } internal -void parser_deinit() +void parser_deinit(Context* ctx) { Array(Token) null_array = { nullptr }; - _ctx->Lexer_Tokens = null_array; + ctx->Lexer_Tokens = null_array; } #pragma region Helper Macros -#define check_parse_args( def ) _check_parse_args(def, stringize(_func_) ) -bool _check_parse_args( Str def, char const* func_name ) +#define check_parse_args( def ) _check_parse_args(& ctx->parser, def, stringize(_func_) ) +bool _check_parse_args(ParseContext* parser, Str def, char const* func_name ) { if ( def.Len <= 0 ) { log_failure( c_str_fmt_buf("gen::%s: length must greater than 0", func_name) ); - parser_pop(& _ctx->parser); + parser_pop(parser); return false; } if ( def.Ptr == nullptr ) { log_failure( c_str_fmt_buf("gen::%s: def was null", func_name) ); - parser_pop(& _ctx->parser); + parser_pop(parser); return false; } return true; } -# define currtok_noskip (* lex_current( & _ctx->parser.Tokens, lex_dont_skip_formatting )) -# define currtok (* lex_current( & _ctx->parser.Tokens, lex_skip_formatting )) -# define peektok (* lex_peek(_ctx->parser.Tokens, lex_skip_formatting)) -# define prevtok (* lex_previous( _ctx->parser.Tokens, lex_dont_skip_formatting)) -# define nexttok (* lex_next( _ctx->parser.Tokens, lex_skip_formatting )) -# define nexttok_noskip (* lex_next( _ctx->parser.Tokens, lex_dont_skip_formatting)) -# define eat( Type_ ) lex__eat( & _ctx->parser.Tokens, Type_ ) -# define left ( array_num(_ctx->parser.Tokens.Arr) - _ctx->parser.Tokens.Idx ) +# define currtok_noskip (* lex_current( & ctx->parser.Tokens, lex_dont_skip_formatting )) +# define currtok (* lex_current( & ctx->parser.Tokens, lex_skip_formatting )) +# define peektok (* lex_peek(ctx->parser.Tokens, lex_skip_formatting)) +# define prevtok (* lex_previous( ctx->parser.Tokens, lex_dont_skip_formatting)) +# define nexttok (* lex_next( ctx->parser.Tokens, lex_skip_formatting )) +# define nexttok_noskip (* lex_next( ctx->parser.Tokens, lex_dont_skip_formatting)) +# define eat( Type_ ) lex__eat(ctx, & ctx->parser.Tokens, Type_ ) +# define left ( array_num(ctx->parser.Tokens.Arr) - ctx->parser.Tokens.Idx ) #if GEN_COMPILER_CPP # define def_assign( ... ) { __VA_ARGS__ } @@ -171,7 +171,6 @@ bool _check_parse_args( Str def, char const* func_name ) # define def_assign( ... ) __VA_ARGS__ #endif - #ifdef check #define CHECK_WAS_DEFINED #pragma push_macro("check") @@ -181,12 +180,6 @@ bool _check_parse_args( Str def, char const* func_name ) # define check_noskip( Type_ ) ( left && currtok_noskip.Type == Type_ ) # define check( Type_ ) ( left && currtok.Type == Type_ ) -// TODO(Ed): Don't do this anymore, we need a better initializer. -# define push_scope() \ - Str null_name = {}; \ - ParseStackNode scope = { nullptr, {nullptr, 0}, lex_current( & _ctx->parser.Tokens, lex_dont_skip_formatting ), null_name, txt( __func__ ), { nullptr} }; \ - parser_push( & _ctx->parser, & scope ) - #if GEN_COMPILER_CPP # define NullScope { nullptr, {nullptr, 0}, lex_current( & ctx->parser.Tokens, lex_dont_skip_formatting ), Str{nullptr, 0}, txt( __func__ ), { nullptr} } #else @@ -203,45 +196,45 @@ internal CodeComment parse_comment (Context* ctx); internal Code parse_complicated_definition (Context* ctx, TokType which); internal CodeBody parse_class_struct_body (Context* ctx, TokType which, Token name); internal Code parse_class_struct (Context* ctx, TokType which, bool inplace_def); -internal Code parse_expression (); +internal Code parse_expression (Context* ctx); internal Code parse_forward_or_definition (Context* ctx, TokType which, bool is_inplace); internal CodeFn parse_function_after_name (Context* ctx, ModuleFlag mflags, CodeAttributes attributes, CodeSpecifiers specifiers, CodeTypename ret_type, Token name); internal Code parse_function_body (Context* ctx); -internal CodeBody parse_global_nspace (Context* ctx, CodeType which ); +internal CodeBody parse_global_nspace (Context* ctx, CodeType which); internal Code parse_global_nspace_constructor_destructor(Context* ctx, CodeSpecifiers specifiers); internal Token parse_identifier (Context* ctx, bool* possible_member_function); internal CodeInclude parse_include (Context* ctx); internal Code parse_macro_as_definiton (Context* ctx, CodeAttributes attributes, CodeSpecifiers specifiers); -internal CodeOperator parse_operator_after_ret_type (Context* ctx, ModuleFlag mflags, CodeAttributes attributes, CodeSpecifiers specifiers, CodeTypename ret_type ); -internal Code parse_operator_function_or_variable( bool expects_function, CodeAttributes attributes, CodeSpecifiers specifiers ); -internal CodePragma parse_pragma (); -internal CodeParams parse_params ( bool use_template_capture ); -internal CodePreprocessCond parse_preprocess_cond (); -internal Code parse_simple_preprocess ( TokType which ); -internal Code parse_static_assert (); -internal void parse_template_args ( Token* token ); -internal CodeVar parse_variable_after_name ( ModuleFlag mflags, CodeAttributes attributes, CodeSpecifiers specifiers, CodeTypename type, Str name ); -internal CodeVar parse_variable_declaration_list (); +internal CodeOperator parse_operator_after_ret_type (Context* ctx, ModuleFlag mflags, CodeAttributes attributes, CodeSpecifiers specifiers, CodeTypename ret_type); +internal Code parse_operator_function_or_variable(Context* ctx, bool expects_function, CodeAttributes attributes, CodeSpecifiers specifiers); +internal CodePragma parse_pragma (Context* ctx); +internal CodeParams parse_params (Context* ctx, bool use_template_capture); +internal CodePreprocessCond parse_preprocess_cond (Context* ctx); +internal Code parse_simple_preprocess (Context* ctx, TokType which); +internal Code parse_static_assert (Context* ctx); +internal void parse_template_args (Context* ctx, Token* token ); +internal CodeVar parse_variable_after_name (Context* ctx, ModuleFlag mflags, CodeAttributes attributes, CodeSpecifiers specifiers, CodeTypename type, Str name); +internal CodeVar parse_variable_declaration_list (Context* ctx); -internal CodeClass parser_parse_class ( bool inplace_def ); -internal CodeConstructor parser_parse_constructor ( CodeSpecifiers specifiers ); -internal CodeDefine parser_parse_define (); -internal CodeDestructor parser_parse_destructor ( CodeSpecifiers specifiers ); -internal CodeEnum parser_parse_enum ( bool inplace_def ); -internal CodeBody parser_parse_export_body (); -internal CodeBody parser_parse_extern_link_body(); -internal CodeExtern parser_parse_extern_link (); -internal CodeFriend parser_parse_friend (); -internal CodeFn parser_parse_function (); -internal CodeNS parser_parse_namespace (); -internal CodeOpCast parser_parse_operator_cast ( CodeSpecifiers specifiers ); -internal CodeStruct parser_parse_struct ( bool inplace_def ); -internal CodeVar parser_parse_variable (); -internal CodeTemplate parser_parse_template (); -internal CodeTypename parser_parse_type ( bool from_template, bool* is_function ); -internal CodeTypedef parser_parse_typedef (); -internal CodeUnion parser_parse_union ( bool inplace_def ); -internal CodeUsing parser_parse_using (); +internal CodeClass parser_parse_class (Context* ctx, bool inplace_def ); +internal CodeConstructor parser_parse_constructor (Context* ctx, CodeSpecifiers specifiers ); +internal CodeDefine parser_parse_define (Context* ctx); +internal CodeDestructor parser_parse_destructor (Context* ctx, CodeSpecifiers specifiers ); +internal CodeEnum parser_parse_enum (Context* ctx, bool inplace_def ); +internal CodeBody parser_parse_export_body (Context* ctx); +internal CodeBody parser_parse_extern_link_body(Context* ctx); +internal CodeExtern parser_parse_extern_link (Context* ctx); +internal CodeFriend parser_parse_friend (Context* ctx); +internal CodeFn parser_parse_function (Context* ctx); +internal CodeNS parser_parse_namespace (Context* ctx); +internal CodeOpCast parser_parse_operator_cast (Context* ctx, CodeSpecifiers specifiers ); +internal CodeStruct parser_parse_struct (Context* ctx, bool inplace_def ); +internal CodeVar parser_parse_variable (Context* ctx); +internal CodeTemplate parser_parse_template (Context* ctx); +internal CodeTypename parser_parse_type (Context* ctx, bool from_template, bool* is_function ); +internal CodeTypedef parser_parse_typedef (Context* ctx); +internal CodeUnion parser_parse_union (Context* ctx, bool inplace_def ); +internal CodeUsing parser_parse_using (Context* ctx); constexpr bool parser_inplace_def = true; constexpr bool parser_not_inplace_def = false; @@ -259,9 +252,9 @@ constexpr bool parser_strip_formatting_dont_preserve_newlines = false; It has edge case failures that prevent it from being used in function bodies. */ internal -StrBuilder parser_strip_formatting( Str raw_text, bool preserve_newlines ) +StrBuilder parser_strip_formatting(Context* ctx, Str raw_text, bool preserve_newlines ) { - StrBuilder content = strbuilder_make_reserve( _ctx->Allocator_Temp, raw_text.Len ); + StrBuilder content = strbuilder_make_reserve( ctx->Allocator_Temp, raw_text.Len ); if ( raw_text.Len == 0 ) return content; @@ -527,14 +520,14 @@ Code parse_array_decl(Context* ctx) if ( left == 0 ) { - log_failure( "Error, unexpected end of array declaration ( '[]' scope started )\n%s", parser_to_strbuilder(ctx->parser) ); + log_failure( "Error, unexpected end of array declaration ( '[]' scope started )\n%s", parser_to_strbuilder(ctx->parser, ctx->Allocator_Temp) ); parser_pop(& ctx->parser); return InvalidCode; } if ( currtok.Type == Tok_BraceSquare_Close ) { - log_failure( "Error, empty array expression in definition\n%s", parser_to_strbuilder(ctx->parser) ); + log_failure( "Error, empty array expression in definition\n%s", parser_to_strbuilder(ctx->parser, ctx->Allocator_Temp) ); parser_pop(& ctx->parser); return InvalidCode; } @@ -557,14 +550,14 @@ Code parse_array_decl(Context* ctx) if ( left == 0 ) { - log_failure( "Error, unexpected end of array declaration, expected ]\n%s", parser_to_strbuilder(ctx->parser) ); + log_failure( "Error, unexpected end of array declaration, expected ]\n%s", parser_to_strbuilder(ctx->parser, ctx->Allocator_Temp) ); parser_pop(& ctx->parser); return InvalidCode; } if ( currtok.Type != Tok_BraceSquare_Close ) { - log_failure( "%s: Error, expected ] in array declaration, not %s\n%s", toktype_to_str( currtok.Type ), parser_to_strbuilder(ctx->parser) ); + log_failure( "%s: Error, expected ] in array declaration, not %s\n%s", toktype_to_str( currtok.Type ), parser_to_strbuilder(ctx->parser, ctx->Allocator_Temp) ); parser_pop(& ctx->parser); return InvalidCode; } @@ -686,7 +679,7 @@ CodeAttributes parse_attributes(Context* ctx) Str attribute_txt = { start.Text.Ptr, len }; parser_pop(& ctx->parser); - StrBuilder name_stripped = parser_strip_formatting( attribute_txt, parser_strip_formatting_dont_preserve_newlines ); + StrBuilder name_stripped = parser_strip_formatting(ctx, attribute_txt, parser_strip_formatting_dont_preserve_newlines ); Code result = make_code(); result->Type = CT_PlatformAttributes; @@ -704,7 +697,7 @@ internal Code parse_class_struct(Context* ctx, TokType which, bool inplace_def) { if ( which != Tok_Decl_Class && which != Tok_Decl_Struct ) { - log_failure( "Error, expected class or struct, not %s\n%s", toktype_to_str( which ), parser_to_strbuilder(ctx->parser) ); + log_failure( "Error, expected class or struct, not %s\n%s", toktype_to_str( which ), parser_to_strbuilder(ctx->parser, ctx->Allocator_Temp) ); return InvalidCode; } @@ -896,12 +889,12 @@ CodeBody parse_class_struct_body(Context* ctx, TokType which, Token name) break; } case Tok_Decl_Friend: { - member = cast(Code, parser_parse_friend()); + member = cast(Code, parser_parse_friend(ctx)); // friend break; } case Tok_Decl_Operator: { - member = cast(Code, parser_parse_operator_cast(NullCode)); + member = cast(Code, parser_parse_operator_cast(ctx, NullCode)); // operator () break; } @@ -911,12 +904,12 @@ CodeBody parse_class_struct_body(Context* ctx, TokType which, Token name) break; } case Tok_Decl_Template: { - member = cast(Code, parser_parse_template()); + member = cast(Code, parser_parse_template(ctx)); // template< ... > break; } case Tok_Decl_Typedef: { - member = cast(Code, parser_parse_typedef()); + member = cast(Code, parser_parse_typedef(ctx)); // typedef break; } @@ -926,7 +919,7 @@ CodeBody parse_class_struct_body(Context* ctx, TokType which, Token name) break; } case Tok_Decl_Using: { - member = cast(Code, parser_parse_using()); + member = cast(Code, parser_parse_using(ctx)); // using break; } @@ -938,12 +931,12 @@ CodeBody parse_class_struct_body(Context* ctx, TokType which, Token name) // return InvalidCode; //} - member = cast(Code, parser_parse_destructor(NullCode)); + member = cast(Code, parser_parse_destructor(ctx, NullCode)); // ~() break; } case Tok_Preprocess_Define: { - member = cast(Code, parser_parse_define()); + member = cast(Code, parser_parse_define(ctx)); // #define break; } @@ -958,7 +951,7 @@ CodeBody parse_class_struct_body(Context* ctx, TokType which, Token name) case Tok_Preprocess_IfDef: case Tok_Preprocess_IfNotDef: case Tok_Preprocess_ElIf: - member = cast(Code, parse_preprocess_cond()); + member = cast(Code, parse_preprocess_cond(ctx)); // # break; @@ -976,7 +969,7 @@ CodeBody parse_class_struct_body(Context* ctx, TokType which, Token name) } case Tok_Preprocess_Macro_Stmt: { - member = cast(Code, parse_simple_preprocess( Tok_Preprocess_Macro_Stmt )); + member = cast(Code, parse_simple_preprocess(ctx, Tok_Preprocess_Macro_Stmt )); break; } @@ -987,19 +980,19 @@ CodeBody parse_class_struct_body(Context* ctx, TokType which, Token name) // break; case Tok_Preprocess_Pragma: { - member = cast(Code, parse_pragma()); + member = cast(Code, parse_pragma(ctx)); // #pragma break; } case Tok_Preprocess_Unsupported: { - member = cast(Code, parse_simple_preprocess( Tok_Preprocess_Unsupported )); + member = cast(Code, parse_simple_preprocess(ctx, Tok_Preprocess_Unsupported)); // # break; } case Tok_StaticAssert: { - member = parse_static_assert(); + member = parse_static_assert(ctx); // static_assert break; } @@ -1008,7 +1001,7 @@ CodeBody parse_class_struct_body(Context* ctx, TokType which, Token name) { if ( ! tok_is_attribute(currtok)) { - log_failure("Unbounded macro expression residing in class/struct body\n%S", parser_to_strbuilder(ctx->parser)); + log_failure("Unbounded macro expression residing in class/struct body\n%S", parser_to_strbuilder(ctx->parser, ctx->Allocator_Temp)); return InvalidCode; } } @@ -1049,7 +1042,7 @@ CodeBody parse_class_struct_body(Context* ctx, TokType which, Token name) break; default: - log_failure( "Invalid specifier %S for class/struct member\n%S", spec_to_str(spec), strbuilder_to_str( parser_to_strbuilder(ctx->parser)) ); + log_failure( "Invalid specifier %S for class/struct member\n%S", spec_to_str(spec), strbuilder_to_str( parser_to_strbuilder(ctx->parser, ctx->Allocator_Temp)) ); parser_pop(& ctx->parser); return InvalidCode; } @@ -1090,14 +1083,14 @@ CodeBody parse_class_struct_body(Context* ctx, TokType which, Token name) if ( currtok.Type == Tok_Operator && currtok.Text.Ptr[0] == '~' ) { - member = cast(Code, parser_parse_destructor( specifiers )); + member = cast(Code, parser_parse_destructor(ctx, specifiers )); // ~() break; } if ( currtok.Type == Tok_Decl_Operator ) { - member = cast(Code, parser_parse_operator_cast( specifiers )); + member = cast(Code, parser_parse_operator_cast(ctx, specifiers )); // operator () break; } @@ -1119,13 +1112,13 @@ CodeBody parse_class_struct_body(Context* ctx, TokType which, Token name) { if ( c_str_compare_len( name.Text.Ptr, currtok.Text.Ptr, name.Text.Len ) == 0 ) { - member = cast(Code, parser_parse_constructor( specifiers )); + member = cast(Code, parser_parse_constructor(ctx, specifiers )); // () break; } } - member = parse_operator_function_or_variable( expects_function, attributes, specifiers ); + member = parse_operator_function_or_variable(ctx, expects_function, attributes, specifiers ); // operator ... // or // ... @@ -1146,7 +1139,7 @@ CodeBody parse_class_struct_body(Context* ctx, TokType which, Token name) if ( member == Code_Invalid ) { - log_failure( "Failed to parse member\n%s", parser_to_strbuilder(ctx->parser) ); + log_failure( "Failed to parse member\n%s", parser_to_strbuilder(ctx->parser, ctx->Allocator_Temp) ); parser_pop(& ctx->parser); return InvalidCode; } @@ -1211,7 +1204,7 @@ Code parse_complicated_definition(Context* ctx, TokType which) if (is_fn_def) { // Function definition with on return type - Code result = parse_operator_function_or_variable(false, NullCode, NullCode); + Code result = parse_operator_function_or_variable(ctx, false, NullCode, NullCode); // (...) ... { ... } parser_pop(& ctx->parser); return result; @@ -1244,13 +1237,13 @@ Code parse_complicated_definition(Context* ctx, TokType which) // Forward declaration with trailing specifiers for a procedure tok = tokens.Arr[spec_idx]; - Code result = parse_operator_function_or_variable( false, NullCode, NullCode ); + Code result = parse_operator_function_or_variable(ctx, false, NullCode, NullCode); // , or Name> ... parser_pop(& ctx->parser); return result; } - log_failure( "Unsupported or bad member definition after %s declaration\n%s", toktype_to_str(which), parser_to_strbuilder(ctx->parser) ); + log_failure( "Unsupported or bad member definition after %s declaration\n%s", toktype_to_str(which), parser_to_strbuilder(ctx->parser, ctx->Allocator_Temp) ); parser_pop(& ctx->parser); return InvalidCode; } @@ -1273,7 +1266,7 @@ Code parse_complicated_definition(Context* ctx, TokType which) Token name = parse_identifier(ctx, nullptr); ctx->parser.Scope->Name = name.Text; - CodeVar result = parse_variable_after_name(ModuleFlag_None, NullCode, NullCode, type, name.Text); + CodeVar result = parse_variable_after_name(ctx, ModuleFlag_None, NullCode, NullCode, type, name.Text); parser_pop(& ctx->parser); return (Code) result; } @@ -1292,7 +1285,7 @@ Code parse_complicated_definition(Context* ctx, TokType which) // : ; // : ; ok_to_parse = true; - Code result = cast(Code, parser_parse_enum( ! parser_inplace_def)); + Code result = cast(Code, parser_parse_enum(ctx, ! parser_inplace_def)); parser_pop(& ctx->parser); return result; } @@ -1305,12 +1298,12 @@ Code parse_complicated_definition(Context* ctx, TokType which) if ( ! ok_to_parse ) { - log_failure( "Unsupported or bad member definition after %s declaration\n%s", toktype_to_str(which), parser_to_strbuilder(ctx->parser) ); + log_failure( "Unsupported or bad member definition after %s declaration\n%s", toktype_to_str(which), parser_to_strbuilder(ctx->parser, ctx->Allocator_Temp) ); parser_pop(& ctx->parser); return InvalidCode; } - Code result = parse_operator_function_or_variable( false, NullCode, NullCode ); + Code result = parse_operator_function_or_variable(ctx, false, NullCode, NullCode ); // , or Name> ... parser_pop(& ctx->parser); return result; @@ -1324,7 +1317,7 @@ Code parse_complicated_definition(Context* ctx, TokType which) && ( tokens.Arr[idx - 4].Type != which)) ) { - log_failure( "Unsupported or bad member definition after %s declaration\n%s", toktype_to_str(which), parser_to_strbuilder(ctx->parser) ); + log_failure( "Unsupported or bad member definition after %s declaration\n%s", toktype_to_str(which), parser_to_strbuilder(ctx->parser, ctx->Allocator_Temp) ); parser_pop(& ctx->parser); return InvalidCode; } @@ -1332,7 +1325,7 @@ Code parse_complicated_definition(Context* ctx, TokType which) // Its a forward declaration of an enum class // : ; // : ; - Code result = cast(Code, parser_parse_enum( ! parser_inplace_def)); + Code result = cast(Code, parser_parse_enum(ctx, ! parser_inplace_def)); parser_pop(& ctx->parser); return result; } @@ -1347,14 +1340,14 @@ Code parse_complicated_definition(Context* ctx, TokType which) else if ( tok.Type == Tok_BraceSquare_Close ) { // Its an array definition - Code result = parse_operator_function_or_variable( false, NullCode, NullCode ); + Code result = parse_operator_function_or_variable(ctx, false, NullCode, NullCode ); // [ ... ]; parser_pop(& ctx->parser); return result; } else { - log_failure( "Unsupported or bad member definition after %s declaration\n%SB", toktype_to_str(which).Ptr, parser_to_strbuilder(ctx->parser) ); + log_failure( "Unsupported or bad member definition after %s declaration\n%SB", toktype_to_str(which).Ptr, parser_to_strbuilder(ctx->parser, ctx->Allocator_Temp) ); parser_pop(& ctx->parser); return InvalidCode; } @@ -1375,7 +1368,7 @@ Code parse_assignment_expression(Context* ctx) if ( currtok.Type == Tok_Statement_End && currtok.Type != Tok_Comma ) { - log_failure( "Expected expression after assignment operator\n%s", parser_to_strbuilder(ctx->parser) ); + log_failure( "Expected expression after assignment operator\n%s", parser_to_strbuilder(ctx->parser, ctx->Allocator_Temp) ); parser_pop(& ctx->parser); return InvalidCode; } @@ -1413,25 +1406,25 @@ Code parse_forward_or_definition(Context* ctx, TokType which, bool is_inplace ) switch ( which ) { case Tok_Decl_Class: - result = cast(Code, parser_parse_class( is_inplace )); + result = cast(Code, parser_parse_class(ctx, is_inplace )); return result; case Tok_Decl_Enum: - result = cast(Code, parser_parse_enum( is_inplace )); + result = cast(Code, parser_parse_enum(ctx, is_inplace )); return result; case Tok_Decl_Struct: - result = cast(Code, parser_parse_struct( is_inplace )); + result = cast(Code, parser_parse_struct(ctx, is_inplace )); return result; case Tok_Decl_Union: - result = cast(Code, parser_parse_union( is_inplace )); + result = cast(Code, parser_parse_union(ctx, is_inplace )); return result; default: log_failure( "Error, wrong token type given to parse_complicated_definition " "(only supports class, enum, struct, union) \n%s" - , parser_to_strbuilder(ctx->parser) ); + , parser_to_strbuilder(ctx->parser, ctx->Allocator_Temp) ); return InvalidCode; } @@ -1450,7 +1443,7 @@ CodeFn parse_function_after_name(Context* ctx ParseStackNode scope = NullScope; parser_push(& ctx->parser, & scope ); - CodeParams params = parse_params(parser_use_parenthesis); + CodeParams params = parse_params(ctx, parser_use_parenthesis); // ( ) Code suffix_specs = NullCode; @@ -1462,7 +1455,7 @@ CodeFn parse_function_after_name(Context* ctx Macro* macro = lookup_macro( currtok.Text ); if (macro && tok_is_specifier(currtok)) { - suffix_specs = parse_simple_preprocess(Tok_Preprocess_Macro_Expr); + suffix_specs = parse_simple_preprocess(ctx, Tok_Preprocess_Macro_Expr); continue; } if ( specifiers == nullptr ) @@ -1565,7 +1558,7 @@ CodeFn parse_function_after_name(Context* ctx default: { - log_failure("Body must be either of Function_Body or Untyped type, %s\n%s", code_debug_str(body), parser_to_strbuilder(ctx->parser)); + log_failure("Body must be either of Function_Body or Untyped type, %s\n%s", code_debug_str(body), parser_to_strbuilder(ctx->parser, ctx->Allocator_Temp)); parser_pop(& ctx->parser); return InvalidCode; } @@ -1679,7 +1672,7 @@ CodeBody parse_global_nspace(Context* ctx, CodeType which) { case Tok_Comma: { - log_failure("Dangling comma found: %SB\nContext:\n%SB", tok_to_strbuilder(currtok), parser_to_strbuilder(ctx->parser)); + log_failure("Dangling comma found: %SB\nContext:\n%SB", tok_to_strbuilder(currtok), parser_to_strbuilder(ctx->parser, ctx->Allocator_Temp)); parser_pop( & ctx->parser); return InvalidCode; } @@ -1713,14 +1706,14 @@ CodeBody parse_global_nspace(Context* ctx, CodeType which) case Tok_Decl_Extern_Linkage: if ( which == CT_Extern_Linkage_Body ) - log_failure( "Nested extern linkage\n%s", parser_to_strbuilder(ctx->parser) ); + log_failure( "Nested extern linkage\n%s", parser_to_strbuilder(ctx->parser, ctx->Allocator_Temp) ); - member = cast(Code, parser_parse_extern_link()); + member = cast(Code, parser_parse_extern_link(ctx)); // extern "..." { ... } break; case Tok_Decl_Namespace: - member = cast(Code, parser_parse_namespace()); + member = cast(Code, parser_parse_namespace(ctx)); // namespace { ... } break; @@ -1730,12 +1723,12 @@ CodeBody parse_global_nspace(Context* ctx, CodeType which) break; case Tok_Decl_Template: - member = cast(Code, parser_parse_template()); + member = cast(Code, parser_parse_template(ctx)); // template<...> ... break; case Tok_Decl_Typedef: - member = cast(Code, parser_parse_typedef()); + member = cast(Code, parser_parse_typedef(ctx)); // typedef ... break; @@ -1745,12 +1738,12 @@ CodeBody parse_global_nspace(Context* ctx, CodeType which) break; case Tok_Decl_Using: - member = cast(Code, parser_parse_using()); + member = cast(Code, parser_parse_using(ctx)); // using ... break; case Tok_Preprocess_Define: - member = cast(Code, parser_parse_define()); + member = cast(Code, parser_parse_define(ctx)); // #define ... break; @@ -1763,7 +1756,7 @@ CodeBody parse_global_nspace(Context* ctx, CodeType which) case Tok_Preprocess_IfDef: case Tok_Preprocess_IfNotDef: case Tok_Preprocess_ElIf: - member = cast(Code, parse_preprocess_cond()); + member = cast(Code, parse_preprocess_cond(ctx)); // # ... break; @@ -1780,33 +1773,33 @@ CodeBody parse_global_nspace(Context* ctx, CodeType which) break; case Tok_Preprocess_Macro_Stmt: { - member = cast(Code, parse_simple_preprocess( Tok_Preprocess_Macro_Stmt )); + member = cast(Code, parse_simple_preprocess(ctx, Tok_Preprocess_Macro_Stmt )); break; } case Tok_Preprocess_Pragma: { - member = cast(Code, parse_pragma()); + member = cast(Code, parse_pragma(ctx)); // #pragma ... } break; case Tok_Preprocess_Unsupported: { - member = cast(Code, parse_simple_preprocess( Tok_Preprocess_Unsupported )); + member = cast(Code, parse_simple_preprocess(ctx, Tok_Preprocess_Unsupported )); // # ... } break; case Tok_StaticAssert: { - member = cast(Code, parse_static_assert()); + member = cast(Code, parse_static_assert(ctx)); // static_assert( , ... ); } break; case Tok_Module_Export: { if ( which == CT_Export_Body ) - log_failure( "Nested export declaration\n%s", parser_to_strbuilder(ctx->parser) ); + log_failure( "Nested export declaration\n%s", parser_to_strbuilder(ctx->parser, ctx->Allocator_Temp) ); - member = cast(Code, parser_parse_export_body()); + member = cast(Code, parser_parse_export_body(ctx)); // export { ... } } break; @@ -1822,7 +1815,7 @@ CodeBody parse_global_nspace(Context* ctx, CodeType which) { if ( ! tok_is_attribute(currtok)) { - log_failure("Unbounded macro expression residing in class/struct body\n%S", parser_to_strbuilder(ctx->parser)); + log_failure("Unbounded macro expression residing in class/struct body\n%S", parser_to_strbuilder(ctx->parser, ctx->Allocator_Temp)); return InvalidCode; } } @@ -1865,7 +1858,7 @@ CodeBody parse_global_nspace(Context* ctx, CodeType which) default: Str spec_str = spec_to_str(spec); - log_failure( "Invalid specifier %S for variable\n%S", spec_str, strbuilder_to_str( parser_to_strbuilder(ctx->parser)) ); + log_failure( "Invalid specifier %S for variable\n%S", spec_str, strbuilder_to_str( parser_to_strbuilder(ctx->parser, ctx->Allocator_Temp)) ); parser_pop(& ctx->parser); return InvalidCode; } @@ -1932,20 +1925,20 @@ CodeBody parse_global_nspace(Context* ctx, CodeType which) if ( found_operator_cast_outside_class_implmentation ) { - member = cast(Code, parser_parse_operator_cast( specifiers )); + member = cast(Code, parser_parse_operator_cast(ctx, specifiers )); // ::operator () { ... } break; } } - member = parse_operator_function_or_variable( expects_function, attributes, specifiers ); + member = parse_operator_function_or_variable(ctx, expects_function, attributes, specifiers ); // ... } } if ( member == Code_Invalid ) { - log_failure( "Failed to parse member\nToken: %SB\nContext:\n%SB", tok_to_strbuilder(currtok_noskip), parser_to_strbuilder(_ctx->parser) ); + log_failure( "Failed to parse member\nToken: %SB\nContext:\n%SB", tok_to_strbuilder(currtok_noskip), parser_to_strbuilder(ctx->parser, ctx->Allocator_Temp) ); parser_pop(& ctx->parser); return InvalidCode; } @@ -2085,11 +2078,11 @@ Code parse_global_nspace_constructor_destructor(Context* ctx, CodeSpecifiers spe if (possible_destructor) { // :: ~ ( - result = cast(Code, parser_parse_destructor( specifiers )); + result = cast(Code, parser_parse_destructor(ctx, specifiers )); } else { // :: ( - result = cast(Code, parser_parse_constructor( specifiers )); + result = cast(Code, parser_parse_constructor(ctx, specifiers )); } } @@ -2118,7 +2111,7 @@ Token parse_identifier(Context* ctx, bool* possible_member_function) if (check(Tok_Identifier) || accept_as_identifier) { if (is_decarator) { - Code name_macro = parse_simple_preprocess(currtok.Type); + Code name_macro = parse_simple_preprocess(ctx, currtok.Type); name.Text.Len = ( ( sptr )prevtok.Text.Ptr + prevtok.Text.Len ) - ( sptr )name.Text.Ptr; } else { @@ -2127,7 +2120,7 @@ Token parse_identifier(Context* ctx, bool* possible_member_function) } // - parse_template_args( & name ); + parse_template_args(ctx, & name); //