-
Notifications
You must be signed in to change notification settings - Fork 21
New issue
Have a question about this project? Sign up for a free GitHub account to open an issue and contact its maintainers and the community.
By clicking “Sign up for GitHub”, you agree to our terms of service and privacy statement. We’ll occasionally send you account related emails.
Already on GitHub? Sign in to your account
Add variadic functions #69
Comments
I think I would prefer to do something similar to Rust's take on variadic functions. I don't think functions themselves should take a variable number of arguments, but I think we should allow the user to make macros that handle variadic arguments at compile time. Right now I'm laying down the groundwork for user defined macros for expressions, statements, and declarations. I also think that some form of static polymorphism could be done using a good enough macro system. |
Hm, im not familiar with how rust macro's work but that sounds pretty cool. Also makes it easier to do the variadic stuff at compile time as the runtime wont have to deal with it. |
I'm thinking something like this: #[std]
#[macro print(arg, args[argc]) {
if is_type(arg, num) {
putnum(arg as num)
} else if is_type(arg, &char) {
putstr(arg as &char)
} else if is_type(arg, char) {
putchar(arg as char)
} else if is_type(arg, bool) {
putbool(arg as bool)
}
if argc > 0 {
$print($args)
} else {
putchar('\n')
}
}] This example defines a macro The To use |
This seems like a good start. I think it would be a great replacement to having to write out all the |
To do macros best, I think we need to move away from having the parser spit out IR code. The parser should generate an AST node, which could then be transformed into the top level of IR. This would allow us to do AST transformations without worrying about messing with any side-effecting declarations (like issue #68). Additionally, macros wouldnt need to be expanded to blocks of MIR or HIR, they could all be dealt with exclusively in the AST! |
I am a bit confused on what the difference between the AST and the current TIR would be. Are they not already a tree like structure? |
Yes, TIR is a tree like structure, but it's a very strict structure. Information about the movability of objects and their members needs to be extracted here for use in HIR, copy and drop methods need to be added based on the movability, etc. An AST for Oak, however, could be implemented very loosely. An enum AstNode {
Identifier(String),
MacroArg(String), // Such as $arg in the example above
MacroCall(String, Vec<Self>), // Such as $print($args) in the example above
FunctionDef {
name: Box<Self>,
args: Vec<Self>,
return_type: Box<Self>,
body: Box<Self>
},
Block(Vec<Self>), // A list of `AstNode` objects between brackets
...
} This way, macros could act on ANY part of the AST. This could look like something like this: #[macro make_fn(name, type, body) {
fn $name() -> $type $body
}] This could make the macro system incredibly powerful, I think. |
All the AST would need to compute is
|
Additionally, we could add some compiler optimizations similar to Haskells GHC plugin system (with a TON of work) #[optimize (if (true) $then_body else $else_body) -> ($then_body)] This could be really difficult, but possible? |
Right, an AST with interchangable nodes would make optimizing easier. The attempt in PR #77 is very limited in that regard. |
Im wondering what would be needed to add variadic functions. As it would greatly help to implement
printf
and change the following code:The text was updated successfully, but these errors were encountered: