remove passes and achieve the same by translating
This commit is contained in:
88
lib/arm/translator.rb
Normal file
88
lib/arm/translator.rb
Normal file
@ -0,0 +1,88 @@
|
||||
module Arm
|
||||
class Translator
|
||||
|
||||
# don't replace labels
|
||||
def translate_Label code
|
||||
nil
|
||||
end
|
||||
|
||||
# Arm stores the return address in a register (not on the stack)
|
||||
# The register is called link , or lr for short .
|
||||
# Maybe because it provides the "link" back to the caller
|
||||
# the vm defines a register for the location, so we store it there.
|
||||
def translate_SaveReturn code
|
||||
ArmMachine.str( :lr , code.register , 4 * code.index )
|
||||
end
|
||||
|
||||
def translate_GetSlot code
|
||||
# times 4 because arm works in bytes, but vm in words
|
||||
ArmMachine.ldr( code.register , code.array , 4 * code.index )
|
||||
end
|
||||
|
||||
def translate_RegisterTransfer code
|
||||
# Register machine convention is from => to
|
||||
# But arm has the receiver/result as the first
|
||||
ArmMachine.mov( code.to , code.from)
|
||||
end
|
||||
|
||||
def translate_SetSlot code
|
||||
# times 4 because arm works in bytes, but vm in words
|
||||
ArmMachine.str( code.register , code.array , 4 * code.index )
|
||||
end
|
||||
|
||||
def translate_FunctionCall code
|
||||
ArmMachine.call( code.method )
|
||||
end
|
||||
|
||||
def translate_FunctionReturn code
|
||||
ArmMachine.ldr( :pc , code.register , 4 * code.index )
|
||||
end
|
||||
|
||||
def translate_LoadConstant code
|
||||
constant = code.constant
|
||||
if constant.is_a?(Parfait::Object) or constant.is_a? Symbol
|
||||
return ArmMachine.add( code.register , constant )
|
||||
else
|
||||
return ArmMachine.mov( code.register , code.constant )
|
||||
end
|
||||
end
|
||||
|
||||
# This implements branch logic, which is simply assembler branch
|
||||
#
|
||||
# The only target for a call is a Block, so we just need to get the address for the code
|
||||
# and branch to it.
|
||||
def translate_Branch code
|
||||
ArmMachine.b( code.block )
|
||||
end
|
||||
def translate_Syscall code
|
||||
call_codes = { :putstring => 4 , :exit => 1 }
|
||||
int_code = call_codes[code.name]
|
||||
raise "Not implemented syscall, #{code.name}" unless int_code
|
||||
send( code.name , int_code )
|
||||
end
|
||||
|
||||
def putstring int_code
|
||||
codes = ArmMachine.ldr( :r1 , Register.message_reg, 4 * Register.resolve_index(:message , :receiver))
|
||||
codes.append ArmMachine.add( :r1 , :r1 , 8 )
|
||||
codes.append ArmMachine.mov( :r0 , 1 )
|
||||
codes.append ArmMachine.mov( :r2 , 12 ) # String length, obvious TODO
|
||||
syscall(int_code , codes )
|
||||
end
|
||||
|
||||
def exit int_code
|
||||
codes = Register::Label.new(nil , "exit")
|
||||
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.append ArmMachine.mov( :r7 , int_code )
|
||||
codes.append ArmMachine.swi( 0 )
|
||||
codes
|
||||
end
|
||||
|
||||
end
|
||||
end
|
Reference in New Issue
Block a user