2024-12-10 19:31:50 -05:00
# General Docs
2023-07-29 05:52:06 -04:00
2024-12-10 19:31:50 -05:00
[Top ](../Readme.md )
2023-07-29 05:52:06 -04:00
2024-12-10 19:31:50 -05:00
Contains:
2023-07-29 05:52:06 -04:00
2024-12-12 10:24:21 -05:00
* [AST_Design ](./AST_Design.md ): Overview of ASTs
2024-12-10 19:31:50 -05:00
* [AST Types ](./AST_Types.md ): Listing of all AST types along with their Code type interface.
* [Parsing ](./Parsing.md ): Overview of the parsing interface.
* [Parser Algo ](./Parser_Algo.md ): In-depth breakdown of the parser's implementation.
2023-07-29 05:52:06 -04:00
2024-12-10 19:31:50 -05:00
### *CURRENTLY UNSUPPORTED*
2023-07-29 05:52:06 -04:00
**There is no support for validating expressions.**
2024-12-10 16:38:01 -05:00
Its a [todo ](https://github.com/Ed94/gencpp/issues/49 )
2023-07-29 05:52:06 -04:00
2024-12-10 16:38:01 -05:00
**Only trivial template support is provided.**
The intention is for only simple, non-recursive substitution.
The parameters of the template are treated like regular parameter AST entries.
2023-07-29 05:52:06 -04:00
This means that the typename entry for the parameter AST would be either:
* `class`
* `typename`
* A fundamental type, function, or pointer type.
2024-12-10 16:38:01 -05:00
***Concepts and Constraints are not supported***
Its a [todo ](https://github.com/Ed94/gencpp/issues/21 )
2023-07-29 05:52:06 -04:00
2024-12-10 16:38:01 -05:00
### Feature Macros:
* `GEN_DEFINE_ATTRIBUTE_TOKENS` : Allows user to define their own attribute macros for use in parsing.
2024-12-10 19:31:50 -05:00
* This can be generated using base.cpp.
2024-12-10 16:38:01 -05:00
* `GEN_DEFINE_LIBRARY_CORE_CONSTANTS` : Optional typename codes as they are non-standard to C/C++ and not necessary to library usage
* `GEN_DONT_ENFORCE_GEN_TIME_GUARD` : By default, the library ( gen.hpp/ gen.cpp ) expects the macro `GEN_TIME` to be defined, this disables that.
* `GEN_ENFORCE_STRONG_CODE_TYPES` : Enforces casts to filtered code types.
* `GEN_EXPOSE_BACKEND` : Will expose symbols meant for internal use only.
* `GEN_ROLL_OWN_DEPENDENCIES` : Optional override so that user may define the dependencies themselves.
* `GEN_DONT_ALLOW_INVALID_CODE` (Not implemented yet) : Will fail when an invalid code is constructed, parsed, or serialized.
2024-12-10 19:31:50 -05:00
* `GEN_C_LIKE_PP` : Setting to `<true or 1>` Will prevent usage of function defnitions using references and structs with member functions. Structs will still have user-defined operator conversions, for-range support, and other operator overloads
2023-07-29 05:52:06 -04:00
### The Data & Interface
2023-08-08 11:56:42 -04:00
As mentioned in root readme, the user is provided Code objects by calling the constructor's functions to generate them or find existing matches.
2023-07-29 05:52:06 -04:00
2024-12-10 19:31:50 -05:00
The AST is managed by the library and provided to the user via its interface.
2023-07-29 05:52:06 -04:00
However, the user may specifiy memory configuration.
2024-12-10 16:56:42 -05:00
[Data layout of AST struct (Subject to heavily change with upcoming todos) ](../base/components/ast.hpp#L396-461 )
2024-12-10 16:58:04 -05:00
https://github.com/Ed94/gencpp/blob/eea4ebf5c40d5d87baa465abfb1be30845b2377e/base/components/ast.hpp#L396 -L461
2023-07-29 05:52:06 -04:00
2024-12-12 10:24:21 -05:00
*`StringCahced` is a typedef for `StrC` (a string slice), to denote it is an interned string*
2024-12-10 19:31:50 -05:00
*`CodeType` is enum taggin the type of code. Has an underlying type of `u32` *
*`OperatorT` is a typedef for `EOperator::Type` which has an underlying type of `u32` *
*`String` is the dynamically allocated string type for the library*
2023-07-29 05:52:06 -04:00
2024-12-10 19:31:50 -05:00
AST widths are setup to be AST_POD_Size.
2023-07-29 05:52:06 -04:00
The width dictates how much the static array can hold before it must give way to using an allocated array:
```cpp
constexpr static
2024-12-10 19:31:50 -05:00
int AST_ArrSpecs_Cap =
2023-07-29 05:52:06 -04:00
(
AST_POD_Size
2024-12-10 19:31:50 -05:00
- sizeof(Code)
2023-07-29 05:52:06 -04:00
- sizeof(StringCached)
2024-12-10 19:31:50 -05:00
- sizeof(Code) * 2
- sizeof(Token*)
- sizeof(Code)
- sizeof(CodeType)
2023-07-29 05:52:06 -04:00
- sizeof(ModuleFlag)
2023-08-08 11:56:42 -04:00
- sizeof(u32)
2023-07-29 05:52:06 -04:00
)
2024-12-10 19:31:50 -05:00
/ sizeof(Specifier) - 1;
2023-07-29 05:52:06 -04:00
```
*Ex: If the AST_POD_Size is 128 the capacity of the static array is 20.*
Data Notes:
* The allocator definitions used are exposed to the user incase they want to dictate memory usage
2023-08-08 11:56:42 -04:00
* You'll find the memory handling in `init` , `deinit` , `reset` , `gen_string_allocator` , `get_cached_string` , `make_code` .
2024-12-10 19:31:50 -05:00
* Allocators are defined with the `AllocatorInfo` structure found in [`memory.hpp` ](../base/dependencies/memory.hpp )
2023-09-11 18:34:37 -04:00
* Most of the work is just defining the allocation procedure:
```cpp
2024-10-27 18:58:37 -04:00
void* ( void* allocator_data, AllocType type, ssize size, ssize alignment, void* old_memory, ssize old_size, u64 flags );
2023-09-11 18:34:37 -04:00
```
2023-07-29 05:52:06 -04:00
* ASTs are wrapped for the user in a Code struct which is a wrapper for a AST* type.
2024-12-10 19:31:50 -05:00
* Code types have member symbols but their data layout is enforced to be POD types.
2023-07-29 05:52:06 -04:00
* This library treats memory failures as fatal.
* Cached Strings are stored in their own set of arenas. AST constructors use cached strings for names, and content.
* `StringArenas` , `StringCache` , `Allocator_StringArena` , and `Allocator_StringTable` are the associated containers or allocators.
* Strings used for serialization and file buffers are not contained by those used for cached strings.
* They are currently using `GlobalAllocator` , which are tracked array of arenas that grows as needed (adds buckets when one runs out).
* Memory within the buckets is not reused, so its inherently wasteful.
2024-12-10 19:31:50 -05:00
* I will be augmenting the default allocator with virtual memory & a slab allocator in the [future ](https://github.com/Ed94/gencpp/issues/12 )
* Intrusive linked lists used children nodes on bodies, and parameters.
2023-07-29 05:52:06 -04:00
* Its intended to generate the AST in one go and serialize after. The constructors and serializer are designed to be a "one pass, front to back" setup.
2024-12-10 19:31:50 -05:00
* Allocations can be tuned by defining the folloiwng macros (will be moved to runtime configuration in the future):
2023-07-29 05:52:06 -04:00
* `GEN_GLOBAL_BUCKET_SIZE` : Size of each bucket area for the global allocator
2023-08-08 11:56:42 -04:00
* `GEN_CODEPOOL_NUM_BLOCKS` : Number of blocks per code pool in the code allocator
* `GEN_SIZE_PER_STRING_ARENA` : Size per arena used with string caching.
2023-07-29 05:52:06 -04:00
* `GEN_MAX_COMMENT_LINE_LENGTH` : Longest length a comment can have per line.
* `GEN_MAX_NAME_LENGTH` : Max length of any identifier.
* `GEN_MAX_UNTYPED_STR_LENGTH` : Max content length for any untyped code.
2024-12-10 19:31:50 -05:00
* `TokenMap_FixedArena` : token_fmt_va uses local_persit memory of this arena type for the hashtable.
2023-08-08 11:56:42 -04:00
* `GEN_LEX_ALLOCATOR_SIZE`
* `GEN_BUILDER_STR_BUFFER_RESERVE`
2023-07-29 05:52:06 -04:00
The following CodeTypes are used which the user may optionally use strong typing with if they enable: `GEN_ENFORCE_STRONG_CODE_TYPES`
2024-12-10 19:31:50 -05:00
* CodeBody : Has support for `for : range` iterating across Code objects.
2023-07-29 05:52:06 -04:00
* CodeAttributes
* CodeComment
* CodeClass
* CodeConstructor
* CodeDefine
* CodeDestructor
* CodeEnum
* CodeExec
* CodeExtern
* CodeInclude
* CodeFriend
* CodeFn
* CodeModule
2023-08-06 14:58:43 -04:00
* CodeNS
2023-07-29 05:52:06 -04:00
* CodeOperator
* CodeOpCast
2024-12-11 13:33:35 -05:00
* CodeParams : Has support for `for : range` iterating across parameters.
2023-08-08 11:56:42 -04:00
* CodePreprocessCond
* CodePragma
2024-12-10 19:31:50 -05:00
* CodeSpecifiers : Has support for `for : range` iterating across specifiers.
2023-07-29 05:52:06 -04:00
* CodeStruct
* CodeTemplate
2024-12-10 19:31:50 -05:00
* CodeTypename
2023-07-29 05:52:06 -04:00
* CodeTypedef
* CodeUnion
* CodeUsing
* CodeVar
Each Code boy has an associated "filtered AST" with the naming convention: `AST_<CodeName>`
Unrelated fields of the AST for that node type are omitted and only necessary padding members are defined otherwise.
Retrieving a raw version of the ast can be done using the `raw()` function defined in each AST.
## There are three sets of interfaces for Code AST generation the library provides
* Upfront
* Parsing
* Untyped
### Upfront Construction
All component ASTs must be previously constructed, and provided on creation of the code AST.
2023-08-08 11:56:42 -04:00
The construction will fail and return CodeInvalid otherwise.
2023-07-29 05:52:06 -04:00
Interface :``
* def_attributes
* *This is pre-appended right before the function symbol, or placed after the class or struct keyword for any flavor of attributes used.*
* *Its up to the user to use the desired attribute formatting: `[[]]` (standard), `__declspec` (Microsoft), or `__attribute__` (GNU).*
* def_comment
* def_class
* def_constructor
2023-08-08 11:56:42 -04:00
* def_define
2023-07-29 05:52:06 -04:00
* def_destructor
* def_enum
* def_execution
* *This is equivalent to untyped_str, except that its intended for use only in execution scopes.*
* def_extern_link
* def_friend
* def_function
* def_include
* def_module
* def_namespace
* def_operator
* def_operator_cast
* def_param
* def_params
2023-09-11 18:34:37 -04:00
* def_pragma
2023-08-08 11:56:42 -04:00
* def_preprocess_cond
2023-07-29 05:52:06 -04:00
* def_specifier
* def_specifiers
* def_struct
* def_template
* def_type
* def_typedef
* def_union
* def_using
* def_using_namespace
* def_variable
Bodies:
* def_body
* def_class_body
* def_enum_body
* def_export_body
* def_extern_link_body
* def_function_body
* *Use this for operator bodies as well*
* def_global_body
* def_namespace_body
* def_struct_body
* def_union_body
Usage:
```cpp
< name > = def_< function type > ( ... );
Code < name >
{
...
< name > = def_< function name > ( ... );
}
```
When using the body functions, its recommended to use the args macro to auto determine the number of arguments for the varadic:
2024-12-10 19:31:50 -05:00
2023-07-29 05:52:06 -04:00
```cpp
def_global_body( args( ht_entry, array_ht_entry, hashtable ));
// instead of:
def_global_body( 3, ht_entry, array_ht_entry, hashtable );
```
2024-12-10 16:38:01 -05:00
If a more incremental approach is desired for the body ASTs, `Code def_body( CodeT type )` can be used to create an empty body.
2023-07-29 05:52:06 -04:00
When the members have been populated use: `AST::validate_body` to verify that the members are valid entires for that type.
### Parse construction
A string provided to the API is parsed for the intended language construct.
Interface :
* parse_class
* parse_constructor
* parse_destructor
* parse_enum
* parse_export_body
* parse_extern_link
* parse_friend
* Purposefully are only support forward declares with this constructor.
* parse_function
* parse_global_body
* parse_namespace
* parse_operator
* parse_operator_cast
* parse_struct
* parse_template
* parse_type
* parse_typedef
* parse_union
* parse_using
* parse_variable
Usage:
```cpp
Code < name > = parse_< function name > ( string with code );
Code < name > = def_< function name > ( ..., parse_< function name > (
< string with code >
));
```
### Untyped constructions
Code ASTs are constructed using unvalidated strings.
Interface :
* token_fmt_va
* token_fmt
* untyped_str
* untyped_fmt
* untyped_token_fmt
2024-12-10 16:38:01 -05:00
During serialization any untyped Code AST has its string value directly injected inline of whatever context the content existed as an entry within.
2023-07-29 05:52:06 -04:00
Even though these 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 (ex: variable assignment).
These restrictions help prevent abuse of untyped code to some extent.
Usage Conventions:
```cpp
Code < name > = def_variable( < type > , < name > , untyped_< function name > (
< string with code >
));
Code < name > = untyped_str( code(
< some code without " " quotes >
));
```
Optionally, `code_str` , and `code_fmt` macros can be used so that the code macro doesn't have to be used:
2024-12-10 19:31:50 -05:00
2023-07-29 05:52:06 -04:00
```cpp
Code < name > = code_str( < some code without " " quotes > )
```
Template metaprogramming in the traditional sense becomes possible with the use of `token_fmt` and parse constructors:
```cpp
2023-08-09 18:47:59 -04:00
StrC value = txt("Something");
2023-07-29 05:52:06 -04:00
char const* template_str = txt(
Code with < key > to replace with token_values
...
);
char const* gen_code_str = token_fmt( "key", value, template_str );
Code < name > = parse_< function name > ( gen_code_str );
```
## Predefined Codes
The following are provided predefined by the library as they are commonly used:
* `access_public`
* `access_protected`
* `access_private`
2023-08-08 11:56:42 -04:00
* `attrib_api_export`
* `attrib_api_import`
2023-07-29 05:52:06 -04:00
* `module_global_fragment`
* `module_private_fragment`
2023-08-08 11:56:42 -04:00
* `fmt_newline`
2023-09-11 18:34:37 -04:00
* `pragma_once`
2024-12-10 19:31:50 -05:00
* `param_varaidc` (Used for varadic definitions)
2023-07-29 05:52:06 -04:00
* `preprocess_else`
* `preprocess_endif`
* `spec_const`
* `spec_consteval`
* `spec_constexpr`
* `spec_constinit`
* `spec_extern_linkage` (extern)
* `spec_final`
2023-09-11 18:34:37 -04:00
* `spec_forceinline`
2023-07-29 05:52:06 -04:00
* `spec_global` (global macro)
* `spec_inline`
* `spec_internal_linkage` (internal macro)
* `spec_local_persist` (local_persist macro)
* `spec_mutable`
2023-08-08 11:56:42 -04:00
* `spec_neverinline`
2024-12-10 19:31:50 -05:00
* `spec_noexcept`
2023-07-29 05:52:06 -04:00
* `spec_override`
* `spec_ptr`
2023-08-08 11:56:42 -04:00
* `spec_pure`
2023-07-29 05:52:06 -04:00
* `spec_ref`
* `spec_register`
* `spec_rvalue`
* `spec_static_member` (static)
* `spec_thread_local`
* `spec_virtual`
* `spec_volatile`
* `t_empty` (Used for varaidc macros)
* `t_auto`
* `t_void`
* `t_int`
* `t_bool`
* `t_char`
* `t_wchar_t`
* `t_class`
* `t_typename`
Optionally the following may be defined if `GEN_DEFINE_LIBRARY_CODE_CONSTANTS` is defined
* `t_b32`
* `t_s8`
* `t_s16`
* `t_s32`
* `t_s64`
* `t_u8`
* `t_u16`
* `t_u32`
* `t_u64`
2024-12-10 19:31:50 -05:00
* `t_ssize` (ssize_t)
* `t_usize` (size_t)
2023-07-29 05:52:06 -04:00
* `t_f32`
* `t_f64`
## Extent of operator overload validation
The AST and constructors will be able to validate that the arguments provided for the operator type match the expected form:
* If return type must match a parameter
* If number of parameters is correct
* If added as a member symbol to a class or struct, that operator matches the requirements for the class (types match up)
2024-04-17 18:29:30 -04:00
* There is no support for validating new & delete operations (yet)
2023-07-29 05:52:06 -04:00
The user is responsible for making sure the code types provided are correct
and have the desired specifiers assigned to them beforehand.
## Code generation and modification
2024-12-10 19:31:50 -05:00
There are two provided auxillary interfaces:
2023-07-29 05:52:06 -04:00
* Builder
* Scanner
### Builder is a similar object to the jai language's string_builder
* The purpose of it is to generate a file.
2023-09-11 18:34:37 -04:00
* A file is specified and opened for writing using the open( file_path) function.
2023-07-29 05:52:06 -04:00
* The code is provided via print( code ) function will be serialized to its buffer.
* When all serialization is finished, use the write() command to write the buffer to the file.
2023-09-11 18:34:37 -04:00
### Scanner Auxillary Interface
2023-07-29 05:52:06 -04:00
2024-12-11 13:33:35 -05:00
* The purpose is to scan or parse files
* Some with two basic functions to convert a fil to code: `scan_file` and `parse_file`
* `scan_file` : Merely grabs the file and stores it in an untyped Code.
* `parse_file` : Will parse the file using `parse_global_body` and return a `CodeBody` .
* Two basic functions for grabbing columns from a CSV: `parse_csv_one_column` and `parse_csv_two_columns`