257 lines
5.2 KiB
ArmAsm
257 lines
5.2 KiB
ArmAsm
|
|
.ifndef _data_s_file
|
|
_data_s_file:
|
|
nop # avoid confusing gdb
|
|
|
|
# Format of the info tables:
|
|
# - code
|
|
# ----- code pointer
|
|
# - 8B helper information for eval/apply (generally this is 0, and only gets used for FUN/PAP)
|
|
# - 8B pointer to scavenge
|
|
# - 8B pointer to evacuate
|
|
#
|
|
# Evacuate interface:
|
|
# in: %rbp what to evacuate
|
|
# out: %rbp where the thing is now
|
|
# ret: jump to _gc_evacuate_ret
|
|
# Notes:
|
|
# - IND thunks skip themselves on evacuate
|
|
# - checking if whether stuff is already in write region are managed by
|
|
# _gc_evacuate prelude
|
|
#
|
|
# Scavenge interface:
|
|
# in: %rbp what to scavenge
|
|
# out: %rbp next thing to scavenge in memory
|
|
# ret: jump to _gc_scavenge_ret
|
|
#
|
|
# Saved registers during the GC process:
|
|
# - _uskel_gc uses %r8-%r11
|
|
# - scavenges use %r12-%r15
|
|
# - %rax-%rdx is scratch and evacuate use
|
|
|
|
# Simple values and boxed machine integers
|
|
# | ptr | value |
|
|
INT_evacuate:
|
|
pushq 010(%rbp)
|
|
pushq $INT_code
|
|
mov %rsp,%rbp
|
|
jmp _gc_evacuate_ret
|
|
INT_scavenge:
|
|
add $020, %rbp
|
|
jmp _gc_scavenge_ret
|
|
|
|
INT_info_table:
|
|
cell INT_evacuate
|
|
cell INT_scavenge
|
|
cell 0
|
|
INT_code:
|
|
continue
|
|
|
|
# Indirection
|
|
# | ptr | indptr |
|
|
IND_evacuate:
|
|
mov 010(%rbp), %rbp
|
|
jmp _gc_evacuate
|
|
IND_scavenge:
|
|
# this should never be triggered but let's play it safe
|
|
add $020, %rbp
|
|
jmp _gc_scavenge_ret
|
|
|
|
IND_info:
|
|
cell IND_evacuate
|
|
cell IND_scavenge
|
|
cell 0
|
|
IND_code:
|
|
enter 010(%rbp)
|
|
|
|
# Blackhole (contains the original thunkptr for debugging purposes)
|
|
# | ptr | orig_thunkptr |
|
|
BLE_evacuate:
|
|
pushq 010(%rbp)
|
|
pushq $BLE_code
|
|
mov %rsp,%rbp
|
|
jmp _gc_evacuate_ret
|
|
BLE_scavenge:
|
|
add $020, %rbp
|
|
jmp _gc_scavenge_ret
|
|
|
|
BLE_info_table:
|
|
cell BLE_evacuate
|
|
cell BLE_scavenge
|
|
cell 0
|
|
BLE_code:
|
|
# if we hit this, we've got a pure loop in a program, and it is never
|
|
# going to actually progress. So let's just shoot it down.
|
|
mov 0, %rax
|
|
jmp BLE_code
|
|
# this might eventually generate an actual IO-style exception or something.
|
|
|
|
# List
|
|
# | ptr | 0 | # [] case
|
|
# | ptr | 1 | a | b | # (a:b) case
|
|
LIST_evacuate:
|
|
cmpq $0, 010(%rbp)
|
|
je LIST_evacuate_nil
|
|
pushq 030(%rbp)
|
|
pushq 020(%rbp)
|
|
pushq $1
|
|
pushq $LIST_code
|
|
mov %rsp, %rbp
|
|
jmp _gc_evacuate_ret
|
|
|
|
LIST_evacuate_nil:
|
|
pushq $0
|
|
pushq $LIST_code
|
|
mov %rsp, %rbp
|
|
jmp _gc_evacuate_ret
|
|
LIST_scavenge:
|
|
cmpq $0, 010(%rbp)
|
|
je LIST_scavenge_nil
|
|
mov %rbp, %r15
|
|
|
|
mov $LIST_scavenge1, %rsi
|
|
mov 020(%r15), %rbp
|
|
jmp _gc_evacuate
|
|
LIST_scavenge1:
|
|
mov %rbp, 020(%r15)
|
|
|
|
mov $LIST_scavenge2, %rsi
|
|
mov 030(%r15), %rbp
|
|
jmp _gc_evacuate
|
|
LIST_scavenge2:
|
|
mov %rbp, 030(%r15)
|
|
|
|
mov %r15, %rbp
|
|
add $040, %rbp
|
|
jmp _gc_scavenge_ret
|
|
|
|
LIST_scavenge_nil:
|
|
add $020, %rbp
|
|
jmp _gc_scavenge_ret
|
|
|
|
LIST_info_table:
|
|
cell LIST_evacuate
|
|
cell LIST_scavenge
|
|
cell 0
|
|
LIST_code:
|
|
continue
|
|
|
|
# FUN objects
|
|
# | ptr | thunkptr | args | arg[0] | arg[1] | ... | arg[args] |
|
|
FUN_evacuate:
|
|
mov 020(%rbp), %rbx # rbx = count of arguments
|
|
mov %rbx, %rcx # rcx = count of arguments for later looping
|
|
lea 030(%rbp, %rbx, 010), %rdx # rdx = address of the arguments
|
|
FUN_evacuate_one:
|
|
sub $1, %rcx
|
|
jl FUN_evacuate_fini
|
|
sub $010, %rdx
|
|
pushq (%rdx)
|
|
jmp FUN_evacuate_one
|
|
FUN_evacuate_fini:
|
|
pushq %rbx
|
|
pushq 010(%rbp)
|
|
pushq 000(%rbp)
|
|
mov %rsp, %rbp
|
|
jmp _gc_evacuate_ret
|
|
FUN_scavenge:
|
|
mov 020(%rbp), %r13 # r13 = count of arguments (for looping)
|
|
mov %rbp, %r15 # r15 = scavengee ptr
|
|
lea 030(%rbp, %r13, 010), %r14 # r14 = address of argument
|
|
|
|
FUN_scavenge_one:
|
|
sub $1, %r13
|
|
jl FUN_scavenge_fini
|
|
sub $010, %r14
|
|
mov (%r14), %rbp
|
|
mov $FUN_scavenge_one_cont, %rsi
|
|
jmp _gc_evacuate
|
|
FUN_scavenge_one_cont:
|
|
mov %rbp, (%r14)
|
|
jmp FUN_scavenge_one
|
|
|
|
FUN_scavenge_fini:
|
|
mov %r15, %rbp # restore rbp
|
|
mov 020(%rbp), %r14
|
|
lea 030(%rbp, %r14, 010), %rbp
|
|
jmp _gc_scavenge_ret
|
|
|
|
# Info tables for FUN objects.
|
|
FUN0_info_table:
|
|
cell FUN_evacuate
|
|
cell FUN_scavenge
|
|
cell 0
|
|
FUN0_code:
|
|
continue
|
|
|
|
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
|
|
|
|
FUN4_info_table:
|
|
cell FUN_evacuate
|
|
cell FUN_scavenge
|
|
cell 4
|
|
FUN4_code:
|
|
continue
|
|
# add more funN here as needed
|
|
|
|
# THU objects (gc implementation only, actual THU data are created 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:
|
|
mov 010(%rbp), %rbx # rbx = count of arguments
|
|
mov %rbx, %rcx # rcx = count of arguments for later looping
|
|
lea 020(%rbp, %rbx, 010), %rdx # rdx = address of the argument
|
|
THU_evacuate_one:
|
|
sub $1, %rcx
|
|
jl THU_evacuate_fini
|
|
sub $010, %rdx
|
|
pushq (%rdx)
|
|
jmp THU_evacuate_one
|
|
THU_evacuate_fini:
|
|
pushq %rbx
|
|
pushq 000(%rbp)
|
|
mov %rsp, %rbp
|
|
jmp _gc_evacuate_ret
|
|
THU_scavenge:
|
|
mov 010(%rbp), %r13 # r13 = count of arguments (for looping)
|
|
mov %rbp, %r15 # r15 = scavengee ptr
|
|
lea 020(%rbp, %r13, 010), %r14 # r14 = address of argument
|
|
|
|
THU_scavenge_one:
|
|
sub $1, %r13
|
|
jl THU_scavenge_fini
|
|
sub $010, %r14
|
|
mov (%r14), %rbp
|
|
mov $THU_scavenge_one_cont, %rsi
|
|
jmp _gc_evacuate
|
|
THU_scavenge_one_cont:
|
|
mov %rbp, (%r14)
|
|
jmp THU_scavenge_one
|
|
|
|
THU_scavenge_fini:
|
|
mov %r15, %rbp # restore rbp
|
|
mov 010(%rbp), %r14
|
|
lea 020(%rbp, %r14, 010), %rbp
|
|
jmp _gc_scavenge_ret
|
|
|
|
.endif # _data_s_file
|