Artus is a statically-typed toy programming language with your typical programming paradigms. It supports programs across multiple source files by the way of a modular package system, and compiles down to most instruction sets through LLVM.
Artus uses CMake (version 3.31) version to produce an executable, and requires linking against LLVM 18.1.8-4.
To fully compile and link source programs, the compiler shells out to clang
and
thus requires it on the system if binaries are desired without an external linker.
Identifier | Name | Example |
---|---|---|
bool |
boolean | 0 , 1 |
char |
char/8-bit | 'a' , 256 |
i32 |
32-bit signed | 2^31 |
i64 |
64-bit signed | 2^63 |
u8 |
8-bit unsigned | 2^8 |
u32 |
32-bit unsigned | 2^32 |
u64 |
64-bit unsigned | 2^64 |
str |
string | "abc" |
f64 |
floating point | 3.14 |
Operator | Precedence | Use |
---|---|---|
= += -= *= /= |
1 | Assignment |
&& || ^^ |
2 | Logical Comparison (and, or, xor) |
== != |
3 | Equality Comparison |
< > <= >= |
4 | Inequalities |
+ - |
5 | Additive Ops |
* / |
6 | Multiplicative Ops |
# & ! - |
7 | Unary Ops |
import with: import std::io
// Prints string <s>
fn @print(s: #str) -> void
// Prints string <s> with a newline
fn @println(s: #str) -> void
// Reads a string into pointer s
fn @read_str(s: #str) -> void
import with: import std::memory
// allocate memory
fn @malloc(size: i32) -> #void
// free allocated memory
fn @free(ptr: #void) -> void
// main.artus
import utils
fn @main() -> i64 {
ret @foo
}
// utils.artus
fn @foo() -> i64 {
ret 0
}
/// ptr address-of
mut x: i64 = 5
mut y: #i64 = &x
/// ptr dereference
#y = 0
// array initialization
mut nums: i64[3] = [0, 1, 2]
// array access
fix a: i64 = nums[0]
function-decl:
'fn' '@' <identifier> '(' [param-list] ')' '->' [type] <stmt>
| 'fn' '@' <identifier> '(' [param-list] ')' <stmt>
| 'fn' '@' <identifier> '->' [type] <stmt>
| 'fn' '@' <identifier> <stmt>
param-list:
param [',' param-list]
param:
'mut' <identifier> ':' [type]
| <identifier> ':' [type]
type: <identifier>
// main
fn @main() -> i64 { ... }
// otherwise
fn @foo(a: i64, mut b: i64[2]) -> i64 { ... }
// function calls, empty:
@foo
// function calls, with arguments
@foo(1, [2, 3])
var-decl:
'fix' <identifier> ':' [type] '=' <expr>
| 'mut' <identifier> ':' [type]
| 'mut' <identifier> ':' [type] '=' <expr>
type: <identifier>
// immutable assignment
fix x: i64 = 0
// mutable assignment
mut y: i64 = 1
if-stmt:
'if' <expr> <stmt>
| 'if' <expr> <stmt> 'else' <stmt>
if cond {
...
}
if cond {
...
} else {
...
}
if cond {
...
} else if cond {
...
} else {
...
}
while-stmt:
'while' <expr> <stmt>
while cond {
...
}
until-stmt:
'until' <expr> <stmt>
until cond {
...
}
match-stmt:
'match' <expr> '{' [match-case-list] '}'
match-case-list:
match-case [',' match-case-list]
match-case:
case-stmt
| default-stmt
case-stmt:
<expr> => <stmt>
default-stmt:
'_' => <stmt>
fix x: i64 = 0
match x {
0 => ...,
1 => { ... },
_ => { ... },
}