2
0
gingerBill 9 жил өмнө
parent
commit
ba238c569a

+ 7 - 7
examples/basic.odin

@@ -1,23 +1,23 @@
-putchar :: proc(c : i32) -> i32 #foreign
+putchar :: proc(c: i32) -> i32 #foreign
 
-print_string :: proc(s : string) {
+print_string :: proc(s: string) {
 	for i := 0; i < len(s); i++ {
 		c := cast(i32)s[i];
 		putchar(c);
 	}
 }
 
-string_byte_reverse :: proc(s : string) {
+string_byte_reverse :: proc(s: string) {
 	n := len(s);
 	for i := 0; i < n/2; i++ {
 		s[i], s[n-1-i] = s[n-1-i], s[i];
 	}
 }
 
-print_int :: proc(i : int, base : int) {
-	NUM_TO_CHAR_TABLE := "0123456789ABCDEFGHIJKLMNOPQRSTUVWXYZabcdefghijklmnopqrstuvwxyz@$";
+print_int :: proc(i, base: int) {
+	NUM_TO_CHAR_TABLE :: "0123456789ABCDEFGHIJKLMNOPQRSTUVWXYZabcdefghijklmnopqrstuvwxyz@$";
 
-	buf : [21]byte;
+	buf: [21]u8;
 	len := 0;
 	negative := false;
 	if i < 0 {
@@ -26,7 +26,7 @@ print_int :: proc(i : int, base : int) {
 	}
 	if i > 0 {
 		for i > 0 {
-			c : byte = NUM_TO_CHAR_TABLE[i % base];
+			c : u8 = NUM_TO_CHAR_TABLE[i % base];
 			buf[len] = c;
 			len++;
 			i /= base;

+ 260 - 143
examples/main.ll

@@ -4,16 +4,132 @@
 
 define void @main() {
 "entry - 0":
-	call void @print_int(i64 123, i64 10)
-	%0 = getelementptr inbounds [1 x i8], [1 x i8]* @.str0, i64 0, i64 0
-	%1 = alloca %-string, align 8 
-	store %-string zeroinitializer, %-string* %1
-	%2 = getelementptr inbounds %-string, %-string* %1, i64 0, i32 0
-	%3 = getelementptr inbounds %-string, %-string* %1, i64 0, i32 1
-	store i8* %0, i8** %2
-	store i64 1, i64* %3
-	%4 = load %-string, %-string* %1
-	call void @print_string(%-string %4)
+	%0 = alloca i64, align 8 ; a
+	store i64 zeroinitializer, i64* %0
+	%1 = getelementptr inbounds [6 x i8], [6 x i8]* @.str0, i64 0, i64 0
+	%2 = getelementptr i8, i8* %1, i64 1
+	%3 = load i8, i8* %2
+	%4 = zext i8 %3 to i64
+	store i64 %4, i64* %0
+	%5 = load i64, i64* %0
+	call void @print_int(i64 %5, i64 10)
+	%6 = getelementptr inbounds [1 x i8], [1 x i8]* @.str1, i64 0, i64 0
+	%7 = alloca %-string, align 8 
+	store %-string zeroinitializer, %-string* %7
+	%8 = getelementptr inbounds %-string, %-string* %7, i64 0, i32 0
+	%9 = getelementptr inbounds %-string, %-string* %7, i64 0, i32 1
+	store i8* %6, i8** %8
+	store i64 1, i64* %9
+	%10 = load %-string, %-string* %7
+	call void @print_string(%-string %10)
+	%11 = getelementptr inbounds [23 x i8], [23 x i8]* @.str2, i64 0, i64 0
+	%12 = alloca %-string, align 8 
+	store %-string zeroinitializer, %-string* %12
+	%13 = getelementptr inbounds %-string, %-string* %12, i64 0, i32 0
+	%14 = getelementptr inbounds %-string, %-string* %12, i64 0, i32 1
+	store i8* %11, i8** %13
+	store i64 23, i64* %14
+	%15 = load %-string, %-string* %12
+	call void @print_string(%-string %15)
+	%16 = getelementptr inbounds [21 x i8], [21 x i8]* @.str3, i64 0, i64 0
+	%17 = alloca %-string, align 8 
+	store %-string zeroinitializer, %-string* %17
+	%18 = getelementptr inbounds %-string, %-string* %17, i64 0, i32 0
+	%19 = getelementptr inbounds %-string, %-string* %17, i64 0, i32 1
+	store i8* %16, i8** %18
+	store i64 21, i64* %19
+	%20 = load %-string, %-string* %17
+	call void @print_string(%-string %20)
+	%21 = getelementptr inbounds [22 x i8], [22 x i8]* @.str4, i64 0, i64 0
+	%22 = alloca %-string, align 8 
+	store %-string zeroinitializer, %-string* %22
+	%23 = getelementptr inbounds %-string, %-string* %22, i64 0, i32 0
+	%24 = getelementptr inbounds %-string, %-string* %22, i64 0, i32 1
+	store i8* %21, i8** %23
+	store i64 22, i64* %24
+	%25 = load %-string, %-string* %22
+	call void @print_string(%-string %25)
+	%26 = getelementptr inbounds [23 x i8], [23 x i8]* @.str5, i64 0, i64 0
+	%27 = alloca %-string, align 8 
+	store %-string zeroinitializer, %-string* %27
+	%28 = getelementptr inbounds %-string, %-string* %27, i64 0, i32 0
+	%29 = getelementptr inbounds %-string, %-string* %27, i64 0, i32 1
+	store i8* %26, i8** %28
+	store i64 23, i64* %29
+	%30 = load %-string, %-string* %27
+	call void @print_string(%-string %30)
+	%31 = getelementptr inbounds [20 x i8], [20 x i8]* @.str6, i64 0, i64 0
+	%32 = alloca %-string, align 8 
+	store %-string zeroinitializer, %-string* %32
+	%33 = getelementptr inbounds %-string, %-string* %32, i64 0, i32 0
+	%34 = getelementptr inbounds %-string, %-string* %32, i64 0, i32 1
+	store i8* %31, i8** %33
+	store i64 20, i64* %34
+	%35 = load %-string, %-string* %32
+	call void @print_string(%-string %35)
+	%36 = getelementptr inbounds [37 x i8], [37 x i8]* @.str7, i64 0, i64 0
+	%37 = alloca %-string, align 8 
+	store %-string zeroinitializer, %-string* %37
+	%38 = getelementptr inbounds %-string, %-string* %37, i64 0, i32 0
+	%39 = getelementptr inbounds %-string, %-string* %37, i64 0, i32 1
+	store i8* %36, i8** %38
+	store i64 37, i64* %39
+	%40 = load %-string, %-string* %37
+	call void @print_string(%-string %40)
+	%41 = getelementptr inbounds [21 x i8], [21 x i8]* @.str8, i64 0, i64 0
+	%42 = alloca %-string, align 8 
+	store %-string zeroinitializer, %-string* %42
+	%43 = getelementptr inbounds %-string, %-string* %42, i64 0, i32 0
+	%44 = getelementptr inbounds %-string, %-string* %42, i64 0, i32 1
+	store i8* %41, i8** %43
+	store i64 21, i64* %44
+	%45 = load %-string, %-string* %42
+	call void @print_string(%-string %45)
+	%46 = getelementptr inbounds [33 x i8], [33 x i8]* @.str9, i64 0, i64 0
+	%47 = alloca %-string, align 8 
+	store %-string zeroinitializer, %-string* %47
+	%48 = getelementptr inbounds %-string, %-string* %47, i64 0, i32 0
+	%49 = getelementptr inbounds %-string, %-string* %47, i64 0, i32 1
+	store i8* %46, i8** %48
+	store i64 33, i64* %49
+	%50 = load %-string, %-string* %47
+	call void @print_string(%-string %50)
+	%51 = getelementptr inbounds [29 x i8], [29 x i8]* @.stra, i64 0, i64 0
+	%52 = alloca %-string, align 8 
+	store %-string zeroinitializer, %-string* %52
+	%53 = getelementptr inbounds %-string, %-string* %52, i64 0, i32 0
+	%54 = getelementptr inbounds %-string, %-string* %52, i64 0, i32 1
+	store i8* %51, i8** %53
+	store i64 29, i64* %54
+	%55 = load %-string, %-string* %52
+	call void @print_string(%-string %55)
+	%56 = getelementptr inbounds [24 x i8], [24 x i8]* @.strb, i64 0, i64 0
+	%57 = alloca %-string, align 8 
+	store %-string zeroinitializer, %-string* %57
+	%58 = getelementptr inbounds %-string, %-string* %57, i64 0, i32 0
+	%59 = getelementptr inbounds %-string, %-string* %57, i64 0, i32 1
+	store i8* %56, i8** %58
+	store i64 24, i64* %59
+	%60 = load %-string, %-string* %57
+	call void @print_string(%-string %60)
+	%61 = getelementptr inbounds [42 x i8], [42 x i8]* @.strc, i64 0, i64 0
+	%62 = alloca %-string, align 8 
+	store %-string zeroinitializer, %-string* %62
+	%63 = getelementptr inbounds %-string, %-string* %62, i64 0, i32 0
+	%64 = getelementptr inbounds %-string, %-string* %62, i64 0, i32 1
+	store i8* %61, i8** %63
+	store i64 42, i64* %64
+	%65 = load %-string, %-string* %62
+	call void @print_string(%-string %65)
+	%66 = getelementptr inbounds [21 x i8], [21 x i8]* @.strd, i64 0, i64 0
+	%67 = alloca %-string, align 8 
+	store %-string zeroinitializer, %-string* %67
+	%68 = getelementptr inbounds %-string, %-string* %67, i64 0, i32 0
+	%69 = getelementptr inbounds %-string, %-string* %67, i64 0, i32 1
+	store i8* %66, i8** %68
+	store i64 21, i64* %69
+	%70 = load %-string, %-string* %67
+	call void @print_string(%-string %70)
 	ret void
 }
 
@@ -32,10 +148,10 @@ define void @print_string(%-string %s) {
 "for.body - 1":
 	%2 = alloca i32, align 4 ; c
 	store i32 zeroinitializer, i32* %2
-	%3 = load i64, i64* %1
-	%4 = getelementptr inbounds %-string, %-string* %0, i64 0, i32 0
-	%5 = load i8*, i8** %4
-	%6 = getelementptr i8, i8* %5, i64 %3
+	%3 = getelementptr inbounds %-string, %-string* %0, i64 0, i32 0
+	%4 = load i8*, i8** %3
+	%5 = load i64, i64* %1
+	%6 = getelementptr i8, i8* %4, i64 %5
 	%7 = load i8, i8* %6
 	%8 = zext i8 %7 to i32
 	store i32 %8, i32* %2
@@ -76,29 +192,29 @@ define void @string_byte_reverse(%-string %s) {
 	br label %"for.loop - 2"
 
 "for.body - 1":
-	%5 = load i64, i64* %4
-	%6 = getelementptr inbounds %-string, %-string* %0, i64 0, i32 0
-	%7 = load i8*, i8** %6
-	%8 = getelementptr i8, i8* %7, i64 %5
-	%9 = load i64, i64* %4
-	%10 = load i64, i64* %1
-	%11 = sub i64 %10, 1
-	%12 = sub i64 %11, %9
-	%13 = getelementptr inbounds %-string, %-string* %0, i64 0, i32 0
-	%14 = load i8*, i8** %13
-	%15 = getelementptr i8, i8* %14, i64 %12
-	%16 = load i64, i64* %4
-	%17 = load i64, i64* %1
-	%18 = sub i64 %17, 1
-	%19 = sub i64 %18, %16
-	%20 = getelementptr inbounds %-string, %-string* %0, i64 0, i32 0
-	%21 = load i8*, i8** %20
-	%22 = getelementptr i8, i8* %21, i64 %19
+	%5 = getelementptr inbounds %-string, %-string* %0, i64 0, i32 0
+	%6 = load i8*, i8** %5
+	%7 = load i64, i64* %4
+	%8 = getelementptr i8, i8* %6, i64 %7
+	%9 = getelementptr inbounds %-string, %-string* %0, i64 0, i32 0
+	%10 = load i8*, i8** %9
+	%11 = load i64, i64* %4
+	%12 = load i64, i64* %1
+	%13 = sub i64 %12, 1
+	%14 = sub i64 %13, %11
+	%15 = getelementptr i8, i8* %10, i64 %14
+	%16 = getelementptr inbounds %-string, %-string* %0, i64 0, i32 0
+	%17 = load i8*, i8** %16
+	%18 = load i64, i64* %4
+	%19 = load i64, i64* %1
+	%20 = sub i64 %19, 1
+	%21 = sub i64 %20, %18
+	%22 = getelementptr i8, i8* %17, i64 %21
 	%23 = load i8, i8* %22
-	%24 = load i64, i64* %4
-	%25 = getelementptr inbounds %-string, %-string* %0, i64 0, i32 0
-	%26 = load i8*, i8** %25
-	%27 = getelementptr i8, i8* %26, i64 %24
+	%24 = getelementptr inbounds %-string, %-string* %0, i64 0, i32 0
+	%25 = load i8*, i8** %24
+	%26 = load i64, i64* %4
+	%27 = getelementptr i8, i8* %25, i64 %26
 	%28 = load i8, i8* %27
 	store i8 %23, i8* %8
 	store i8 %28, i8* %15
@@ -129,139 +245,140 @@ define void @print_int(i64 %i, i64 %base) {
 	%1 = alloca i64, align 8 ; base
 	store i64 zeroinitializer, i64* %1
 	store i64 %base, i64* %1
-	%2 = alloca %-string, align 8 ; NUM_TO_CHAR_TABLE
-	store %-string zeroinitializer, %-string* %2
-	%3 = getelementptr inbounds [64 x i8], [64 x i8]* @.str1, i64 0, i64 0
-	%4 = alloca %-string, align 8 
-	store %-string zeroinitializer, %-string* %4
-	%5 = getelementptr inbounds %-string, %-string* %4, i64 0, i32 0
-	%6 = getelementptr inbounds %-string, %-string* %4, i64 0, i32 1
-	store i8* %3, i8** %5
-	store i64 64, i64* %6
-	%7 = load %-string, %-string* %4
-	store %-string %7, %-string* %2
-	%8 = alloca [21 x i8], align 1 ; buf
-	store [21 x i8] zeroinitializer, [21 x i8]* %8
-	%9 = alloca i64, align 8 ; len
-	store i64 zeroinitializer, i64* %9
-	store i64 0, i64* %9
-	%10 = alloca i1, align 1 ; negative
-	store i1 zeroinitializer, i1* %10
-	store i1 false, i1* %10
-	%11 = load i64, i64* %0
-	%12 = icmp slt i64 %11, 0
-	br i1 %12, label %"if.then - 1", label %"if.done - 2"
+	%2 = alloca [21 x i8], align 1 ; buf
+	store [21 x i8] zeroinitializer, [21 x i8]* %2
+	%3 = alloca i64, align 8 ; len
+	store i64 zeroinitializer, i64* %3
+	store i64 0, i64* %3
+	%4 = alloca i1, align 1 ; negative
+	store i1 zeroinitializer, i1* %4
+	store i1 false, i1* %4
+	%5 = load i64, i64* %0
+	%6 = icmp slt i64 %5, 0
+	br i1 %6, label %"if.then - 1", label %"if.done - 2"
 
 "if.then - 1":
-	store i1 true, i1* %10
-	%13 = load i64, i64* %0
-	%14 = sub i64 0, %13
-	store i64 %14, i64* %0
+	store i1 true, i1* %4
+	%7 = load i64, i64* %0
+	%8 = sub i64 0, %7
+	store i64 %8, i64* %0
 	br label %"if.done - 2"
 
 "if.done - 2":
-	%15 = load i64, i64* %0
-	%16 = icmp sgt i64 %15, 0
-	br i1 %16, label %"if.then - 3", label %"if.else - 4"
+	%9 = load i64, i64* %0
+	%10 = icmp sgt i64 %9, 0
+	br i1 %10, label %"if.then - 3", label %"if.else - 4"
 
 "if.then - 3":
 	br label %"for.loop - 6"
 
 "if.else - 4":
-	%17 = load i64, i64* %9
-	%18 = getelementptr inbounds [21 x i8], [21 x i8]* %8, i64 0, i64 0
-	%19 = getelementptr i8, i8* %18, i64 %17
-	store i8 0, i8* %19
-	%20 = load i64, i64* %9
-	%21 = add i64 %20, 1
-	store i64 %21, i64* %9
+	%11 = getelementptr inbounds [21 x i8], [21 x i8]* %2, i64 0, i64 0
+	%12 = load i64, i64* %3
+	%13 = getelementptr i8, i8* %11, i64 %12
+	store i8 0, i8* %13
+	%14 = load i64, i64* %3
+	%15 = add i64 %14, 1
+	store i64 %15, i64* %3
 	br label %"if.done - 8"
 
 "for.body - 5":
-	%22 = alloca i8, align 1 ; c
-	store i8 zeroinitializer, i8* %22
-	%23 = load i64, i64* %1
-	%24 = load i64, i64* %0
-	%25 = srem i64 %24, %23
-	%26 = getelementptr inbounds %-string, %-string* %2, i64 0, i32 0
-	%27 = load i8*, i8** %26
-	%28 = getelementptr i8, i8* %27, i64 %25
-	%29 = load i8, i8* %28
-	store i8 %29, i8* %22
-	%30 = load i64, i64* %9
-	%31 = getelementptr inbounds [21 x i8], [21 x i8]* %8, i64 0, i64 0
-	%32 = getelementptr i8, i8* %31, i64 %30
-	%33 = load i8, i8* %22
-	store i8 %33, i8* %32
-	%34 = load i64, i64* %9
-	%35 = add i64 %34, 1
-	store i64 %35, i64* %9
-	%36 = load i64, i64* %1
-	%37 = load i64, i64* %0
-	%38 = sdiv i64 %37, %36
-	store i64 %38, i64* %0
+	%16 = alloca i8, align 1 ; c
+	store i8 zeroinitializer, i8* %16
+	%17 = getelementptr inbounds [64 x i8], [64 x i8]* @.stre, i64 0, i64 0
+	%18 = load i64, i64* %1
+	%19 = load i64, i64* %0
+	%20 = srem i64 %19, %18
+	%21 = getelementptr i8, i8* %17, i64 %20
+	%22 = load i8, i8* %21
+	store i8 %22, i8* %16
+	%23 = getelementptr inbounds [21 x i8], [21 x i8]* %2, i64 0, i64 0
+	%24 = load i64, i64* %3
+	%25 = getelementptr i8, i8* %23, i64 %24
+	%26 = load i8, i8* %16
+	store i8 %26, i8* %25
+	%27 = load i64, i64* %3
+	%28 = add i64 %27, 1
+	store i64 %28, i64* %3
+	%29 = load i64, i64* %1
+	%30 = load i64, i64* %0
+	%31 = sdiv i64 %30, %29
+	store i64 %31, i64* %0
 	br label %"for.loop - 6"
 
 "for.loop - 6":
-	%39 = load i64, i64* %0
-	%40 = icmp sgt i64 %39, 0
-	br i1 %40, label %"for.body - 5", label %"for.done - 7"
+	%32 = load i64, i64* %0
+	%33 = icmp sgt i64 %32, 0
+	br i1 %33, label %"for.body - 5", label %"for.done - 7"
 
 "for.done - 7":
 	br label %"if.done - 8"
 
 "if.done - 8":
-	%41 = load i1, i1* %10
-	br i1 %41, label %"if.then - 9", label %"if.done - 10"
+	%34 = load i1, i1* %4
+	br i1 %34, label %"if.then - 9", label %"if.done - 10"
 
 "if.then - 9":
-	%42 = load i64, i64* %9
-	%43 = getelementptr inbounds [21 x i8], [21 x i8]* %8, i64 0, i64 0
-	%44 = getelementptr i8, i8* %43, i64 %42
-	store i8 0, i8* %44
-	%45 = load i64, i64* %9
-	%46 = add i64 %45, 1
-	store i64 %46, i64* %9
+	%35 = getelementptr inbounds [21 x i8], [21 x i8]* %2, i64 0, i64 0
+	%36 = load i64, i64* %3
+	%37 = getelementptr i8, i8* %35, i64 %36
+	store i8 0, i8* %37
+	%38 = load i64, i64* %3
+	%39 = add i64 %38, 1
+	store i64 %39, i64* %3
 	br label %"if.done - 10"
 
 "if.done - 10":
-	%47 = alloca %-string, align 8 ; str
-	store %-string zeroinitializer, %-string* %47
-	%48 = load i64, i64* %9
-	%49 = sub i64 %48, 0
-	%50 = sub i64 21, 0
-	%51 = getelementptr inbounds [21 x i8], [21 x i8]* %8, i64 0, i64 0
-	%52 = getelementptr i8, i8* %51, i64 0
-	%53 = alloca {i8*, i64, i64}, align 8 
-	store {i8*, i64, i64} zeroinitializer, {i8*, i64, i64}* %53
-	%54 = getelementptr inbounds {i8*, i64, i64}, {i8*, i64, i64}* %53, i64 0, i32 0
-	store i8* %52, i8** %54
-	%55 = getelementptr inbounds {i8*, i64, i64}, {i8*, i64, i64}* %53, i64 0, i32 1
-	store i64 %49, i64* %55
-	%56 = getelementptr inbounds {i8*, i64, i64}, {i8*, i64, i64}* %53, i64 0, i32 2
-	store i64 %50, i64* %56
-	%57 = load {i8*, i64, i64}, {i8*, i64, i64}* %53
-	%58 = alloca {i8*, i64, i64}, align 8 
-	store {i8*, i64, i64} zeroinitializer, {i8*, i64, i64}* %58
-	store {i8*, i64, i64} %57, {i8*, i64, i64}* %58
-	%59 = getelementptr inbounds {i8*, i64, i64}, {i8*, i64, i64}* %58, i64 0, i32 0
-	%60 = load i8*, i8** %59
-	%61 = getelementptr inbounds {i8*, i64, i64}, {i8*, i64, i64}* %58, i64 0, i32 1
-	%62 = load i64, i64* %61
-	%63 = alloca %-string, align 8 
-	store %-string zeroinitializer, %-string* %63
-	%64 = getelementptr inbounds %-string, %-string* %63, i64 0, i32 0
-	%65 = getelementptr inbounds %-string, %-string* %63, i64 0, i32 1
-	store i8* %60, i8** %64
-	store i64 %62, i64* %65
-	%66 = load %-string, %-string* %63
-	store %-string %66, %-string* %47
-	%67 = load %-string, %-string* %47
-	call void @string_byte_reverse(%-string %67)
-	%68 = load %-string, %-string* %47
-	call void @print_string(%-string %68)
+	%40 = alloca %-string, align 8 ; str
+	store %-string zeroinitializer, %-string* %40
+	%41 = load i64, i64* %3
+	%42 = sub i64 %41, 0
+	%43 = sub i64 21, 0
+	%44 = getelementptr inbounds [21 x i8], [21 x i8]* %2, i64 0, i64 0
+	%45 = getelementptr i8, i8* %44, i64 0
+	%46 = alloca {i8*, i64, i64}, align 8 
+	store {i8*, i64, i64} zeroinitializer, {i8*, i64, i64}* %46
+	%47 = getelementptr inbounds {i8*, i64, i64}, {i8*, i64, i64}* %46, i64 0, i32 0
+	store i8* %45, i8** %47
+	%48 = getelementptr inbounds {i8*, i64, i64}, {i8*, i64, i64}* %46, i64 0, i32 1
+	store i64 %42, i64* %48
+	%49 = getelementptr inbounds {i8*, i64, i64}, {i8*, i64, i64}* %46, i64 0, i32 2
+	store i64 %43, i64* %49
+	%50 = load {i8*, i64, i64}, {i8*, i64, i64}* %46
+	%51 = alloca {i8*, i64, i64}, align 8 
+	store {i8*, i64, i64} zeroinitializer, {i8*, i64, i64}* %51
+	store {i8*, i64, i64} %50, {i8*, i64, i64}* %51
+	%52 = getelementptr inbounds {i8*, i64, i64}, {i8*, i64, i64}* %51, i64 0, i32 0
+	%53 = load i8*, i8** %52
+	%54 = getelementptr inbounds {i8*, i64, i64}, {i8*, i64, i64}* %51, i64 0, i32 1
+	%55 = load i64, i64* %54
+	%56 = alloca %-string, align 8 
+	store %-string zeroinitializer, %-string* %56
+	%57 = getelementptr inbounds %-string, %-string* %56, i64 0, i32 0
+	%58 = getelementptr inbounds %-string, %-string* %56, i64 0, i32 1
+	store i8* %53, i8** %57
+	store i64 %55, i64* %58
+	%59 = load %-string, %-string* %56
+	store %-string %59, %-string* %40
+	%60 = load %-string, %-string* %40
+	call void @string_byte_reverse(%-string %60)
+	%61 = load %-string, %-string* %40
+	call void @print_string(%-string %61)
 	ret void
 }
 
[email protected] = global [1 x i8] c"\0A"
[email protected] = global [64 x i8] c"0123456789ABCDEFGHIJKLMNOPQRSTUVWXYZabcdefghijklmnopqrstuvwxyz\40$"
[email protected] = global [6 x i8] c"Hello\0A"
[email protected] = global [1 x i8] c"\0A"
[email protected] = global [23 x i8] c"Chinese\20-\20\E4\BD\A0\E5\A5\BD\E4\B8\96\E7\95\8C\0A"
[email protected] = global [21 x i8] c"Dutch\20-\20Hello\20wereld\0A"
[email protected] = global [22 x i8] c"English\20-\20Hello\20world\0A"
[email protected] = global [23 x i8] c"French\20-\20Bonjour\20monde\0A"
[email protected] = global [20 x i8] c"German\20-\20Hallo\20Welt\0A"
[email protected] = global [37 x i8] c"Greek\20-\20\CE\B3\CE\B5\CE\B9\CE\AC\20\CF\83\CE\BF\CF\85\20\CE\BA\CF\8C\CF\83\CE\BC\CE\BF\CF\82\0A"
[email protected] = global [21 x i8] c"Italian\20-\20Ciao\20mondo\0A"
[email protected] = global [33 x i8] c"Japanese\20-\20\E3\81\93\E3\82\93\E3\81\AB\E3\81\A1\E3\81\AF\E4\B8\96\E7\95\8C\0A"
[email protected] = global [29 x i8] c"Korean\20-\20\EC\97\AC\EB\B3\B4\EC\84\B8\EC\9A\94\20\EC\84\B8\EA\B3\84\0A"
[email protected] = global [24 x i8] c"Portuguese\20-\20Ol\C3\A1\20mundo\0A"
[email protected] = global [42 x i8] c"Russian\20-\20\D0\97\D0\B4\D1\80\D0\B0\D0\B2\D1\81\D1\82\D0\B2\D1\83\D0\BB\D1\82\D0\B5\20\D0\BC\D0\B8\D1\80\0A"
[email protected] = global [21 x i8] c"Spanish\20-\20Hola\20mundo\0A"
[email protected] = global [64 x i8] c"0123456789ABCDEFGHIJKLMNOPQRSTUVWXYZabcdefghijklmnopqrstuvwxyz\40$"

+ 15 - 1
examples/main.odin

@@ -1,6 +1,20 @@
 import "basic"
 
 main :: proc() {
-	print_int(123, 10);
+	a := cast(int)"Hello\n"[1];
+	print_int(a, 10);
 	print_string("\n");
+
+	print_string("Chinese - 你好世界\n");
+	print_string("Dutch - Hello wereld\n");
+	print_string("English - Hello world\n");
+	print_string("French - Bonjour monde\n");
+	print_string("German - Hallo Welt\n");
+	print_string("Greek - γειά σου κόσμος\n");
+	print_string("Italian - Ciao mondo\n");
+	print_string("Japanese - こんにちは世界\n");
+	print_string("Korean - 여보세요 세계\n");
+	print_string("Portuguese - Olá mundo\n");
+	print_string("Russian - Здравствулте мир\n");
+	print_string("Spanish - Hola mundo\n");
 }

+ 9 - 5
src/checker/checker.cpp

@@ -456,8 +456,9 @@ void add_type_and_value(CheckerInfo *i, AstNode *expression, AddressingMode mode
 	}
 
 	TypeAndValue tv = {};
-	tv.type = type;
+	tv.type  = type;
 	tv.value = value;
+	tv.mode  = mode;
 	map_set(&i->types, hash_pointer(expression), tv);
 }
 
@@ -522,7 +523,8 @@ void pop_procedure(Checker *c) {
 }
 
 void add_curr_ast_file(Checker *c, AstFile *file) {
-	gb_zero_item(&c->error_collector);
+	TokenPos zero_pos = {};
+	c->error_collector.prev = zero_pos;
 	c->curr_ast_file = file;
 }
 
@@ -672,10 +674,12 @@ void check_parsed_files(Checker *c) {
 		u64 key = entry->key;
 		AstNode *expr = cast(AstNode *)cast(uintptr)key;
 		ExpressionInfo *info = &entry->value;
-		if (is_type_typed(info->type)) {
-			GB_PANIC("%s (type %s) is typed!", expr_to_string(expr), info->type);
+		if (info != NULL && expr != NULL) {
+			if (is_type_typed(info->type)) {
+				GB_PANIC("%s (type %s) is typed!", expr_to_string(expr), info->type);
+			}
+			add_type_and_value(&c->info, expr, info->mode, info->type, info->value);
 		}
-		add_type_and_value(&c->info, expr, info->mode, info->type, info->value);
 	}
 }
 

+ 21 - 13
src/checker/expr.cpp

@@ -11,6 +11,7 @@ void           convert_to_typed        (Checker *c, Operand *operand, Type *targ
 gbString       expr_to_string          (AstNode *expression);
 void           check_entity_decl       (Checker *c, Entity *e, DeclInfo *decl, Type *named_type);
 void           check_proc_body         (Checker *c, Token token, DeclInfo *decl, Type *type, AstNode *body);
+void           update_expr_type        (Checker *c, AstNode *e, Type *type, b32 final);
 
 
 void check_struct_type(Checker *c, Type *struct_type, AstNode *node) {
@@ -664,7 +665,10 @@ void check_comparison(Checker *c, Operand *x, Operand *y, Token op) {
 	    y->mode == Addressing_Constant) {
 		x->value = make_exact_value_bool(compare_exact_values(op, x->value, y->value));
 	} else {
-		// TODO(bill): What should I do?
+		x->mode = Addressing_Value;
+
+		update_expr_type(c, x->expr, default_type(x->type), true);
+		update_expr_type(c, y->expr, default_type(y->type), true);
 	}
 
 	x->type = t_untyped_bool;
@@ -770,33 +774,35 @@ void check_binary_expr(Checker *c, Operand *x, AstNode *node) {
 }
 
 
-void update_expr_type(Checker *c, AstNode *e, Type *type) {
-	ExpressionInfo *found = map_get(&c->info.untyped, hash_pointer(e));
-	if (!found)
+void update_expr_type(Checker *c, AstNode *e, Type *type, b32 final) {
+	u64 key = hash_pointer(e);
+	ExpressionInfo *found = map_get(&c->info.untyped, key);
+	if (found == NULL)
 		return;
 
 	switch (e->kind) {
 	case_ast_node(ue, UnaryExpr, e);
 		if (found->value.kind != ExactValue_Invalid)
 			break;
-		update_expr_type(c, ue->expr, type);
-		break;
+		update_expr_type(c, ue->expr, type, final);
 	case_end;
 
 	case_ast_node(be, BinaryExpr, e);
 		if (found->value.kind != ExactValue_Invalid)
 			break;
 		if (!token_is_comparison(be->op)) {
-			update_expr_type(c, be->left,  type);
-			update_expr_type(c, be->right, type);
+			update_expr_type(c, be->left,  type, final);
+			update_expr_type(c, be->right, type, final);
 		}
 	case_end;
 	}
 
-	if (is_type_untyped(type)) {
+	if (!final && is_type_untyped(type)) {
 		found->type = get_base_type(type);
+		map_set(&c->info.untyped, key, *found);
 	} else {
-		found->type = type;
+		map_remove(&c->info.untyped, key);
+		add_type_and_value(&c->info, e, found->mode, type, found->value);
 	}
 }
 
@@ -838,7 +844,7 @@ void convert_to_typed(Checker *c, Operand *operand, Type *target_type) {
 		if (is_type_numeric(x) && is_type_numeric(y)) {
 			if (x < y) {
 				operand->type = target_type;
-				update_expr_type(c, operand->expr, target_type);
+				update_expr_type(c, operand->expr, target_type, false);
 			}
 		} else if (x != y) {
 			convert_untyped_error(c, operand, target_type);
@@ -1479,10 +1485,10 @@ b32 check_castable_to(Checker *c, Operand *operand, Type *y) {
 	}
 
 	// []byte/[]u8 <-> string
-	if (is_type_byte_slice(xb) && is_type_string(yb)) {
+	if (is_type_u8_slice(xb) && is_type_string(yb)) {
 		return true;
 	}
-	if (is_type_string(xb) && is_type_byte_slice(yb)) {
+	if (is_type_string(xb) && is_type_u8_slice(yb)) {
 		return true;
 	}
 
@@ -1722,6 +1728,8 @@ ExpressionKind check__expr_base(Checker *c, Operand *o, AstNode *node, Type *typ
 				if (o->mode == Addressing_Constant) {
 					max_count = o->value.value_string.len;
 				}
+				if (o->mode != Addressing_Variable)
+					o->mode = Addressing_Value;
 				o->type = t_u8;
 			}
 			break;

+ 7 - 7
src/checker/type.cpp

@@ -26,7 +26,7 @@ enum BasicKind {
 
 	Basic_Count,
 
-	Basic_byte = Basic_u8,
+	// Basic_byte = Basic_u8,
 	Basic_rune = Basic_i32,
 };
 
@@ -242,7 +242,7 @@ gb_global Type basic_types[] = {
 };
 
 gb_global Type basic_type_aliases[] = {
-	{Type_Basic, {Basic_byte, BasicFlag_Integer | BasicFlag_Unsigned, STR_LIT("byte")}},
+	// {Type_Basic, {Basic_byte, BasicFlag_Integer | BasicFlag_Unsigned, STR_LIT("byte")}},
 	{Type_Basic, {Basic_rune, BasicFlag_Integer,                      STR_LIT("rune")}},
 };
 
@@ -268,7 +268,7 @@ gb_global Type *t_untyped_float   = &basic_types[Basic_UntypedFloat];
 gb_global Type *t_untyped_pointer = &basic_types[Basic_UntypedPointer];
 gb_global Type *t_untyped_string  = &basic_types[Basic_UntypedString];
 gb_global Type *t_untyped_rune    = &basic_types[Basic_UntypedRune];
-gb_global Type *t_byte            = &basic_type_aliases[Basic_byte];
+// gb_global Type *t_byte            = &basic_type_aliases[Basic_byte];
 gb_global Type *t_rune            = &basic_type_aliases[Basic_rune];
 
 
@@ -343,9 +343,9 @@ b32 is_type_rawptr(Type *t) {
 		return t->basic.kind == Basic_rawptr;
 	return false;
 }
-b32 is_type_byte(Type *t) {
+b32 is_type_u8(Type *t) {
 	if (t->kind == Type_Basic)
-		return t->basic.kind == Basic_byte;
+		return t->basic.kind == Basic_u8;
 	return false;
 }
 b32 is_type_slice(Type *t) {
@@ -353,9 +353,9 @@ b32 is_type_slice(Type *t) {
 }
 
 
-b32 is_type_byte_slice(Type *t) {
+b32 is_type_u8_slice(Type *t) {
 	if (t->kind == Type_Slice)
-		return is_type_byte(t->slice.elem);
+		return is_type_u8(t->slice.elem);
 	return false;
 }
 

+ 3 - 3
src/codegen/codegen.cpp

@@ -7,14 +7,14 @@ struct ssaGen {
 };
 
 b32 ssa_gen_init(ssaGen *s, Checker *c) {
-	if (c->error_collector.count > 0)
+	if (c->error_collector.count != 0)
 		return false;
 
 	gb_for_array(i, c->parser->files) {
 		AstFile *f = &c->parser->files[i];
-		if (f->error_collector.count > 0)
+		if (f->error_collector.count != 0)
 			return false;
-		if (f->tokenizer.error_count > 0)
+		if (f->tokenizer.error_count != 0)
 			return false;
 	}
 

+ 9 - 5
src/codegen/print_llvm.cpp

@@ -440,12 +440,13 @@ void ssa_print_instr(gbFile *f, ssaModule *m, ssaValue *value) {
 
 	case ssaInstr_Call: {
 		auto *call = &instr->call;
-		if (call->type) {
+		Type *result_type = call->type->proc.results;
+		if (result_type) {
 			ssa_fprintf(f, "%%%d = ", value->id);
 		}
 		ssa_fprintf(f, "call ");
-		if (call->type) {
-			ssa_print_type(f, m->sizes, call->type);
+		if (result_type) {
+			ssa_print_type(f, m->sizes, result_type);
 		} else {
 			ssa_fprintf(f, "void");
 		}
@@ -454,14 +455,17 @@ void ssa_print_instr(gbFile *f, ssaModule *m, ssaValue *value) {
 
 
 		ssa_fprintf(f, "(");
+		auto *params = &call->type->proc.params->tuple;
 		for (isize i = 0; i < call->arg_count; i++) {
-			ssaValue *arg = call->args[i];
-			Type *t = ssa_value_type(arg);
+			Entity *e = params->variables[i];
+			GB_ASSERT(e != NULL);
+			Type *t = e->type;
 			if (i > 0) {
 				ssa_fprintf(f, ", ");
 			}
 			ssa_print_type(f, m->sizes, t);
 			ssa_fprintf(f, " ");
+			ssaValue *arg = call->args[i];
 			ssa_print_value(f, m, arg, t);
 		}
 		ssa_fprintf(f, ")\n");

+ 49 - 38
src/codegen/ssa.cpp

@@ -7,6 +7,7 @@ struct ssaValue;
 struct ssaModule {
 	CheckerInfo *info;
 	BaseTypeSizes sizes;
+	gbArena     arena;
 	gbAllocator allocator;
 
 	String layout;
@@ -244,7 +245,10 @@ ssaLvalue ssa_make_lvalue_address(ssaValue *value, AstNode *expr) {
 
 
 void ssa_module_init(ssaModule *m, Checker *c) {
-	m->allocator = gb_heap_allocator();
+	isize token_count = c->parser->total_token_count;
+	isize arena_size = 3 * token_count * gb_size_of(ssaValue);
+	gb_arena_init_from_allocator(&m->arena, gb_heap_allocator(), arena_size);
+	m->allocator = gb_arena_allocator(&m->arena);
 	m->info = &c->info;
 	m->sizes = c->sizes;
 
@@ -255,6 +259,7 @@ void ssa_module_init(ssaModule *m, Checker *c) {
 void ssa_module_destroy(ssaModule *m) {
 	map_destroy(&m->values);
 	map_destroy(&m->members);
+	gb_arena_free(&m->arena);
 }
 
 void ssa_module_add_value(ssaModule *m, Entity *e, ssaValue *v) {
@@ -279,6 +284,8 @@ Type *ssa_instr_type(ssaInstr *instr) {
 		return instr->binary_op.type;
 	case ssaInstr_Conv:
 		return instr->conv.to;
+	case ssaInstr_Call:
+		return instr->call.type;
 	}
 	return NULL;
 }
@@ -303,6 +310,9 @@ void ssa_instr_set_type(ssaInstr *instr, Type *type) {
 	case ssaInstr_Conv:
 		instr->conv.to = type;
 		break;
+	case ssaInstr_Call:
+		instr->call.type = type;
+		break;
 	}
 }
 
@@ -355,8 +365,6 @@ ssaValue *ssa_emit_conv(ssaProcedure *proc, ssaValue *value, Type *a_type);
 
 
 
-
-
 ssaValue *ssa_alloc_value(gbAllocator a, ssaValueKind kind) {
 	ssaValue *v = gb_alloc_item(a, ssaValue);
 	v->kind = kind;
@@ -733,7 +741,7 @@ void ssa_end_procedure_body(ssaProcedure *proc) {
 			case ssaInstr_Unreachable:
 				continue;
 			case ssaInstr_Call:
-				if (instr->call.type == NULL) {
+				if (instr->call.type->proc.results == NULL) {
 					continue;
 				}
 				break;
@@ -993,7 +1001,7 @@ ssaValue *ssa_emit_string(ssaProcedure *proc, ssaValue *elem, ssaValue *len) {
 	Type *t_u8_ptr = ssa_value_type(elem);
 	GB_ASSERT(t_u8_ptr->kind == Type_Pointer);
 
-	GB_ASSERT(is_type_byte(t_u8_ptr->pointer.elem));
+	GB_ASSERT(is_type_u8(t_u8_ptr->pointer.elem));
 
 	ssaValue *str = ssa_add_local_generated(proc, t_string);
 	ssaValue *str_elem = ssa_emit_struct_gep(proc, str, v_zero32, t_u8_ptr);
@@ -1074,14 +1082,14 @@ ssaValue *ssa_emit_conv(ssaProcedure *proc, ssaValue *value, Type *t) {
 
 
 	// []byte/[]u8 <-> string
-	if (is_type_byte_slice(src) && is_type_string(dst)) {
+	if (is_type_u8_slice(src) && is_type_string(dst)) {
 		ssaValue *slice = ssa_add_local_generated(proc, src);
 		ssa_emit_store(proc, slice, value);
 		ssaValue *elem = ssa_slice_elem(proc, slice);
 		ssaValue *len  = ssa_slice_len(proc, slice);
 		return ssa_emit_string(proc, elem, len);
 	}
-	if (is_type_string(src) && is_type_byte_slice(dst)) {
+	if (is_type_string(src) && is_type_u8_slice(dst)) {
 		ssaValue *str = ssa_add_local_generated(proc, src);
 		ssa_emit_store(proc, str, value);
 		ssaValue *elem = ssa_string_elem(proc, str);
@@ -1276,6 +1284,7 @@ ssaValue *ssa_build_single_expr(ssaProcedure *proc, AstNode *expr, TypeAndValue
 #endif
 
 		ssaValue *call = ssa_make_instr_call(proc, value, args, arg_count, tv->type);
+		ssa_value_set_type(call, proc_type_);
 		return ssa_emit(proc, call);
 	case_end;
 
@@ -1318,26 +1327,25 @@ ssaValue *ssa_build_expr(ssaProcedure *proc, AstNode *expr) {
 	expr = unparen_expr(expr);
 
 	TypeAndValue *tv = map_get(&proc->module->info->types, hash_pointer(expr));
-	if (tv) {
-		if (tv->value.kind != ExactValue_Invalid) {
-			if (tv->value.kind == ExactValue_String) {
-				ssaValue *array = ssa_add_global_string_array(proc, tv->value);
-				ssaValue *elem = ssa_array_elem(proc, array);
-				return ssa_emit_string(proc, elem, ssa_array_len(proc, array));
-			}
-			return ssa_make_value_constant(proc->module->allocator, tv->type, tv->value);
-		}
+	GB_ASSERT_NOT_NULL(tv);
 
-		ssaValue *value = NULL;
-		if (tv->mode == Addressing_Variable) {
-			value = ssa_lvalue_load(ssa_build_addr(proc, expr), proc);
-		} else {
-			value = ssa_build_single_expr(proc, expr, tv);
+	if (tv->value.kind != ExactValue_Invalid) {
+		if (tv->value.kind == ExactValue_String) {
+			ssaValue *array = ssa_add_global_string_array(proc, tv->value);
+			ssaValue *elem = ssa_array_elem(proc, array);
+			return ssa_emit_string(proc, elem, ssa_array_len(proc, array));
 		}
+		return ssa_make_value_constant(proc->module->allocator, tv->type, tv->value);
+	}
 
-		return value;
+	ssaValue *value = NULL;
+	if (tv->mode == Addressing_Variable) {
+		value = ssa_lvalue_load(ssa_build_addr(proc, expr), proc);
+	} else {
+		value = ssa_build_single_expr(proc, expr, tv);
 	}
-	return NULL;
+
+	return value;
 }
 
 
@@ -1383,33 +1391,35 @@ ssaLvalue ssa_build_addr(ssaProcedure *proc, AstNode *expr) {
 	case_ast_node(ie, IndexExpr, expr);
 		ssaValue *v = NULL;
 		Type *t = get_base_type(type_of_expr(proc->module->info, ie->expr));
+		ssaValue *elem = NULL;
 		switch (t->kind) {
 		case Type_Array: {
 			ssaValue *array = ssa_lvalue_address(ssa_build_addr(proc, ie->expr), proc);
-			ssaValue *index = ssa_emit_conv(proc, ssa_build_expr(proc, ie->index), t_int);
-			ssaValue *elem = ssa_array_elem(proc, array);
-			v = ssa_emit_ptr_offset(proc, elem, index);
+			elem = ssa_array_elem(proc, array);
 		} break;
 		case Type_Slice: {
 			ssaValue *slice = ssa_lvalue_address(ssa_build_addr(proc, ie->expr), proc);
-			ssaValue *index = ssa_emit_conv(proc, ssa_build_expr(proc, ie->index), t_int);
-			ssaValue *elem = ssa_slice_elem(proc, slice);
-			v = ssa_emit_ptr_offset(proc, elem, index);
+			elem = ssa_slice_elem(proc, slice);
 		} break;
 		case Type_Basic: { // Basic_string
-			ssaValue *str = ssa_lvalue_address(ssa_build_addr(proc, ie->expr), proc);
-			ssaValue *index = ssa_emit_conv(proc, ssa_build_expr(proc, ie->index), t_int);
-			ssaValue *elem = ssa_string_elem(proc, str);
-			v = ssa_emit_ptr_offset(proc, elem, index);
+			TypeAndValue *tv = map_get(&proc->module->info->types, hash_pointer(ie->expr));
+			if (tv->mode == Addressing_Constant) {
+				ssaValue *array = ssa_add_global_string_array(proc, tv->value);
+				elem = ssa_array_elem(proc, array);
+			} else {
+				ssaLvalue lval = ssa_build_addr(proc, ie->expr);
+				ssaValue *str = ssa_lvalue_address(lval, proc);
+				elem = ssa_string_elem(proc, str);
+			}
 		} break;
 		case Type_Pointer: {
-			ssaValue *ptr = ssa_emit_load(proc, ssa_lvalue_address(ssa_build_addr(proc, ie->expr), proc));
-			ssaValue *index = ssa_emit_conv(proc, ssa_build_expr(proc, ie->index), t_int);
-			v = ssa_emit_ptr_offset(proc, ptr, index);
+			elem = ssa_emit_load(proc, ssa_lvalue_address(ssa_build_addr(proc, ie->expr), proc));
 		} break;
 		}
 
-		// NOTE(bill): lvalue address encodes the pointer, thus the deref
+		ssaValue *index = ssa_emit_conv(proc, ssa_build_expr(proc, ie->index), t_int);
+		v = ssa_emit_ptr_offset(proc, elem, index);
+
 		ssa_value_set_type(v, type_deref(ssa_value_type(v)));
 		return ssa_make_lvalue_address(v, expr);
 	case_end;
@@ -1425,7 +1435,8 @@ ssaLvalue ssa_build_addr(ssaProcedure *proc, AstNode *expr) {
 	case_end;
 	}
 
-	GB_PANIC("Unexpected address expression");
+	GB_PANIC("Unexpected address expression\n"
+	         "\tAstNode: %.*s\n", LIT(ast_node_strings[expr->kind]));
 
 	ssaLvalue blank = {ssaLvalue_Blank};
 	return blank;

+ 1 - 1
src/main.cpp

@@ -27,11 +27,11 @@ int main(int argc, char **argv) {
 				// print_ast(parser.files[0].declarations, 0);
 
 				Checker checker = {};
+
 				init_checker(&checker, &parser);
 				defer (destroy_checker(&checker));
 
 				check_parsed_files(&checker);
-
 				ssaGen ssa = {};
 				if (ssa_gen_init(&ssa, &checker)) {
 					defer (ssa_gen_destroy(&ssa));

+ 3 - 0
src/parser.cpp

@@ -62,6 +62,7 @@ struct Parser {
 	gbArray(AstFile) files;
 	gbArray(String) imports;
 	isize import_index;
+	isize total_token_count;
 };
 
 enum DeclKind {
@@ -2137,8 +2138,10 @@ ParseFileError parse_files(Parser *p, char *init_filename) {
 		}
 		parse_file(p, &file);
 		gb_array_append(p->files, file);
+		p->total_token_count += gb_array_count(file.tokens);
 	}
 
+
 	return ParseFile_None;
 }
 

+ 88 - 88
src/tokenizer.cpp

@@ -26,98 +26,98 @@ b32 rune_is_whitespace(Rune r) {
 }
 
 #define TOKEN_KINDS \
-	TOKEN_KIND(Invalid, "Invalid"), \
-	TOKEN_KIND(EOF, "EOF"), \
+	TOKEN_KIND(Token_Invalid, "Invalid"), \
+	TOKEN_KIND(Token_EOF, "EOF"), \
 \
-TOKEN_KIND(_LiteralBegin, "_LiteralBegin"), \
-	TOKEN_KIND(Identifier, "Identifier"), \
-	TOKEN_KIND(Integer, "Integer"), \
-	TOKEN_KIND(Float, "Float"), \
-	TOKEN_KIND(Rune, "Rune"), \
-	TOKEN_KIND(String, "String"), \
-TOKEN_KIND(_LiteralEnd, "_LiteralEnd"), \
+TOKEN_KIND(Token__LiteralBegin, "_LiteralBegin"), \
+	TOKEN_KIND(Token_Identifier, "Identifier"), \
+	TOKEN_KIND(Token_Integer, "Integer"), \
+	TOKEN_KIND(Token_Float, "Float"), \
+	TOKEN_KIND(Token_Rune, "Rune"), \
+	TOKEN_KIND(Token_String, "String"), \
+TOKEN_KIND(Token__LiteralEnd, "_LiteralEnd"), \
 \
-TOKEN_KIND(_OperatorBegin, "_OperatorBegin"), \
-	TOKEN_KIND(Eq, "="), \
-	TOKEN_KIND(Not, "!"), \
-	TOKEN_KIND(Hash, "#"), \
-	TOKEN_KIND(At, "@"), \
-	TOKEN_KIND(Pointer, "^"), \
-	TOKEN_KIND(Add, "+"), \
-	TOKEN_KIND(Sub, "-"), \
-	TOKEN_KIND(Mul, "*"), \
-	TOKEN_KIND(Quo, "/"), \
-	TOKEN_KIND(Mod, "%"), \
-	TOKEN_KIND(And, "&"), \
-	TOKEN_KIND(Or, "|"), \
-	TOKEN_KIND(Xor, "~"), \
-	TOKEN_KIND(AndNot, "&~"), \
-TOKEN_KIND(_AssignOpBegin, "_AssignOpBegin"), \
-	TOKEN_KIND(AddEq, "+="), \
-	TOKEN_KIND(SubEq, "-="), \
-	TOKEN_KIND(MulEq, "*="), \
-	TOKEN_KIND(QuoEq, "/="), \
-	TOKEN_KIND(ModEq, "%="), \
-	TOKEN_KIND(AndEq, "&="), \
-	TOKEN_KIND(OrEq, "|="), \
-	TOKEN_KIND(XorEq, "~="), \
-	TOKEN_KIND(AndNotEq, "&~="), \
-TOKEN_KIND(_AssignOpEnd, "_AssignOpEnd"), \
-	TOKEN_KIND(Increment, "++"), \
-	TOKEN_KIND(Decrement, "--"), \
-	TOKEN_KIND(ArrowRight, "->"), \
-	TOKEN_KIND(ArrowLeft, "<-"), \
-	TOKEN_KIND(CmpAnd, "&&"), \
-	TOKEN_KIND(CmpOr, "||"), \
-	TOKEN_KIND(CmpAndEq, "&&="), \
-	TOKEN_KIND(CmpOrEq, "||="), \
+TOKEN_KIND(Token__OperatorBegin, "_OperatorBegin"), \
+	TOKEN_KIND(Token_Eq, "="), \
+	TOKEN_KIND(Token_Not, "!"), \
+	TOKEN_KIND(Token_Hash, "#"), \
+	TOKEN_KIND(Token_At, "@"), \
+	TOKEN_KIND(Token_Pointer, "^"), \
+	TOKEN_KIND(Token_Add, "+"), \
+	TOKEN_KIND(Token_Sub, "-"), \
+	TOKEN_KIND(Token_Mul, "*"), \
+	TOKEN_KIND(Token_Quo, "/"), \
+	TOKEN_KIND(Token_Mod, "%"), \
+	TOKEN_KIND(Token_And, "&"), \
+	TOKEN_KIND(Token_Or, "|"), \
+	TOKEN_KIND(Token_Xor, "~"), \
+	TOKEN_KIND(Token_AndNot, "&~"), \
+TOKEN_KIND(Token__AssignOpBegin, "_AssignOpBegin"), \
+	TOKEN_KIND(Token_AddEq, "+="), \
+	TOKEN_KIND(Token_SubEq, "-="), \
+	TOKEN_KIND(Token_MulEq, "*="), \
+	TOKEN_KIND(Token_QuoEq, "/="), \
+	TOKEN_KIND(Token_ModEq, "%="), \
+	TOKEN_KIND(Token_AndEq, "&="), \
+	TOKEN_KIND(Token_OrEq, "|="), \
+	TOKEN_KIND(Token_XorEq, "~="), \
+	TOKEN_KIND(Token_AndNotEq, "&~="), \
+TOKEN_KIND(Token__AssignOpEnd, "_AssignOpEnd"), \
+	TOKEN_KIND(Token_Increment, "++"), \
+	TOKEN_KIND(Token_Decrement, "--"), \
+	TOKEN_KIND(Token_ArrowRight, "->"), \
+	TOKEN_KIND(Token_ArrowLeft, "<-"), \
+	TOKEN_KIND(Token_CmpAnd, "&&"), \
+	TOKEN_KIND(Token_CmpOr, "||"), \
+	TOKEN_KIND(Token_CmpAndEq, "&&="), \
+	TOKEN_KIND(Token_CmpOrEq, "||="), \
 \
-TOKEN_KIND(_ComparisonBegin, "_ComparisonBegin"), \
-	TOKEN_KIND(CmpEq, "=="), \
-	TOKEN_KIND(NotEq, "!="), \
-	TOKEN_KIND(Lt, "<"), \
-	TOKEN_KIND(Gt, ">"), \
-	TOKEN_KIND(LtEq, "<="), \
-	TOKEN_KIND(GtEq, ">="), \
-TOKEN_KIND(_ComparisonEnd, "_ComparisonEnd"), \
+TOKEN_KIND(Token__ComparisonBegin, "_ComparisonBegin"), \
+	TOKEN_KIND(Token_CmpEq, "=="), \
+	TOKEN_KIND(Token_NotEq, "!="), \
+	TOKEN_KIND(Token_Lt, "<"), \
+	TOKEN_KIND(Token_Gt, ">"), \
+	TOKEN_KIND(Token_LtEq, "<="), \
+	TOKEN_KIND(Token_GtEq, ">="), \
+TOKEN_KIND(Token__ComparisonEnd, "_ComparisonEnd"), \
 \
-	TOKEN_KIND(OpenParen, "("), \
-	TOKEN_KIND(CloseParen, ")"), \
-	TOKEN_KIND(OpenBracket, "["), \
-	TOKEN_KIND(CloseBracket, "]"), \
-	TOKEN_KIND(OpenBrace, "{"), \
-	TOKEN_KIND(CloseBrace, "}"), \
-	TOKEN_KIND(Colon, ":"), \
-	TOKEN_KIND(Semicolon, ";"), \
-	TOKEN_KIND(Period, "."), \
-	TOKEN_KIND(Comma, ","), \
-	TOKEN_KIND(Ellipsis, "..."), \
-TOKEN_KIND(_OperatorEnd, "_OperatorEnd"), \
+	TOKEN_KIND(Token_OpenParen, "("), \
+	TOKEN_KIND(Token_CloseParen, ")"), \
+	TOKEN_KIND(Token_OpenBracket, "["), \
+	TOKEN_KIND(Token_CloseBracket, "]"), \
+	TOKEN_KIND(Token_OpenBrace, "{"), \
+	TOKEN_KIND(Token_CloseBrace, "}"), \
+	TOKEN_KIND(Token_Colon, ":"), \
+	TOKEN_KIND(Token_Semicolon, ";"), \
+	TOKEN_KIND(Token_Period, "."), \
+	TOKEN_KIND(Token_Comma, ","), \
+	TOKEN_KIND(Token_Ellipsis, "..."), \
+TOKEN_KIND(Token__OperatorEnd, "_OperatorEnd"), \
 \
-TOKEN_KIND(_KeywordBegin, "_KeywordBegin"), \
-	TOKEN_KIND(type, "type"), \
-	TOKEN_KIND(alias, "alias"), \
-	TOKEN_KIND(proc, "proc"), \
-	TOKEN_KIND(match, "match"), \
-	TOKEN_KIND(break, "break"), \
-	TOKEN_KIND(continue, "continue"), \
-	TOKEN_KIND(fallthrough, "fallthrough"), \
-	TOKEN_KIND(case, "case"), \
-	TOKEN_KIND(if, "if"), \
-	TOKEN_KIND(else, "else"), \
-	TOKEN_KIND(for, "for"), \
-	TOKEN_KIND(defer, "defer"), \
-	TOKEN_KIND(return, "return"), \
-	TOKEN_KIND(import, "import"), \
-	TOKEN_KIND(cast, "cast"), \
-	TOKEN_KIND(struct, "struct"), \
-	TOKEN_KIND(union, "union"), \
-	TOKEN_KIND(enum, "enum"), \
-TOKEN_KIND(_KeywordEnd, "_KeywordEnd"), \
-	TOKEN_KIND(Count, "")
+TOKEN_KIND(Token__KeywordBegin, "_KeywordBegin"), \
+	TOKEN_KIND(Token_type, "type"), \
+	TOKEN_KIND(Token_alias, "alias"), \
+	TOKEN_KIND(Token_proc, "proc"), \
+	TOKEN_KIND(Token_match, "match"), \
+	TOKEN_KIND(Token_break, "break"), \
+	TOKEN_KIND(Token_continue, "continue"), \
+	TOKEN_KIND(Token_fallthrough, "fallthrough"), \
+	TOKEN_KIND(Token_case, "case"), \
+	TOKEN_KIND(Token_if, "if"), \
+	TOKEN_KIND(Token_else, "else"), \
+	TOKEN_KIND(Token_for, "for"), \
+	TOKEN_KIND(Token_defer, "defer"), \
+	TOKEN_KIND(Token_return, "return"), \
+	TOKEN_KIND(Token_import, "import"), \
+	TOKEN_KIND(Token_cast, "cast"), \
+	TOKEN_KIND(Token_struct, "struct"), \
+	TOKEN_KIND(Token_union, "union"), \
+	TOKEN_KIND(Token_enum, "enum"), \
+TOKEN_KIND(Token__KeywordEnd, "_KeywordEnd"), \
+	TOKEN_KIND(Token_Count, "")
 
 enum TokenKind {
-#define TOKEN_KIND(e, s) GB_JOIN2(Token_, e)
+#define TOKEN_KIND(e, s) e
 	TOKEN_KINDS
 #undef TOKEN_KIND
 };
@@ -162,10 +162,10 @@ Token empty_token = {Token_Invalid};
 
 struct ErrorCollector {
 	TokenPos prev;
-	isize count;
+	i64 count;
 };
 
-void error(ErrorCollector *ec, Token token, char *fmt, ...) {
+gb_no_inline void error(ErrorCollector *ec, Token token, char *fmt, ...) {
 	ec->count++;
 	// NOTE(bill): Duplicate error, skip it
 	if (!token_pos_are_equal(ec->prev, token.pos)) {
@@ -181,7 +181,7 @@ void error(ErrorCollector *ec, Token token, char *fmt, ...) {
 	}
 }
 
-void warning(Token token, char *fmt, ...) {
+gb_no_inline void warning(Token token, char *fmt, ...) {
 	va_list va;
 	va_start(va, fmt);
 	gb_printf_err("%.*s(%td:%td) Warning: %s\n",