mirror of
https://github.com/Ed94/gencpp.git
synced 2024-11-10 02:54:53 -08:00
Updated parsing constructors with impl to use the lexer.
This commit is contained in:
parent
86cd0e1fb7
commit
4adb4654e9
@ -254,4 +254,3 @@ sw fatal(char const *fmt, ...)
|
|||||||
return -1;
|
return -1;
|
||||||
#endif
|
#endif
|
||||||
}
|
}
|
||||||
|
|
||||||
|
645
project/gen.cpp
645
project/gen.cpp
@ -2602,7 +2602,7 @@ namespace gen
|
|||||||
# define eat( Type_ ) \
|
# define eat( Type_ ) \
|
||||||
if ( curr_tok.Type != Type_ ) \
|
if ( curr_tok.Type != Type_ ) \
|
||||||
{ \
|
{ \
|
||||||
String token_str = string_make_length( g_allocator, curr_tok.Address, curr_tok.Length ); \
|
String token_str = string_make_length( g_allocator, curr_tok.Text, curr_tok.Length ); \
|
||||||
log_failure( "gen::" txt(context) ": expected %s, got %s", txt(Type_), curr_tok.Type ); \
|
log_failure( "gen::" txt(context) ": expected %s, got %s", txt(Type_), curr_tok.Type ); \
|
||||||
return Code::Invalid; \
|
return Code::Invalid; \
|
||||||
} \
|
} \
|
||||||
@ -2642,6 +2642,7 @@ namespace gen
|
|||||||
Entry( Decl_Friend, "friend" ) \
|
Entry( Decl_Friend, "friend" ) \
|
||||||
Entry( Decl_Namespace, "namespace" ) \
|
Entry( Decl_Namespace, "namespace" ) \
|
||||||
Entry( Decl_Struct, "struct" ) \
|
Entry( Decl_Struct, "struct" ) \
|
||||||
|
Entry( Decl_Typedef, "typedef" ) \
|
||||||
Entry( Decl_Using, "using" ) \
|
Entry( Decl_Using, "using" ) \
|
||||||
Entry( Decl_Union, "union" ) \
|
Entry( Decl_Union, "union" ) \
|
||||||
Entry( Identifier, "__SymID__" ) \
|
Entry( Identifier, "__SymID__" ) \
|
||||||
@ -2664,7 +2665,11 @@ namespace gen
|
|||||||
Entry( Spec_Volatile, "volatile") \
|
Entry( Spec_Volatile, "volatile") \
|
||||||
Entry( Star, "*" ) \
|
Entry( Star, "*" ) \
|
||||||
Entry( Statement_End, ";" ) \
|
Entry( Statement_End, ";" ) \
|
||||||
Entry( String, "__String__" )
|
Entry( String, "__String__" ) \
|
||||||
|
Entry( Type_Unsigned, "unsigned" ) \
|
||||||
|
Entry( Type_Signed, "signed" ) \
|
||||||
|
Entry( Type_Short, "short" ) \
|
||||||
|
Entry( Type_Long, "long" )
|
||||||
|
|
||||||
enum class TokType : u32
|
enum class TokType : u32
|
||||||
{
|
{
|
||||||
@ -2678,7 +2683,7 @@ namespace gen
|
|||||||
|
|
||||||
struct Token
|
struct Token
|
||||||
{
|
{
|
||||||
char const* Address;
|
char const* Text;
|
||||||
s32 Length;
|
s32 Length;
|
||||||
TokType Type;
|
TokType Type;
|
||||||
};
|
};
|
||||||
@ -2702,6 +2707,12 @@ namespace gen
|
|||||||
return TokType::Invalid;
|
return TokType::Invalid;
|
||||||
}
|
}
|
||||||
|
|
||||||
|
inline
|
||||||
|
bool tok_is_specifier( Token const& tok )
|
||||||
|
{
|
||||||
|
return tok.Type >= TokType::Spec_API && tok.Type <= TokType::Spec_Volatile;
|
||||||
|
}
|
||||||
|
|
||||||
Arena LexAllocator;
|
Arena LexAllocator;
|
||||||
|
|
||||||
Array(Token) lex( s32 length, char const* content)
|
Array(Token) lex( s32 length, char const* content)
|
||||||
@ -2744,7 +2755,7 @@ namespace gen
|
|||||||
switch ( current )
|
switch ( current )
|
||||||
{
|
{
|
||||||
case '.':
|
case '.':
|
||||||
token.Address = scanner;
|
token.Text = scanner;
|
||||||
token.Length = 1;
|
token.Length = 1;
|
||||||
token.Type = TokType::Access_MemberSymbol;
|
token.Type = TokType::Access_MemberSymbol;
|
||||||
|
|
||||||
@ -2753,7 +2764,7 @@ namespace gen
|
|||||||
goto FoundToken;
|
goto FoundToken;
|
||||||
|
|
||||||
case '&' :
|
case '&' :
|
||||||
token.Address = scanner;
|
token.Text = scanner;
|
||||||
token.Length = 1;
|
token.Length = 1;
|
||||||
token.Type = TokType::Ampersand;
|
token.Type = TokType::Ampersand;
|
||||||
|
|
||||||
@ -2772,7 +2783,7 @@ namespace gen
|
|||||||
goto FoundToken;
|
goto FoundToken;
|
||||||
|
|
||||||
case ':':
|
case ':':
|
||||||
token.Address = scanner;
|
token.Text = scanner;
|
||||||
token.Length = 1;
|
token.Length = 1;
|
||||||
token.Type = TokType::Assign_Classifer;
|
token.Type = TokType::Assign_Classifer;
|
||||||
|
|
||||||
@ -2788,19 +2799,19 @@ namespace gen
|
|||||||
goto FoundToken;
|
goto FoundToken;
|
||||||
|
|
||||||
case '{':
|
case '{':
|
||||||
token.Address = scanner;
|
token.Text = scanner;
|
||||||
token.Length = 1;
|
token.Length = 1;
|
||||||
token.Type = TokType::BraceCurly_Open;
|
token.Type = TokType::BraceCurly_Open;
|
||||||
goto FoundToken;
|
goto FoundToken;
|
||||||
|
|
||||||
case '}':
|
case '}':
|
||||||
token.Address = scanner;
|
token.Text = scanner;
|
||||||
token.Length = 1;
|
token.Length = 1;
|
||||||
token.Type = TokType::BraceCurly_Close;
|
token.Type = TokType::BraceCurly_Close;
|
||||||
goto FoundToken;
|
goto FoundToken;
|
||||||
|
|
||||||
case '[':
|
case '[':
|
||||||
token.Address = scanner;
|
token.Text = scanner;
|
||||||
token.Length = 1;
|
token.Length = 1;
|
||||||
token.Type = TokType::BraceSquare_Open;
|
token.Type = TokType::BraceSquare_Open;
|
||||||
if ( left )
|
if ( left )
|
||||||
@ -2817,13 +2828,13 @@ namespace gen
|
|||||||
goto FoundToken;
|
goto FoundToken;
|
||||||
|
|
||||||
case ']':
|
case ']':
|
||||||
token.Address = scanner;
|
token.Text = scanner;
|
||||||
token.Length = 1;
|
token.Length = 1;
|
||||||
token.Type = TokType::BraceSquare_Close;
|
token.Type = TokType::BraceSquare_Close;
|
||||||
goto FoundToken;
|
goto FoundToken;
|
||||||
|
|
||||||
case '(':
|
case '(':
|
||||||
token.Address = scanner;
|
token.Text = scanner;
|
||||||
token.Length = 1;
|
token.Length = 1;
|
||||||
token.Type = TokType::Capture_Start;
|
token.Type = TokType::Capture_Start;
|
||||||
if ( left )
|
if ( left )
|
||||||
@ -2840,13 +2851,13 @@ namespace gen
|
|||||||
goto FoundToken;
|
goto FoundToken;
|
||||||
|
|
||||||
case ')':
|
case ')':
|
||||||
token.Address = scanner;
|
token.Text = scanner;
|
||||||
token.Length = 1;
|
token.Length = 1;
|
||||||
token.Type = TokType::Capture_End;
|
token.Type = TokType::Capture_End;
|
||||||
goto FoundToken;
|
goto FoundToken;
|
||||||
|
|
||||||
case '\'':
|
case '\'':
|
||||||
token.Address = scanner;
|
token.Text = scanner;
|
||||||
token.Length = 1;
|
token.Length = 1;
|
||||||
token.Type = TokType::Char;
|
token.Type = TokType::Char;
|
||||||
|
|
||||||
@ -2866,25 +2877,25 @@ namespace gen
|
|||||||
goto FoundToken;
|
goto FoundToken;
|
||||||
|
|
||||||
case ',':
|
case ',':
|
||||||
token.Address = scanner;
|
token.Text = scanner;
|
||||||
token.Length = 1;
|
token.Length = 1;
|
||||||
token.Type = TokType::Comma;
|
token.Type = TokType::Comma;
|
||||||
goto FoundToken;
|
goto FoundToken;
|
||||||
|
|
||||||
case '*':
|
case '*':
|
||||||
token.Address = scanner;
|
token.Text = scanner;
|
||||||
token.Length = 1;
|
token.Length = 1;
|
||||||
token.Type = TokType::Star;
|
token.Type = TokType::Star;
|
||||||
goto FoundToken;
|
goto FoundToken;
|
||||||
|
|
||||||
case ';':
|
case ';':
|
||||||
token.Address = scanner;
|
token.Text = scanner;
|
||||||
token.Length = 1;
|
token.Length = 1;
|
||||||
token.Type = TokType::Statement_End;
|
token.Type = TokType::Statement_End;
|
||||||
goto FoundToken;
|
goto FoundToken;
|
||||||
|
|
||||||
case '"':
|
case '"':
|
||||||
token.Address = scanner;
|
token.Text = scanner;
|
||||||
token.Length = 1;
|
token.Length = 1;
|
||||||
token.Type = TokType::String;
|
token.Type = TokType::String;
|
||||||
|
|
||||||
@ -2925,7 +2936,7 @@ namespace gen
|
|||||||
case '<':
|
case '<':
|
||||||
case '>':
|
case '>':
|
||||||
case '|':
|
case '|':
|
||||||
token.Address = scanner;
|
token.Text = scanner;
|
||||||
token.Length = 1;
|
token.Length = 1;
|
||||||
token.Type = TokType::Operator;
|
token.Type = TokType::Operator;
|
||||||
|
|
||||||
@ -2950,7 +2961,7 @@ namespace gen
|
|||||||
|
|
||||||
// Dash is unfortunatlly a bit more complicated...
|
// Dash is unfortunatlly a bit more complicated...
|
||||||
case '-':
|
case '-':
|
||||||
token.Address = scanner;
|
token.Text = scanner;
|
||||||
token.Length = 1;
|
token.Length = 1;
|
||||||
token.Type = TokType::Operator;
|
token.Type = TokType::Operator;
|
||||||
if ( left )
|
if ( left )
|
||||||
@ -2986,7 +2997,7 @@ namespace gen
|
|||||||
goto FoundToken;
|
goto FoundToken;
|
||||||
|
|
||||||
case '/':
|
case '/':
|
||||||
token.Address = scanner;
|
token.Text = scanner;
|
||||||
token.Length = 1;
|
token.Length = 1;
|
||||||
token.Type = TokType::Operator;
|
token.Type = TokType::Operator;
|
||||||
|
|
||||||
@ -3026,7 +3037,7 @@ namespace gen
|
|||||||
|
|
||||||
if ( char_is_alpha( current ) || current == '_' )
|
if ( char_is_alpha( current ) || current == '_' )
|
||||||
{
|
{
|
||||||
token.Address = scanner;
|
token.Text = scanner;
|
||||||
token.Length = 1;
|
token.Length = 1;
|
||||||
move_forward();
|
move_forward();
|
||||||
|
|
||||||
@ -3059,7 +3070,7 @@ namespace gen
|
|||||||
continue;
|
continue;
|
||||||
}
|
}
|
||||||
|
|
||||||
TokType type = get_token_type( token.Address, token.Length );
|
TokType type = get_token_type( token.Text, token.Length );
|
||||||
|
|
||||||
if ( type != TokType::Invalid )
|
if ( type != TokType::Invalid )
|
||||||
{
|
{
|
||||||
@ -3071,7 +3082,7 @@ namespace gen
|
|||||||
// Its most likely an identifier...
|
// Its most likely an identifier...
|
||||||
|
|
||||||
|
|
||||||
String tok_str = zpl::string_sprintf_buf( g_allocator, "%s", token.Address, token.Length );
|
String tok_str = zpl::string_sprintf_buf( g_allocator, "%s", token.Text, token.Length );
|
||||||
|
|
||||||
log_failure( "Failed to lex token %s", tok_str );
|
log_failure( "Failed to lex token %s", tok_str );
|
||||||
|
|
||||||
@ -3089,6 +3100,7 @@ namespace gen
|
|||||||
|
|
||||||
Code parse_class( s32 length, char const* def )
|
Code parse_class( s32 length, char const* def )
|
||||||
{
|
{
|
||||||
|
# define context parse_class
|
||||||
Array(Token) tokens = lex( length, def );
|
Array(Token) tokens = lex( length, def );
|
||||||
|
|
||||||
if ( tokens == nullptr || array_count( tokens ) == 0 )
|
if ( tokens == nullptr || array_count( tokens ) == 0 )
|
||||||
@ -3105,17 +3117,6 @@ namespace gen
|
|||||||
|
|
||||||
Token& curr_token = * tokens;
|
Token& curr_token = * tokens;
|
||||||
|
|
||||||
# define eat( Type_ ) \
|
|
||||||
if ( curr_token.Type != Type_ ) \
|
|
||||||
{ \
|
|
||||||
String token_str = string_make_length( g_allocator, curr_token.Address, curr_token.Length ); \
|
|
||||||
log_failure( "gen::parse_class: expected %s, got %s", txt(Type_), curr_token.Type ); \
|
|
||||||
return Code::Invalid; \
|
|
||||||
} \
|
|
||||||
tokens++; \
|
|
||||||
curr_token = * tokens; \
|
|
||||||
left--
|
|
||||||
|
|
||||||
s32 left = array_count( tokens );
|
s32 left = array_count( tokens );
|
||||||
do
|
do
|
||||||
{
|
{
|
||||||
@ -3135,128 +3136,106 @@ namespace gen
|
|||||||
}
|
}
|
||||||
}
|
}
|
||||||
while ( left--, left > 0 );
|
while ( left--, left > 0 );
|
||||||
|
|
||||||
# undef eat
|
|
||||||
}
|
}
|
||||||
|
|
||||||
Code parse_enum( s32 length, char const* def )
|
Code parse_enum( s32 length, char const* def )
|
||||||
{
|
{
|
||||||
check_parse_args( parse_enum, length, def );
|
# define context parse_type
|
||||||
|
|
||||||
|
check_parse_args( parse_typedef, length, def );
|
||||||
|
|
||||||
|
Array(Token) tokens = lex( length, def );
|
||||||
|
|
||||||
|
if ( tokens == nullptr )
|
||||||
|
{
|
||||||
|
log_failure( "gen::parse_typedef: no tokens found for provided definition" );
|
||||||
|
return Code::Invalid;
|
||||||
|
}
|
||||||
|
|
||||||
|
s32 left = array_count( tokens );
|
||||||
|
|
||||||
|
SpecifierT specs_found[16] { ESpecifier::Num_Specifiers };
|
||||||
|
s32 num_specifiers = 0;
|
||||||
|
|
||||||
|
Token* name = nullptr;
|
||||||
|
Code array_expr = { nullptr };
|
||||||
|
Code type = { nullptr };
|
||||||
|
Token body = { nullptr, 0, TokType::Invalid };
|
||||||
|
|
||||||
// Just in case someone gets a vulkan-level enum in here...
|
|
||||||
char entries_code[ kilobytes(128) ] { 0 };
|
char entries_code[ kilobytes(128) ] { 0 };
|
||||||
s32 entries_length = 0;
|
s32 entries_length = 0;
|
||||||
|
|
||||||
char const* name = nullptr;
|
|
||||||
s32 name_length = 0;
|
|
||||||
|
|
||||||
s32 left = length;
|
|
||||||
char const* scanner = def;
|
|
||||||
|
|
||||||
char const* word = scanner;
|
|
||||||
s32 word_length = 0;
|
|
||||||
|
|
||||||
Code type = { nullptr };
|
|
||||||
|
|
||||||
bool is_enum_class = false;
|
bool is_enum_class = false;
|
||||||
|
|
||||||
SkipWhitespace_Checked( parse_enum, "enum definition was empty" );
|
eat( TokType::Decl_Enum );
|
||||||
|
|
||||||
GetWord();
|
if ( curr_tok.Type == TokType::Decl_Class )
|
||||||
if ( word_length != 4 || str_compare( word, "enum", word_length ) != 0 )
|
|
||||||
{
|
|
||||||
log_failure( "gen::parse_enum: enum definition did not start with `enum`" );
|
|
||||||
return Code::Invalid;
|
|
||||||
}
|
|
||||||
|
|
||||||
SkipWhitespace_Checked( parse_enum, "enum definition did not have a name" );
|
|
||||||
|
|
||||||
GetWord();
|
|
||||||
if ( word_length == 5 && str_compare( word, "class", word_length ) == 0 )
|
|
||||||
{
|
{
|
||||||
|
eat( TokType::Decl_Class);
|
||||||
is_enum_class = true;
|
is_enum_class = true;
|
||||||
|
|
||||||
SkipWhitespace_Checked( parse_enum, "enum definition did not have a name" );
|
|
||||||
}
|
}
|
||||||
|
|
||||||
GetWord();
|
if ( curr_tok.Type != TokType::Identifier )
|
||||||
name = word;
|
|
||||||
name_length = word_length;
|
|
||||||
|
|
||||||
SkipWhitespace();
|
|
||||||
|
|
||||||
if ( current == ':' )
|
|
||||||
{
|
{
|
||||||
move_forward();
|
log_failure( "gen::parse_enum: expected identifier for enum name" );
|
||||||
SkipWhitespace();
|
|
||||||
|
|
||||||
GetWord();
|
|
||||||
type = def_type( word_length, word, type );
|
|
||||||
}
|
|
||||||
|
|
||||||
SkipWhitespace();
|
|
||||||
|
|
||||||
if ( current == ';' )
|
|
||||||
{
|
|
||||||
goto Finished;
|
|
||||||
}
|
|
||||||
|
|
||||||
if ( current != '{' )
|
|
||||||
{
|
|
||||||
log_failure( "gen::parse_enum: enum definition did not have a body" );
|
|
||||||
return Code::Invalid;
|
|
||||||
}
|
|
||||||
move_forward();
|
|
||||||
|
|
||||||
char const* body_start = scanner;
|
|
||||||
do
|
|
||||||
{
|
|
||||||
if ( current == '}' )
|
|
||||||
{
|
|
||||||
move_forward();
|
|
||||||
break;
|
|
||||||
}
|
|
||||||
|
|
||||||
SkipWhitespace_Checked( parse_enum, "enum definition did not have a body" );
|
|
||||||
|
|
||||||
GetWord();
|
|
||||||
if ( word_length == 0 )
|
|
||||||
{
|
|
||||||
log_failure( "gen::parse_enum: enum definition did not have a body" );
|
|
||||||
return Code::Invalid;
|
return Code::Invalid;
|
||||||
}
|
}
|
||||||
|
|
||||||
entries_length += word_length;
|
name = tokens;
|
||||||
|
eat( TokType::Identifier );
|
||||||
|
|
||||||
if ( entries_length >= kilobytes(128) )
|
if ( curr_tok.Type == TokType::Assign_Classifer )
|
||||||
{
|
{
|
||||||
log_failure( "gen::parse_enum: enum definition had too many entries" );
|
eat( TokType::Assign_Classifer );
|
||||||
|
|
||||||
|
type = parse_type( curr_tok.Length, curr_tok.Text );
|
||||||
|
if ( type == Code::Invalid )
|
||||||
|
{
|
||||||
|
log_failure( "gen::parse_enum: failed to parse enum type" );
|
||||||
return Code::Invalid;
|
return Code::Invalid;
|
||||||
}
|
}
|
||||||
}
|
}
|
||||||
while(1);
|
|
||||||
|
|
||||||
Finished:
|
if ( curr_tok.Type == TokType::BraceCurly_Open )
|
||||||
|
{
|
||||||
|
eat( TokType::BraceCurly_Open );
|
||||||
|
|
||||||
|
body = curr_tok;
|
||||||
|
|
||||||
|
while ( curr_tok.Type != TokType::BraceCurly_Close )
|
||||||
|
{
|
||||||
|
body.Length += curr_tok.Length;
|
||||||
|
|
||||||
|
eat( curr_tok.Type );
|
||||||
|
}
|
||||||
|
|
||||||
|
eat( TokType::BraceCurly_Close );
|
||||||
|
}
|
||||||
|
else
|
||||||
|
{
|
||||||
|
eat( TokType::Statement_End );
|
||||||
|
}
|
||||||
|
|
||||||
using namespace ECode;
|
using namespace ECode;
|
||||||
|
|
||||||
Code
|
Code
|
||||||
result = make_code();
|
result = make_code();
|
||||||
|
|
||||||
if ( entries_length )
|
if ( body.Length )
|
||||||
{
|
{
|
||||||
memcopy( entries_code, body_start, entries_length );
|
memcopy( entries_code, body.Text, body.Length );
|
||||||
|
|
||||||
Code body = untyped_str( entries_length, entries_code );
|
Code untyped_body = untyped_str( entries_length, entries_code );
|
||||||
|
|
||||||
result->Type = is_enum_class ? Enum_Class : Enum;
|
result->Type = is_enum_class ? Enum_Class : Enum;
|
||||||
result->add_entry( body );
|
result->add_entry( untyped_body );
|
||||||
}
|
}
|
||||||
else
|
else
|
||||||
{
|
{
|
||||||
result->Type = is_enum_class ? Enum_Class_Fwd : Enum_Fwd;
|
result->Type = is_enum_class ? Enum_Class_Fwd : Enum_Fwd;
|
||||||
}
|
}
|
||||||
|
|
||||||
result->Name = get_cached_string( name, name_length );
|
result->Name = get_cached_string( name->Text, name->Length );
|
||||||
|
|
||||||
if ( type )
|
if ( type )
|
||||||
result->add_entry( type );
|
result->add_entry( type );
|
||||||
@ -3272,19 +3251,7 @@ namespace gen
|
|||||||
|
|
||||||
Code parse_friend( s32 length, char const* def )
|
Code parse_friend( s32 length, char const* def )
|
||||||
{
|
{
|
||||||
# define curr_tok tokens[0]
|
|
||||||
# define context parse_friend
|
# define context parse_friend
|
||||||
|
|
||||||
# define eat( Type_ ) \
|
|
||||||
if ( curr_tok.Type != Type_ ) \
|
|
||||||
{ \
|
|
||||||
String token_str = string_make_length( g_allocator, curr_tok.Address, curr_tok.Length ); \
|
|
||||||
log_failure( "gen::" txt(context) ": expected %s, got %s", txt(Type_), curr_tok.Type ); \
|
|
||||||
return Code::Invalid; \
|
|
||||||
} \
|
|
||||||
tokens++; \
|
|
||||||
left--
|
|
||||||
|
|
||||||
check_parse_args( parse_friend, length, def );
|
check_parse_args( parse_friend, length, def );
|
||||||
|
|
||||||
Array(Token) tokens = lex( length, def );
|
Array(Token) tokens = lex( length, def );
|
||||||
@ -3302,7 +3269,7 @@ namespace gen
|
|||||||
// If its a function declaration, it will have a return type, followed by a name, followed by a parameter list.
|
// If its a function declaration, it will have a return type, followed by a name, followed by a parameter list.
|
||||||
// If its a simple type, it will have a type, followed by a name.
|
// If its a simple type, it will have a type, followed by a name.
|
||||||
|
|
||||||
# undef eat
|
return Code::Invalid;
|
||||||
}
|
}
|
||||||
|
|
||||||
Code parse_global_body( s32 length, char const* def )
|
Code parse_global_body( s32 length, char const* def )
|
||||||
@ -3478,6 +3445,8 @@ namespace gen
|
|||||||
|
|
||||||
char const name [LengthID] { 0 };
|
char const name [LengthID] { 0 };
|
||||||
char const parent[LengthID] { 0 };
|
char const parent[LengthID] { 0 };
|
||||||
|
|
||||||
|
return Code::Invalid;
|
||||||
}
|
}
|
||||||
|
|
||||||
Code parse_variable( s32 length, char const* def )
|
Code parse_variable( s32 length, char const* def )
|
||||||
@ -3486,13 +3455,98 @@ namespace gen
|
|||||||
}
|
}
|
||||||
|
|
||||||
inline
|
inline
|
||||||
bool parse_type_helper_tok( char const* func_name
|
bool parse_type_helper( char const* func_name
|
||||||
|
, Token& name
|
||||||
, s32& left, Array(Token)& tokens
|
, s32& left, Array(Token)& tokens
|
||||||
, u8& num_specifiers, SpecifierT* specs_found
|
, s32& num_specifiers, SpecifierT* specs_found
|
||||||
, Code& array_expr
|
, Code& array_expr
|
||||||
)
|
)
|
||||||
{
|
{
|
||||||
|
while ( left && tok_is_specifier( curr_tok ) )
|
||||||
|
{
|
||||||
|
SpecifierT spec = ESpecifier::to_type( curr_tok.Text, curr_tok.Length );
|
||||||
|
|
||||||
|
if ( spec != ESpecifier::Const
|
||||||
|
&& spec < ESpecifier::Type_Signed )
|
||||||
|
{
|
||||||
|
log_failure( "%s: Error, invalid specifier used in type definition: %s", func_name, curr_tok.Text );
|
||||||
|
return false;
|
||||||
|
}
|
||||||
|
|
||||||
|
specs_found[num_specifiers] = spec;
|
||||||
|
num_specifiers++;
|
||||||
|
eat( curr_tok.Type );
|
||||||
|
}
|
||||||
|
|
||||||
|
if ( left == 0 )
|
||||||
|
{
|
||||||
|
log_failure( "%s: Error, unexpected end of type definition", func_name );
|
||||||
|
return false;
|
||||||
|
}
|
||||||
|
|
||||||
|
name = curr_tok;
|
||||||
|
eat( TokType::Identifier );
|
||||||
|
|
||||||
|
while ( left && tok_is_specifier( curr_tok ) )
|
||||||
|
{
|
||||||
|
SpecifierT spec = ESpecifier::to_type( curr_tok.Text, curr_tok.Length );
|
||||||
|
|
||||||
|
if ( spec != ESpecifier::Const
|
||||||
|
&& spec != ESpecifier::Ref
|
||||||
|
&& spec != ESpecifier::RValue
|
||||||
|
&& spec < ESpecifier::Type_Signed )
|
||||||
|
{
|
||||||
|
log_failure( "%s: Error, invalid specifier used in type definition: %s", func_name, curr_tok.Text );
|
||||||
|
return false;
|
||||||
|
}
|
||||||
|
|
||||||
|
specs_found[num_specifiers] = spec;
|
||||||
|
num_specifiers++;
|
||||||
|
eat( curr_tok.Type );
|
||||||
|
}
|
||||||
|
|
||||||
|
if ( left && curr_tok.Type == TokType::BraceSquare_Open )
|
||||||
|
{
|
||||||
|
eat( TokType::BraceSquare_Open );
|
||||||
|
|
||||||
|
if ( left == 0 )
|
||||||
|
{
|
||||||
|
log_failure( "%s: Error, unexpected end of type definition", func_name );
|
||||||
|
return false;
|
||||||
|
}
|
||||||
|
|
||||||
|
if ( curr_tok.Type == TokType::BraceSquare_Close )
|
||||||
|
{
|
||||||
|
eat( TokType::BraceSquare_Close );
|
||||||
|
return true;
|
||||||
|
}
|
||||||
|
|
||||||
|
Token
|
||||||
|
untyped_tok = curr_tok;
|
||||||
|
|
||||||
|
while ( left && curr_tok.Type != TokType::BraceSquare_Close )
|
||||||
|
{
|
||||||
|
untyped_tok.Length += curr_tok.Length;
|
||||||
|
}
|
||||||
|
|
||||||
|
array_expr = untyped_str( untyped_tok.Length, untyped_tok.Text );
|
||||||
|
|
||||||
|
if ( left == 0 )
|
||||||
|
{
|
||||||
|
log_failure( "%s: Error, unexpected end of type definition", func_name );
|
||||||
|
return false;
|
||||||
|
}
|
||||||
|
|
||||||
|
if ( curr_tok.Type != TokType::BraceSquare_Close )
|
||||||
|
{
|
||||||
|
log_failure( "%s: Error, expected ] in type definition", func_name );
|
||||||
|
return false;
|
||||||
|
}
|
||||||
|
|
||||||
|
eat( TokType::BraceSquare_Close );
|
||||||
|
}
|
||||||
|
|
||||||
|
return true;
|
||||||
}
|
}
|
||||||
|
|
||||||
Code parse_type( s32 length, char const* def )
|
Code parse_type( s32 length, char const* def )
|
||||||
@ -3505,7 +3559,7 @@ namespace gen
|
|||||||
|
|
||||||
if ( tokens == nullptr )
|
if ( tokens == nullptr )
|
||||||
{
|
{
|
||||||
log_failure( "gen::parse_friend: no tokens found for provided definition" );
|
log_failure( "gen::parse_type: no tokens found for provided definition" );
|
||||||
return Code::Invalid;
|
return Code::Invalid;
|
||||||
}
|
}
|
||||||
|
|
||||||
@ -3517,7 +3571,8 @@ namespace gen
|
|||||||
SpecifierT specs_found[16] { ESpecifier::Num_Specifiers };
|
SpecifierT specs_found[16] { ESpecifier::Num_Specifiers };
|
||||||
u8 num_specifiers = 0;
|
u8 num_specifiers = 0;
|
||||||
|
|
||||||
bool helper_result = parse_type_helper_tok( txt(parse_type)
|
bool helper_result = parse_type_helper( txt(parse_type)
|
||||||
|
, * name
|
||||||
, left, tokens
|
, left, tokens
|
||||||
, num_specifiers, specs_found
|
, num_specifiers, specs_found
|
||||||
, array_expr
|
, array_expr
|
||||||
@ -3531,154 +3586,7 @@ namespace gen
|
|||||||
Code
|
Code
|
||||||
result = make_code();
|
result = make_code();
|
||||||
result->Type = Typename;
|
result->Type = Typename;
|
||||||
result->Name = get_cached_string( name->Address, name->Length );
|
result->Name = get_cached_string( name->Text, name->Length );
|
||||||
|
|
||||||
if (num_specifiers)
|
|
||||||
{
|
|
||||||
Code specifiers = def_specifiers( num_specifiers, specs_found );
|
|
||||||
|
|
||||||
result->add_entry( specifiers );
|
|
||||||
}
|
|
||||||
|
|
||||||
if ( array_expr )
|
|
||||||
result->add_entry( array_expr );
|
|
||||||
|
|
||||||
result.lock();
|
|
||||||
return result;
|
|
||||||
}
|
|
||||||
|
|
||||||
inline
|
|
||||||
bool parse__type_helper( char const* func_name
|
|
||||||
, s32& length, char const* def
|
|
||||||
, s32& name_length, char const*& name
|
|
||||||
, u8& num_specifiers, SpecifierT* specs_found
|
|
||||||
, Code& array_expr)
|
|
||||||
{
|
|
||||||
s32 left = length;
|
|
||||||
char const* scanner = def;
|
|
||||||
|
|
||||||
char const* word = scanner;
|
|
||||||
s32 word_length = 0;
|
|
||||||
|
|
||||||
// Find all left-hand specifiers and the typename.
|
|
||||||
do
|
|
||||||
{
|
|
||||||
// Clearing any whitespace
|
|
||||||
if ( left <= 0 )
|
|
||||||
{
|
|
||||||
log_failure( "gen::%s: Error, reached end of string before finding typename", func_name );
|
|
||||||
return false;
|
|
||||||
}
|
|
||||||
|
|
||||||
CheckForSpecifiers();
|
|
||||||
break;
|
|
||||||
}
|
|
||||||
while (1);
|
|
||||||
|
|
||||||
name = scanner;
|
|
||||||
name_length = word_length;
|
|
||||||
|
|
||||||
// Find all right-hand specifiers.
|
|
||||||
do
|
|
||||||
{
|
|
||||||
SkipWhitespace();
|
|
||||||
if ( left <= 0 )
|
|
||||||
break;
|
|
||||||
|
|
||||||
if ( current == '*')
|
|
||||||
{
|
|
||||||
specs_found[num_specifiers] = ESpecifier::Ptr;
|
|
||||||
num_specifiers++;
|
|
||||||
move_forward();
|
|
||||||
continue;
|
|
||||||
}
|
|
||||||
|
|
||||||
if ( current == '&')
|
|
||||||
{
|
|
||||||
move_forward();
|
|
||||||
|
|
||||||
if ( current == '&')
|
|
||||||
{
|
|
||||||
specs_found[num_specifiers] = ESpecifier::RValue;
|
|
||||||
num_specifiers++;
|
|
||||||
move_forward();
|
|
||||||
continue;
|
|
||||||
}
|
|
||||||
|
|
||||||
specs_found[num_specifiers] = ESpecifier::Ref;
|
|
||||||
num_specifiers++;
|
|
||||||
continue;
|
|
||||||
}
|
|
||||||
|
|
||||||
if ( current == '[')
|
|
||||||
{
|
|
||||||
move_forward();
|
|
||||||
|
|
||||||
SkipWhitespace();
|
|
||||||
if ( left <= 0 )
|
|
||||||
{
|
|
||||||
log_failure( "gen::%s: Error, reached end of string before finding array expression", func_name );
|
|
||||||
return false;
|
|
||||||
}
|
|
||||||
|
|
||||||
word = scanner;
|
|
||||||
word_length = 0;
|
|
||||||
GetWord();
|
|
||||||
|
|
||||||
array_expr = untyped_str( word_length, word );
|
|
||||||
|
|
||||||
if ( left <= 0 )
|
|
||||||
{
|
|
||||||
log_failure( "gen::%s: Error, reached end of string before finding ']' for array expression", func_name );
|
|
||||||
return false;
|
|
||||||
}
|
|
||||||
|
|
||||||
if ( current == ']')
|
|
||||||
{
|
|
||||||
move_forward();
|
|
||||||
}
|
|
||||||
|
|
||||||
num_specifiers++;
|
|
||||||
continue;
|
|
||||||
}
|
|
||||||
|
|
||||||
word = scanner;
|
|
||||||
word_length = 0;
|
|
||||||
|
|
||||||
CheckForSpecifiers();
|
|
||||||
break;
|
|
||||||
}
|
|
||||||
while (1);
|
|
||||||
}
|
|
||||||
|
|
||||||
Code parse_type_non_lex( s32 length, char const* def )
|
|
||||||
{
|
|
||||||
check_parse_args( parse_type, length, def );
|
|
||||||
|
|
||||||
SpecifierT specs_found[16] { ESpecifier::Num_Specifiers };
|
|
||||||
u8 num_specifiers = 0;
|
|
||||||
|
|
||||||
char const* name = nullptr;
|
|
||||||
s32 name_length = 0;
|
|
||||||
|
|
||||||
Code array_expr = { nullptr };
|
|
||||||
|
|
||||||
bool helper_result = parse__type_helper( txt(parse_type)
|
|
||||||
, length, def
|
|
||||||
, name_length, name
|
|
||||||
, num_specifiers, specs_found
|
|
||||||
, array_expr
|
|
||||||
);
|
|
||||||
|
|
||||||
if ( ! helper_result )
|
|
||||||
return Code::Invalid;
|
|
||||||
|
|
||||||
using namespace ECode;
|
|
||||||
|
|
||||||
Code
|
|
||||||
result = make_code();
|
|
||||||
result->Type = Typename;
|
|
||||||
result->Name = get_cached_string( name, name_length );
|
|
||||||
|
|
||||||
if (num_specifiers)
|
if (num_specifiers)
|
||||||
{
|
{
|
||||||
@ -3696,41 +3604,32 @@ namespace gen
|
|||||||
|
|
||||||
Code parse_typedef( s32 length, char const* def )
|
Code parse_typedef( s32 length, char const* def )
|
||||||
{
|
{
|
||||||
|
# define context parse_type
|
||||||
|
|
||||||
check_parse_args( parse_typedef, length, def );
|
check_parse_args( parse_typedef, length, def );
|
||||||
|
|
||||||
using namespace ECode;
|
Array(Token) tokens = lex( length, def );
|
||||||
|
|
||||||
|
if ( tokens == nullptr )
|
||||||
|
{
|
||||||
|
log_failure( "gen::parse_typedef: no tokens found for provided definition" );
|
||||||
|
return Code::Invalid;
|
||||||
|
}
|
||||||
|
|
||||||
|
s32 left = array_count( tokens );
|
||||||
|
|
||||||
|
Token* name = nullptr;
|
||||||
|
Code array_expr = { nullptr };
|
||||||
|
Code type = { nullptr };
|
||||||
|
|
||||||
SpecifierT specs_found[16] { ESpecifier::Num_Specifiers };
|
SpecifierT specs_found[16] { ESpecifier::Num_Specifiers };
|
||||||
s32 num_specifiers = 0;
|
s32 num_specifiers = 0;
|
||||||
|
|
||||||
char const* name = nullptr;
|
eat( TokType::Decl_Typedef );
|
||||||
s32 name_length = 0;
|
|
||||||
|
|
||||||
s32 left = length;
|
bool helper_result = parse_type_helper( txt(parse_typedef)
|
||||||
char const* scanner = def;
|
, * name
|
||||||
|
, left, tokens
|
||||||
Code array_expr = { nullptr };
|
|
||||||
Code type = { nullptr };
|
|
||||||
|
|
||||||
char const* word = scanner;
|
|
||||||
s32 word_length = 0;
|
|
||||||
|
|
||||||
do
|
|
||||||
{
|
|
||||||
SkipWhitespace_Checked( parse_typedef, "Error, reached end of string before finding typename" );
|
|
||||||
|
|
||||||
GetWord();
|
|
||||||
if ( str_compare( word, "typedef", word_length ) != 0 )
|
|
||||||
{
|
|
||||||
log_failure( "gen::parse_typedef: Error, expected 'typedef' but found '%.*s'", word_length, word );
|
|
||||||
return Code::Invalid;
|
|
||||||
}
|
|
||||||
|
|
||||||
// Determining the typename inline
|
|
||||||
|
|
||||||
bool helper_result = parse__type_helper( txt(parse_typedef)
|
|
||||||
, left, scanner
|
|
||||||
, name_length, name
|
|
||||||
, num_specifiers, specs_found
|
, num_specifiers, specs_found
|
||||||
, array_expr
|
, array_expr
|
||||||
);
|
);
|
||||||
@ -3738,35 +3637,25 @@ namespace gen
|
|||||||
if ( ! helper_result )
|
if ( ! helper_result )
|
||||||
return Code::Invalid;
|
return Code::Invalid;
|
||||||
|
|
||||||
type = def_type( name_length, name, def_specifiers( num_specifiers, specs_found ) );
|
type = def_type( name->Length, name->Text, def_specifiers( num_specifiers, specs_found ) );
|
||||||
|
|
||||||
// End typename
|
if ( curr_tok.Type != TokType::Identifier )
|
||||||
|
|
||||||
SkipWhitespace_Checked( parse_typedef, "Error, reached end of string before finding name" );
|
|
||||||
|
|
||||||
GetWord();
|
|
||||||
name = word;
|
|
||||||
name_length = word_length;
|
|
||||||
|
|
||||||
SkipWhitespace_Checked( parse_typedef, "Error, reached end of string before finding ';'" );
|
|
||||||
|
|
||||||
if ( current == ';')
|
|
||||||
{
|
{
|
||||||
move_forward();
|
log_failure( "gen::parse_typedef: Error, expected identifier for typedef" );
|
||||||
break;
|
|
||||||
}
|
|
||||||
|
|
||||||
log_failure( "gen::parse_typedef: Error, expected ';' for typedef" );
|
|
||||||
return Code::Invalid;
|
return Code::Invalid;
|
||||||
}
|
}
|
||||||
while (1);
|
|
||||||
|
name = tokens;
|
||||||
|
eat( TokType::Identifier );
|
||||||
|
|
||||||
|
eat( TokType::Statement_End );
|
||||||
|
|
||||||
using namespace ECode;
|
using namespace ECode;
|
||||||
|
|
||||||
Code
|
Code
|
||||||
result = make_code();
|
result = make_code();
|
||||||
result->Type = Typedef;
|
result->Type = Typedef;
|
||||||
result->Name = get_cached_string( name, name_length );
|
result->Name = get_cached_string( name->Text, name->Length );
|
||||||
|
|
||||||
result->add_entry( type );
|
result->add_entry( type );
|
||||||
|
|
||||||
@ -3776,92 +3665,68 @@ namespace gen
|
|||||||
|
|
||||||
Code parse_using( s32 length, char const* def )
|
Code parse_using( s32 length, char const* def )
|
||||||
{
|
{
|
||||||
|
# define context parse_type
|
||||||
|
|
||||||
check_parse_args( parse_typedef, length, def );
|
check_parse_args( parse_typedef, length, def );
|
||||||
|
|
||||||
using namespace ECode;
|
Array(Token) tokens = lex( length, def );
|
||||||
|
|
||||||
|
if ( tokens == nullptr )
|
||||||
|
{
|
||||||
|
log_failure( "gen::parse_typedef: no tokens found for provided definition" );
|
||||||
|
return Code::Invalid;
|
||||||
|
}
|
||||||
|
|
||||||
|
s32 left = array_count( tokens );
|
||||||
|
|
||||||
SpecifierT specs_found[16] { ESpecifier::Num_Specifiers };
|
SpecifierT specs_found[16] { ESpecifier::Num_Specifiers };
|
||||||
s32 num_specifiers = 0;
|
s32 num_specifiers = 0;
|
||||||
|
|
||||||
char const* name = nullptr;
|
Token* name = nullptr;
|
||||||
s32 name_length = 0;
|
|
||||||
|
|
||||||
s32 left = length;
|
|
||||||
char const* scanner = def;
|
|
||||||
|
|
||||||
Code array_expr = { nullptr };
|
Code array_expr = { nullptr };
|
||||||
Code type = { nullptr };
|
Code type = { nullptr };
|
||||||
|
|
||||||
char const* word = scanner;
|
|
||||||
s32 word_length = 0;
|
|
||||||
|
|
||||||
bool is_namespace = false;
|
bool is_namespace = false;
|
||||||
|
|
||||||
SkipWhitespace_Checked( parse_using, "Error, reached end of string before finding 'using'" );
|
eat( TokType::Decl_Using );
|
||||||
|
|
||||||
GetWord();
|
if ( curr_tok.Type == TokType::Decl_Namespace )
|
||||||
if ( str_compare( word, "using", word_length ) != 0 )
|
|
||||||
{
|
{
|
||||||
log_failure( "gen::parse_using: Error, expected 'using' but found '%.*s'", word_length, word );
|
is_namespace = true;
|
||||||
|
eat( TokType::Decl_Namespace );
|
||||||
|
}
|
||||||
|
|
||||||
|
eat( TokType::Identifier );
|
||||||
|
|
||||||
|
if ( curr_tok.Type != TokType::Statement_End )
|
||||||
|
{
|
||||||
|
if ( is_namespace )
|
||||||
|
{
|
||||||
|
log_failure( "gen::parse_using: Error, expected ; after identifier for a using namespace declaration" );
|
||||||
return Code::Invalid;
|
return Code::Invalid;
|
||||||
}
|
}
|
||||||
|
|
||||||
SkipWhitespace_Checked( parse_using, "Error, reached end of string before finding 'namespace' or typename" );
|
bool helper_result = parse_type_helper( txt(parse_using)
|
||||||
|
, * name
|
||||||
GetWord();
|
, left, tokens
|
||||||
if ( str_compare( word, "namespace", word_length ) == 0 )
|
|
||||||
{
|
|
||||||
is_namespace = true;
|
|
||||||
|
|
||||||
SkipWhitespace_Checked( parse_using, "Error, reached end of string before finding name" );
|
|
||||||
|
|
||||||
GetWord();
|
|
||||||
name = word;
|
|
||||||
name_length = word_length;
|
|
||||||
|
|
||||||
SkipWhitespace_Checked( parse_using, "Error, reached end of string before finding end statement \';\'");
|
|
||||||
|
|
||||||
if ( current == ';' )
|
|
||||||
{
|
|
||||||
goto Finished;
|
|
||||||
};
|
|
||||||
}
|
|
||||||
else
|
|
||||||
{
|
|
||||||
name = word;
|
|
||||||
name_length = word_length;
|
|
||||||
}
|
|
||||||
|
|
||||||
move_forward();
|
|
||||||
SkipWhitespace_Checked( parse_using, "Error, reached end of string before finding end statement ';'" );
|
|
||||||
|
|
||||||
if ( current == '=' )
|
|
||||||
{
|
|
||||||
move_forward();
|
|
||||||
SkipWhitespace_Checked( parse_using, "Error, reached end of string before finding the 'typename definition after the = sign" );
|
|
||||||
|
|
||||||
bool helper_result = parse__type_helper( txt(parse_using)
|
|
||||||
, left, scanner
|
|
||||||
, name_length, name
|
|
||||||
, num_specifiers, specs_found
|
, num_specifiers, specs_found
|
||||||
, array_expr
|
, array_expr
|
||||||
);
|
);
|
||||||
|
|
||||||
if ( helper_result )
|
if ( ! helper_result )
|
||||||
{
|
|
||||||
type = def_type( name_length, name, def_specifiers( num_specifiers, specs_found ) );
|
|
||||||
}
|
|
||||||
else
|
|
||||||
{
|
|
||||||
return Code::Invalid;
|
return Code::Invalid;
|
||||||
|
|
||||||
|
type = def_type( name->Length, name->Text, def_specifiers( num_specifiers, specs_found ) );
|
||||||
}
|
}
|
||||||
}
|
|
||||||
Finished:
|
eat( TokType::Statement_End );
|
||||||
|
|
||||||
|
using namespace ECode;
|
||||||
|
|
||||||
Code
|
Code
|
||||||
result = make_code();
|
result = make_code();
|
||||||
result->Type = is_namespace ? Using : Using_Namespace;
|
result->Type = is_namespace ? Using : Using_Namespace;
|
||||||
result->Name = get_cached_string( name, name_length );
|
result->Name = get_cached_string( name->Text, name->Length );
|
||||||
|
|
||||||
result->add_entry( type );
|
result->add_entry( type );
|
||||||
|
|
||||||
@ -3870,7 +3735,6 @@ namespace gen
|
|||||||
}
|
}
|
||||||
|
|
||||||
|
|
||||||
|
|
||||||
s32 parse_classes( s32 length, char const* class_defs, Code* out_class_codes )
|
s32 parse_classes( s32 length, char const* class_defs, Code* out_class_codes )
|
||||||
{
|
{
|
||||||
not_implemented( parse_classes );
|
not_implemented( parse_classes );
|
||||||
@ -3933,6 +3797,7 @@ namespace gen
|
|||||||
result->Type = ECode::Untyped;
|
result->Type = ECode::Untyped;
|
||||||
result->Content = result->Name;
|
result->Content = result->Name;
|
||||||
|
|
||||||
|
result.lock();
|
||||||
return result;
|
return result;
|
||||||
}
|
}
|
||||||
|
|
||||||
@ -3952,6 +3817,7 @@ namespace gen
|
|||||||
result->Type = ECode::Untyped;
|
result->Type = ECode::Untyped;
|
||||||
result->Content = get_cached_string( buf, length );
|
result->Content = get_cached_string( buf, length );
|
||||||
|
|
||||||
|
result.lock();
|
||||||
return result;
|
return result;
|
||||||
}
|
}
|
||||||
|
|
||||||
@ -3972,7 +3838,6 @@ namespace gen
|
|||||||
result->Content = get_cached_string( buf, length );
|
result->Content = get_cached_string( buf, length );
|
||||||
|
|
||||||
result.lock();
|
result.lock();
|
||||||
|
|
||||||
return result;
|
return result;
|
||||||
}
|
}
|
||||||
# pragma endregion Untyped Constructors
|
# pragma endregion Untyped Constructors
|
||||||
|
@ -606,7 +606,11 @@ namespace gen
|
|||||||
Entry( RValue, && ) \
|
Entry( RValue, && ) \
|
||||||
Entry( Static_Member, static ) \
|
Entry( Static_Member, static ) \
|
||||||
Entry( Thread_Local, thread_local ) \
|
Entry( Thread_Local, thread_local ) \
|
||||||
Entry( Volatile, volatile )
|
Entry( Volatile, volatile ) \
|
||||||
|
Entry( Type_Signed, signed ) \
|
||||||
|
Entry( Type_Unsigned, unsigned ) \
|
||||||
|
Entry( Type_Short, short ) \
|
||||||
|
Entry( Type_Long, long )
|
||||||
|
|
||||||
enum Type : u32
|
enum Type : u32
|
||||||
{
|
{
|
||||||
|
Loading…
Reference in New Issue
Block a user