rubyx/lib/arm/stack_instruction.rb

72 lines
2.0 KiB
Ruby
Raw Normal View History

require_relative "instruction"
2014-05-03 22:18:04 +03:00
module Arm
# ADDRESSING MODE 4
2014-05-03 22:18:04 +03:00
class StackInstruction < Vm::StackInstruction
include Arm::Constants
# arm intrucioons are pretty sensible, and always 4 bytes (thumb not supported)
def length
4
end
2014-05-07 14:58:38 +03:00
def initialize(attributes)
super(attributes)
@update_status_flag = 0
@condition_code = :al
2014-05-07 14:58:38 +03:00
@opcode = attributes[:opcode]
@args = [attributes[:left] , attributes[:right] , attributes[:extra]]
@operand = 0
@update_status_flag= 0
@rn = :r0 # register zero = zero bit pattern
# downward growing, decrement before memory access
# official ARM style stack as used by gas
@write_base = 1
if (opcode == :push)
@pre_post_index = 1
@up_down = 0
@is_pop = 0
else #pop
@pre_post_index = 0
@up_down = 1
@is_pop = 1
end
end
attr_accessor :pre_post_index, :up_down,
:update_status_flag, :write_base, :is_pop, :rn
def assemble(io)
build
instuction_class = 0b10 # OPC_STACK
cond = @condition_code.is_a?(Symbol) ? COND_CODES[@condition_code] : @condition_code
@rn = :sp # sp register
#assemble of old
val = @operand
val |= (reg_code(@rn) << 16)
val |= (is_pop << 16+4) #20
val |= (write_base << 16+4+ 1)
val |= (update_status_flag << 16+4+ 1+1)
val |= (up_down << 16+4+ 1+1+1)
val |= (pre_post_index << 16+4+ 1+1+1+1)#24
val |= (instuction_class << 16+4+ 1+1+1+1 +2)
val |= (cond << 16+4+ 1+1+1+1 +2+2)
io.write_uint32 val
end
private
# Build representation for source value
def build
if (@args.is_a?(Array))
@operand = 0
@args.each do |reg |
next unless reg
@operand |= (1 << reg_code(reg))
end
else
raise "invalid operand argument #{@args.inspect} #{inspect}"
end
end
end
end