2015-06-23 18:55:54 +02:00
|
|
|
module Arm
|
|
|
|
|
|
|
|
class SyscallImplementation
|
2015-06-25 15:31:09 +02:00
|
|
|
CALLS_CODES = { :putstring => 4 , :exit => 0 }
|
2015-06-23 18:55:54 +02:00
|
|
|
def run block
|
|
|
|
block.codes.dup.each do |code|
|
|
|
|
next unless code.is_a? Register::Syscall
|
2015-06-25 15:31:09 +02:00
|
|
|
new_codes = []
|
|
|
|
int_code = CALLS_CODES[code.name]
|
|
|
|
raise "Not implemented syscall, #{code.name}" unless int_code
|
|
|
|
send( code.name , int_code , new_codes )
|
2015-06-23 18:55:54 +02:00
|
|
|
block.replace(code , new_codes )
|
|
|
|
end
|
|
|
|
end
|
2015-06-25 15:31:09 +02:00
|
|
|
|
|
|
|
def putstring int_code , codes
|
|
|
|
codes << ArmMachine.mov( :r1 , 20 ) # String length, obvious TODO
|
2015-06-27 20:16:46 +02:00
|
|
|
codes << ArmMachine.ldr( :r0 , Register::RegisterReference.message_reg, Virtual::SELF_INDEX)
|
2015-06-25 15:31:09 +02:00
|
|
|
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
|
2015-06-23 18:55:54 +02:00
|
|
|
end
|
2015-06-25 15:31:09 +02:00
|
|
|
|
2015-06-24 15:07:27 +02:00
|
|
|
Virtual.machine.add_pass "Arm::SyscallImplementation"
|
2015-06-23 18:55:54 +02:00
|
|
|
end
|