Win32 test

This commit is contained in:
gingerBill
2016-08-15 21:22:44 +01:00
parent dcbb2fcfbd
commit 50fd9548b9
13 changed files with 803 additions and 432 deletions
+2 -55
View File
@@ -1,15 +1,6 @@
// CRT
putchar :: proc(c: i32) -> i32 #foreign
heap_alloc :: proc(sz: int) -> rawptr #foreign "malloc"
heap_free :: proc(ptr: rawptr) #foreign "free"
mem_compare :: proc(dst, src : rawptr, len: int) -> i32 #foreign "memcmp"
mem_copy :: proc(dst, src : rawptr, len: int) -> i32 #foreign "memcpy"
mem_move :: proc(dst, src : rawptr, len: int) -> i32 #foreign "memmove"
debug_trap :: proc() #foreign "llvm.debugtrap"
#load "runtime.odin"
TWO_HEARTS :: '💕';
print_string :: proc(s: string) {
for i := 0; i < len(s); i++ {
@@ -137,47 +128,3 @@ print_bool :: proc(b : bool) {
}
}
// Runtime procedures
__string_eq :: proc(a, b : string) -> bool {
if len(a) != len(b) {
return false;
}
if ^a[0] == ^b[0] {
return true;
}
return mem_compare(^a[0], ^b[0], len(a)) == 0;
}
__string_ne :: proc(a, b : string) -> bool {
return !__string_eq(a, b);
}
__string_cmp :: proc(a, b : string) -> int {
min_len := len(a);
if len(b) < min_len {
min_len = len(b);
}
for i := 0; i < min_len; i++ {
x := a[i];
y := b[i];
if x < y {
return -1;
} else if x > y {
return +1;
}
}
if len(a) < len(b) {
return -1;
} else if len(a) > len(b) {
return +1;
}
return 0;
}
__string_lt :: proc(a, b : string) -> bool { return __string_cmp(a, b) < 0; }
__string_gt :: proc(a, b : string) -> bool { return __string_cmp(a, b) > 0; }
__string_le :: proc(a, b : string) -> bool { return __string_cmp(a, b) <= 0; }
__string_ge :: proc(a, b : string) -> bool { return __string_cmp(a, b) >= 0; }
+394 -188
View File
@@ -1,57 +1,224 @@
%.string = type {i8*, i64} ; Basic_string
%.rawptr = type i8* ; Basic_rawptr
%HANDLE = type %.rawptr
%HWND = type %.rawptr
%HDC = type %.rawptr
%HINSTANCE = type %.rawptr
%HICON = type %.rawptr
%HCURSOR = type %.rawptr
%HMENU = type %.rawptr
%WPARAM = type i64
%LPARAM = type i64
%LRESULT = type i64
%ATOM = type i16
%POINT = type {i32, i32}
%BOOL = type i32
%WNDPROC = type %LRESULT (%HWND, i32, %WPARAM, %LPARAM)*
%WNDCLASSEXA = type {i32, i32, %WNDPROC, i32, i32, %HINSTANCE, %HICON, %HCURSOR, %HANDLE, i8*, i8*, %HICON}
%MSG = type {%HWND, i32, %WPARAM, %LPARAM, i32, %POINT}
declare void @llvm.memmove.p0i8.p0i8.i64(i8*, i8*, i64, i32, i1) argmemonly nounwind
define void @main() {
entry.-.0:
%0 = getelementptr inbounds [8 x i8], [8 x i8]* @.str1, 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 8, i64* %3
%4 = load %.string, %.string* %1, align 8
call void @print_string(%.string %4)
call void @"main$\E4\B8\96\E7\95\8C-1"()
%0 = alloca %WNDCLASSEXA, align 8 ; wc
store %WNDCLASSEXA zeroinitializer, %WNDCLASSEXA* %0
%1 = alloca %HINSTANCE, align 8 ; instance
store %HINSTANCE zeroinitializer, %HINSTANCE* %1
%2 = call %HINSTANCE @GetModuleHandleA(i8* null)
store %HINSTANCE %2, %HINSTANCE* %1
%3 = alloca i8*, align 8 ; class_name
store i8* zeroinitializer, i8** %3
%4 = getelementptr inbounds [18 x i8], [18 x i8]* @.str0, i64 0, i64 0
%5 = alloca %.string, align 8
store %.string zeroinitializer, %.string* %5
%6 = getelementptr inbounds %.string, %.string* %5, i64 0, i32 0
%7 = getelementptr inbounds %.string, %.string* %5, i64 0, i32 1
store i8* %4, i8** %6
store i64 18, i64* %7
%8 = load %.string, %.string* %5, align 8
%9 = call i8* @main$to_c_string-0(%.string %8)
store i8* %9, i8** %3
%10 = alloca i8*, align 8 ; title
store i8* zeroinitializer, i8** %10
%11 = getelementptr inbounds [18 x i8], [18 x i8]* @.str1, 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 18, i64* %14
%15 = load %.string, %.string* %12, align 8
%16 = call i8* @main$to_c_string-0(%.string %15)
store i8* %16, i8** %10
%17 = getelementptr inbounds %WNDCLASSEXA, %WNDCLASSEXA* %0, i64 0, i32 0
store i32 80, i32* %17
%18 = getelementptr inbounds %WNDCLASSEXA, %WNDCLASSEXA* %0, i64 0, i32 1
store i32 3, i32* %18
%19 = getelementptr inbounds %WNDCLASSEXA, %WNDCLASSEXA* %0, i64 0, i32 5
%20 = load %HINSTANCE, %HINSTANCE* %1, align 8
store %HINSTANCE %20, %HINSTANCE* %19
%21 = getelementptr inbounds %WNDCLASSEXA, %WNDCLASSEXA* %0, i64 0, i32 10
%22 = load i8*, i8** %3, align 8
store i8* %22, i8** %21
%23 = getelementptr inbounds %WNDCLASSEXA, %WNDCLASSEXA* %0, i64 0, i32 2
store %WNDPROC @main$1, %WNDPROC* %23
%24 = getelementptr inbounds %WNDCLASSEXA, %WNDCLASSEXA* %0
%25 = call %ATOM @RegisterClassExA(%WNDCLASSEXA* %24)
%26 = icmp eq i16 %25, 0
br i1 %26, label %if.then.-.1, label %if.done.-.2
if.then.-.1:
ret void
if.done.-.2:
%27 = alloca %HWND, align 8 ; hwnd
store %HWND zeroinitializer, %HWND* %27
%28 = load i8*, i8** %3, align 8
%29 = load i8*, i8** %10, align 8
%30 = load %HINSTANCE, %HINSTANCE* %1, align 8
%31 = call %HWND @CreateWindowExA(i32 0, i8* %28, i8* %29, i32 281673728, i32 0, i32 0, i32 854, i32 480, %HWND null, %HMENU null, %HINSTANCE %30, %.rawptr null)
store %HWND %31, %HWND* %27
%32 = load %HWND, %HWND* %27, align 8
%33 = icmp eq %.rawptr %32, null
br i1 %33, label %if.then.-.3, label %if.done.-.4
if.then.-.3:
%34 = getelementptr inbounds [14 x i8], [14 x i8]* @.str2, i64 0, i64 0
%35 = alloca %.string, align 8
store %.string zeroinitializer, %.string* %35
%36 = getelementptr inbounds %.string, %.string* %35, i64 0, i32 0
%37 = getelementptr inbounds %.string, %.string* %35, i64 0, i32 1
store i8* %34, i8** %36
store i64 14, i64* %37
%38 = load %.string, %.string* %35, align 8
call void @print_string(%.string %38)
%39 = call i32 @GetLastError()
%40 = zext i32 %39 to i64
call void @print_int(i64 %40)
%41 = getelementptr inbounds [1 x i8], [1 x i8]* @.str3, 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 1, i64* %44
%45 = load %.string, %.string* %42, align 8
call void @print_string(%.string %45)
ret void
if.done.-.4:
%46 = alloca %MSG, align 8 ; msg
store %MSG zeroinitializer, %MSG* %46
br label %for.body.-.5
for.body.-.5:
%47 = alloca %BOOL, align 4 ; ok
store %BOOL zeroinitializer, %BOOL* %47
%48 = getelementptr inbounds %MSG, %MSG* %46
%49 = call %BOOL @PeekMessageA(%MSG* %48, %HWND null, i32 0, i32 0, i32 1)
store %BOOL %49, %BOOL* %47
%50 = load %BOOL, %BOOL* %47, align 4
%51 = icmp eq i32 %50, 0
br i1 %51, label %if.then.-.6, label %if.done.-.7
if.then.-.6:
br label %for.done.-.11
if.done.-.7:
%52 = getelementptr inbounds %MSG, %MSG* %46, i64 0, i32 1
%53 = load i32, i32* %52, align 4
%54 = icmp eq i32 %53, 18
br i1 %54, label %if.then.-.8, label %if.else.-.9
if.then.-.8:
ret void
if.else.-.9:
%55 = getelementptr inbounds %MSG, %MSG* %46
%56 = call %BOOL @TranslateMessage(%MSG* %55)
%57 = getelementptr inbounds %MSG, %MSG* %46
%58 = call %LRESULT @DispatchMessageA(%MSG* %57)
br label %if.done.-.10
if.done.-.10:
br label %for.body.-.5
for.done.-.11:
ret void
}
define void @main$nl-0() {
define i8* @main$to_c_string-0(%.string %s) {
entry.-.0:
call void @print_rune(i32 10)
ret void
%0 = alloca %.string, align 8 ; s
store %.string zeroinitializer, %.string* %0
store %.string %s, %.string* %0
%1 = alloca i8*, align 8 ; c_str
store i8* zeroinitializer, i8** %1
%2 = getelementptr inbounds %.string, %.string* %0, i64 0, i32 1
%3 = load i64, i64* %2, align 8
%4 = add i64 %3, 1
%5 = call %.rawptr @malloc(i64 %4)
%6 = bitcast %.rawptr %5 to i8*
store i8* %6, i8** %1
%7 = load i8*, i8** %1, align 8
%8 = getelementptr inbounds %.string, %.string* %0, i64 0, i32 0
%9 = load i8*, i8** %8, align 8
%10 = getelementptr i8, i8* %9, i64 0
%11 = getelementptr inbounds i8, i8* %10
%12 = getelementptr inbounds %.string, %.string* %0, i64 0, i32 1
%13 = load i64, i64* %12, align 8
%14 = call i32 @memcpy(%.rawptr %7, %.rawptr %11, i64 %13)
%15 = load i8*, i8** %1, align 8
%16 = getelementptr inbounds %.string, %.string* %0, i64 0, i32 1
%17 = load i64, i64* %16, align 8
%18 = getelementptr i8, i8* %15, i64 %17
store i8 0, i8* %18
%19 = load i8*, i8** %1, align 8
ret i8* %19
}
define void @"main$\E4\B8\96\E7\95\8C-1"() {
define %LRESULT @main$1(%HWND %hwnd, i32 %msg, %WPARAM %wparam, %LPARAM %lparam) noinline {
entry.-.0:
%0 = getelementptr inbounds [9 x i8], [9 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 9, i64* %3
%4 = load %.string, %.string* %1, align 8
call void @print_string(%.string %4)
ret void
%0 = alloca %HWND, align 8 ; hwnd
store %HWND zeroinitializer, %HWND* %0
store %HWND %hwnd, %HWND* %0
%1 = alloca i32, align 4 ; msg
store i32 zeroinitializer, i32* %1
store i32 %msg, i32* %1
%2 = alloca %WPARAM, align 8 ; wparam
store %WPARAM zeroinitializer, %WPARAM* %2
store %WPARAM %wparam, %WPARAM* %2
%3 = alloca %LPARAM, align 8 ; lparam
store %LPARAM zeroinitializer, %LPARAM* %3
store %LPARAM %lparam, %LPARAM* %3
%4 = load i32, i32* %1, align 4
%5 = icmp eq i32 %4, 2
br i1 %5, label %if.then.-.1, label %cmp-or.-.3
if.then.-.1:
call void @ExitProcess(i32 0)
ret %LRESULT 0
cmp-or.-.2:
%6 = load i32, i32* %1, align 4
%7 = icmp eq i32 %6, 18
br i1 %7, label %if.then.-.1, label %if.done.-.4
cmp-or.-.3:
%8 = load i32, i32* %1, align 4
%9 = icmp eq i32 %8, 16
br i1 %9, label %if.then.-.1, label %cmp-or.-.2
if.done.-.4:
%10 = load %HWND, %HWND* %0, align 8
%11 = load i32, i32* %1, align 4
%12 = load %WPARAM, %WPARAM* %2, align 8
%13 = load %LPARAM, %LPARAM* %3, align 8
%14 = call %LRESULT @DefWindowProcA(%HWND %10, i32 %11, %WPARAM %12, %LPARAM %13)
ret i64 %14
}
declare i32 @putchar(i32 %c) ; foreign procedure
declare %.rawptr @malloc(i64 %sz) ; foreign procedure
declare void @free(%.rawptr %ptr) ; foreign procedure
declare i32 @memcmp(%.rawptr %dst, %.rawptr %src, i64 %len) ; foreign procedure
declare i32 @memcpy(%.rawptr %dst, %.rawptr %src, i64 %len) ; foreign procedure
declare i32 @memmove(%.rawptr %dst, %.rawptr %src, i64 %len) ; foreign procedure
declare void @llvm.debugtrap() ; foreign procedure
define void @print_string(%.string %s) {
entry.-.0:
%0 = alloca %.string, align 8 ; s
@@ -63,30 +230,30 @@ for.init.-.1:
%1 = alloca i64, align 8 ; i
store i64 zeroinitializer, i64* %1
store i64 0, i64* %1
br label %for.loop.-.2
br label %for.loop.-.3
for.loop.-.2:
%2 = load i64, i64* %1, align 8
%3 = getelementptr inbounds %.string, %.string* %0, i64 0, i32 1
%4 = load i64, i64* %3, align 8
%5 = icmp slt i64 %2, %4
br i1 %5, label %for.body.-.4, label %for.done.-.5
for.body.-.2:
%2 = getelementptr inbounds %.string, %.string* %0, i64 0, i32 0
%3 = load i8*, i8** %2, align 8
%4 = load i64, i64* %1, align 8
%5 = getelementptr i8, i8* %3, i64 %4
%6 = load i8, i8* %5, align 1
%7 = zext i8 %6 to i32
%8 = call i32 @putchar(i32 %7)
br label %for.post.-.4
for.post.-.3:
%6 = load i64, i64* %1, align 8
%7 = add i64 %6, 1
store i64 %7, i64* %1
br label %for.loop.-.2
for.loop.-.3:
%9 = load i64, i64* %1, align 8
%10 = getelementptr inbounds %.string, %.string* %0, i64 0, i32 1
%11 = load i64, i64* %10, align 8
%12 = icmp slt i64 %9, %11
br i1 %12, label %for.body.-.2, label %for.done.-.5
for.body.-.4:
%8 = getelementptr inbounds %.string, %.string* %0, i64 0, i32 0
%9 = load i8*, i8** %8, align 8
%10 = load i64, i64* %1, align 8
%11 = getelementptr i8, i8* %9, i64 %10
%12 = load i8, i8* %11, align 1
%13 = zext i8 %12 to i32
%14 = call i32 @putchar(i32 %13)
br label %for.post.-.3
for.post.-.4:
%13 = load i64, i64* %1, align 8
%14 = add i64 %13, 1
store i64 %14, i64* %1
br label %for.loop.-.3
for.done.-.5:
ret void
@@ -108,49 +275,49 @@ for.init.-.1:
%4 = alloca i64, align 8 ; i
store i64 zeroinitializer, i64* %4
store i64 0, i64* %4
br label %for.loop.-.2
br label %for.loop.-.3
for.loop.-.2:
%5 = load i64, i64* %4, align 8
%6 = load i64, i64* %1, align 8
%7 = sdiv i64 %6, 2
%8 = icmp slt i64 %5, %7
br i1 %8, label %for.body.-.4, label %for.done.-.5
for.body.-.2:
%5 = getelementptr inbounds {i8*, i64, i64}, {i8*, i64, i64}* %0, i64 0, i32 0
%6 = load i8*, i8** %5, align 8
%7 = load i64, i64* %4, align 8
%8 = getelementptr i8, i8* %6, i64 %7
%9 = getelementptr inbounds {i8*, i64, i64}, {i8*, i64, i64}* %0, i64 0, i32 0
%10 = load i8*, i8** %9, align 8
%11 = load i64, i64* %4, align 8
%12 = load i64, i64* %1, align 8
%13 = sub i64 %12, 1
%14 = sub i64 %13, %11
%15 = getelementptr i8, i8* %10, i64 %14
%16 = getelementptr inbounds {i8*, i64, i64}, {i8*, i64, i64}* %0, i64 0, i32 0
%17 = load i8*, i8** %16, align 8
%18 = load i64, i64* %4, align 8
%19 = load i64, i64* %1, align 8
%20 = sub i64 %19, 1
%21 = sub i64 %20, %18
%22 = getelementptr i8, i8* %17, i64 %21
%23 = load i8, i8* %22, align 1
%24 = getelementptr inbounds {i8*, i64, i64}, {i8*, i64, i64}* %0, i64 0, i32 0
%25 = load i8*, i8** %24, align 8
%26 = load i64, i64* %4, align 8
%27 = getelementptr i8, i8* %25, i64 %26
%28 = load i8, i8* %27, align 1
store i8 %23, i8* %8
store i8 %28, i8* %15
br label %for.post.-.4
for.post.-.3:
%9 = load i64, i64* %4, align 8
%10 = add i64 %9, 1
store i64 %10, i64* %4
br label %for.loop.-.2
for.loop.-.3:
%29 = load i64, i64* %4, align 8
%30 = load i64, i64* %1, align 8
%31 = sdiv i64 %30, 2
%32 = icmp slt i64 %29, %31
br i1 %32, label %for.body.-.2, label %for.done.-.5
for.body.-.4:
%11 = getelementptr inbounds {i8*, i64, i64}, {i8*, i64, i64}* %0, i64 0, i32 0
%12 = load i8*, i8** %11, align 8
%13 = load i64, i64* %4, align 8
%14 = getelementptr i8, i8* %12, i64 %13
%15 = getelementptr inbounds {i8*, i64, i64}, {i8*, i64, i64}* %0, i64 0, i32 0
%16 = load i8*, i8** %15, align 8
%17 = load i64, i64* %4, align 8
%18 = load i64, i64* %1, align 8
%19 = sub i64 %18, 1
%20 = sub i64 %19, %17
%21 = getelementptr i8, i8* %16, i64 %20
%22 = getelementptr inbounds {i8*, i64, i64}, {i8*, i64, i64}* %0, i64 0, i32 0
%23 = load i8*, i8** %22, align 8
%24 = load i64, i64* %4, align 8
%25 = load i64, i64* %1, align 8
%26 = sub i64 %25, 1
%27 = sub i64 %26, %24
%28 = getelementptr i8, i8* %23, i64 %27
%29 = load i8, i8* %28, align 1
%30 = getelementptr inbounds {i8*, i64, i64}, {i8*, i64, i64}* %0, i64 0, i32 0
%31 = load i8*, i8** %30, align 8
%32 = load i64, i64* %4, align 8
%33 = getelementptr i8, i8* %31, i64 %32
%34 = load i8, i8* %33, align 1
store i8 %29, i8* %14
store i8 %34, i8* %21
br label %for.post.-.3
for.post.-.4:
%33 = load i64, i64* %4, align 8
%34 = add i64 %33, 1
store i64 %34, i64* %4
br label %for.loop.-.3
for.done.-.5:
ret void
@@ -420,32 +587,32 @@ if.then.-.3:
br label %if.done.-.4
if.done.-.4:
br label %for.loop.-.5
br label %for.loop.-.6
for.loop.-.5:
%16 = load i64, i64* %0, align 8
%17 = icmp sgt i64 %16, 0
br i1 %17, label %for.body.-.6, label %for.done.-.7
for.body.-.5:
%16 = getelementptr inbounds [65 x i8], [65 x i8]* %2, i64 0, i64 0
%17 = load i64, i64* %3, align 8
%18 = getelementptr i8, i8* %16, i64 %17
%19 = getelementptr inbounds [64 x i8], [64 x i8]* @.str4, i64 0, i64 0
%20 = load i64, i64* %1, align 8
%21 = load i64, i64* %0, align 8
%22 = srem i64 %21, %20
%23 = getelementptr i8, i8* %19, i64 %22
%24 = load i8, i8* %23, align 1
store i8 %24, i8* %18
%25 = load i64, i64* %3, align 8
%26 = add i64 %25, 1
store i64 %26, i64* %3
%27 = load i64, i64* %1, align 8
%28 = load i64, i64* %0, align 8
%29 = sdiv i64 %28, %27
store i64 %29, i64* %0
br label %for.loop.-.6
for.body.-.6:
%18 = getelementptr inbounds [65 x i8], [65 x i8]* %2, i64 0, i64 0
%19 = load i64, i64* %3, align 8
%20 = getelementptr i8, i8* %18, i64 %19
%21 = getelementptr inbounds [64 x i8], [64 x i8]* @.str2, i64 0, i64 0
%22 = load i64, i64* %1, align 8
%23 = load i64, i64* %0, align 8
%24 = srem i64 %23, %22
%25 = getelementptr i8, i8* %21, i64 %24
%26 = load i8, i8* %25, align 1
store i8 %26, i8* %20
%27 = load i64, i64* %3, align 8
%28 = add i64 %27, 1
store i64 %28, i64* %3
%29 = load i64, i64* %1, align 8
for.loop.-.6:
%30 = load i64, i64* %0, align 8
%31 = sdiv i64 %30, %29
store i64 %31, i64* %0
br label %for.loop.-.5
%31 = icmp sgt i64 %30, 0
br i1 %31, label %for.body.-.5, label %for.done.-.7
for.done.-.7:
%32 = load i1, i1* %4, align 1
@@ -562,32 +729,32 @@ if.then.-.3:
br label %if.done.-.4
if.done.-.4:
br label %for.loop.-.5
br label %for.loop.-.6
for.loop.-.5:
%16 = load i64, i64* %0, align 8
%17 = icmp ugt i64 %16, 0
br i1 %17, label %for.body.-.6, label %for.done.-.7
for.body.-.5:
%16 = getelementptr inbounds [65 x i8], [65 x i8]* %2, i64 0, i64 0
%17 = load i64, i64* %3, align 8
%18 = getelementptr i8, i8* %16, i64 %17
%19 = getelementptr inbounds [64 x i8], [64 x i8]* @.str5, i64 0, i64 0
%20 = load i64, i64* %1, align 8
%21 = load i64, i64* %0, align 8
%22 = urem i64 %21, %20
%23 = getelementptr i8, i8* %19, i64 %22
%24 = load i8, i8* %23, align 1
store i8 %24, i8* %18
%25 = load i64, i64* %3, align 8
%26 = add i64 %25, 1
store i64 %26, i64* %3
%27 = load i64, i64* %1, align 8
%28 = load i64, i64* %0, align 8
%29 = udiv i64 %28, %27
store i64 %29, i64* %0
br label %for.loop.-.6
for.body.-.6:
%18 = getelementptr inbounds [65 x i8], [65 x i8]* %2, i64 0, i64 0
%19 = load i64, i64* %3, align 8
%20 = getelementptr i8, i8* %18, i64 %19
%21 = getelementptr inbounds [64 x i8], [64 x i8]* @.str3, i64 0, i64 0
%22 = load i64, i64* %1, align 8
%23 = load i64, i64* %0, align 8
%24 = urem i64 %23, %22
%25 = getelementptr i8, i8* %21, i64 %24
%26 = load i8, i8* %25, align 1
store i8 %26, i8* %20
%27 = load i64, i64* %3, align 8
%28 = add i64 %27, 1
store i64 %28, i64* %3
%29 = load i64, i64* %1, align 8
for.loop.-.6:
%30 = load i64, i64* %0, align 8
%31 = udiv i64 %30, %29
store i64 %31, i64* %0
br label %for.loop.-.5
%31 = icmp ugt i64 %30, 0
br i1 %31, label %for.body.-.5, label %for.done.-.7
for.done.-.7:
%32 = load i1, i1* %4, align 1
@@ -660,7 +827,7 @@ entry.-.0:
br i1 %1, label %if.then.-.1, label %if.else.-.2
if.then.-.1:
%2 = getelementptr inbounds [4 x i8], [4 x i8]* @.str4, i64 0, i64 0
%2 = getelementptr inbounds [4 x i8], [4 x i8]* @.str6, i64 0, i64 0
%3 = alloca %.string, align 8
store %.string zeroinitializer, %.string* %3
%4 = getelementptr inbounds %.string, %.string* %3, i64 0, i32 0
@@ -672,7 +839,7 @@ if.then.-.1:
br label %if.done.-.3
if.else.-.2:
%7 = getelementptr inbounds [5 x i8], [5 x i8]* @.str5, i64 0, i64 0
%7 = getelementptr inbounds [5 x i8], [5 x i8]* @.str7, i64 0, i64 0
%8 = alloca %.string, align 8
store %.string zeroinitializer, %.string* %8
%9 = getelementptr inbounds %.string, %.string* %8, i64 0, i32 0
@@ -687,6 +854,43 @@ if.done.-.3:
ret void
}
declare %HANDLE @GetStdHandle(i32 %h) ; foreign
declare i32 @CloseHandle(%HANDLE %h) ; foreign
declare i32 @WriteFileA(%HANDLE %h, %.rawptr %buf, i32 %len, i32* %written_result, %.rawptr %overlapped) ; foreign
declare i32 @GetLastError() ; foreign
declare void @ExitProcess(i32 %exit_code) ; foreign
declare %HWND @GetDesktopWindow() ; foreign
declare i32 @GetCursorPos(%POINT* %p) ; foreign
declare i32 @ScreenToClient(%HWND %h, %POINT* %p) ; foreign
declare %HINSTANCE @GetModuleHandleA(i8* %module_name) ; foreign
declare i32 @QueryPerformanceFrequency(i64* %result) ; foreign
declare i32 @QueryPerformanceCounter(i64* %result) ; foreign
define void @sleep_ms(i32 %ms) {
entry.-.0:
%0 = alloca i32, align 4 ; ms
store i32 zeroinitializer, i32* %0
store i32 %ms, i32* %0
%1 = load i32, i32* %0, align 4
%2 = call i32 @Sleep(i32 %1)
ret void
}
declare i32 @Sleep(i32 %ms) declare void @OutputDebugStringA(i8* %c_str) ; foreign
declare %ATOM @RegisterClassExA(%WNDCLASSEXA* %wc) ; foreign
declare %HWND @CreateWindowExA(i32 %ex_style, i8* %class_name, i8* %title, i32 %style, i32 %x, i32 %y, i32 %w, i32 %h, %HWND %parent, %HMENU %menu, %HINSTANCE %instance, %.rawptr %param) ; foreign
declare %BOOL @ShowWindow(%HWND %hwnd, i32 %cmd_show) ; foreign
declare %BOOL @UpdateWindow(%HWND %hwnd) ; foreign
declare %BOOL @PeekMessageA(%MSG* %msg, %HWND %hwnd, i32 %msg_filter_min, i32 %msg_filter_max, i32 %remove_msg) ; foreign
declare %BOOL @TranslateMessage(%MSG* %msg) ; foreign
declare %LRESULT @DispatchMessageA(%MSG* %msg) ; foreign
declare %LRESULT @DefWindowProcA(%HWND %hwnd, i32 %msg, %WPARAM %wparam, %LPARAM %lparam) ; foreign
declare i32 @putchar(i32 %c) ; foreign
declare %.rawptr @malloc(i64 %sz) ; foreign
declare void @free(%.rawptr %ptr) ; foreign
declare i32 @memcmp(%.rawptr %dst, %.rawptr %src, i64 %len) ; foreign
declare i32 @memcpy(%.rawptr %dst, %.rawptr %src, i64 %len) ; foreign
declare i32 @memmove(%.rawptr %dst, %.rawptr %src, i64 %len) ; foreign
declare void @llvm.debugtrap() ; foreign
define i1 @__string_eq(%.string %a, %.string %b) {
entry.-.0:
%0 = alloca %.string, align 8 ; a
@@ -783,48 +987,48 @@ for.init.-.3:
%11 = alloca i64, align 8 ; i
store i64 zeroinitializer, i64* %11
store i64 0, i64* %11
br label %for.loop.-.4
br label %for.loop.-.5
for.loop.-.4:
%12 = load i64, i64* %11, align 8
%13 = load i64, i64* %2, align 8
%14 = icmp slt i64 %12, %13
br i1 %14, label %for.body.-.6, label %for.done.-.12
for.post.-.5:
for.body.-.4:
%12 = alloca i8, align 1 ; x
store i8 zeroinitializer, i8* %12
%13 = getelementptr inbounds %.string, %.string* %0, i64 0, i32 0
%14 = load i8*, i8** %13, align 8
%15 = load i64, i64* %11, align 8
%16 = add i64 %15, 1
store i64 %16, i64* %11
br label %for.loop.-.4
%16 = getelementptr i8, i8* %14, i64 %15
%17 = load i8, i8* %16, align 1
store i8 %17, i8* %12
%18 = alloca i8, align 1 ; y
store i8 zeroinitializer, i8* %18
%19 = getelementptr inbounds %.string, %.string* %1, i64 0, i32 0
%20 = load i8*, i8** %19, align 8
%21 = load i64, i64* %11, align 8
%22 = getelementptr i8, i8* %20, i64 %21
%23 = load i8, i8* %22, align 1
store i8 %23, i8* %18
%24 = load i8, i8* %12, align 1
%25 = load i8, i8* %18, align 1
%26 = icmp ult i8 %24, %25
br i1 %26, label %if.then.-.7, label %if.else.-.8
for.body.-.6:
%17 = alloca i8, align 1 ; x
store i8 zeroinitializer, i8* %17
%18 = getelementptr inbounds %.string, %.string* %0, i64 0, i32 0
%19 = load i8*, i8** %18, align 8
%20 = load i64, i64* %11, align 8
%21 = getelementptr i8, i8* %19, i64 %20
%22 = load i8, i8* %21, align 1
store i8 %22, i8* %17
%23 = alloca i8, align 1 ; y
store i8 zeroinitializer, i8* %23
%24 = getelementptr inbounds %.string, %.string* %1, i64 0, i32 0
%25 = load i8*, i8** %24, align 8
%26 = load i64, i64* %11, align 8
%27 = getelementptr i8, i8* %25, i64 %26
%28 = load i8, i8* %27, align 1
store i8 %28, i8* %23
%29 = load i8, i8* %17, align 1
%30 = load i8, i8* %23, align 1
%31 = icmp ult i8 %29, %30
br i1 %31, label %if.then.-.7, label %if.else.-.8
for.loop.-.5:
%27 = load i64, i64* %11, align 8
%28 = load i64, i64* %2, align 8
%29 = icmp slt i64 %27, %28
br i1 %29, label %for.body.-.4, label %for.done.-.12
for.post.-.6:
%30 = load i64, i64* %11, align 8
%31 = add i64 %30, 1
store i64 %31, i64* %11
br label %for.loop.-.5
if.then.-.7:
ret i64 -1
if.else.-.8:
%32 = load i8, i8* %17, align 1
%33 = load i8, i8* %23, align 1
%32 = load i8, i8* %12, align 1
%33 = load i8, i8* %18, align 1
%34 = icmp ugt i8 %32, %33
br i1 %34, label %if.then.-.9, label %if.done.-.10
@@ -835,7 +1039,7 @@ if.done.-.10:
br label %if.done.-.11
if.done.-.11:
br label %for.post.-.5
br label %for.post.-.6
for.done.-.12:
%35 = getelementptr inbounds %.string, %.string* %0, i64 0, i32 1
@@ -926,9 +1130,11 @@ entry.-.0:
ret i1 %5
}
@.str0 = global [9 x i8] c"\E6\97\A5\E6\9C\AC\E8\AA\9E"
@.str1 = global [8 x i8] c"Hellope\0A"
@.str2 = global [64 x i8] c"0123456789ABCDEFGHIJKLMNOPQRSTUVWXYZabcdefghijklmnopqrstuvwxyz\40$"
@.str3 = global [64 x i8] c"0123456789ABCDEFGHIJKLMNOPQRSTUVWXYZabcdefghijklmnopqrstuvwxyz\40$"
@.str4 = global [4 x i8] c"true"
@.str5 = global [5 x i8] c"false"
@.str0 = global [18 x i8] c"Odin-Language-Demo"
@.str1 = global [18 x i8] c"Odin-Language-Demo"
@.str2 = global [14 x i8] c"GetLastError\3A\20"
@.str3 = global [1 x i8] c"\0A"
@.str4 = global [64 x i8] c"0123456789ABCDEFGHIJKLMNOPQRSTUVWXYZabcdefghijklmnopqrstuvwxyz\40$"
@.str5 = global [64 x i8] c"0123456789ABCDEFGHIJKLMNOPQRSTUVWXYZabcdefghijklmnopqrstuvwxyz\40$"
@.str6 = global [4 x i8] c"true"
@.str7 = global [5 x i8] c"false"
+55 -40
View File
@@ -1,49 +1,64 @@
#load "basic.odin"
TWO_HEARTS :: '💕';
#load "win32.odin"
main :: proc() {
nl :: proc() { print_rune('\n'); }
世界 :: proc() { print_string(`日本語`); }
wc: WNDCLASSEXA;
instance := GetModuleHandleA(null);
print_string("Hellope\n");
世界();
/*
DATA_SIZE :: 100;
data := malloc(DATA_SIZE);
slice := (data as ^u8)[:0:DATA_SIZE];
for i := 0; i < cap(slice); i++ {
ok := append(^slice, (i*i) as u8 % 8);
// Yuck!
to_c_string :: proc(s: string) -> ^u8 {
c_str := heap_alloc(len(s)+1) as ^u8;
mem_copy(c_str, ^s[0], len(s));
c_str[len(s)] = 0;
return c_str;
}
for i := 0; i < len(slice); i++ {
print_int(slice[i] as int);
print_string(", ");
if (i+1) % 8 == 0 {
print_string("\n");
class_name := to_c_string("Odin-Language-Demo");
title := to_c_string("Odin-Language-Demo");
wc.cbSize = size_of(WNDCLASSEXA) as u32;
wc.style = CS_VREDRAW | CS_HREDRAW;
wc.hInstance = instance;
wc.className = class_name;
wc.wndProc = proc(hwnd: HWND, msg: u32, wparam: WPARAM, lparam: LPARAM) -> LRESULT #no_inline {
if (msg == WM_DESTROY ||
msg == WM_CLOSE ||
msg == WM_QUIT) {
ExitProcess(0);
return 0;
}
// HACK(bill): Compiler bug
return DefWindowProcA(hwnd, msg, wparam, lparam);
};
if RegisterClassExA(^wc) == 0 {
return;
}
hwnd := CreateWindowExA(0,
class_name, title,
WS_VISIBLE | WS_OVERLAPPED | WS_CAPTION | WS_SYSMENU | WS_MINIMIZEBOX,
0, 0, 854, 480,
null, null, instance, null);
if hwnd == null {
print_string("GetLastError: ");
print_int(GetLastError() as int);
print_string("\n");
return;
}
msg: MSG;
for {
ok := PeekMessageA(^msg, null, 0, 0, PM_REMOVE);
if ok == 0 { break; }
if msg.message == WM_QUIT {
return;
} else {
_ = TranslateMessage(^msg);
_ = DispatchMessageA(^msg);
}
}
print_string("\n");
free(data);
*/
}
// print_hello :: proc() {
// 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");
// }
+52
View File
@@ -0,0 +1,52 @@
putchar :: proc(c: i32) -> i32 #foreign
heap_alloc :: proc(sz: int) -> rawptr #foreign "malloc"
heap_free :: proc(ptr: rawptr) #foreign "free"
mem_compare :: proc(dst, src : rawptr, len: int) -> i32 #foreign "memcmp"
mem_copy :: proc(dst, src : rawptr, len: int) -> i32 #foreign "memcpy"
mem_move :: proc(dst, src : rawptr, len: int) -> i32 #foreign "memmove"
debug_trap :: proc() #foreign "llvm.debugtrap"
__string_eq :: proc(a, b : string) -> bool {
if len(a) != len(b) {
return false;
}
if ^a[0] == ^b[0] {
return true;
}
return mem_compare(^a[0], ^b[0], len(a)) == 0;
}
__string_ne :: proc(a, b : string) -> bool {
return !__string_eq(a, b);
}
__string_cmp :: proc(a, b : string) -> int {
min_len := len(a);
if len(b) < min_len {
min_len = len(b);
}
for i := 0; i < min_len; i++ {
x := a[i];
y := b[i];
if x < y {
return -1;
} else if x > y {
return +1;
}
}
if len(a) < len(b) {
return -1;
} else if len(a) > len(b) {
return +1;
}
return 0;
}
__string_lt :: proc(a, b : string) -> bool { return __string_cmp(a, b) < 0; }
__string_gt :: proc(a, b : string) -> bool { return __string_cmp(a, b) > 0; }
__string_le :: proc(a, b : string) -> bool { return __string_cmp(a, b) <= 0; }
__string_ge :: proc(a, b : string) -> bool { return __string_cmp(a, b) >= 0; }
+99
View File
@@ -0,0 +1,99 @@
STD_INPUT_HANDLE :: -10;
STD_OUTPUT_HANDLE :: -11;
STD_ERROR_HANDLE :: -12;
CS_VREDRAW :: 1;
CS_HREDRAW :: 2;
CW_USEDEFAULT :: 0x80000000;
WS_OVERLAPPED :: 0;
WS_MAXIMIZEBOX :: 0x00010000;
WS_MINIMIZEBOX :: 0x00020000;
WS_THICKFRAME :: 0x00040000;
WS_SYSMENU :: 0x00080000;
WS_CAPTION :: 0x00C00000;
WS_VISIBLE :: 0x10000000;
WS_OVERLAPPEDWINDOW :: WS_OVERLAPPED|WS_CAPTION|WS_SYSMENU|WS_THICKFRAME|WS_MINIMIZEBOX|WS_MAXIMIZEBOX;
WM_DESTROY :: 0x02;
WM_CLOSE :: 0x10;
WM_QUIT :: 0x12;
PM_REMOVE :: 1;
type HANDLE: rawptr;
type HWND: HANDLE;
type HDC: HANDLE;
type HINSTANCE: HANDLE;
type HICON: HANDLE;
type HCURSOR: HANDLE;
type HMENU: HANDLE;
type WPARAM: uint;
type LPARAM: int;
type LRESULT: int;
type ATOM: i16;
type POINT: struct { x, y: i32 }
type BOOL: i32;
type WNDPROC: proc(hwnd: HWND, msg: u32, wparam: WPARAM, lparam: LPARAM) -> LRESULT;
type WNDCLASSEXA: struct {
cbSize, style: u32,
wndProc: WNDPROC,
cbClsExtra, cbWndExtra: i32,
hInstance: HINSTANCE,
hIcon: HICON,
hCursor: HCURSOR,
hbrBackground: HANDLE,
menuName, className: ^u8,
hIconSm: HICON,
}
type MSG: struct {
hwnd: HWND,
message: u32,
wparam: WPARAM,
lparam: LPARAM,
time: u32,
pt: POINT,
}
GetStdHandle :: proc(h: i32) -> HANDLE #foreign
CloseHandle :: proc(h: HANDLE) -> i32 #foreign
WriteFileA :: proc(h: HANDLE, buf: rawptr, len: i32, written_result: ^i32, overlapped: rawptr) -> i32 #foreign
GetLastError :: proc() -> i32 #foreign
ExitProcess :: proc(exit_code: u32) #foreign
GetDesktopWindow :: proc() -> HWND #foreign
GetCursorPos :: proc(p: ^POINT) -> i32 #foreign
ScreenToClient :: proc(h: HWND, p: ^POINT) -> i32 #foreign
GetModuleHandleA :: proc(module_name: ^u8) -> HINSTANCE #foreign
QueryPerformanceFrequency :: proc(result: ^i64) -> i32 #foreign
QueryPerformanceCounter :: proc(result: ^i64) -> i32 #foreign
sleep_ms :: proc(ms: i32) {
Sleep :: proc(ms: i32) -> i32 #foreign
Sleep(ms);
}
OutputDebugStringA :: proc(c_str: ^u8) #foreign
RegisterClassExA :: proc(wc: ^WNDCLASSEXA) -> ATOM #foreign
CreateWindowExA :: proc(ex_style: u32,
class_name, title: ^u8,
style: u32,
x, y, w, h: i32,
parent: HWND, menu: HMENU, instance: HINSTANCE,
param: rawptr) -> HWND #foreign
ShowWindow :: proc(hwnd: HWND, cmd_show: i32) -> BOOL #foreign
UpdateWindow :: proc(hwnd: HWND) -> BOOL #foreign
PeekMessageA :: proc(msg: ^MSG, hwnd: HWND,
msg_filter_min, msg_filter_max, remove_msg: u32) -> BOOL #foreign
TranslateMessage :: proc(msg: ^MSG) -> BOOL #foreign
DispatchMessageA :: proc(msg: ^MSG) -> LRESULT #foreign
DefWindowProcA :: proc(hwnd: HWND, msg: u32, wparam: WPARAM, lparam: LPARAM) -> LRESULT #foreign
+6 -2
View File
@@ -3,9 +3,13 @@
rem call clang -c -emit-llvm -DGB_IMPLEMENTATION -DGB_DEF=GB_DLL_EXPORT ..\src\gb\gb.h
pushd ..\examples
call ..\bin\odin.exe ..\examples/main.odin ^
&& ..\misc\llvm-bin\opt.exe -mem2reg ..\examples/main.ll -o ..\examples/main.bc ^
&& ..\misc\llvm-bin\lli.exe ..\examples/main.bc
&& opt -mem2reg main.ll -o main.bc ^
&& clang main.bc -o main.exe ^
-Wno-override-module -lkernel32.lib -luser32.lib ^
&& main.exe
popd
rem && llvm-dis ..\examples/main.bc -o - ^
rem call ..\misc\llvm-bin\opt.exe -mem2reg ..\examples/output.ll > ..\examples/main.bc
+7 -2
View File
@@ -452,7 +452,8 @@ void add_type_and_value(CheckerInfo *i, AstNode *expression, AddressingMode mode
if (mode == Addressing_Constant) {
GB_ASSERT(value.kind != ExactValue_Invalid);
GB_ASSERT(type == t_invalid || is_type_constant_type(type));
GB_ASSERT_MSG(type != t_invalid || is_type_constant_type(type),
"type: %s", type_to_string(type));
}
TypeAndValue tv = {};
@@ -473,7 +474,11 @@ void add_entity(Checker *c, Scope *scope, AstNode *identifier, Entity *entity) {
if (!are_strings_equal(entity->token.string, make_string("_"))) {
Entity *insert_entity = scope_insert_entity(scope, entity);
if (insert_entity) {
error(&c->error_collector, entity->token, "Redeclared entity in this scope: %.*s", LIT(entity->token.string));
error(&c->error_collector, entity->token,
"Redeclararation of `%.*s` in this scope\n"
"\tat %.*s(%td:%td)",
LIT(entity->token.string),
LIT(entity->token.pos.file), entity->token.pos.line, entity->token.pos.column);
return;
}
}
+9 -9
View File
@@ -1807,7 +1807,7 @@ ExpressionKind check__expr_base(Checker *c, Operand *o, AstNode *node, Type *typ
case Token_Float: t = t_untyped_float; break;
case Token_String: t = t_untyped_string; break;
case Token_Rune: t = t_untyped_rune; break;
default: GB_PANIC("Unknown literal"); break;
default: GB_PANIC("Unknown literal"); break;
}
o->mode = Addressing_Constant;
o->type = t;
@@ -1815,16 +1815,16 @@ ExpressionKind check__expr_base(Checker *c, Operand *o, AstNode *node, Type *typ
case_end;
case_ast_node(pl, ProcLit, node);
auto curr_context = c->context;
c->context.scope = c->global_scope;
check_open_scope(c, pl->type);
c->context.decl = make_declaration_info(c->allocator, c->context.scope);
defer ({
check_close_scope(c);
c->context = curr_context;
});
Type *proc_type = check_type(c, pl->type);
if (proc_type != NULL) {
auto context = c->context;
c->context.scope = c->global_scope;
check_open_scope(c, pl->type);
c->context.decl = make_declaration_info(c->allocator, c->context.scope);
defer ({
c->context = context;
check_close_scope(c);
});
check_proc_body(c, empty_token, c->context.decl, proc_type, pl->body);
o->mode = Addressing_Value;
o->type = proc_type;
+2
View File
@@ -91,6 +91,8 @@ void ssa_gen_code(ssaGen *s) {
name = pd->foreign_name;
}
ssaValue *p = ssa_make_value_procedure(a, m, e->type, decl->type_expr, body, name);
p->proc.tags = pd->tags;
map_set(&m->values, hash_pointer(e), p);
map_set(&m->members, hash_string(name), p);
} break;
+53 -29
View File
@@ -174,21 +174,22 @@ void ssa_print_type(gbFile *f, BaseTypeSizes s, Type *t) {
ssa_fprintf(f, "}");
}
break;
case Type_Proc:
case Type_Proc: {
if (t->proc.result_count == 0) {
ssa_fprintf(f, "void");
} else {
ssa_print_type(f, s, t->proc.results);
}
ssa_fprintf(f, " (");
auto *params = &t->proc.params->tuple;
for (isize i = 0; i < t->proc.param_count; i++) {
if (i > 0) {
ssa_fprintf(f, ", ");
}
ssa_print_type(f, s, &t->proc.params[i]);
ssa_print_type(f, s, params->variables[i]->type);
}
ssa_fprintf(f, ")*");
break;
} break;
}
}
@@ -506,7 +507,7 @@ void ssa_print_instr(gbFile *f, ssaModule *m, ssaValue *value) {
case ssaInstr_Call: {
auto *call = &instr->call;
Type *result_type = call->type->proc.results;
Type *result_type = call->type;
if (result_type) {
ssa_fprintf(f, "%%%d = ", value->id);
}
@@ -521,18 +522,22 @@ 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++) {
Entity *e = params->variables[i];
GB_ASSERT(e != NULL);
Type *t = e->type;
if (i > 0) {
ssa_fprintf(f, ", ");
if (call->arg_count > 0) {
Type *proc_type = get_base_type(ssa_value_type(call->value));
GB_ASSERT(proc_type->kind == Type_Proc);
auto *params = &proc_type->proc.params->tuple;
for (isize i = 0; i < call->arg_count; i++) {
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_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");
@@ -650,9 +655,18 @@ void ssa_print_proc(gbFile *f, ssaModule *m, ssaProcedure *proc) {
ssa_fprintf(f, ") ");
if (proc->body == NULL) {
ssa_fprintf(f, "; foreign procedure\n\n");
} else {
if (proc->tags != 0) {
if (proc->tags & ProcTag_inline)
ssa_fprintf(f, "alwaysinline ");
if (proc->tags & ProcTag_no_inline)
ssa_fprintf(f, "noinline ");
if (proc->tags & ProcTag_foreign)
ssa_fprintf(f, "; foreign\n");
}
if (proc->body != NULL) {
ssa_fprintf(f, "{\n");
gb_for_array(i, proc->blocks) {
ssaBlock *block = proc->blocks[i];
@@ -682,6 +696,7 @@ void ssa_print_type_name(gbFile *f, ssaModule *m, ssaValue *v) {
ssa_fprintf(f, "\n");
}
void ssa_print_llvm_ir(gbFile *f, ssaModule *m) {
if (m->layout.len > 0) {
ssa_fprintf(f, "target datalayout = \"%.*s\"\n", LIT(m->layout));
@@ -695,17 +710,6 @@ void ssa_print_llvm_ir(gbFile *f, ssaModule *m) {
ssa_print_encoded_local(f, make_string(".rawptr"));
ssa_fprintf(f, " = type i8* ; Basic_rawptr\n\n");
ssa_fprintf(f, "declare void @llvm.memmove.p0i8.p0i8.");
ssa_print_type(f, m->sizes, t_int);
ssa_fprintf(f, "(i8*, i8*, ");
ssa_print_type(f, m->sizes, t_int);
ssa_fprintf(f, ", i32, i1) argmemonly nounwind \n\n");
gb_for_array(i, m->nested_type_names) {
ssaValue *v = m->nested_type_names[i];
ssa_print_type_name(f, m, v);
}
gb_for_array(member_index, m->members.entries) {
auto *entry = &m->members.entries[member_index];
ssaValue *v = entry->value;
@@ -716,7 +720,27 @@ void ssa_print_llvm_ir(gbFile *f, ssaModule *m) {
ssa_print_type(f, m->sizes, get_base_type(v->type_name.type));
ssa_fprintf(f, "\n");
} break;
}
}
gb_for_array(i, m->nested_type_names) {
ssaValue *v = m->nested_type_names[i];
ssa_print_type_name(f, m, v);
}
ssa_fprintf(f, "declare void @llvm.memmove.p0i8.p0i8.");
ssa_print_type(f, m->sizes, t_int);
ssa_fprintf(f, "(i8*, i8*, ");
ssa_print_type(f, m->sizes, t_int);
ssa_fprintf(f, ", i32, i1) argmemonly nounwind \n\n");
gb_for_array(member_index, m->members.entries) {
auto *entry = &m->members.entries[member_index];
ssaValue *v = entry->value;
switch (v->kind) {
case ssaValue_Global: {
auto *g = &v->global;
ssa_print_encoded_global(f, g->entity->token.string);
+17 -12
View File
@@ -46,6 +46,7 @@ struct ssaProcedure {
Type * type;
AstNode * type_expr;
AstNode * body;
u64 tags;
gbArray(ssaBlock *) blocks;
ssaBlock * curr_block;
@@ -318,13 +319,13 @@ Type *ssa_instr_type(ssaInstr *instr) {
case ssaInstr_Select:
return ssa_value_type(instr->select.true_value);
case ssaInstr_Call: {
Type *pt = instr->call.type;
GB_ASSERT(pt->kind == Type_Proc);
auto *tuple = &pt->proc.results->tuple;
if (tuple->variable_count != 1)
return pt->proc.results;
else
return tuple->variables[0]->type;
Type *pt = get_base_type(instr->call.type);
if (pt != NULL) {
if (pt->kind == Type_Tuple && pt->tuple.variable_count == 1)
return pt->tuple.variables[0]->type;
return pt;
}
return NULL;
}
case ssaInstr_CopyMemory:
return t_int;
@@ -891,7 +892,7 @@ void ssa_end_procedure_body(ssaProcedure *proc) {
case ssaInstr_CopyMemory:
continue;
case ssaInstr_Call:
if (instr->call.type->proc.results == NULL) {
if (instr->call.type == NULL) {
continue;
}
break;
@@ -1459,6 +1460,8 @@ ssaValue *ssa_build_single_expr(ssaProcedure *proc, AstNode *expr, TypeAndValue
ssaValue *value = ssa_make_value_procedure(proc->module->allocator,
proc->module, type, pl->type, pl->body, name);
value->proc.tags = pl->tags;
gb_array_append(proc->children, &value->proc);
ssa_build_proc(value, proc);
@@ -1693,8 +1696,7 @@ ssaValue *ssa_build_single_expr(ssaProcedure *proc, AstNode *expr, TypeAndValue
}
}
ssaValue *call = ssa_make_instr_call(proc, value, args, arg_count, tv->type);
ssa_value_set_type(call, proc_type_);
ssaValue *call = ssa_make_instr_call(proc, value, args, arg_count, type->results);
return ssa_emit(proc, call);
case_end;
@@ -2043,6 +2045,8 @@ void ssa_build_stmt(ssaProcedure *proc, AstNode *node) {
if (proc->children == NULL) {
gb_array_init(proc->children, gb_heap_allocator());
}
if (pd->body != NULL) {
// NOTE(bill): Generate a new name
// parent$name-guid
@@ -2059,6 +2063,8 @@ void ssa_build_stmt(ssaProcedure *proc, AstNode *node) {
ssaValue *value = ssa_make_value_procedure(proc->module->allocator,
proc->module, e->type, pd->type, pd->body, name);
value->proc.tags = pd->tags;
ssa_module_add_value(proc->module, e, value);
gb_array_append(proc->children, &value->proc);
ssa_build_proc(value, proc);
@@ -2273,7 +2279,7 @@ void ssa_build_stmt(ssaProcedure *proc, AstNode *node) {
proc->curr_block = init;
ssa_build_stmt(proc, fs->init);
}
ssaBlock *body = ssa__make_block(proc, node, make_string("for.body"));
ssaBlock *body = ssa_add_block(proc, node, make_string("for.body"));
ssaBlock *done = ssa__make_block(proc, node, make_string("for.done")); // NOTE(bill): Append later
ssaBlock *loop = body;
@@ -2289,7 +2295,6 @@ void ssa_build_stmt(ssaProcedure *proc, AstNode *node) {
proc->curr_block = loop;
if (loop != body) {
ssa_build_cond(proc, fs->cond, body, done);
gb_array_append(proc->blocks, body);
proc->curr_block = body;
}
+106 -95
View File
@@ -87,6 +87,7 @@ enum ProcTag {
AST_NODE_KIND(ProcLit, struct { \
AstNode *type; \
AstNode *body; \
u64 tags; \
}) \
AST_NODE_KIND(CompoundLit, struct { \
AstNode *type; \
@@ -559,23 +560,24 @@ gb_inline AstNode *make_basic_lit(AstFile *f, Token basic_lit) {
return result;
}
gb_inline AstNode *make_identifier(AstFile *f, Token token, AstEntity *entity = NULL) {
gb_inline AstNode *make_ident(AstFile *f, Token token, AstEntity *entity = NULL) {
AstNode *result = make_node(f, AstNode_Ident);
result->Ident.token = token;
result->Ident.entity = entity;
return result;
}
gb_inline AstNode *make_procedure_literal(AstFile *f, AstNode *type, AstNode *body) {
gb_inline AstNode *make_proc_lit(AstFile *f, AstNode *type, AstNode *body, u64 tags) {
AstNode *result = make_node(f, AstNode_ProcLit);
result->ProcLit.type = type;
result->ProcLit.body = body;
result->ProcLit.tags = tags;
return result;
}
gb_inline AstNode *make_compound_literal(AstFile *f, AstNode *type, AstNode *elem_list, isize elem_count,
Token open, Token close) {
gb_inline AstNode *make_compound_lit(AstFile *f, AstNode *type, AstNode *elem_list, isize elem_count,
Token open, Token close) {
AstNode *result = make_node(f, AstNode_CompoundLit);
result->CompoundLit.type = type;
result->CompoundLit.elem_list = elem_list;
@@ -845,16 +847,12 @@ gb_internal void add_ast_entity(AstFile *f, AstScope *scope, AstNode *declaratio
AstEntity *insert_entity = ast_scope_insert(scope, *entity);
if (insert_entity != NULL && !is_blank_ident(insert_entity->token.string)) {
ast_file_err(f, entity->token,
"There is already a previous declaration of `%.*s` in the current scope at\n"
"\t%.*s(%td:%td)",
"There is already a previous declaration of `%.*s` in the current scope\n"
"at \t%.*s(%td:%td)",
LIT(insert_entity->token.string),
LIT(insert_entity->token.pos.file),
insert_entity->token.pos.line,
insert_entity->token.pos.column);
gb_printf_err("Hashes\n");
gb_printf_err("%16llx - %.*s\n", hash_string(insert_entity->token.string), LIT(insert_entity->token.string));
gb_printf_err("%16llx - %.*s\n\n", hash_string(entity->token.string), LIT(entity->token.string));
}
}
}
@@ -911,7 +909,7 @@ AstNode *parse_identifier(AstFile *f) {
token.string = make_string("_");
expect_token(f, Token_Identifier);
}
return make_identifier(f, token);
return make_ident(f, token);
}
AstNode *parse_tag_expr(AstFile *f, AstNode *expression) {
@@ -971,7 +969,7 @@ AstNode *parse_literal_value(AstFile *f, AstNode *type) {
f->expr_level--;
Token close = expect_token(f, Token_CloseBrace);
return make_compound_literal(f, type, element_list, element_count, open, close);
return make_compound_lit(f, type, element_list, element_count, open, close);
}
AstNode *parse_value(AstFile *f) {
@@ -984,6 +982,93 @@ AstNode *parse_value(AstFile *f) {
AstNode *parse_identifier_or_type(AstFile *f);
void check_proc_add_tag(AstFile *f, AstNode *tag_expr, u64 *tags, ProcTag tag, String tag_name) {
if (*tags & tag) {
ast_file_err(f, ast_node_token(tag_expr), "Procedure tag already used: %.*s", LIT(tag_name));
}
*tags |= tag;
}
b32 is_foreign_name_valid(String name) {
// TODO(bill): is_foreign_name_valid
if (name.len == 0)
return false;
isize offset = 0;
while (offset < name.len) {
Rune rune;
isize remaining = name.len - offset;
isize width = gb_utf8_decode(name.text+offset, remaining, &rune);
if (rune == GB_RUNE_INVALID && width == 1) {
return false;
} else if (rune == GB_RUNE_BOM && remaining > 0) {
return false;
}
if (offset == 0) {
switch (rune) {
case '-':
case '$':
case '.':
case '_':
break;
default:
if (!rune_is_letter(rune))
return false;
break;
}
} else {
switch (rune) {
case '-':
case '$':
case '.':
case '_':
break;
default:
if (!rune_is_letter(rune) && !rune_is_digit(rune)) {
return false;
}
break;
}
}
offset += width;
}
return true;
}
void parse_proc_tags(AstFile *f, u64 *tags, String *foreign_name) {
// TODO(bill): Add this to procedure literals too
while (f->cursor[0].kind == Token_Hash) {
AstNode *tag_expr = parse_tag_expr(f, NULL);
ast_node(te, TagExpr, tag_expr);
String tag_name = te->name.string;
if (are_strings_equal(tag_name, make_string("foreign"))) {
check_proc_add_tag(f, tag_expr, tags, ProcTag_foreign, tag_name);
if (f->cursor[0].kind == Token_String) {
*foreign_name = f->cursor[0].string;
// TODO(bill): Check if valid string
if (!is_foreign_name_valid(*foreign_name)) {
ast_file_err(f, ast_node_token(tag_expr), "Invalid alternative foreign procedure name");
}
next_token(f);
}
} else if (are_strings_equal(tag_name, make_string("inline"))) {
check_proc_add_tag(f, tag_expr, tags, ProcTag_inline, tag_name);
} else if (are_strings_equal(tag_name, make_string("no_inline"))) {
check_proc_add_tag(f, tag_expr, tags, ProcTag_no_inline, tag_name);
} else {
ast_file_err(f, ast_node_token(tag_expr), "Unknown procedure tag");
}
}
if ((*tags & ProcTag_inline) && (*tags & ProcTag_no_inline)) {
ast_file_err(f, f->cursor[0], "You cannot apply both `inline` and `no_inline` to a procedure");
}
}
AstNode *parse_operand(AstFile *f, b32 lhs) {
AstNode *operand = NULL; // Operand
switch (f->cursor[0].kind) {
@@ -1024,6 +1109,13 @@ AstNode *parse_operand(AstFile *f, b32 lhs) {
AstScope *scope = NULL;
AstNode *type = parse_proc_type(f, &scope);
u64 tags = 0;
String foreign_name = {};
parse_proc_tags(f, &tags, &foreign_name);
if (tags & ProcTag_foreign) {
ast_file_err(f, f->cursor[0], "#foreign cannot be applied to procedure literals");
}
if (f->cursor[0].kind != Token_OpenBrace) {
return type;
} else {
@@ -1036,7 +1128,7 @@ AstNode *parse_operand(AstFile *f, b32 lhs) {
f->expr_level--;
f->curr_scope = curr_scope;
return make_procedure_literal(f, type, body);
return make_proc_lit(f, type, body, tags);
}
}
@@ -1605,60 +1697,7 @@ AstNode *parse_body(AstFile *f, AstScope *scope) {
return make_block_stmt(f, statement_list, statement_list_count, open, close);
}
b32 is_foreign_name_valid(String name) {
// TODO(bill): is_foreign_name_valid
if (name.len == 0)
return false;
isize offset = 0;
while (offset < name.len) {
Rune rune;
isize remaining = name.len - offset;
isize width = gb_utf8_decode(name.text+offset, remaining, &rune);
if (rune == GB_RUNE_INVALID && width == 1) {
return false;
} else if (rune == GB_RUNE_BOM && remaining > 0) {
return false;
}
if (offset == 0) {
switch (rune) {
case '-':
case '$':
case '.':
case '_':
break;
default:
if (!rune_is_letter(rune))
return false;
break;
}
} else {
switch (rune) {
case '-':
case '$':
case '.':
case '_':
break;
default:
if (!rune_is_letter(rune) && !rune_is_digit(rune)) {
return false;
}
break;
}
}
offset += width;
}
return true;
}
void check_proc_add_tag(AstFile *f, AstNode *tag_expr, u64 *tags, ProcTag tag, String tag_name) {
if (*tags & tag) {
ast_file_err(f, ast_node_token(tag_expr), "Procedure tag already used: %.*s", LIT(tag_name));
}
*tags |= tag;
}
AstNode *parse_proc_decl(AstFile *f, Token proc_token, AstNode *name) {
AstNode *param_list = NULL;
@@ -1673,36 +1712,8 @@ AstNode *parse_proc_decl(AstFile *f, Token proc_token, AstNode *name) {
AstNode *body = NULL;
u64 tags = 0;
String foreign_name = {};
while (f->cursor[0].kind == Token_Hash) {
AstNode *tag_expr = parse_tag_expr(f, NULL);
ast_node(te, TagExpr, tag_expr);
String tag_name = te->name.string;
if (are_strings_equal(tag_name, make_string("foreign"))) {
check_proc_add_tag(f, tag_expr, &tags, ProcTag_foreign, tag_name);
if (f->cursor[0].kind == Token_String) {
foreign_name = f->cursor[0].string;
// TODO(bill): Check if valid string
if (!is_foreign_name_valid(foreign_name)) {
ast_file_err(f, ast_node_token(tag_expr), "Invalid alternative foreign procedure name");
}
next_token(f);
}
} else if (are_strings_equal(tag_name, make_string("inline"))) {
check_proc_add_tag(f, tag_expr, &tags, ProcTag_inline, tag_name);
} else if (are_strings_equal(tag_name, make_string("no_inline"))) {
check_proc_add_tag(f, tag_expr, &tags, ProcTag_no_inline, tag_name);
} else {
ast_file_err(f, ast_node_token(tag_expr), "Unknown procedure tag");
}
}
b32 is_inline = (tags & ProcTag_inline) != 0;
b32 is_no_inline = (tags & ProcTag_no_inline) != 0;
if (is_inline && is_no_inline) {
ast_file_err(f, f->cursor[0], "You cannot apply both `inline` and `no_inline` to a procedure");
}
parse_proc_tags(f, &tags, &foreign_name);
if (f->cursor[0].kind == Token_OpenBrace) {
if ((tags & ProcTag_foreign) != 0) {
+1
View File
@@ -85,6 +85,7 @@ TOKEN_KIND(Token__KeywordBegin, "_KeywordBegin"), \
TOKEN_KIND(Token_continue, "continue"), \
TOKEN_KIND(Token_fallthrough, "fallthrough"), \
TOKEN_KIND(Token_case, "case"), \
TOKEN_KIND(Token_then, "then"), \
TOKEN_KIND(Token_if, "if"), \
TOKEN_KIND(Token_else, "else"), \
TOKEN_KIND(Token_for, "for"), \