2014-04-25 18:37:19 +03:00
|
|
|
require_relative "instruction"
|
2014-04-23 19:59:55 +03:00
|
|
|
|
|
|
|
module Asm
|
2014-04-23 23:40:35 +03:00
|
|
|
# ADDRESSING MODE 4
|
|
|
|
class StackInstruction < Instruction
|
2014-04-23 19:59:55 +03:00
|
|
|
|
2014-04-25 15:07:47 +03:00
|
|
|
def initialize(opcode , condition_code , update_status , args)
|
|
|
|
super(opcode , condition_code , update_status , args)
|
2014-04-23 23:40:35 +03:00
|
|
|
@update_status_flag= 0
|
|
|
|
@rn = reg "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
|
2014-04-23 19:59:55 +03:00
|
|
|
end
|
2014-04-23 23:40:35 +03:00
|
|
|
end
|
2014-04-25 13:29:12 +03:00
|
|
|
attr_accessor :pre_post_index, :up_down,
|
|
|
|
:update_status_flag, :write_base, :is_pop, :rn
|
2014-04-23 23:40:35 +03:00
|
|
|
|
2014-04-25 11:56:53 +03:00
|
|
|
def assemble(io)
|
2014-04-23 23:40:35 +03:00
|
|
|
build
|
2014-04-25 13:29:12 +03:00
|
|
|
instuction_class = 0b10 # OPC_STACK
|
2014-04-25 15:07:47 +03:00
|
|
|
cond = @condition_code.is_a?(Symbol) ? COND_CODES[@condition_code] : @condition_code
|
2014-04-23 23:40:35 +03:00
|
|
|
rn = reg "sp" # sp register
|
|
|
|
#assemble of old
|
|
|
|
val = operand
|
|
|
|
val |= (rn.bits << 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
|
2014-04-25 13:29:12 +03:00
|
|
|
val |= (instuction_class << 16+4+ 1+1+1+1 +2)
|
2014-04-23 23:40:35 +03:00
|
|
|
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 |
|
|
|
|
@operand |= (1 << reg.bits)
|
2014-04-23 19:59:55 +03:00
|
|
|
end
|
2014-04-23 23:40:35 +03:00
|
|
|
else
|
|
|
|
raise Asm::AssemblyError.new("invalid operand argument #{args.inspect}")
|
2014-04-23 19:59:55 +03:00
|
|
|
end
|
|
|
|
end
|
2014-04-23 23:40:35 +03:00
|
|
|
end
|
2014-04-23 19:59:55 +03:00
|
|
|
end
|