Files
Odin/examples/runtime.odin
2016-08-30 00:04:14 +01:00

392 lines
9.2 KiB
Odin

#load "win32.odin"
debug_trap :: proc() #foreign "llvm.debugtrap"
// TODO(bill): make custom heap procedures
heap_alloc :: proc(len: int) -> rawptr {
return HeapAlloc(GetProcessHeap(), HEAP_ZERO_MEMORY, len);
}
heap_free :: proc(ptr: rawptr) {
_ = HeapFree(GetProcessHeap(), 0, ptr);
}
memory_compare :: proc(dst, src: rawptr, len: int) -> int {
s1, s2: ^u8 = dst, src;
for i := 0; i < len; i++ {
a := ptr_offset(s1, i)^;
b := ptr_offset(s2, i)^;
if a != b {
return (a - b) as int;
}
}
return 0;
}
memory_copy :: proc(dst, src: rawptr, n: int) #inline {
if dst == src {
return;
}
v128b :: type {4}u32;
static_assert(align_of(v128b) == 16);
d, s: ^u8 = dst, src;
for ; s as uint % 16 != 0 && n != 0; n-- {
d^ = s^;
d, s = ptr_offset(d, 1), ptr_offset(s, 1);
}
if d as uint % 16 == 0 {
for ; n >= 16; d, s, n = ptr_offset(d, 16), ptr_offset(s, 16), n-16 {
(d as ^v128b)^ = (s as ^v128b)^;
}
if n&8 != 0 {
(d as ^u64)^ = (s as ^u64)^;
d, s = ptr_offset(d, 8), ptr_offset(s, 8);
}
if n&4 != 0 {
(d as ^u32)^ = (s as ^u32)^;
d, s = ptr_offset(d, 4), ptr_offset(s, 4);
}
if n&2 != 0 {
(d as ^u16)^ = (s as ^u16)^;
d, s = ptr_offset(d, 2), ptr_offset(s, 2);
}
if n&1 != 0 {
d^ = s^;
d, s = ptr_offset(d, 1), ptr_offset(s, 1);
}
return;
}
// IMPORTANT NOTE(bill): Little endian only
LS :: proc(a, b: u32) -> u32 #inline { return a << b; }
RS :: proc(a, b: u32) -> u32 #inline { return a >> b; }
/* NOTE(bill): Big endian version
LS :: proc(a, b: u32) -> u32 #inline { return a >> b; }
RS :: proc(a, b: u32) -> u32 #inline { return a << b; }
*/
w, x: u32;
if d as uint % 4 == 1 {
w = (s as ^u32)^;
d^ = s^; d = ptr_offset(d, 1); s = ptr_offset(s, 1);
d^ = s^; d = ptr_offset(d, 1); s = ptr_offset(s, 1);
d^ = s^; d = ptr_offset(d, 1); s = ptr_offset(s, 1);
n -= 3;
for n > 16 {
d32 := d as ^u32;
s32 := ptr_offset(s, 1) as ^u32;
x = s32^; d32^ = LS(w, 24) | RS(x, 8);
d32, s32 = ptr_offset(d32, 1), ptr_offset(s32, 1);
w = s32^; d32^ = LS(x, 24) | RS(w, 8);
d32, s32 = ptr_offset(d32, 1), ptr_offset(s32, 1);
x = s32^; d32^ = LS(w, 24) | RS(x, 8);
d32, s32 = ptr_offset(d32, 1), ptr_offset(s32, 1);
w = s32^; d32^ = LS(x, 24) | RS(w, 8);
d32, s32 = ptr_offset(d32, 1), ptr_offset(s32, 1);
d, s, n = ptr_offset(d, 16), ptr_offset(s, 16), n-16;
}
} else if d as uint % 4 == 2 {
w = (s as ^u32)^;
d^ = s^; d = ptr_offset(d, 1); s = ptr_offset(s, 1);
d^ = s^; d = ptr_offset(d, 1); s = ptr_offset(s, 1);
n -= 2;
for n > 17 {
d32 := d as ^u32;
s32 := ptr_offset(s, 2) as ^u32;
x = s32^; d32^ = LS(w, 16) | RS(x, 16);
d32, s32 = ptr_offset(d32, 1), ptr_offset(s32, 1);
w = s32^; d32^ = LS(x, 16) | RS(w, 16);
d32, s32 = ptr_offset(d32, 1), ptr_offset(s32, 1);
x = s32^; d32^ = LS(w, 16) | RS(x, 16);
d32, s32 = ptr_offset(d32, 1), ptr_offset(s32, 1);
w = s32^; d32^ = LS(x, 16) | RS(w, 16);
d32, s32 = ptr_offset(d32, 1), ptr_offset(s32, 1);
d, s, n = ptr_offset(d, 16), ptr_offset(s, 16), n-16;
}
} else if d as uint % 4 == 3 {
w = (s as ^u32)^;
d^ = s^;
n -= 1;
for n > 18 {
d32 := d as ^u32;
s32 := ptr_offset(s, 3) as ^u32;
x = s32^; d32^ = LS(w, 8) | RS(x, 24);
d32, s32 = ptr_offset(d32, 1), ptr_offset(s32, 1);
w = s32^; d32^ = LS(x, 8) | RS(w, 24);
d32, s32 = ptr_offset(d32, 1), ptr_offset(s32, 1);
x = s32^; d32^ = LS(w, 8) | RS(x, 24);
d32, s32 = ptr_offset(d32, 1), ptr_offset(s32, 1);
w = s32^; d32^ = LS(x, 8) | RS(w, 24);
d32, s32 = ptr_offset(d32, 1), ptr_offset(s32, 1);
d, s, n = ptr_offset(d, 16), ptr_offset(s, 16), n-16;
}
}
if n&16 != 0 {
(d as ^v128b)^ = (s as ^v128b)^;
d, s = ptr_offset(d, 16), ptr_offset(s, 16);
}
if n&8 != 0 {
(d as ^u64)^ = (s as ^u64)^;
d, s = ptr_offset(d, 8), ptr_offset(s, 8);
}
if n&4 != 0 {
(d as ^u32)^ = (s as ^u32)^;
d, s = ptr_offset(d, 4), ptr_offset(s, 4);
}
if n&2 != 0 {
(d as ^u16)^ = (s as ^u16)^;
d, s = ptr_offset(d, 2), ptr_offset(s, 2);
}
if n&1 != 0 {
d^ = s^;
}
}
memory_move :: proc(dst, src: rawptr, n: int) #inline {
d, s: ^u8 = dst, src;
if d == s {
return;
}
if d >= ptr_offset(s, n) || ptr_offset(d, n) <= s {
memory_copy(d, s, n);
return;
}
// TODO(bill): Vectorize the shit out of this
if d < s {
if s as int % size_of(int) == d as int % size_of(int) {
for d as int % size_of(int) != 0 {
if n == 0 {
return;
}
n--;
d^ = s^;
d, s = ptr_offset(d, 1), ptr_offset(s, 1);
}
di, si := d as ^int, s as ^int;
for n >= size_of(int) {
di^ = si^;
di, si = ptr_offset(di, 1), ptr_offset(si, 1);
n -= size_of(int);
}
}
for ; n > 0; n-- {
d^ = s^;
d, s = ptr_offset(d, 1), ptr_offset(s, 1);
}
} else {
if s as int % size_of(int) == d as int % size_of(int) {
for ptr_offset(d, n) as int % size_of(int) != 0 {
if n == 0 {
return;
}
n--;
d^ = s^;
d, s = ptr_offset(d, 1), ptr_offset(s, 1);
}
for n >= size_of(int) {
n -= size_of(int);
di := ptr_offset(d, n) as ^int;
si := ptr_offset(s, n) as ^int;
di^ = si^;
}
for ; n > 0; n-- {
d^ = s^;
d, s = ptr_offset(d, 1), ptr_offset(s, 1);
}
}
for n > 0 {
n--;
dn := ptr_offset(d, n);
sn := ptr_offset(s, n);
dn^ = sn^;
}
}
}
__string_eq :: proc(a, b : string) -> bool {
if len(a) != len(b) {
return false;
}
if ^a[0] == ^b[0] {
return true;
}
return memory_compare(^a[0], ^b[0], len(a)) == 0;
}
__string_ne :: proc(a, b : string) -> bool #inline {
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 #inline { return __string_cmp(a, b) < 0; }
__string_gt :: proc(a, b : string) -> bool #inline { return __string_cmp(a, b) > 0; }
__string_le :: proc(a, b : string) -> bool #inline { return __string_cmp(a, b) <= 0; }
__string_ge :: proc(a, b : string) -> bool #inline { return __string_cmp(a, b) >= 0; }
AllocationMode :: type enum {
ALLOC,
DEALLOC,
DEALLOC_ALL,
RESIZE,
}
AllocatorProc :: type proc(allocator_data: rawptr, mode: AllocationMode,
size, alignment: int,
old_memory: rawptr, old_size: int, flags: u64) -> rawptr;
Allocator :: type struct {
procedure: AllocatorProc;
data: rawptr;
}
Context :: type struct {
thread_id: i32;
user_index: i32;
user_data: rawptr;
allocator: Allocator;
}
#thread_local context: Context;
DEFAULT_ALIGNMENT :: 2*size_of(int);
__check_context :: proc() {
static_assert(AllocationMode.ALLOC == 0);
static_assert(AllocationMode.DEALLOC == 1);
static_assert(AllocationMode.DEALLOC_ALL == 2);
static_assert(AllocationMode.RESIZE == 3);
if context.allocator.procedure == null {
context.allocator = __default_allocator();
}
}
alloc :: proc(size: int) -> rawptr #inline { return alloc_align(size, DEFAULT_ALIGNMENT); }
alloc_align :: proc(size, alignment: int) -> rawptr #inline {
__check_context();
a := context.allocator;
return a.procedure(a.data, AllocationMode.ALLOC, size, alignment, null, 0, 0);
}
dealloc :: proc(ptr: rawptr) #inline {
__check_context();
a := context.allocator;
_ = a.procedure(a.data, AllocationMode.DEALLOC, 0, 0, ptr, 0, 0);
}
dealloc_all :: proc(ptr: rawptr) #inline {
__check_context();
a := context.allocator;
_ = a.procedure(a.data, AllocationMode.DEALLOC_ALL, 0, 0, ptr, 0, 0);
}
resize :: proc(ptr: rawptr, old_size, new_size: int) -> rawptr #inline { return resize_align(ptr, old_size, new_size, DEFAULT_ALIGNMENT); }
resize_align :: proc(ptr: rawptr, old_size, new_size, alignment: int) -> rawptr #inline {
__check_context();
a := context.allocator;
return a.procedure(a.data, AllocationMode.RESIZE, new_size, alignment, ptr, old_size, 0);
}
default_resize_align :: proc(old_memory: rawptr, old_size, new_size, alignment: int) -> rawptr {
if old_memory == null {
return alloc_align(new_size, alignment);
}
if new_size == 0 {
dealloc(old_memory);
return null;
}
if new_size < old_size {
new_size = old_size;
}
if old_size == new_size {
return old_memory;
}
new_memory := alloc_align(new_size, alignment);
if new_memory == null {
return null;
}
_ = copy(slice_ptr(new_memory as ^u8, new_size), slice_ptr(old_memory as ^u8, old_size));
dealloc(old_memory);
return new_memory;
}
__default_allocator_proc :: proc(allocator_data: rawptr, mode: AllocationMode,
size, alignment: int,
old_memory: rawptr, old_size: int, flags: u64) -> rawptr {
if mode == AllocationMode.ALLOC {
return heap_alloc(size);
} else if mode == AllocationMode.RESIZE {
return default_resize_align(old_memory, old_size, size, alignment);
} else if mode == AllocationMode.DEALLOC {
heap_free(old_memory);
} else if mode == AllocationMode.DEALLOC_ALL {
// NOTE(bill): Does nothing
}
return null;
}
__default_allocator :: proc() -> Allocator {
return Allocator{
__default_allocator_proc,
null,
};
}