rubyx/lib/arm/memory_instruction.rb

100 lines
3.3 KiB
Ruby
Raw Normal View History

require_relative "nodes"
require_relative "instruction"
2014-05-03 22:18:04 +03:00
module Arm
# ADDRESSING MODE 2
# Implemented: immediate offset with offset=0
2014-05-03 22:18:04 +03:00
class MemoryInstruction < Vm::MemoryInstruction
include Arm::Constants
2014-05-07 14:58:38 +03:00
def initialize(attributes)
super(attributes)
2014-05-06 21:36:28 +03:00
@update_status_flag = 0
@condition_code = :al
2014-05-07 14:58:38 +03:00
@opcode = attributes[:opcode]
@args = [attributes[:left] , attributes[:right] ]
2014-05-06 21:36:28 +03:00
@operand = 0
@i = 0 #I flag (third bit)
@pre_post_index = 0 #P flag
@add_offset = 0 #U flag
@byte_access = opcode.to_s[-1] == "b" ? 1 : 0 #B (byte) flag
@w = 0 #W flag
@is_load = opcode.to_s[0] == "l" ? 1 : 0 #L (load) flag
2014-05-06 21:36:28 +03:00
@rn = :r0 # register zero = zero bit pattern
@rd = :r0 # register zero = zero bit pattern
end
attr_accessor :i, :pre_post_index, :add_offset,
:byte_access, :w, :is_load, :rn, :rd
# arm intrucioons are pretty sensible, and always 4 bytes (thumb not supported)
def length
4
end
# Build representation for target address
def build
if( @is_load )
2014-05-06 21:36:28 +03:00
@rd = @args[0]
arg = @args[1]
else #store
2014-05-06 21:36:28 +03:00
@rd = @args[1]
arg = @args[0]
end
#str / ldr are _serious instructions. With BIG possibilities not half are implemented
2014-05-06 21:36:28 +03:00
if (arg.is_a?(Symbol)) #symbol is register
@rn = arg
2014-05-07 14:58:38 +03:00
if @attributes[:offset]
@operand = @attributes[:offset]
if (@operand < 0)
@add_offset = 0
#TODO test/check/understand
@operand *= -1
else
@add_offset = 1
end
if (@operand.abs > 4095)
raise "reference offset too large/small (max 4095) #{arg} #{inspect}"
end
end
2014-05-06 21:36:28 +03:00
elsif (arg.is_a?(Vm::StringLiteral)) #use pc relative
@rn = :pc
@operand = arg.position - self.position - 8 #stringtable is after code
@add_offset = 1
if (@operand.abs > 4095)
raise "reference offset too large/small (max 4095) #{arg} #{inspect}"
end
elsif (arg.is_a?(Arm::Label) or arg.is_a?(Arm::NumLiteral))
@pre_post_index = 1
@rn = pc
@use_addrtable_reloc = true
@addrtable_reloc_target = arg
else
raise "invalid operand argument #{arg.inspect} #{inspect}"
end
end
def assemble(io)
build
#not sure about these 2 constants. They produce the correct output for str r0 , r1
# but i can't help thinking that that is because they are not used in that instruction and
# so it doesn't matter. Will see
@add_offset = 1
@pre_post_index = 1
instuction_class = 0b01 # OPC_MEMORY_ACCESS
2014-05-06 21:36:28 +03:00
val = @operand.is_a?(Symbol) ? reg_code(@operand) : @operand
val |= (reg_code(rd) << 12 )
val |= (reg_code(rn) << 12+4) #16
val |= (is_load << 12+4 +4)
val |= (w << 12+4 +4+1)
val |= (byte_access << 12+4 +4+1+1)
val |= (add_offset << 12+4 +4+1+1+1)
val |= (pre_post_index << 12+4 +4+1+1+1+1)#24
val |= (i << 12+4 +4+1+1+1+1 +1)
val |= (instuction_class<<12+4 +4+1+1+1+1 +1+1)
val |= (cond_bit_code << 12+4 +4+1+1+1+1 +1+1+2)
io.write_uint32 val
2014-05-06 21:36:28 +03:00
end
end
end