39 lines
1.1 KiB
Ruby
39 lines
1.1 KiB
Ruby
module Arm
|
|
|
|
class SyscallImplementation
|
|
CALLS_CODES = { :putstring => 4 , :exit => 0 }
|
|
def run block
|
|
block.codes.dup.each do |code|
|
|
next unless code.is_a? Register::Syscall
|
|
new_codes = []
|
|
int_code = CALLS_CODES[code.name]
|
|
raise "Not implemented syscall, #{code.name}" unless int_code
|
|
send( code.name , int_code , new_codes )
|
|
block.replace(code , new_codes )
|
|
end
|
|
end
|
|
|
|
def putstring int_code , codes
|
|
codes << ArmMachine.mov( :r0 , 1 ) # stdout == 1
|
|
codes << ArmMachine.ldr( :r1 , Register.message_reg, 4 * Register.resolve_index(:message , :receiver))
|
|
codes << ArmMachine.mov( :r2 , 20 ) # String length, obvious TODO
|
|
syscall(int_code , codes )
|
|
end
|
|
|
|
def exit int_code , codes
|
|
syscall int_code , codes
|
|
end
|
|
|
|
private
|
|
|
|
# syscall is always triggered by swi(0)
|
|
# The actual code (ie the index of the kernel function) is in r7
|
|
def syscall int_code , codes
|
|
codes << ArmMachine.mov( :r7 , int_code )
|
|
codes << ArmMachine.swi( 0 )
|
|
end
|
|
end
|
|
|
|
Virtual.machine.add_pass "Arm::SyscallImplementation"
|
|
end
|