2015-10-24 10:42:36 +02:00
|
|
|
module Arm
|
2018-03-26 18:42:15 +02:00
|
|
|
# A translator is cpu specific and translates from risc instructions to a given
|
2018-03-28 11:50:07 +02:00
|
|
|
# cpu. This one transltes to Arm Instructions.
|
2015-10-24 10:42:36 +02:00
|
|
|
class Translator
|
|
|
|
|
2015-10-24 16:11:18 +02:00
|
|
|
# translator should translate from register instructio set to it's own (arm eg)
|
|
|
|
# for each instruction we call the translator with translate_XXX
|
|
|
|
# with XXX being the class name.
|
|
|
|
# the result is replaced in the stream
|
2018-03-11 11:41:15 +01:00
|
|
|
def translate( instruction )
|
2015-10-24 16:11:18 +02:00
|
|
|
class_name = instruction.class.name.split("::").last
|
|
|
|
self.send( "translate_#{class_name}".to_sym , instruction)
|
|
|
|
end
|
|
|
|
|
2018-03-11 11:41:15 +01:00
|
|
|
def translate_Label( code )
|
2018-03-25 18:38:59 +02:00
|
|
|
Risc::Label.new( code.source , code.name )
|
2015-10-24 10:42:36 +02:00
|
|
|
end
|
|
|
|
|
2015-11-14 23:35:12 +01:00
|
|
|
# arm indexes are
|
|
|
|
# in bytes, so *4
|
2015-11-15 19:42:07 +01:00
|
|
|
# if an instruction is passed in we get the index with index function
|
2018-03-11 11:41:15 +01:00
|
|
|
def arm_index( index )
|
2017-01-19 08:02:29 +01:00
|
|
|
index = index.index if index.is_a?(Risc::Instruction)
|
2015-11-14 23:35:12 +01:00
|
|
|
raise "index error 0" if index == 0
|
2015-11-15 19:42:07 +01:00
|
|
|
index * 4
|
2015-11-14 23:35:12 +01:00
|
|
|
end
|
2015-10-24 10:42:36 +02:00
|
|
|
# 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.
|
2016-12-29 20:24:11 +01:00
|
|
|
def translate_SaveReturn( code )
|
2015-11-14 23:35:12 +01:00
|
|
|
ArmMachine.str( :lr , code.register , arm_index(code) )
|
2015-10-24 10:42:36 +02:00
|
|
|
end
|
|
|
|
|
2018-03-21 11:18:04 +01:00
|
|
|
def translate_Transfer( code )
|
2017-01-19 08:02:29 +01:00
|
|
|
# Risc machine convention is from => to
|
2015-11-19 09:09:55 +01:00
|
|
|
# But arm has the receiver/result as the first
|
|
|
|
ArmMachine.mov( code.to , code.from)
|
|
|
|
end
|
|
|
|
|
2016-12-25 17:05:39 +01:00
|
|
|
def translate_SlotToReg( code )
|
2016-12-15 17:21:08 +01:00
|
|
|
ArmMachine.ldr( *slot_args_for(code) )
|
2015-10-24 10:42:36 +02:00
|
|
|
end
|
|
|
|
|
2016-12-25 17:02:39 +01:00
|
|
|
def translate_RegToSlot( code )
|
2016-12-15 17:21:08 +01:00
|
|
|
ArmMachine.str( *slot_args_for(code) )
|
|
|
|
end
|
|
|
|
|
|
|
|
def slot_args_for( code )
|
2015-11-08 16:10:36 +01:00
|
|
|
if(code.index.is_a? Numeric)
|
2016-12-15 17:21:08 +01:00
|
|
|
[ code.register , code.array , arm_index(code) ]
|
2015-11-08 16:10:36 +01:00
|
|
|
else
|
2016-12-15 17:21:08 +01:00
|
|
|
[ code.register , code.array , code.index , :shift_lsl => 2]
|
2015-11-08 16:10:36 +01:00
|
|
|
end
|
2015-10-24 10:42:36 +02:00
|
|
|
end
|
|
|
|
|
2016-12-15 17:21:08 +01:00
|
|
|
def byte_args_for( code )
|
|
|
|
args = slot_args_for( code )
|
|
|
|
args.pop if(code.index.is_a? Numeric)
|
|
|
|
args
|
|
|
|
end
|
|
|
|
|
2018-03-28 11:50:07 +02:00
|
|
|
def translate_ByteToReg( code )
|
2016-12-15 17:21:08 +01:00
|
|
|
ArmMachine.ldrb( *byte_args_for(code) )
|
2015-11-19 09:09:55 +01:00
|
|
|
end
|
|
|
|
|
2018-03-28 11:50:07 +02:00
|
|
|
def translate_RegToByte( code )
|
2016-12-15 17:21:08 +01:00
|
|
|
ArmMachine.strb( *byte_args_for(code) )
|
2015-11-19 09:09:55 +01:00
|
|
|
end
|
|
|
|
|
2018-03-28 11:50:07 +02:00
|
|
|
def translate_FunctionCall( code )
|
|
|
|
ArmMachine.b( code.method.binary )
|
2015-10-24 10:42:36 +02:00
|
|
|
end
|
|
|
|
|
|
|
|
def translate_FunctionReturn code
|
2018-03-28 11:50:07 +02:00
|
|
|
ArmMachine.mov( :pc , code.register)
|
2015-10-24 10:42:36 +02:00
|
|
|
end
|
|
|
|
|
2018-03-28 11:50:07 +02:00
|
|
|
def translate_LoadConstant( code )
|
2015-10-24 10:42:36 +02:00
|
|
|
constant = code.constant
|
2018-03-28 11:50:07 +02:00
|
|
|
constant = constant.to_cpu(self) if constant.is_a?(Risc::Label)
|
2018-03-31 11:38:30 +02:00
|
|
|
return ArmMachine.add( code.register , constant )
|
|
|
|
end
|
|
|
|
|
|
|
|
def translate_LoadData( code )
|
|
|
|
return ArmMachine.mov( code.register , code.constant )
|
2015-10-24 10:42:36 +02:00
|
|
|
end
|
|
|
|
|
2018-03-30 16:09:02 +02:00
|
|
|
def translate_OperatorInstruction( code )
|
2015-11-12 19:02:14 +01:00
|
|
|
left = code.left
|
|
|
|
right = code.right
|
|
|
|
case code.operator.to_s
|
|
|
|
when "+"
|
|
|
|
c = ArmMachine.add(left , left , right)
|
|
|
|
when "-"
|
|
|
|
c = ArmMachine.sub(left , left , right)
|
|
|
|
when "&"
|
|
|
|
c = ArmMachine.and(left , left , right)
|
|
|
|
when "|"
|
|
|
|
c = ArmMachine.orr(left , left , right)
|
|
|
|
when "*"
|
|
|
|
c = ArmMachine.mul(left , right , left) #arm rule about left not being result, lukily commutative
|
2015-11-13 23:20:03 +01:00
|
|
|
when ">>"
|
|
|
|
c = ArmMachine.mov(left , left , :shift_asr => right) #arm rule about left not being result, lukily commutative
|
|
|
|
when "<<"
|
|
|
|
c = ArmMachine.mov(left , left , :shift_lsl => right) #arm rule about left not being result, lukily commutative
|
2015-11-12 19:02:14 +01:00
|
|
|
else
|
|
|
|
raise "unimplemented '#{code.operator}' #{code}"
|
|
|
|
end
|
|
|
|
c
|
|
|
|
end
|
|
|
|
|
2015-10-24 10:42:36 +02:00
|
|
|
# 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.
|
2018-03-28 11:50:07 +02:00
|
|
|
def translate_Branch( code )
|
|
|
|
ArmMachine.b( code.label.to_cpu(self) )
|
2015-10-24 10:42:36 +02:00
|
|
|
end
|
2015-10-24 16:11:18 +02:00
|
|
|
|
2018-03-28 11:50:07 +02:00
|
|
|
def translate_IsPlus( code )
|
|
|
|
ArmMachine.bpl( code.label.to_cpu(self) )
|
2015-11-13 23:20:03 +01:00
|
|
|
end
|
|
|
|
|
2018-03-28 11:50:07 +02:00
|
|
|
def translate_IsMinus( code )
|
|
|
|
ArmMachine.bmi( code.label.to_cpu(self) )
|
2015-11-13 23:20:03 +01:00
|
|
|
end
|
|
|
|
|
2018-03-28 11:50:07 +02:00
|
|
|
def translate_IsZero( code )
|
|
|
|
ArmMachine.beq( code.label.to_cpu(self) )
|
2018-04-02 18:31:08 +02:00
|
|
|
end
|
|
|
|
|
|
|
|
def translate_IsNotZero( code )
|
|
|
|
ArmMachine.bne( code.label.to_cpu(self) )
|
2015-11-13 23:20:03 +01:00
|
|
|
end
|
|
|
|
|
2018-03-28 11:50:07 +02:00
|
|
|
def translate_IsOverflow( code )
|
|
|
|
ArmMachine.bvs( code.label.to_cpu(self))
|
2015-11-13 23:20:03 +01:00
|
|
|
end
|
|
|
|
|
2018-03-28 11:50:07 +02:00
|
|
|
def translate_Syscall( code )
|
2015-10-24 10:42:36 +02:00
|
|
|
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
|
|
|
|
|
2018-03-28 11:50:07 +02:00
|
|
|
def putstring( int_code )
|
2015-11-16 17:03:29 +01:00
|
|
|
codes = ArmMachine.add( :r1 , :r1 , 12 ) # adjust for object header
|
2015-11-15 10:28:16 +01:00
|
|
|
codes.append ArmMachine.mov( :r0 , 1 ) # write to stdout == 1
|
2015-10-24 10:42:36 +02:00
|
|
|
syscall(int_code , codes )
|
|
|
|
end
|
|
|
|
|
2018-03-28 11:50:07 +02:00
|
|
|
def exit( int_code )
|
2017-01-19 08:02:29 +01:00
|
|
|
codes = ArmMachine.ldr( :r0 , :r0 , arm_index(Risc.resolve_to_index(:Message , :return_value)) )
|
2015-10-24 10:42:36 +02:00
|
|
|
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
|