2016-12-14 13:43:13 +02:00
|
|
|
module Arm
|
|
|
|
# ADDRESSING MODE 4
|
|
|
|
|
2018-03-26 20:04:39 +03:00
|
|
|
class StackInstruction < Instruction
|
2016-12-14 13:43:13 +02:00
|
|
|
|
|
|
|
def initialize(first , attributes)
|
|
|
|
super(nil)
|
|
|
|
@attributes = attributes
|
|
|
|
@first = first
|
|
|
|
@attributes[:update_status] = 0 if @attributes[:update_status] == nil
|
|
|
|
@attributes[:condition_code] = :al if @attributes[:condition_code] == nil
|
|
|
|
@attributes[:opcode] = attributes[:opcode]
|
2016-12-14 21:53:26 +02:00
|
|
|
@rn = :sp # sp register
|
2016-12-14 13:43:13 +02:00
|
|
|
end
|
|
|
|
|
2016-12-14 21:53:26 +02:00
|
|
|
# don't overwrite instance variables, to make assembly repeatable
|
2016-12-14 13:43:13 +02:00
|
|
|
def assemble(io)
|
2016-12-14 21:53:26 +02:00
|
|
|
operand = 0
|
|
|
|
raise "invalid operand argument #{inspect}" unless (@first.is_a?(Array))
|
|
|
|
@first.each do |r|
|
|
|
|
raise "nil register in push, index #{r}- #{inspect}" if r.nil?
|
|
|
|
operand = operand | (1 << reg_code(r))
|
2016-12-14 13:43:13 +02:00
|
|
|
end
|
|
|
|
val = operand
|
|
|
|
val = val | (reg_code(@rn) << 16)
|
2016-12-14 19:57:09 +02:00
|
|
|
val = val | (is_pop << 16 + 4) #20
|
2016-12-14 21:53:26 +02:00
|
|
|
val = val | (1 << 16 + 4 + 1)
|
2016-12-14 19:57:09 +02:00
|
|
|
val = val | (@attributes[:update_status] << 16 + 4 + 1 + 1)
|
|
|
|
val = val | (up_down << 16 + 4 + 1 + 1 + 1)
|
|
|
|
val = val | (pre_post_index << 16 + 4 + 1 + 1 + 1 + 1)#24
|
2016-12-15 12:38:22 +02:00
|
|
|
val = val | instruction_code
|
2016-12-14 19:57:09 +02:00
|
|
|
val = val | (cond << 16 + 4 + 1 + 1 + 1 + 1 + 2 + 2)
|
2016-12-31 18:45:22 +02:00
|
|
|
io.write_unsigned_int_32 val
|
2016-12-14 13:43:13 +02:00
|
|
|
end
|
|
|
|
|
2016-12-14 21:53:26 +02:00
|
|
|
def cond
|
|
|
|
if @attributes[:condition_code].is_a?(Symbol)
|
|
|
|
COND_CODES[@attributes[:condition_code]]
|
|
|
|
else
|
|
|
|
@attributes[:condition_code]
|
|
|
|
end
|
|
|
|
end
|
|
|
|
def instuction_class
|
|
|
|
0b10 # OPC_STACK
|
|
|
|
end
|
|
|
|
def up_down
|
|
|
|
(opcode == :push) ? 0 : 1
|
|
|
|
end
|
|
|
|
alias :is_pop :up_down
|
|
|
|
|
|
|
|
def pre_post_index
|
|
|
|
(opcode == :push) ? 1 : 0
|
|
|
|
end
|
|
|
|
|
2016-12-14 13:43:13 +02:00
|
|
|
def regs
|
|
|
|
@first
|
|
|
|
end
|
2016-12-14 21:53:26 +02:00
|
|
|
|
2016-12-14 13:43:13 +02:00
|
|
|
def to_s
|
2018-07-03 19:15:36 +03:00
|
|
|
"#{opcode} [#{@first.join(', ') }] #{super}"
|
2016-12-14 13:43:13 +02:00
|
|
|
end
|
|
|
|
end
|
|
|
|
|
|
|
|
end
|