MetaC or meta-c (for domain name reasons) is a language that aims to be compatible with C99.
You can use a project written in C99 and you should be able to compile it with the metac
compiler.
It improves on C by providing more extensive meta programming capabilities than C.
NOTE: This project is not currently finished and some items described below might not be functional.
Upon startup metac
will try to read a file named metac_compiler_interface.h
and search for struct definitions in there.
Currently, it's expected that only one struct is defined.
It will then put an instance of that struct into the .
namespace/scope which is accessible as .Compiler
.
Right now, the .compiler
does not get populated with current compiler-state information, but this is supposed to change.
Within the language, you can access a struct of type metac_compiler_t
by using the special name .Compiler
.
It exposes the functions that you can call.
During compilation you can have functions run and those functions can access the compiler via the API. This allows you to use powerful meta-programming techniques.
In MetaC, types are expressions.
So in EE
-mode (Expression Evaluation) you can type int
and the return value will be int
.
Or you could write double
and the type-value double
will be returned.
At the time of writing MetaC doesn't do type checking on operations so you can enter expressions such as
int - 1
which will result in an integer value (268435467).
That is because at the time of writing, types are represented as a 32 bit wide bitfield.
The type kind is in the leftmost (most-significant) bits and an index is in the rightmost (least-significant) bits.
This is also the reason why sizeof(typeof(void))
is 4; since the runtime representation of a type is 4 bytes in size (uint32_t).
We can force an integer to be interpreted as a type by assigning it to a variable of type type
:
:ds
type T;
:ee
T=int+1,T
If you execute the code above in the repl the result should be long
.
Similarly, if you execute
:ds
type T;
:ee
T=int-1,T
the result should be short
.
At the moment, this may seem like a quirky toy feature, but it will be one of the corner-stones of meta-programming in MetaC.
_yield
is a keyword in MetaC that has the effect of pausing the execution of the function you are currently in and returning a value.
When you then call that same function again it will not start at the beginning but rather it will continue after the point where you yielded.
All local variables are preserved.
So is any global
state which is annotated with @TaskLocal
.
First, the auto-generated code has to be generated.
Just execute ./gen_code.sh
in the directory you cloned the repository into.
Then, to build the repl simply cd repl
and compile the file linenoise_repl.c
.
An example command line would look like gcc -Os linenoise_repl.c -o repl -lpthread -lm
.
I have only tested this on x86_64
and aarch64
with Ubuntu 16.04 and Ubuntu 18.04 so anything else might run into unforeseen trouble.
If it does fail to compile please open an issue.
NOTE: when you supply the defines
PRINT_BYTECODE=1
andPRINT_CODE=1
the REPL will print the first 24 instructions when evaluating expressions.
Foreign language blocks (FLBs) are a tool to integrate other languages with MetaC.
They are essentially heredoc strings with a language tag that allows source processors to pick out the block it is interested in.
An FLB starts with \n@FLB{\n
for an untagged block or with \n@(FLB, "some-tag-string"){\n
for a tagged foreign language block
and it ends with \n}FLB\n
.
Even though the {
syntax does suggest nesting, foreign language blocks do not nest currently.
This may be subject to change in the future.
The REPL is mostly a development and debugging tool for me
and it serves to mitigate the current shortcomings of the unfinished parser.
Therefore, it has modes which change the parser behavior.
By default, it starts in :ee
the "Expression Evaluation" mode.
In this mode, expressions given on the command line will be parsed, semantically analyzed and evaluated.
The value resulting from this evaluation will be returned.
Then, there is :es
the "Expression Semantic" mode.
In this mode, the expression will be parsed and semantically analyzed.
The type of the expression will be returned.
Another important mode is the :ds
("Declaration Semantic") mode.
In this mode, a declaration/definition will be parsed and inserted into the global scope.
Note that previous definitions will shadow the following ones, meaning
EE> :ds
DS> int x;
DS> :ee
EE> sizeof(x)
EE> :ds
DS> int x[16];
DS> int x2[16];
DS> :ee
EE> sizeof(x)
EE> sizeof(x2)
will result in the output
4
4
64
showing that x cannot be modified after defining it.
This might be unusual for a REPL but this REPL is mainly a tool to develop the semantic and code generation portion of the compiler without having to finish the parser.
The other modes can be seen when typing :h
.
They are mostly for debugging/developing the parser.