fibs work
This commit is contained in:
parent
46d84e91eb
commit
74cad99337
14
Makefile
14
Makefile
|
@ -1,11 +1,17 @@
|
|||
|
||||
all: uskel
|
||||
SYMSRCS=$(wildcard *.s)
|
||||
OBJS=$(SYMSRCS:.s=.o)
|
||||
PROGS=$(SYMSRCS:.s=)
|
||||
|
||||
.SUFFIXES:
|
||||
|
||||
all: $(PROGS)
|
||||
|
||||
clean:
|
||||
rm -f uskel.o uskel
|
||||
rm -f $(OBJS) $(PROGS)
|
||||
|
||||
uskel.o: uskel.s $(wildcard *.s)
|
||||
%.o: %.s $(wildcard include/*.s)
|
||||
as $< -o $@
|
||||
|
||||
uskel: uskel.o
|
||||
%: %.o
|
||||
ld $@.o -o $@
|
||||
|
|
48
exit_123.s
48
exit_123.s
|
@ -1,6 +1,10 @@
|
|||
|
||||
.include "include/uskel.s"
|
||||
|
||||
.include "include/data.s"
|
||||
|
||||
# || -> cont
|
||||
.func main
|
||||
.thunkcode main
|
||||
# push a new integer
|
||||
pushq $100
|
||||
pushq $INT_code
|
||||
|
@ -27,44 +31,6 @@
|
|||
mov %rsp, %rsi
|
||||
enter %r13
|
||||
|
||||
# exitcode -> | cont (unused, should be 0) |
|
||||
.func main_exit
|
||||
mov 0x8(%rsi), %rdi # result to syscall exitcode
|
||||
mov $0x3c, %rax # syscall 60
|
||||
syscall # exit %rdi
|
||||
|
||||
# | arg1 | arg2 | -> cont
|
||||
.func plus
|
||||
# push a thunk for finishing the plus
|
||||
push %rsi # cont
|
||||
push %rbp # ret (self)
|
||||
mov 0x18(%rbp), %rax
|
||||
push %rax # arg2
|
||||
pushq $3
|
||||
pushq $plus_step1
|
||||
|
||||
mov %rsp, %rsi # continue to the new thunk
|
||||
enter 0x10(%rbp) # evaluate arg1
|
||||
|
||||
# arg1 -> | arg2 | ret | cont |
|
||||
.func plus_step1
|
||||
# this is guaranteed to be entered only once (it's a cont), so we can rewrite the thunk in place
|
||||
mov 0x10(%rbp), %rax
|
||||
movq $plus_fini, 0x00(%rbp)
|
||||
mov %rsi, 0x10(%rbp)
|
||||
|
||||
mov %rbp, %rsi # continue on the rewritten thunk
|
||||
enter %rax # evaluate arg1
|
||||
|
||||
# arg2 -> | arg1 | ret | cont |
|
||||
.func plus_fini
|
||||
mov 0x8(%rsi), %rax # arg 2
|
||||
mov 0x10(%rbp), %rsi # location of arg1
|
||||
add 0x8(%rsi), %rax # arg 1
|
||||
|
||||
mov 0x18(%rbp), %rsi # save result to the original plus thunk
|
||||
mov %rax, 0x08(%rsi)
|
||||
movq $INT_code, 0x00(%rsi)
|
||||
|
||||
enter 0x20(%rbp) # eval cont, returning %rsi
|
||||
.include "include/main_exit.s"
|
||||
|
||||
.include "include/intops.s"
|
||||
|
|
81
fibs.s
Normal file
81
fibs.s
Normal file
|
@ -0,0 +1,81 @@
|
|||
|
||||
.include "include/uskel.s"
|
||||
|
||||
.include "include/listops.s"
|
||||
.include "include/intops.s"
|
||||
.include "include/io.s"
|
||||
.include "include/main_exit.s"
|
||||
|
||||
# | lag1 | lag2 | -> cont
|
||||
.thunkcode fibs
|
||||
# next value
|
||||
pushq 030(%rbp)
|
||||
pushq 020(%rbp)
|
||||
pushq $2
|
||||
pushq $plus
|
||||
mov %rsp, %r11
|
||||
|
||||
# fib call with the next value
|
||||
push %r11
|
||||
pushq 030(%rbp)
|
||||
pushq $2
|
||||
pushq $fibs
|
||||
mov %rsp, %r12
|
||||
|
||||
# cons list with lag1
|
||||
push %r12
|
||||
push 020(%rbp)
|
||||
pushq $1
|
||||
pushq $LIST_code
|
||||
mov %rsp, %r13
|
||||
|
||||
# replace self with IND
|
||||
mov %r13, 010(%rbp)
|
||||
movq $IND_code, 0(%rbp)
|
||||
|
||||
mov %rsi, %rbp
|
||||
mov %r13, %rsi
|
||||
enter_rbp
|
||||
|
||||
# || -> cont
|
||||
.thunkcode fibs0
|
||||
pushq $1
|
||||
pushq $INT_code
|
||||
mov %rsp, %r12
|
||||
pushq $0
|
||||
pushq $INT_code
|
||||
mov %rsp, %r11
|
||||
|
||||
push %r12
|
||||
push %r11
|
||||
pushq $2
|
||||
pushq $fibs
|
||||
|
||||
enter %rsp
|
||||
|
||||
.thunkcode main
|
||||
pushq $0
|
||||
pushq $fibs0
|
||||
mov %rsp, %r12
|
||||
|
||||
pushq $20
|
||||
pushq $INT_code
|
||||
mov %rsp, %r11
|
||||
|
||||
push %r12
|
||||
push %r11
|
||||
pushq $2
|
||||
pushq $list_int_index
|
||||
mov %rsp, %r11
|
||||
|
||||
push %r11
|
||||
pushq $1
|
||||
pushq $print
|
||||
mov %rsp, %r11
|
||||
|
||||
push %rsi
|
||||
pushq $1
|
||||
pushq $main_exit
|
||||
|
||||
mov %rsp, %rsi
|
||||
enter %r11
|
99
include/data.s
Normal file
99
include/data.s
Normal file
|
@ -0,0 +1,99 @@
|
|||
|
||||
.ifndef _data_s_file
|
||||
_data_s_file:
|
||||
|
||||
# Simple values and boxed machine integers
|
||||
# | ptr | value |
|
||||
CON_evacuate1:
|
||||
retq
|
||||
CON_scavenge1:
|
||||
add $0x10, %rsi
|
||||
retq
|
||||
INT_info_table:
|
||||
cell CON_evacuate1
|
||||
cell CON_scavenge1
|
||||
cell 0
|
||||
INT_code:
|
||||
continue
|
||||
|
||||
# List
|
||||
# | ptr | 0 |
|
||||
# | ptr | 1 | a | b |
|
||||
LIST_evacuate:
|
||||
# [] | a : b
|
||||
retq #TODO
|
||||
LIST_scavenge:
|
||||
mov 0x8(%rbp), %rax
|
||||
shl $1, %rax
|
||||
add $2, %rax
|
||||
shl $3, %rax
|
||||
add %rax, %rsi
|
||||
retq
|
||||
LIST_info_table:
|
||||
cell LIST_evacuate
|
||||
cell LIST_scavenge
|
||||
cell 0
|
||||
LIST_code:
|
||||
continue
|
||||
|
||||
# FUN/PAP combo objects
|
||||
# | ptr | thunkptr | args | arg[0] | arg[1] | ... | arg[args] |
|
||||
FUN_evacuate:
|
||||
retq #TODO
|
||||
FUN_scavenge:
|
||||
mov 0x10(%rbp), %rax
|
||||
add $3, %rax
|
||||
shl $3, %rax
|
||||
add %rax, %rsi
|
||||
retq
|
||||
|
||||
# Simple info for n-ary functions
|
||||
# TODO continue to add as required
|
||||
fun1_info_table:
|
||||
cell FUN_evacuate
|
||||
cell FUN_scavenge
|
||||
cell 1
|
||||
fun1_code:
|
||||
continue
|
||||
|
||||
fun2_info_table:
|
||||
cell FUN_evacuate
|
||||
cell FUN_scavenge
|
||||
cell 2
|
||||
fun2_code:
|
||||
continue
|
||||
|
||||
fun3_info_table:
|
||||
cell FUN_evacuate
|
||||
cell FUN_scavenge
|
||||
cell 3
|
||||
fun3_code:
|
||||
continue
|
||||
|
||||
# indirection (Q: how to recognize IND and THUNK on return?)
|
||||
# | ptr | indptr |
|
||||
IND_evacuate:
|
||||
retq #TODO
|
||||
IND_scavenge:
|
||||
add $0x10,%rsi
|
||||
retq
|
||||
IND_info:
|
||||
cell IND_evacuate
|
||||
cell IND_scavenge
|
||||
cell 0
|
||||
IND_code:
|
||||
enter 0x8(%rbp)
|
||||
|
||||
# THU objects (gc implementation only, actual THUs are defined by functions)
|
||||
# | ptr | args | arg[0] | arg[1] | ... | arg[args] |
|
||||
# args wouldn't need to be here but let's keep them for gc simplicity
|
||||
THU_evacuate:
|
||||
retq #TODO
|
||||
THU_scavenge:
|
||||
mov 0x8(%rbp), %rax
|
||||
add $2,%rax
|
||||
shl $3,%rax
|
||||
add %rax,%rsi
|
||||
retq
|
||||
|
||||
.endif # _data_s_file
|
25
include/intops.s
Normal file
25
include/intops.s
Normal file
|
@ -0,0 +1,25 @@
|
|||
|
||||
.ifndef _intops_s_file
|
||||
_intops_s_file:
|
||||
|
||||
.include "include/primops.s"
|
||||
|
||||
.primop2 plus
|
||||
mov 010(%rsi), %rax # arg 2
|
||||
mov 020(%rbp), %rsi # location of arg1
|
||||
add 010(%rsi), %rax # arg 1
|
||||
primop2_ret_int %rax
|
||||
|
||||
.primop2 mul
|
||||
mov 010(%rsi), %rax # arg 2
|
||||
mov 020(%rbp), %rsi # location of arg1
|
||||
mulq 010(%rsi) # arg 1 (goes to %rax and %rdx)
|
||||
primop2_ret_int %rax
|
||||
|
||||
.primop2 sub
|
||||
mov 020(%rbp), %rdi # location of arg1
|
||||
mov 010(%rdx), %rax # arg 1
|
||||
sub 010(%rsi), %rax # arg 2
|
||||
primop2_ret_int %rax
|
||||
|
||||
.endif # _intops_s_file
|
44
include/io.s
Normal file
44
include/io.s
Normal file
|
@ -0,0 +1,44 @@
|
|||
|
||||
.ifndef _io_s_file
|
||||
_io_s_file:
|
||||
|
||||
# | int | -> cont
|
||||
.thunkcode print
|
||||
push %rsi
|
||||
push %rbp
|
||||
pushq $2
|
||||
pushq $print_fini
|
||||
|
||||
mov %rsp, %rsi
|
||||
enter 020(%rbp)
|
||||
|
||||
# arg -> | ret | cont |
|
||||
.thunkcode print_fini
|
||||
mov 010(%rsi), %rax
|
||||
|
||||
# make a string
|
||||
mov %rsp, %r15
|
||||
sub $1, %r15
|
||||
movb $0x0a, (%r15)
|
||||
print_fini_next:
|
||||
mov %al, %bl
|
||||
and $1, %bl
|
||||
add $0x30, %bl
|
||||
sub $1, %r15
|
||||
movb %bl, (%r15)
|
||||
shr $1, %rax
|
||||
jnz print_fini_next
|
||||
|
||||
mov $0, %rdi #stdin
|
||||
mov %rsp, %rdx
|
||||
sub %r15, %rdx #size
|
||||
mov %r15, %rsi #buf
|
||||
mov $1, %rax #write
|
||||
syscall
|
||||
|
||||
mov 020(%rbp), %rsi
|
||||
movq $0, 010(%rsi)
|
||||
movq $INT_code, 0(%rsi)
|
||||
enter 030(%rbp)
|
||||
|
||||
.endif # _io_s_file
|
40
include/listops.s
Normal file
40
include/listops.s
Normal file
|
@ -0,0 +1,40 @@
|
|||
|
||||
.ifndef _listops_s_file
|
||||
_listops_s_file:
|
||||
|
||||
.include "include/primops.s"
|
||||
|
||||
# | n | list | -> cont
|
||||
.primop2 list_int_index
|
||||
mov 010(%rsi), %rdx # the list constructor id, must be 1
|
||||
cmp $1, %rdx
|
||||
jne list_int_index_not_found
|
||||
|
||||
mov 020(%rbp), %rcx
|
||||
mov 010(%rcx), %rcx
|
||||
test %rcx, %rcx
|
||||
jz list_int_index_found #we are taking 0, all happy, return it
|
||||
|
||||
#more probably we need to continue, make replacement thunks
|
||||
sub $1, %rcx
|
||||
pushq %rcx
|
||||
pushq $INT_code
|
||||
mov %rsp, %r11
|
||||
|
||||
pushq 030(%rsi) # tail
|
||||
push %r11
|
||||
pushq $2
|
||||
pushq $list_int_index
|
||||
mov %rsp, %r11
|
||||
|
||||
primop2_cont_indirect %r11
|
||||
|
||||
list_int_index_not_found:
|
||||
movq 0, %rax #fault
|
||||
|
||||
list_int_index_found:
|
||||
mov 020(%rsi), %rax #head
|
||||
primop2_cont_indirect %rax
|
||||
|
||||
|
||||
.endif # _listops_s_file
|
|
@ -1,7 +1,14 @@
|
|||
|
||||
.ifndef _macros_s_file
|
||||
_macros_s_file:
|
||||
|
||||
.macro enter_rbp
|
||||
jmp *(%rbp)
|
||||
.endm
|
||||
|
||||
.macro enter x
|
||||
mov \x, %rbp
|
||||
jmp *(%rbp)
|
||||
enter_rbp
|
||||
.endm
|
||||
|
||||
.macro cell x
|
||||
|
@ -23,11 +30,14 @@
|
|||
jmp *(%rbp)
|
||||
.endm
|
||||
|
||||
.macro .makethunk name
|
||||
# this needs data.s
|
||||
.macro .thunkcode name
|
||||
.align 8
|
||||
\name\()_info:
|
||||
__\name\()_info:
|
||||
cell THU_evacuate
|
||||
cell THU_scavenge
|
||||
cell 0
|
||||
\name:
|
||||
.endm
|
||||
|
||||
.endif # _macros_s_file
|
10
include/main_exit.s
Normal file
10
include/main_exit.s
Normal file
|
@ -0,0 +1,10 @@
|
|||
.ifndef _main_exit_s_file
|
||||
_main_exit_s_file:
|
||||
|
||||
# exitcode -> | cont (unused, should be 0) |
|
||||
.thunkcode main_exit
|
||||
mov 0x8(%rsi), %rdi # result to syscall exitcode
|
||||
mov $0x3c, %rax # syscall 60
|
||||
syscall # exit %rdi
|
||||
|
||||
.endif # _main_exit_s_file
|
54
include/primops.s
Normal file
54
include/primops.s
Normal file
|
@ -0,0 +1,54 @@
|
|||
|
||||
.ifndef _primops_s_file
|
||||
_primops_s_file:
|
||||
|
||||
.include "include/data.s"
|
||||
|
||||
.macro .primop2 name
|
||||
# | arg1 | arg2 | -> cont
|
||||
.thunkcode \name
|
||||
# push a thunk for finishing the plus
|
||||
push %rsi # cont
|
||||
push %rbp # ret (self)
|
||||
pushq 030(%rbp)
|
||||
pushq $3
|
||||
pushq $\name\()_step1
|
||||
|
||||
mov %rsp, %rsi # continue to the new thunk
|
||||
enter 0x10(%rbp) # evaluate arg1
|
||||
|
||||
# arg1 -> | arg2 | ret | cont |
|
||||
.thunkcode \name\()_step1
|
||||
# this is guaranteed to be entered only once (it's a cont), so we can rewrite the thunk in place
|
||||
mov 020(%rbp), %rax
|
||||
movq $\name\()_fini, 0(%rbp)
|
||||
mov %rsi, 020(%rbp)
|
||||
|
||||
mov %rbp, %rsi # continue on the rewritten thunk
|
||||
enter %rax # evaluate arg1
|
||||
|
||||
# arg2 -> | arg1 | ret | cont |
|
||||
.thunkcode \name\()_fini
|
||||
# at this point,
|
||||
# arg1 is pointed to by 020(%rbp)
|
||||
# arg2 is pointed to by %rsi
|
||||
.endm
|
||||
|
||||
.macro primop2_ret_int val
|
||||
# the result should now be in %rax
|
||||
mov 030(%rbp), %rsi # save result to the original plus thunk
|
||||
movq \val, 010(%rsi)
|
||||
movq $INT_code, 0(%rsi)
|
||||
|
||||
enter 040(%rbp) # eval cont, returning %rsi
|
||||
.endm
|
||||
|
||||
.macro primop2_cont_indirect new
|
||||
mov 030(%rbp), %rdi # load the original thunk
|
||||
mov 040(%rbp), %rsi # set the continuation
|
||||
movq \new, 010(%rdi) # set the indirect to the new thunk
|
||||
movq $IND_code, 0(%rdi)
|
||||
enter \new # continue evaluating the new thunk
|
||||
.endm
|
||||
|
||||
.endif # _primops_s_file
|
54
include/uskel.s
Normal file
54
include/uskel.s
Normal file
|
@ -0,0 +1,54 @@
|
|||
|
||||
# uskel runtime and start; include this at the top.
|
||||
|
||||
.section .init
|
||||
.global _start
|
||||
_start:
|
||||
jmp _uskel_start
|
||||
|
||||
.include "include/macros.s"
|
||||
|
||||
.section .bss
|
||||
_memory_state:
|
||||
cell 0 # bottom of allocation (grows down)
|
||||
cell 0 # region start
|
||||
cell 0 # region end
|
||||
cell 0 # program entry rsp (aka the actual stack)
|
||||
|
||||
.section .text
|
||||
|
||||
_uskel_alloc_basic_mem:
|
||||
mov $0x100000, %r15 # desired size
|
||||
|
||||
mov $0x9, %rax # mmap
|
||||
mov $0, %rdi # addr = NULL
|
||||
mov %r15, %rsi # len = %rcx
|
||||
mov $0x3, %rdx # prot = PROT_READ 0x1 | PROT_WRITE 0x2
|
||||
mov $0x22, %r10 # flags = MAP_PRIVATE 0x2 | MAP_ANONYMOUS 0x20
|
||||
mov $-1, %r8 # fd = -1
|
||||
mov $0, %r9 # off = 0
|
||||
syscall
|
||||
mov $_memory_state, %rdi
|
||||
mov %rax, 010(%rdi)
|
||||
add %r15, %rax
|
||||
mov %rax, (%rdi)
|
||||
mov %rax, 020(%rdi)
|
||||
retq
|
||||
|
||||
_uskel_start:
|
||||
call _uskel_alloc_basic_mem
|
||||
# use the stack pointer for easy writing to the heap,
|
||||
# but back it up to memory state
|
||||
mov $_memory_state, %rdi
|
||||
mov %rsp, 030(%rdi)
|
||||
mov 0(%rdi), %rsp
|
||||
|
||||
# push a thunk for main
|
||||
pushq $0
|
||||
pushq $main
|
||||
|
||||
mov $0, %rsi # set continuation to exit
|
||||
enter %rsp # run the program
|
||||
# Q: are there gonna be functions that have both the argument AND the cont?
|
||||
# A: No, stuff is either entered as return-continuation (takes res, cont has to be saved) or as forward call (takes cont)
|
||||
# (needs validation)
|
42
print_123.s
Normal file
42
print_123.s
Normal file
|
@ -0,0 +1,42 @@
|
|||
|
||||
.include "include/uskel.s"
|
||||
|
||||
.include "include/data.s"
|
||||
.include "include/io.s"
|
||||
.include "include/intops.s"
|
||||
|
||||
# || -> cont
|
||||
.thunkcode main
|
||||
# push a new integer
|
||||
pushq $100
|
||||
pushq $INT_code
|
||||
mov %rsp, %r11 # backup first arg
|
||||
|
||||
# push another new integer
|
||||
pushq $23
|
||||
pushq $INT_code
|
||||
mov %rsp, %r12 # backup second arg
|
||||
|
||||
# push the plus
|
||||
push %r12
|
||||
push %r11
|
||||
pushq $2
|
||||
pushq $plus
|
||||
mov %rsp, %r11 # backup plus
|
||||
|
||||
# push the print
|
||||
push %r11
|
||||
pushq $1
|
||||
pushq $print
|
||||
mov %rsp, %r11 # backup print
|
||||
|
||||
# push a cont thunk for main_exit
|
||||
push %rsi
|
||||
pushq $1
|
||||
pushq $main_exit
|
||||
|
||||
# evaluate into main_exit
|
||||
mov %rsp, %rsi
|
||||
enter %r11
|
||||
|
||||
.include "include/main_exit.s"
|
232
uskel.s
232
uskel.s
|
@ -1,232 +0,0 @@
|
|||
|
||||
.include "macros.s"
|
||||
|
||||
.section .init
|
||||
.global _start
|
||||
_start:
|
||||
jmp _uskel_start
|
||||
|
||||
.section .bss
|
||||
_memory_state:
|
||||
cell 0 # bottom of allocation (grows down)
|
||||
cell 0 # region start
|
||||
cell 0 # region end
|
||||
|
||||
.section .text
|
||||
|
||||
_uskel_alloc_basic_mem:
|
||||
mov $0x100000, %r15 # desired size
|
||||
|
||||
mov $0x9, %rax # mmap
|
||||
mov $0, %rdi # addr = NULL
|
||||
mov %r15, %rsi # len = %rcx
|
||||
mov $0x3, %rdx # prot = PROT_READ 0x1 | PROT_WRITE 0x2
|
||||
mov $0x22, %r10 # flags = MAP_PRIVATE 0x2 | MAP_ANONYMOUS 0x20
|
||||
mov $-1, %r8 # fd = -1
|
||||
mov $0, %r9 # off = 0
|
||||
syscall
|
||||
mov $_memory_state, %rdi
|
||||
mov %rax, (%rdi)
|
||||
mov %rax, 0x8(%rdi)
|
||||
add %r15, %rax
|
||||
mov %rax, 0x10(%rdi)
|
||||
retq
|
||||
|
||||
_uskel_start:
|
||||
call _uskel_alloc_basic_mem
|
||||
mov _memory_state, %rdi
|
||||
# push a thunk for main
|
||||
mov %rdi, %r15 # backup main for later
|
||||
movq $main, 0x00(%rdi)
|
||||
movq $0, 0x08(%rdi)
|
||||
add $0x10, %rdi
|
||||
# save the memory ptr
|
||||
mov %rdi, _memory_state
|
||||
|
||||
mov $0, %rsi # set continuation to exit
|
||||
enter %r15 # run the program
|
||||
|
||||
# Simple values and boxed machine integers
|
||||
# | ptr | value |
|
||||
CON_evacuate1:
|
||||
retq
|
||||
CON_scavenge1:
|
||||
add $0x10, %rsi
|
||||
retq
|
||||
INT_info_table:
|
||||
cell CON_evacuate1
|
||||
cell CON_scavenge1
|
||||
cell 0
|
||||
INT_code:
|
||||
continue
|
||||
|
||||
# List
|
||||
# | ptr | 0 |
|
||||
# | ptr | 1 | a | b |
|
||||
LIST_evacuate:
|
||||
# [] | a : b
|
||||
retq #TODO
|
||||
LIST_scavenge:
|
||||
mov 0x8(%rbp), %rax
|
||||
shl $1, %rax
|
||||
add $2, %rax
|
||||
shl $3, %rax
|
||||
add %rax, %rsi
|
||||
retq
|
||||
LIST_info_table:
|
||||
cell LIST_evacuate
|
||||
cell LIST_scavenge
|
||||
cell 0
|
||||
LIST_code:
|
||||
continue
|
||||
|
||||
# FUN/PAP combo objects
|
||||
# | ptr | thunkptr | args | arg[0] | arg[1] | ... | arg[args] |
|
||||
FUN_evacuate:
|
||||
retq #TODO
|
||||
FUN_scavenge:
|
||||
mov 0x10(%rbp), %rax
|
||||
add $3, %rax
|
||||
shl $3, %rax
|
||||
add %rax, %rsi
|
||||
retq
|
||||
|
||||
# Simple info for n-ary functions
|
||||
# TODO continue to add as required
|
||||
fun1_info_table:
|
||||
cell FUN_evacuate
|
||||
cell FUN_scavenge
|
||||
cell 1
|
||||
fun1_code:
|
||||
continue
|
||||
|
||||
fun2_info_table:
|
||||
cell FUN_evacuate
|
||||
cell FUN_scavenge
|
||||
cell 2
|
||||
fun2_code:
|
||||
continue
|
||||
|
||||
fun3_info_table:
|
||||
cell FUN_evacuate
|
||||
cell FUN_scavenge
|
||||
cell 3
|
||||
fun3_code:
|
||||
continue
|
||||
|
||||
# indirection (Q: how to recognize IND and THUNK on return?)
|
||||
# | ptr | indptr |
|
||||
IND_evacuate:
|
||||
retq #TODO
|
||||
IND_scavenge:
|
||||
add $0x10,%rsi
|
||||
retq
|
||||
IND_info:
|
||||
cell IND_evacuate
|
||||
cell IND_scavenge
|
||||
cell 0
|
||||
IND_code:
|
||||
enter 0x8(%rbp)
|
||||
|
||||
# THU objects (gc implementation only, actual THUs are defined by functions)
|
||||
# | ptr | args | arg[0] | arg[1] | ... | arg[args] |
|
||||
# args wouldn't need to be here but let's keep them for gc simplicity
|
||||
THU_evacuate:
|
||||
retq #TODO
|
||||
THU_scavenge:
|
||||
mov 0x8(%rbp), %rax
|
||||
add $2,%rax
|
||||
shl $3,%rax
|
||||
add %rax,%rsi
|
||||
retq
|
||||
|
||||
#
|
||||
# Actual code!
|
||||
#
|
||||
|
||||
# || -> cont
|
||||
.makethunk main
|
||||
mov _memory_state, %r15
|
||||
|
||||
# push a new integer
|
||||
mov %r15, %r11 # backup first arg
|
||||
movq $INT_code, 0x00(%r15)
|
||||
movq $100, 0x08(%r15)
|
||||
add $0x10, %r15
|
||||
|
||||
# push another new integer
|
||||
mov %r15, %r12 # backup second arg
|
||||
movq $INT_code, 0x00(%r15)
|
||||
movq $23, 0x08(%r15)
|
||||
add $0x10, %r15
|
||||
|
||||
# push the plus
|
||||
mov %r15, %r13 # backup plus
|
||||
movq $plus, 0x00(%r15)
|
||||
movq $2, 0x08(%r15)
|
||||
mov %r11, 0x10(%r15)
|
||||
mov %r12, 0x18(%r15)
|
||||
add $0x20, %r15
|
||||
|
||||
# push a cont thunk for main_exit
|
||||
mov %r15, %r14 # backup cont thunk
|
||||
movq $main_exit, 0x00(%r15)
|
||||
movq $1, 0x08(%r15)
|
||||
mov %rsi, 0x10(%r15)
|
||||
add $0x18, %r15
|
||||
|
||||
mov %r15, _memory_state
|
||||
|
||||
# evaluate into main_exit
|
||||
mov %r14, %rsi
|
||||
enter %r13
|
||||
|
||||
# exitcode -> | cont (unused, should be 0) |
|
||||
.makethunk main_exit
|
||||
mov 0x8(%rsi), %rdi
|
||||
mov $0x3c, %rax
|
||||
syscall # exit %rdi
|
||||
|
||||
# | arg1 | arg2 | -> cont
|
||||
.makethunk plus
|
||||
# push a thunk for finishing the plus
|
||||
mov _memory_state, %r15
|
||||
mov %r15, %r14 # plus_step1 origin
|
||||
movq $plus_step1, 0x00(%r15)
|
||||
movq $3, 0x08(%r15)
|
||||
mov 0x18(%rbp), %rax
|
||||
mov %rax, 0x10(%r15)
|
||||
mov %rbp, 0x18(%r15)
|
||||
mov %rsi, 0x20(%r15)
|
||||
add $0x28, %r15
|
||||
mov %r15, _memory_state
|
||||
|
||||
# evaluate arg0
|
||||
mov %r14, %rsi
|
||||
enter 0x10(%rbp)
|
||||
|
||||
# arg0 -> | arg1 | ret | cont |
|
||||
.makethunk plus_step1
|
||||
# this is guaranteed to be entered only once (it's a cont), so we can rewrite the thunk in place
|
||||
mov 0x10(%rbp), %rax
|
||||
movq $plus_fini, 0x00(%rbp)
|
||||
mov %rsi, 0x10(%rbp)
|
||||
|
||||
mov %rbp, %rsi # continue on the rewritten thunk
|
||||
enter %rax # evaluate arg1
|
||||
|
||||
# arg1 -> | arg0 | ret | cont |
|
||||
.makethunk plus_fini
|
||||
mov 0x8(%rsi), %rax # arg1
|
||||
mov 0x10(%rbp), %rsi
|
||||
add 0x8(%rsi), %rax # + arg0
|
||||
|
||||
mov 0x18(%rbp), %rsi # rewrite the resulting thunk
|
||||
movq $INT_code, 0x00(%rsi)
|
||||
mov %rax, 0x08(%rsi)
|
||||
# result is in rsi already
|
||||
enter 0x20(%rbp)
|
||||
|
||||
# Q: are there gonna be functions that have both the argument AND the cont?
|
||||
# A: No, either stuff is entered as return-continuation (takes res) or as forward call (takes cont)
|
||||
# (needs validation)
|
Loading…
Reference in a new issue