The GenASM visitor only uses quad mnemonics so each manipulation to the stack is done in 8 byte chunks. This means our stack is always 8 byte aligned which is fine when only calling KLang code. libc functions like 'malloc' or 'free' require the stack to be 16 byte aligned which isn't automatically guaranteed when being 8 byte aligned. This is why we need to be careful when manipulating the stack (e.g. pushq, popq, subq x, %rsp). By assuming that we are stack aligned at the beginning of execution and by tracking the number of pushq operations we can deduce whether we misaligned the stack and align it again by adding another 8 byte to the stack. This re-alignment of the stack is done directly before entering a function body or calling malloc. Since there are no stack manipulations involved when generating the code to calling 'free' there is reason to check for alignment there.
Klang - The Kaiser language
This is the project for Klang - the Kaiser language.
Authors
This code was in equal parts developed by Dennis Kaiser and Marvin Kaiser at the RheinMain University of Applied Sciences for the Compilers course.
Usage
example call to print help java -cp target/klang-1.0-jar-with-dependencies.jar de.hsrm.compiler.Klang.Klang -h
example call: java -cp <jar> <cp> -out <input> <output>
Arguments:
- -out File to write to
- -h Print this help
- --evaluate: Evaluates the given source code
- --pretty: Pretty print the given source code
- --no-compile: Do not compile the source code
- --no-main: Do not generate main function, will be generated as 'start'. Useful for testing
The makefile can be used to perform various functions more easily:
make buildsimply builds the compilermake cleancleans up all generated outputmakegenerates code.s from code.k in root foldermake prettyprettifies code.k and writes to pretty.kmake evalevaluates code.kmake testruns tests from src/test/make testJavaruns JUnit testsmake cleanTestscleans files generated from tests
Boilerplate Example
A simple program in the KLang Language consits of some struct definitions and some function definition and a single expression that is used as the start for the compilation
struct node {
value: int;
tail: node;
}
function makeList(anz: int): node {
if (anz == 0) {
return naught;
}
return create node(anz - 1, makeList(anz - 1));
}
function get(ll: node, index: int): int {
if (index == 0 || ll == naught) {
return ll.value;
} else {
return get(ll.tail, index - 1);
}
}
function sum(list: node, length: int): int {
if (length == 0) {
return list.value;
}
return list.value + sum(list.tail, length -1);
}
function init(pos: int): int {
let n: node = makeList(5);
return sum(n, pos);
}
init(0);
Functionality
The KLang compiler supports generation of AMD64 assembly code, as well as prettifying and evaluating the KLang code.
Simple Expressions
The following simple expressions are supported. Expressions need to be put in paranthesis. When using comparison operators, the expressions evaluate to 0 for false and 1 for true.
- Addition (+)
- Subtraction (-)
- Multiplication (*)
- Division (/)
- Modulo (%)
- Equality (==)
- Less Than (<)
- Less Than Or Equal (<=)
- Greater Than (>)
- Greater Than Or Equal (>=)
- Number Negation (-)
Examples:
5 + 4
8 % 2
8 == 0
Functions
Functions can be defined and called. A function call can be used like any other expression. Recursion is supported aswell as linking agaings c object files since we are following the calling convention
Examples
function fun(x: int, y: int, z: bool): int {
return x;
}
fun(1, 2, 3);
Structs
Structs can be defined, created and destroyed. Structs can reference other structs as well as themselves. You can reference structs that are defined later in the code. Our structs are compatible to c structs. When defining a struct, a constructor function is implicitly defined so that you can create instances of your struct. To denote a non existing reference to a struct, use the reserved word "naught";
Examples
struct myStruct {
a: int;
b: bool;
c: float;
d: myStruct;
}
function add(x: myStruct, y: myStruct): float {
return x.c + y.c;
}
function isOk(x: myStruct, y: myStruct): bool {
return x.b && y.b;
}
function getReferenced(x: myStruct): myStruct {
return x.d;
}
function start(): int {
let x: myStruct = create myStruct(1, false, 42.0, naught);
let y: myStruct = create myStruct(12, true, 13.37, x);
let z: int = add(x, y);
let a: bool = isOk(x, y);
let y2: myStruct = getReferenced(x);
let isSame: bool = y == y2;
destroy y;
destroy x;
return 0;
}
start();
Statements
Several statements are supported:
- if
- variable declaration
- variable assignment
- return
- while
- do while
- for
Examples
function example(x: int, y: int, z: int): int {
let a: int;
let b: int = 0;
if (x == y) {
a = y;
} else if (x == z) {
a = z;
} else {
return b;
}
return a;
}
function whileExample(end: int): int {
let x: int = 0;
while (x < end) {
x = x + 1;
}
return x;
}
function doWhileExample(end: int): int {
let x: int = 0;
do {
x = x + 1;
} while(x < end);
return x;
}
function forExample(end: int): int {
let x: int = 0;
for (let i: int = 0; i < end; i = i + 1) {
x = x + 1;
}
return x;
}
Tail Call Optimized
Recursive tail calls are optimized at compile time.
Statically typed
KLang statically verifies the integrity of your code. These checks include:
- Type checking
- Ensuring that variables and functions in use are declared
- Ensuring that the arguments of a function call match the function definition
- Ensuring that a function returns something
- Ensuring that a function only returns data of the correct type
Primitive Data Types
- Integer "int"
- Boolean "bool"
- Floats "float"
Examples
You can declare types for parameters, return values and variables
function foo(start: int): boolean {
let threshold: int = 10;
return threshold < start;
}
Type annotations are required as per our parsing rules, so this will result in an error while parsing
function bar() {
return 0;
}
This will throw an error since a boolean is returned, but int is declared as the return type
function baz(): int {
return false;
}
This will throw an error since the function "bam" expects one argument but the call to this function provided none
function bam(a: int): int {
return a;
}
bam();
This will throw an error since the first parameter of function "boo" has to be of type bool
function boo(a: bool): bool {
return a;
}
boo(100);