2023-04-03 00:55:28 -07:00
|
|
|
/*
|
|
|
|
gencpp: A simple staged metaprogramming library for C++.
|
2023-04-03 23:04:19 -07:00
|
|
|
|
|
|
|
This library is intended for small-to midsize projects.
|
2023-04-03 00:55:28 -07:00
|
|
|
|
|
|
|
AST type checking supports only a small subset of c++.
|
|
|
|
See the 'ECode' namespace and 'gen API' region to see what is supported.
|
|
|
|
|
2023-04-03 23:04:19 -07:00
|
|
|
WHAT IS NOT PROVIDED:
|
|
|
|
* Macro or template generation : This library is to avoid those, adding support for them
|
|
|
|
adds unnecessary complexity.
|
|
|
|
If you desire define them outside the gen_time scopes.
|
|
|
|
|
|
|
|
* Modern c++ (STL library) features :
|
|
|
|
* Expression validation : Execution expressions are defined using the untyped string API.
|
|
|
|
: There is no parse API for validating expression (possibly will add in the future)
|
|
|
|
|
|
|
|
* Complete file parser DSL : This isn't like the unreal header tool.
|
|
|
|
Code injection to file or based off a file contents is not supported by the api.
|
|
|
|
However nothing is stopping you using the library for that purpose.
|
|
|
|
|
|
|
|
|
|
|
|
There are four different of construction of Code ast's the library provides:
|
|
|
|
* Upfront construction
|
|
|
|
* Incremental construction
|
|
|
|
* Parse construction
|
|
|
|
* Untyped
|
|
|
|
|
|
|
|
Upfront Construction:
|
|
|
|
All component ASTs must be previously constructed, and provided on creation of the code AST.
|
|
|
|
The construction will fail and return InvalidCode otherwise.
|
|
|
|
|
|
|
|
API:
|
|
|
|
* def_forward_decl
|
|
|
|
* def_class
|
|
|
|
* def_global_body
|
|
|
|
* def_proc
|
|
|
|
* def_proc_body
|
|
|
|
* def_namespace
|
|
|
|
* def_namespace_body
|
|
|
|
* def_param
|
|
|
|
* def_params
|
|
|
|
* def_operator
|
|
|
|
* def_specifier
|
|
|
|
* def_specifiers
|
|
|
|
* def_struct
|
|
|
|
* def_struct_body
|
|
|
|
* def_variable
|
|
|
|
* def_type
|
|
|
|
* def_using
|
|
|
|
* def_using_namespace
|
|
|
|
|
|
|
|
Incremental construction:
|
|
|
|
A Code ast is provided but only completed upfront if all components are provided.
|
|
|
|
Components are then added using the AST API for adding ASTs:
|
|
|
|
* code.add( AST* ) // Adds AST with validation.
|
|
|
|
* code.add_entry( AST* ) // Adds AST entry without validation.
|
|
|
|
* code.add_content( AST* ) // Adds AST string content without validation.
|
|
|
|
|
|
|
|
API:
|
|
|
|
* make_forward_decl
|
|
|
|
* make_class
|
|
|
|
* make_global_body
|
|
|
|
* make_proc
|
|
|
|
* make_namespace
|
|
|
|
* make_params
|
|
|
|
* make_operator
|
|
|
|
* make_specifiers
|
|
|
|
* make_struct
|
|
|
|
* make_variable
|
|
|
|
* make_type
|
|
|
|
* make_using
|
|
|
|
* make_using_namespace
|
|
|
|
|
|
|
|
Parse construction:
|
|
|
|
A string provided to the API is parsed for the intended language construct.
|
|
|
|
|
|
|
|
API:
|
|
|
|
* parse_forward_decl
|
|
|
|
* parse_class
|
|
|
|
* parse_glboal_body
|
|
|
|
* parse_proc
|
|
|
|
* parse_namespace
|
|
|
|
* parse_params
|
|
|
|
* parse_operator
|
|
|
|
* parse_specifiers
|
|
|
|
* parse_struct
|
|
|
|
* parse_variable
|
|
|
|
* parse_type
|
|
|
|
* parse_using
|
|
|
|
* parse_using
|
|
|
|
|
|
|
|
The parse API treats any execution scope definitions with no validation and are turned into untyped Code ASTs.
|
|
|
|
This includes the assignmetn of variables; due to the library not yet supporting c/c++ expression parsing.
|
|
|
|
|
|
|
|
Untyped constructions:
|
|
|
|
Code ASTs are constructed using unvalidated strings.
|
|
|
|
|
|
|
|
API:
|
|
|
|
* untyped_str
|
|
|
|
* untyped_fmt
|
|
|
|
* untyped_token_fmt
|
|
|
|
|
|
|
|
During serialization any untyped Code AST is has its string value directly injected inline of
|
|
|
|
whatever context the content existed as an entry within.
|
|
|
|
Even though thse are not validated from somewhat correct c/c++ syntax or components, it doesn't mean that
|
|
|
|
Untyped code can be added as any component of a Code AST:
|
|
|
|
* Untyped code cannot have children, thus there cannot be recursive injection this way.
|
|
|
|
* Untyped code can only be a child of a parent of body AST, or for values of an assignment.
|
|
|
|
These restrictions help prevent abuse of untyped code to some extent.
|
2023-04-03 00:55:28 -07:00
|
|
|
*/
|
|
|
|
|
2023-04-01 19:21:46 -07:00
|
|
|
#pragma once
|
|
|
|
|
|
|
|
#include "Bloat.hpp"
|
|
|
|
|
2023-04-03 23:04:19 -07:00
|
|
|
// Temporarily here for debugging purposes.
|
|
|
|
#define gen_time
|
|
|
|
#define GEN_ENFORCE_READONLY_AST
|
2023-04-03 00:55:28 -07:00
|
|
|
// #define GEN_DEFINE_DSL
|
2023-04-03 23:04:19 -07:00
|
|
|
#define GEN_DEFINE_LIBRARY_CODE_CONSTANTS
|
|
|
|
// #define GEN_BAN_CPP_TEMPLATES
|
|
|
|
// #define GEN_USE_FATAL
|
2023-04-03 00:55:28 -07:00
|
|
|
|
2023-04-01 22:07:44 -07:00
|
|
|
#ifdef gen_time
|
2023-04-01 19:21:46 -07:00
|
|
|
namespace gen
|
|
|
|
{
|
2023-04-03 23:04:19 -07:00
|
|
|
using LogFailType = sw(*)(char const*, ...);
|
2023-04-01 19:21:46 -07:00
|
|
|
|
2023-04-03 23:04:19 -07:00
|
|
|
#ifdef GEN_BAN_CPP_TEMPLATES
|
|
|
|
#define template static_assert("Templates are banned within gen_time scope blocks")
|
|
|
|
#endif
|
2023-04-01 19:21:46 -07:00
|
|
|
|
2023-04-03 23:04:19 -07:00
|
|
|
#ifdef GEN_USE_FATAL
|
|
|
|
ct LogFailType log_failure = fatal;
|
|
|
|
#else
|
|
|
|
ct LogFailType log_failure = log_fmt;
|
|
|
|
#endif
|
2023-04-01 19:21:46 -07:00
|
|
|
|
2023-04-03 00:55:28 -07:00
|
|
|
namespace ECode
|
2023-04-01 19:21:46 -07:00
|
|
|
{
|
2023-04-03 00:55:28 -07:00
|
|
|
enum Type : u8
|
2023-04-01 19:21:46 -07:00
|
|
|
{
|
2023-04-03 23:04:19 -07:00
|
|
|
Invalid, // Used only with improperly created Code nodes
|
|
|
|
Untyped, // User provided raw string
|
|
|
|
Decl_Function, // <specifier> <type> <name> ( <params> )
|
|
|
|
Decl_Type, // <type> <name>;
|
2023-04-03 00:55:28 -07:00
|
|
|
Function, // <type> <name>( <parameters> )
|
|
|
|
Function_Body, // { <body> }
|
2023-04-03 23:04:19 -07:00
|
|
|
Namespace, // Define a namespace
|
|
|
|
Namespace_Body, // { <body> }
|
|
|
|
Parameters, // <type> <param> ...
|
|
|
|
Specifiers, // Used with functions, structs, variables
|
|
|
|
Struct, // struct <specifier> <name> <parent>
|
|
|
|
Struct_Body, // {<body> }
|
|
|
|
Variable, // <type> <name>
|
|
|
|
Typedef, // typedef <type> <alias>
|
|
|
|
Typename, // Typename, used with other types
|
|
|
|
Using, // using <name> = <type>
|
|
|
|
Unit, // Represents a file.
|
2023-04-01 19:21:46 -07:00
|
|
|
|
|
|
|
Num_Types
|
|
|
|
};
|
|
|
|
|
2023-04-03 00:55:28 -07:00
|
|
|
inline
|
|
|
|
char const* str( Type type )
|
2023-04-01 19:21:46 -07:00
|
|
|
{
|
2023-04-03 00:55:28 -07:00
|
|
|
static
|
|
|
|
char const* lookup[Num_Types] = {
|
|
|
|
"Invalid",
|
|
|
|
"Untyped",
|
|
|
|
"Decl_Function",
|
2023-04-03 23:04:19 -07:00
|
|
|
"Decl_Type",
|
2023-04-03 00:55:28 -07:00
|
|
|
"Function",
|
|
|
|
"Function_Body",
|
|
|
|
"Namespace",
|
|
|
|
"Namespace_Body",
|
|
|
|
"Parameters",
|
|
|
|
"Specifiers",
|
|
|
|
"Struct",
|
|
|
|
"Struct_Body",
|
|
|
|
"Variable",
|
|
|
|
"Typedef",
|
|
|
|
"Typename",
|
2023-04-03 23:04:19 -07:00
|
|
|
"Using",
|
|
|
|
"Unit",
|
2023-04-03 00:55:28 -07:00
|
|
|
};
|
2023-04-01 19:21:46 -07:00
|
|
|
|
2023-04-03 00:55:28 -07:00
|
|
|
return lookup[ type ];
|
2023-04-01 19:21:46 -07:00
|
|
|
}
|
2023-04-03 00:55:28 -07:00
|
|
|
}
|
|
|
|
using CodeT = ECode::Type;
|
2023-04-01 19:21:46 -07:00
|
|
|
|
2023-04-03 23:04:19 -07:00
|
|
|
namespace EOperator
|
|
|
|
{
|
|
|
|
enum Type : u8
|
|
|
|
{
|
|
|
|
Add,
|
|
|
|
Subtract,
|
|
|
|
Multiply,
|
|
|
|
Divide,
|
|
|
|
Modulo,
|
|
|
|
|
|
|
|
Num_Ops
|
|
|
|
};
|
|
|
|
|
|
|
|
inline
|
|
|
|
char const* str( Type op )
|
|
|
|
{
|
|
|
|
static
|
|
|
|
char const* lookup[ Num_Ops ] = {
|
|
|
|
"+",
|
|
|
|
"-",
|
|
|
|
"*",
|
|
|
|
"/",
|
|
|
|
};
|
|
|
|
|
|
|
|
return lookup[ op ];
|
|
|
|
}
|
|
|
|
}
|
|
|
|
using OperatorT = EOperator::Type;
|
|
|
|
|
|
|
|
namespace ESpecifier
|
|
|
|
{
|
|
|
|
enum Type : u8
|
|
|
|
{
|
|
|
|
Attribute, // [ <attributes ]
|
|
|
|
Alignas, // alignas(#)
|
|
|
|
Constexpr, // constexpr
|
|
|
|
Const, // const
|
|
|
|
Inline, // inline
|
|
|
|
RValue, //
|
|
|
|
|
|
|
|
C_Linkage, // extern "C"
|
|
|
|
API_Import, // Vendor specific way dynamic import symbol
|
|
|
|
API_Export, // Vendor specific way to dynamic export
|
|
|
|
External_Linkage, // extern
|
|
|
|
Internal_Linkage, // static (within unit file)
|
|
|
|
Static_Member, // static (within sturct/class)
|
|
|
|
Local_Persist, // static (within function)
|
|
|
|
Thread_Local, // thread_local
|
|
|
|
|
|
|
|
Num_Specifiers,
|
|
|
|
Invalid
|
|
|
|
};
|
|
|
|
|
|
|
|
// Specifier to string
|
|
|
|
inline
|
|
|
|
char const* to_str( Type specifier )
|
|
|
|
{
|
|
|
|
static
|
|
|
|
char const* lookup[ Num_Specifiers ] = {
|
|
|
|
"alignas",
|
|
|
|
"constexpr",
|
|
|
|
"const",
|
|
|
|
"inline",
|
|
|
|
|
|
|
|
"extern \"C\"",
|
|
|
|
|
|
|
|
#if defined(ZPL_SYSTEM_WINDOWS) && 0// API_Import and API_Export strings
|
|
|
|
"__declspec(dllexport)",
|
|
|
|
"__declspec(dllimport)",
|
|
|
|
#elif defined(ZPL_SYSTEM_MACOS) || 1
|
|
|
|
"__attribute__ ((visibility (\"default\")))",
|
|
|
|
"__attribute__ ((visibility (\"default\")))",
|
|
|
|
#endif
|
|
|
|
|
|
|
|
"extern",
|
|
|
|
"static",
|
|
|
|
"static",
|
|
|
|
"static",
|
|
|
|
"thread_local"
|
|
|
|
};
|
|
|
|
|
|
|
|
return lookup[ specifier ];
|
|
|
|
}
|
|
|
|
|
|
|
|
Type to_type( char const* str, s32 length )
|
|
|
|
{
|
|
|
|
static
|
|
|
|
u32 keymap[ Num_Specifiers ];
|
|
|
|
do_once_start
|
|
|
|
for ( u32 index = 0; index < Num_Specifiers; index++ )
|
|
|
|
{
|
|
|
|
char const* enum_str = to_str( (Type)index );
|
|
|
|
|
|
|
|
keymap[index] = crc32( enum_str, strnlen(enum_str, 42) );
|
|
|
|
}
|
|
|
|
do_once_end
|
|
|
|
|
|
|
|
u32 hash = crc32(str, length );
|
|
|
|
|
|
|
|
for ( u32 index = 0; index < Num_Specifiers; index++ )
|
|
|
|
{
|
|
|
|
if ( keymap[index] == hash )
|
|
|
|
return (Type)index;
|
|
|
|
}
|
|
|
|
|
|
|
|
return Invalid;
|
|
|
|
}
|
|
|
|
}
|
|
|
|
using SpecifierT = ESpecifier::Type;
|
|
|
|
|
2023-04-03 00:55:28 -07:00
|
|
|
// TODO: If perf needs it, convert layout an SOA format.
|
|
|
|
/*
|
|
|
|
Simple AST POD with functionality to seralize into C++ syntax.
|
2023-04-01 19:21:46 -07:00
|
|
|
|
2023-04-03 00:55:28 -07:00
|
|
|
ASTs are currently stored as an AOS. They are always reconstructed on demand.
|
|
|
|
Thus redundant AST can easily occur.
|
|
|
|
Not sure if its better to store them in a hashmap.
|
|
|
|
*/
|
|
|
|
struct AST
|
|
|
|
{
|
|
|
|
#pragma region Member API
|
2023-04-01 19:21:46 -07:00
|
|
|
forceinline
|
2023-04-03 00:55:28 -07:00
|
|
|
void add( AST* other )
|
2023-04-01 19:21:46 -07:00
|
|
|
{
|
2023-04-03 00:55:28 -07:00
|
|
|
array_append( Entries, other );
|
|
|
|
|
|
|
|
other->Parent = this;
|
2023-04-01 19:21:46 -07:00
|
|
|
}
|
|
|
|
|
|
|
|
forceinline
|
2023-04-03 23:04:19 -07:00
|
|
|
bool has_entries() const
|
2023-04-01 19:21:46 -07:00
|
|
|
{
|
2023-04-03 00:55:28 -07:00
|
|
|
static bool lookup[ ECode::Num_Types] = {
|
2023-04-01 19:21:46 -07:00
|
|
|
false, // Invalid
|
|
|
|
false, // Untyped
|
|
|
|
true, // Decl_Function
|
2023-04-03 23:04:19 -07:00
|
|
|
true, // Decl_Type
|
2023-04-01 19:21:46 -07:00
|
|
|
true, // Function
|
2023-04-03 23:04:19 -07:00
|
|
|
true, // Parameters
|
|
|
|
false, // Specifies
|
|
|
|
true, // Struct
|
|
|
|
true, // Struct_Body
|
2023-04-01 19:21:46 -07:00
|
|
|
true, // Variable
|
2023-04-03 23:04:19 -07:00
|
|
|
true, // Typedef
|
2023-04-01 19:21:46 -07:00
|
|
|
true, // Typename
|
2023-04-03 23:04:19 -07:00
|
|
|
true, // Using
|
2023-04-01 19:21:46 -07:00
|
|
|
};
|
|
|
|
|
|
|
|
return lookup[Type];
|
|
|
|
}
|
|
|
|
|
2023-04-03 00:55:28 -07:00
|
|
|
forceinline
|
2023-04-03 23:04:19 -07:00
|
|
|
bool is_invalid() const
|
2023-04-03 00:55:28 -07:00
|
|
|
{
|
|
|
|
return Type != ECode::Invalid;
|
|
|
|
}
|
|
|
|
|
|
|
|
forceinline
|
2023-04-03 23:04:19 -07:00
|
|
|
char const* type_str() const
|
2023-04-03 00:55:28 -07:00
|
|
|
{
|
|
|
|
return ECode::str( Type );
|
|
|
|
}
|
|
|
|
|
2023-04-03 23:04:19 -07:00
|
|
|
string to_string() const;
|
2023-04-01 19:21:46 -07:00
|
|
|
|
2023-04-03 00:55:28 -07:00
|
|
|
#pragma endregion Member API
|
|
|
|
|
|
|
|
#define Using_Code_POD \
|
|
|
|
CodeT Type; \
|
|
|
|
bool Readonly; \
|
|
|
|
AST* Parent; \
|
|
|
|
string Name; \
|
|
|
|
string Comment; \
|
|
|
|
union { \
|
|
|
|
array(AST*) Entries; \
|
|
|
|
string Content; \
|
|
|
|
};
|
|
|
|
|
|
|
|
Using_Code_POD;
|
|
|
|
};
|
|
|
|
|
|
|
|
struct CodePOD
|
|
|
|
{
|
|
|
|
Using_Code_POD;
|
|
|
|
};
|
|
|
|
|
|
|
|
// Its intended for the AST to have equivalent size to its POD.
|
|
|
|
// All extra functionality within the AST namespace should just be syntatic sugar.
|
|
|
|
static_assert( sizeof(AST) == sizeof(CodePOD), "ERROR: AST IS NOT POD" );
|
|
|
|
|
|
|
|
/*
|
|
|
|
AST* typedef as to not constantly have to add the '*' as this is written often..
|
|
|
|
|
2023-04-03 23:04:19 -07:00
|
|
|
If GEN_ENFORCE_READONLY_AST is defined, readonly assertions will be done on any member dreference,
|
2023-04-03 00:55:28 -07:00
|
|
|
and the 'gen API' related functions. will set their created ASTs to readonly before returning.
|
|
|
|
|
|
|
|
Casting to AST* will bypass.
|
|
|
|
*/
|
|
|
|
struct Code
|
|
|
|
{
|
|
|
|
AST* ast;
|
|
|
|
|
2023-04-03 23:04:19 -07:00
|
|
|
Code body()
|
|
|
|
{
|
|
|
|
if ( ast->Type == ECode::Invalid )
|
|
|
|
fatal("Code::body: Type is invalid, cannot get");
|
|
|
|
|
|
|
|
if ( ast->Entries == nullptr || array_count(ast->Entries) == 0 )
|
|
|
|
fatal("Code::body: Entries of ast not properly setup.");
|
|
|
|
|
|
|
|
return pcast( Code, ast->Entries[0]);
|
|
|
|
}
|
|
|
|
|
2023-04-01 19:21:46 -07:00
|
|
|
forceinline
|
2023-04-03 23:04:19 -07:00
|
|
|
void lock()
|
2023-04-01 19:21:46 -07:00
|
|
|
{
|
2023-04-03 23:04:19 -07:00
|
|
|
#ifdef GEN_ENFORCE_READONLY_AST
|
|
|
|
ast->Readonly = true;
|
|
|
|
#endif
|
2023-04-01 19:21:46 -07:00
|
|
|
}
|
|
|
|
|
2023-04-03 23:04:19 -07:00
|
|
|
forceinline
|
|
|
|
operator bool() const
|
|
|
|
{
|
|
|
|
return ast && ast->is_invalid();
|
|
|
|
}
|
|
|
|
|
|
|
|
bool operator ==( Code other ) const
|
2023-04-01 19:21:46 -07:00
|
|
|
{
|
2023-04-03 00:55:28 -07:00
|
|
|
return ast == other.ast;
|
2023-04-01 19:21:46 -07:00
|
|
|
}
|
|
|
|
|
2023-04-03 00:55:28 -07:00
|
|
|
operator AST*()
|
|
|
|
{
|
|
|
|
return ast;
|
|
|
|
}
|
|
|
|
|
|
|
|
Code& operator =( Code other )
|
2023-04-01 19:21:46 -07:00
|
|
|
{
|
2023-04-03 00:55:28 -07:00
|
|
|
ast = other.ast;
|
|
|
|
|
|
|
|
return *this;
|
|
|
|
}
|
2023-04-01 19:21:46 -07:00
|
|
|
|
2023-04-03 00:55:28 -07:00
|
|
|
forceinline
|
|
|
|
AST* operator ->()
|
|
|
|
{
|
2023-04-03 23:04:19 -07:00
|
|
|
#ifdef GEN_ENFORCE_READONLY_AST
|
2023-04-03 00:55:28 -07:00
|
|
|
if ( ast == nullptr )
|
|
|
|
fatal("Attempt to dereference a nullptr!");
|
2023-04-01 19:21:46 -07:00
|
|
|
|
2023-04-03 00:55:28 -07:00
|
|
|
if ( ast->Readonly )
|
|
|
|
fatal("Attempted to access a member from a readonly ast!");
|
2023-04-03 23:04:19 -07:00
|
|
|
#endif
|
2023-04-01 19:21:46 -07:00
|
|
|
|
2023-04-03 00:55:28 -07:00
|
|
|
return ast;
|
2023-04-01 19:21:46 -07:00
|
|
|
}
|
|
|
|
};
|
2023-04-03 23:04:19 -07:00
|
|
|
static_assert( sizeof(Code) == sizeof(AST*), "ERROR: Code is not POD" );
|
2023-04-01 19:21:46 -07:00
|
|
|
|
2023-04-03 00:55:28 -07:00
|
|
|
// Used when the its desired when omission is allowed in a definition.
|
|
|
|
ct Code UnusedCode = { nullptr };
|
2023-04-01 19:21:46 -07:00
|
|
|
|
2023-04-03 00:55:28 -07:00
|
|
|
// Used internally for the most part to identify invaidly generated code.
|
2023-04-03 23:04:19 -07:00
|
|
|
extern const Code InvalidCode;
|
2023-04-03 00:55:28 -07:00
|
|
|
|
|
|
|
/*
|
|
|
|
Type registy: Used to store Typename ASTs. Types are registered by their string literal value.
|
2023-04-01 19:21:46 -07:00
|
|
|
|
2023-04-03 00:55:28 -07:00
|
|
|
Purely used as a memory optimization.
|
|
|
|
Strings made with the Typename ASTs are stored in thier own arena allocator.
|
|
|
|
TODO: Implement and replace usage of def_type.
|
|
|
|
*/
|
2023-04-03 23:04:19 -07:00
|
|
|
ZPL_TABLE_DECLARE( ZPL_EXTERN, TypeRegistry, type_reg_, Code );
|
2023-04-02 09:35:14 -07:00
|
|
|
|
2023-04-03 00:55:28 -07:00
|
|
|
#pragma region gen API
|
|
|
|
/*
|
|
|
|
Initialize the library.
|
|
|
|
This currently just initializes the CodePool.
|
|
|
|
*/
|
2023-04-02 09:35:14 -07:00
|
|
|
void init();
|
|
|
|
|
2023-04-03 23:04:19 -07:00
|
|
|
#pragma region Upfront
|
2023-04-03 00:55:28 -07:00
|
|
|
/*
|
|
|
|
Foward Declare a type:
|
|
|
|
<specifiers> <type> <name>;
|
|
|
|
*/
|
2023-04-03 23:04:19 -07:00
|
|
|
Code def_fwd_type( Code type, char const* name, Code specifiers = UnusedCode );
|
2023-04-01 19:21:46 -07:00
|
|
|
|
2023-04-03 00:55:28 -07:00
|
|
|
/*
|
|
|
|
Foward Declare a function:
|
|
|
|
<specifiers> <name> ( <params> );
|
|
|
|
*/
|
2023-04-03 23:04:19 -07:00
|
|
|
Code def_fwd_proc( char const* name
|
2023-04-01 19:21:46 -07:00
|
|
|
, Code specifiers
|
|
|
|
, Code params
|
|
|
|
, Code ret_type
|
|
|
|
);
|
|
|
|
|
2023-04-03 00:55:28 -07:00
|
|
|
/*
|
|
|
|
Define an expression:
|
|
|
|
< c/c++ expression >
|
2023-04-03 23:04:19 -07:00
|
|
|
|
|
|
|
TODO: Evalute if you want to validiate at the execution layer during gen_time (dosen't seem necessary)
|
2023-04-03 00:55:28 -07:00
|
|
|
*/
|
2023-04-03 23:04:19 -07:00
|
|
|
// Code def_expression( Code value );
|
2023-04-01 19:21:46 -07:00
|
|
|
|
2023-04-03 00:55:28 -07:00
|
|
|
/*
|
|
|
|
Define a function:
|
|
|
|
<specifiers> <name> ( <params> )
|
|
|
|
{
|
|
|
|
<body>
|
|
|
|
}
|
|
|
|
*/
|
2023-04-03 23:04:19 -07:00
|
|
|
Code def_proc( char const* name
|
2023-04-01 19:21:46 -07:00
|
|
|
, Code specifiers
|
|
|
|
, Code params
|
|
|
|
, Code ret_type
|
|
|
|
, Code body
|
|
|
|
);
|
2023-04-02 09:35:14 -07:00
|
|
|
|
2023-04-03 00:55:28 -07:00
|
|
|
/*
|
|
|
|
Define a fucntion body:
|
|
|
|
{
|
|
|
|
<entry>
|
|
|
|
|
|
|
|
...
|
|
|
|
}
|
|
|
|
|
2023-04-03 23:04:19 -07:00
|
|
|
There will be an empty line separation between entires
|
2023-04-03 00:55:28 -07:00
|
|
|
*/
|
2023-04-03 23:04:19 -07:00
|
|
|
Code def_proc_body( s32 num, ... );
|
|
|
|
Code def_proc_body( s32 num, Code* codes );
|
2023-04-03 00:55:28 -07:00
|
|
|
|
|
|
|
/*
|
|
|
|
Define a namespace;
|
|
|
|
namespace <name>
|
|
|
|
{
|
|
|
|
<body>
|
|
|
|
}
|
|
|
|
*/
|
2023-04-02 09:35:14 -07:00
|
|
|
Code def_namespace( char const* name, Code body );
|
|
|
|
|
2023-04-03 00:55:28 -07:00
|
|
|
/*
|
|
|
|
Define a namespace body:
|
|
|
|
{
|
|
|
|
<entry>
|
2023-04-02 09:35:14 -07:00
|
|
|
|
2023-04-03 00:55:28 -07:00
|
|
|
...
|
|
|
|
}
|
|
|
|
|
2023-04-03 23:04:19 -07:00
|
|
|
There will be an empty line separation between entires
|
2023-04-03 00:55:28 -07:00
|
|
|
*/
|
|
|
|
Code def_namespace_body( s32 num, ... );
|
|
|
|
|
|
|
|
/*
|
|
|
|
Define a set of parameters for a function:
|
|
|
|
<name> <type>, ...
|
|
|
|
*/
|
2023-04-03 23:04:19 -07:00
|
|
|
Code def_params( s32 num, ... );
|
|
|
|
Code def_params( s32 num, char const** params );
|
|
|
|
|
|
|
|
/*
|
|
|
|
Define an operator definition.
|
|
|
|
*/
|
|
|
|
Code def_operator( OperatorT op, Code specifiers, Code params, Code ret_type, Code body );
|
2023-04-03 00:55:28 -07:00
|
|
|
|
|
|
|
/*
|
|
|
|
Define a set of specifiers for a function, struct, type, or varaible
|
2023-04-03 23:04:19 -07:00
|
|
|
|
|
|
|
Note:
|
|
|
|
If alignas is specified the procedure expects the next argument to be the alignment value.
|
|
|
|
If attribute is specified the procedure expects the next argument to be its content as a string.
|
2023-04-03 00:55:28 -07:00
|
|
|
*/
|
|
|
|
Code def_specifiers( s32 num , ... );
|
2023-04-03 23:04:19 -07:00
|
|
|
Code def_specifiers( s32 num, SpecifierT* specs );
|
2023-04-03 00:55:28 -07:00
|
|
|
|
|
|
|
/*
|
|
|
|
Define a struct:
|
|
|
|
struct <specifiers> <name> : <parent>
|
|
|
|
{
|
|
|
|
<body>
|
|
|
|
}
|
|
|
|
*/
|
2023-04-02 09:35:14 -07:00
|
|
|
Code def_struct( char const* name, Code body, Code parent = UnusedCode, Code specifiers = UnusedCode );
|
2023-04-01 19:21:46 -07:00
|
|
|
|
2023-04-03 00:55:28 -07:00
|
|
|
/*
|
|
|
|
Define a struct's body:
|
|
|
|
{
|
|
|
|
<entry>
|
|
|
|
|
|
|
|
...
|
|
|
|
}
|
|
|
|
|
2023-04-03 23:04:19 -07:00
|
|
|
There will be an empty line separation between entires
|
2023-04-03 00:55:28 -07:00
|
|
|
*/
|
|
|
|
Code def_struct_body( s32 num, ... );
|
2023-04-03 23:04:19 -07:00
|
|
|
Code def_struct_body( s32 num, Code* codes );
|
2023-04-03 00:55:28 -07:00
|
|
|
|
|
|
|
/*
|
|
|
|
Define a variable:
|
|
|
|
<specifiers> <type> <name> = <value>;
|
|
|
|
*/
|
2023-04-03 23:04:19 -07:00
|
|
|
Code def_variable( Code type, char const* name, Code value = UnusedCode, Code specifiers = UnusedCode );
|
2023-04-01 19:21:46 -07:00
|
|
|
|
2023-04-03 00:55:28 -07:00
|
|
|
/*
|
2023-04-03 23:04:19 -07:00
|
|
|
Define a typename AST value.
|
|
|
|
Useless by itself, its intended to be used in conjunction with other Code.
|
|
|
|
|
|
|
|
Planned - Not yet Implemented:
|
|
|
|
Typename Codes are not held in the CodePool, instead they are stored in a
|
|
|
|
type registry (hastable where the key is a crc hash of the name string).
|
|
|
|
|
|
|
|
If a key exists the existing code value will be provided.
|
2023-04-03 00:55:28 -07:00
|
|
|
*/
|
2023-04-02 09:35:14 -07:00
|
|
|
Code def_type( char const* name );
|
2023-04-01 19:21:46 -07:00
|
|
|
|
2023-04-03 00:55:28 -07:00
|
|
|
/*
|
|
|
|
Define a using typedef:
|
|
|
|
using <name> = <type>;
|
|
|
|
*/
|
2023-04-02 09:35:14 -07:00
|
|
|
Code def_using( char const* name, Code type );
|
2023-04-01 19:21:46 -07:00
|
|
|
|
2023-04-03 00:55:28 -07:00
|
|
|
/*
|
|
|
|
Define a using namespace:
|
|
|
|
using namespace <name>;
|
|
|
|
|
|
|
|
Can only be used in either a
|
|
|
|
*/
|
|
|
|
Code def_using_namespace( char const* name );
|
2023-04-03 23:04:19 -07:00
|
|
|
#pragma endregion Upfront
|
|
|
|
|
|
|
|
#pragma region Incremental
|
|
|
|
/*
|
|
|
|
Provides an incomplete procedure AST but sets the intended type.
|
|
|
|
Any adds will be type checked.
|
|
|
|
|
|
|
|
Body is automatically made. Use body() to retrieve.
|
|
|
|
*/
|
|
|
|
Code make_proc( char const* name
|
|
|
|
, Code specifiers = UnusedCode
|
|
|
|
, Code params = UnusedCode
|
|
|
|
, Code ret_type = UnusedCode
|
|
|
|
);
|
|
|
|
|
|
|
|
/*
|
|
|
|
Provides an incomplete struct AST but sets the intended type.
|
|
|
|
Any adds will be type checked.
|
|
|
|
|
|
|
|
Body is automatically made. Use body() to retrieve.
|
|
|
|
*/
|
|
|
|
Code make_struct( char const* name, Code parent = UnusedCode, Code specifiers = UnusedCode );
|
|
|
|
|
|
|
|
/*
|
|
|
|
Creates a unit file.
|
|
|
|
|
|
|
|
These represent an encapsulation of a generated file
|
|
|
|
Used this if you need to pass around a group of Code entires at file scope level.
|
|
|
|
|
|
|
|
The name provided is the name of the file.
|
|
|
|
*/
|
|
|
|
Code make_file_body( char const* name );
|
|
|
|
#pragma endregion Incremental
|
|
|
|
|
|
|
|
/*
|
|
|
|
*/
|
|
|
|
Code parse_variable( char const* var_def, s32 length );
|
|
|
|
|
|
|
|
/*
|
|
|
|
*/
|
|
|
|
Code parse_using( char const* using_def, s32 length );
|
|
|
|
|
|
|
|
/*
|
|
|
|
|
|
|
|
*/
|
|
|
|
Code parse_operator( char const* operator_def, s32 length );
|
2023-04-03 00:55:28 -07:00
|
|
|
|
2023-04-03 23:04:19 -07:00
|
|
|
/*
|
|
|
|
Define a procedure by parsing a string.
|
|
|
|
|
|
|
|
Note: This parser only supports the language features the library supports
|
|
|
|
Any other features used and the lex or parse operation will fail.
|
|
|
|
|
|
|
|
This is not a full-on c/c++ parser, it literally only grabs
|
|
|
|
what it needs to reconstruct the Code AST for seralization in the
|
|
|
|
builder, nothing else.
|
|
|
|
*/
|
|
|
|
Code parse_proc( char const* proc_def, s32 length );
|
|
|
|
|
|
|
|
/*
|
|
|
|
Define a struct by parsing a string.
|
|
|
|
|
|
|
|
Note: This parser only supports the language features the library supports
|
|
|
|
Any other features used and the lex or parse operation will fail.
|
|
|
|
|
|
|
|
This is not a full-on c/c++ parser, it literally only grabs
|
|
|
|
what it needs to reconstruct the Code AST for seralization in the
|
|
|
|
builder, nothing else.
|
|
|
|
*/
|
|
|
|
Code parse_struct( char const* struct_def, s32 length );
|
|
|
|
|
|
|
|
/*
|
|
|
|
|
|
|
|
*/
|
|
|
|
s32 parse_vars( char const* vars_def, s32 length, Code* out_vars_codes );
|
|
|
|
|
|
|
|
/*
|
|
|
|
|
|
|
|
*/
|
|
|
|
s32 parse_usings( char const* usings_def, s32 length, Code* out_usings_codes );
|
|
|
|
|
|
|
|
#pragma region Untyped text
|
2023-04-03 00:55:28 -07:00
|
|
|
/*
|
|
|
|
Define an untyped code string.
|
|
|
|
|
|
|
|
Untyped code may be used in bodies of functions, namespaces, or structs
|
|
|
|
or the in places where expressions may be placed.
|
|
|
|
|
|
|
|
Because the code within it is untyped, errors will naturally not be provided.
|
|
|
|
Consider this an a preprocessor define.
|
|
|
|
*/
|
|
|
|
Code untyped_str( char const* str );
|
2023-04-03 23:04:19 -07:00
|
|
|
Code untyped_str( char const* str, s32 length);
|
2023-04-03 00:55:28 -07:00
|
|
|
|
|
|
|
/*
|
|
|
|
Define an untyped code string using traditional 'printf'.
|
|
|
|
|
|
|
|
Untyped code may be used in bodies of functions, namespaces, or structs
|
|
|
|
or the in places where expressions may be placed.
|
|
|
|
|
|
|
|
Because the code within it is untyped, errors will naturally not be provided.
|
|
|
|
Consider this an a preprocessor define.
|
|
|
|
*/
|
2023-04-02 09:35:14 -07:00
|
|
|
Code untyped_fmt( char const* fmt, ... );
|
2023-04-01 19:21:46 -07:00
|
|
|
|
2023-04-03 00:55:28 -07:00
|
|
|
/*
|
|
|
|
Define an untyped code string using token formatting:
|
|
|
|
... { <ID> } ... Will be repalced with value of token ID.
|
|
|
|
|
|
|
|
Values are to provided as: <char const* ID>, <char const* Value>, ...
|
|
|
|
|
|
|
|
num_tokens : The number of ID-Value pairs provided.
|
|
|
|
|
|
|
|
Untyped code may be used in bodies of functions, namespaces, or structs
|
|
|
|
or the in places where expressions may be placed.
|
|
|
|
|
|
|
|
Because the code within it is untyped, errors will naturally not be provided.
|
|
|
|
Consider this an a preprocessor define.
|
|
|
|
*/
|
2023-04-03 23:04:19 -07:00
|
|
|
Code untyped_token_fmt( char const* fmt, s32 num_tokens, ... );
|
|
|
|
#pragma endregion Untyped text
|
2023-04-03 00:55:28 -07:00
|
|
|
|
|
|
|
/*
|
|
|
|
Used to generate the files.
|
|
|
|
This is inspired by jai's usage of the string_builder with #insert.
|
|
|
|
|
|
|
|
Its expected when using this library that Code ast will be serialized with the:
|
|
|
|
Builder::print() proc
|
|
|
|
|
|
|
|
The seralized content of the Code ast will be appended to Buffer within an empty line
|
|
|
|
prepared for a another definition or to add an empty newline to the end of the file.
|
|
|
|
|
|
|
|
Builder::write() should be called when all Code has been seralized for that file.
|
|
|
|
|
|
|
|
The #insert directive is thus represented by an #include of the generated file at your desired line
|
|
|
|
of any file in the target project.
|
|
|
|
*/
|
2023-04-01 22:07:44 -07:00
|
|
|
struct Builder
|
2023-04-01 19:21:46 -07:00
|
|
|
{
|
2023-04-01 22:07:44 -07:00
|
|
|
zpl_file File;
|
|
|
|
string Buffer;
|
2023-04-01 19:21:46 -07:00
|
|
|
|
2023-04-01 22:07:44 -07:00
|
|
|
void print( Code );
|
2023-04-01 19:21:46 -07:00
|
|
|
|
2023-04-01 22:07:44 -07:00
|
|
|
bool open( char const* path );
|
2023-04-01 19:21:46 -07:00
|
|
|
void write();
|
|
|
|
};
|
2023-04-03 00:55:28 -07:00
|
|
|
#pragma endregion gen API
|
2023-04-01 19:21:46 -07:00
|
|
|
}
|
2023-04-01 22:07:44 -07:00
|
|
|
|
2023-04-03 00:55:28 -07:00
|
|
|
#pragma region MACROS
|
|
|
|
# define gen_main main
|
2023-04-03 23:04:19 -07:00
|
|
|
# define __ UnusedCode
|
2023-04-03 00:55:28 -07:00
|
|
|
|
|
|
|
/*
|
|
|
|
gen's Domain Specific Langauge.
|
|
|
|
|
2023-04-03 23:04:19 -07:00
|
|
|
Completely optional, makes the code gen syntax less verbose and cumbersome...
|
|
|
|
Since its C macros ends up looking like a lisp dialect...
|
2023-04-03 00:55:28 -07:00
|
|
|
|
2023-04-03 23:04:19 -07:00
|
|
|
Longforms auto-define the variable.
|
|
|
|
Shorthands are just the function call.
|
2023-04-03 00:55:28 -07:00
|
|
|
|
2023-04-03 23:04:19 -07:00
|
|
|
Anything below the make() macro is intended to be syntacticall used int he follwing format:
|
|
|
|
make( <type>, <name> )
|
|
|
|
{
|
|
|
|
...
|
|
|
|
}
|
2023-04-03 00:55:28 -07:00
|
|
|
|
2023-04-03 23:04:19 -07:00
|
|
|
Where ... are whatever is deemed necessary to produce the definition for the def( <name> ).
|
2023-04-03 00:55:28 -07:00
|
|
|
|
2023-04-03 23:04:19 -07:00
|
|
|
The code macros are used to embed c/c++ to insert into the desired lcoation.
|
2023-04-03 00:55:28 -07:00
|
|
|
*/
|
2023-04-03 23:04:19 -07:00
|
|
|
#ifdef GEN_DEFINE_DSL
|
|
|
|
# define untyped_code( Name_, Value_ ) Code Name_ = gen::untyped_str( txt(Value_) )
|
|
|
|
# define typename( Name_, Value_ ) Code t_##Name_ = gen::def_type( txt(Value_) )
|
|
|
|
# define typename_fmt( Name_, Fmt_, ... ) Code t_##Name_ = gen::def_type( bprintf( Fmt_, __VA_ARGS__ ) )
|
|
|
|
# define using_type( Name_, Type_ ) Code Name_ = gen::def_using( #Name_, t_##Type_ )
|
|
|
|
# define variable( Type_, Name_, ... ) Code Name_ = gen::def_variable( t_##Type_, #Name_, __VA_ARGS__ )
|
|
|
|
|
|
|
|
# define untyped( Value_ ) gen::untyped_str( txt(Value_) )
|
|
|
|
# define code_token( Fmt_, ... ) gen::untyped_token_fmt( Fmt_, VA_NARGS( __VA_ARGS__), __VA_ARGS__ )
|
|
|
|
# define code_fmt( Fmt_, ... ) gen::untyped_fmt( Fmt_, __VA_ARGS__ )
|
|
|
|
# define specifiers( ... ) gen::def_specifiers( VA_NARGS( __VA_ARGS__ ), __VA_ARGS__ )
|
|
|
|
# define type( Value_ ) gen::def_type( txt(Value_) )
|
|
|
|
# define type_fmt( Fmt_, ... ) gen::def_type( bprintf( Fmt_, __VA_ARGS__ ) )
|
|
|
|
# define using( Name_, Type_ ) gen::def_using( #Name_, Type_ )
|
|
|
|
# define var( Type_, Name_, ... ) gen::def_variable( Type_, #Name_, __VA_ARGS__ )
|
|
|
|
|
|
|
|
# define make( Type_, Name_, ... ) Code Name_ = make_##Type_( #Name_, __VA_ARGS__ );
|
|
|
|
# define proc( Name_, Specifiers_, RetType_, Parameters_, Body_ ) Name_ = gen::def_proc( #Name_, Specifiers_, Parameters_, RetType_, Body_ )
|
|
|
|
# define proc_body( ... ) gen::def_proc_body( VA_NARGS( __VA_ARS__ ), __VA_ARGS__ )
|
|
|
|
# define params( ... ) gen::def_params( VA_NARGS( __VA_ARGS__ ) / 2, __VA_ARGS__ )
|
|
|
|
# define struct( Name_, Parent_, Specifiers_, Body_ ) Name_ = gen::def_struct( #Name_, Body_, Parent_, Specifiers_ )
|
|
|
|
# define struct_body( ... ) gen::def_struct_body( VA_NARGS( __VA_ARGS__ ), __VA_ARGS__ )
|
|
|
|
|
|
|
|
# define add_var( Type_, Name_, ... ) add( gen::def_variable( t_##Type_, #Name_, __VA_ARGS__ ) )
|
|
|
|
# define add_untyped( Value_ ) add( gen::untyped_str( txt( Value ) ) )
|
|
|
|
# define add_ret_type( ... )
|
|
|
|
# define add_params( ... )
|
|
|
|
|
|
|
|
# define proc_code( Def_ ) gen::parse_proc( txt( Def_ ), sizeof( txt( Def_ )) )
|
|
|
|
# define struct_code( Def_ ) gen::parse_struct( txt( Def_ ), sizeof( txt( Def_ )) )
|
2023-04-03 00:55:28 -07:00
|
|
|
#endif
|
|
|
|
#pragma endregion MACROS
|
|
|
|
|
|
|
|
#pragma region CONSTANTS
|
2023-04-03 23:04:19 -07:00
|
|
|
#ifdef GEN_DEFINE_LIBRARY_CODE_CONSTANTS
|
2023-04-03 00:55:28 -07:00
|
|
|
namespace gen
|
|
|
|
{
|
|
|
|
// Predefined typename codes.
|
2023-04-03 23:04:19 -07:00
|
|
|
// These are not set until gen::init is called.
|
|
|
|
// This just preloads a bunch of Code types into the code pool.
|
|
|
|
|
|
|
|
extern const Code t_void;
|
2023-04-03 00:55:28 -07:00
|
|
|
|
|
|
|
extern const Code t_bool;
|
2023-04-03 23:04:19 -07:00
|
|
|
extern const Code t_char;
|
|
|
|
extern const Code t_wchar_t;
|
|
|
|
|
|
|
|
extern const Code t_s8;
|
|
|
|
extern const Code t_s16;
|
|
|
|
extern const Code t_s32;
|
|
|
|
extern const Code t_s64;
|
|
|
|
|
|
|
|
extern const Code t_u8;
|
|
|
|
extern const Code t_u16;
|
|
|
|
extern const Code t_u32;
|
|
|
|
extern const Code t_u64;
|
|
|
|
|
2023-04-03 00:55:28 -07:00
|
|
|
extern const Code t_sw;
|
|
|
|
extern const Code t_uw;
|
|
|
|
|
2023-04-03 23:04:19 -07:00
|
|
|
extern const Code t_f32;
|
|
|
|
extern const Code t_f64;
|
|
|
|
|
|
|
|
extern const Code spec_constexpr;
|
2023-04-03 00:55:28 -07:00
|
|
|
extern const Code spec_inline;
|
|
|
|
}
|
2023-04-03 23:04:19 -07:00
|
|
|
#endif
|
2023-04-03 00:55:28 -07:00
|
|
|
#pragma endregion CONSTANTS
|
2023-04-01 19:21:46 -07:00
|
|
|
#endif
|