| 123456789101112131415161718192021222324252627282930313233343536373839404142434445464748495051525354555657585960616263646566676869707172737475767778798081828384858687888990919293949596979899100101102103104105106107108109110111112113114115116117118119120121122123124125126127128129130131132133134135136137138139140141142143144145146147148149150151152153154155156157158159160161162163164165166167168169170171172173174175176177178179180181182183184185186187188189190191192193194195196197198199200201202203204205206207208209210211212213214215216217218219220221222223224225226227228229230231232233234235236237238239240241242243244245246247248249250251252253254255256257258259260261262263264265266267268269270271272273274275276277278279280281282283284285286287288289290291292293294295296297298299300301302303304305306307308309310311312313314315316317318319320321322323324325326327328329330331332333334335336337338339340341342343344345346347348349350351352353354355356357358359360361362363364365366367368369370371372373374375376377378379380381382383384385386387388389390391392393394395396397398399400401402403404405406407408409410411412 | #include "win32.odin"assume :: proc(cond: bool) #foreign "llvm.assume"__debug_trap           :: proc()           #foreign "llvm.debugtrap"__trap                 :: proc()           #foreign "llvm.trap"read_cycle_counter     :: proc() -> u64    #foreign "llvm.readcyclecounter"bit_reverse16 :: proc(b: u16) -> u16 #foreign "llvm.bitreverse.i16"bit_reverse32 :: proc(b: u32) -> u32 #foreign "llvm.bitreverse.i32"bit_reverse64 :: proc(b: u64) -> u64 #foreign "llvm.bitreverse.i64"byte_swap16 :: proc(b: u16) -> u16 #foreign "llvm.bswap.i16"byte_swap32 :: proc(b: u32) -> u32 #foreign "llvm.bswap.i32"byte_swap64 :: proc(b: u64) -> u64 #foreign "llvm.bswap.i64"fmuladd_f32 :: proc(a, b, c: f32) -> f32 #foreign "llvm.fmuladd.f32"fmuladd_f64 :: proc(a, b, c: f64) -> f64 #foreign "llvm.fmuladd.f64"// TODO(bill): make custom heap proceduresheap_alloc   :: proc(len: int)   -> rawptr #foreign "malloc"heap_dealloc :: proc(ptr: rawptr)          #foreign "free"memory_zero :: proc(data: rawptr, len: int) {	d := slice_ptr(data as ^byte, len)	for i := 0; i < len; i++ {		d[i] = 0	}}memory_compare :: proc(dst, src: rawptr, len: int) -> int {	s1, s2: ^byte = 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	#assert(align_of(v128b) == 16)	d, s: ^byte = 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: ^byte = 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_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_ne :: proc(a, b : string) -> bool #inline { return !__string_eq(a, b) }__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 }Allocation_Mode :: type enum {	ALLOC,	DEALLOC,	DEALLOC_ALL,	RESIZE,}Allocator_Proc :: type proc(allocator_data: rawptr, mode: Allocation_Mode,                            size, alignment: int,                            old_memory: rawptr, old_size: int, flags: u64) -> rawptrAllocator :: type struct {	procedure: Allocator_Proc;	data:      rawptr}Context :: type struct {	thread_ptr: rawptr	user_data:  rawptr	user_index: int	allocator: Allocator}#thread_local context: ContextDEFAULT_ALIGNMENT :: 2*size_of(int)__check_context :: proc() {	if context.allocator.procedure == null {		context.allocator = __default_allocator()	}	if context.thread_ptr == null {		// TODO(bill):		// context.thread_ptr = current_thread_pointer()	}}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, Allocation_Mode.ALLOC, size, alignment, null, 0, 0)}dealloc :: proc(ptr: rawptr) #inline {	__check_context()	a := context.allocator	_ = a.procedure(a.data, Allocation_Mode.DEALLOC, 0, 0, ptr, 0, 0)}dealloc_all :: proc(ptr: rawptr) #inline {	__check_context()	a := context.allocator	_ = a.procedure(a.data, Allocation_Mode.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, Allocation_Mode.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 {		return old_memory	}	new_memory := alloc_align(new_size, alignment)	if new_memory == null {		return null	}	memory_copy(new_memory, old_memory, min(old_size, new_size));	dealloc(old_memory)	return new_memory}__default_allocator_proc :: proc(allocator_data: rawptr, mode: Allocation_Mode,                                 size, alignment: int,                                 old_memory: rawptr, old_size: int, flags: u64) -> rawptr {	using Allocation_Mode	match mode {	case ALLOC:		return heap_alloc(size)	case RESIZE:		return default_resize_align(old_memory, old_size, size, alignment)	case DEALLOC:		heap_dealloc(old_memory)	case DEALLOC_ALL:		// NOTE(bill): Does nothing	}	return null}__default_allocator :: proc() -> Allocator {	return Allocator{		__default_allocator_proc,		null,	}}__assert :: proc(msg: string) {	file_write(file_get_standard(File_Standard.ERROR), msg as []byte)	// TODO(bill): Which is better?	// __trap()	__debug_trap()}
 |