2014-04-25 12:28:23 +03:00
|
|
|
require_relative "assembly_error"
|
2014-04-25 13:34:10 +03:00
|
|
|
require_relative "arm_machine"
|
2014-04-14 21:52:16 +03:00
|
|
|
|
2014-04-14 18:09:56 +03:00
|
|
|
module Asm
|
2014-04-14 21:52:16 +03:00
|
|
|
|
2014-04-25 18:37:19 +03:00
|
|
|
class Code ; end
|
|
|
|
|
2014-04-25 12:28:23 +03:00
|
|
|
# Not surprisingly represents an cpu instruction.
|
|
|
|
# This is an abstract base class, with derived classes
|
|
|
|
# Logic / Move / Compare / Stack / Memory (see there)
|
|
|
|
#
|
|
|
|
# Opcode is a (<= three) letter accronym (same as in assembly code). Though in arm, suffixes can
|
|
|
|
# make the opcode longer, we chop those off in the constructor
|
|
|
|
# Argurments are registers or labels or string/num Literals
|
|
|
|
|
|
|
|
class Instruction < Code
|
2014-04-25 13:34:10 +03:00
|
|
|
include ArmMachine
|
2014-04-14 18:09:56 +03:00
|
|
|
|
2014-04-23 23:40:35 +03:00
|
|
|
COND_POSTFIXES = Regexp.union( COND_CODES.keys.collect{|k|k.to_s} ).source
|
2014-04-14 18:09:56 +03:00
|
|
|
|
2014-04-25 15:07:47 +03:00
|
|
|
def initialize(opcode , condition_code , update_status , args)
|
|
|
|
@update_status_flag = update_status
|
|
|
|
@condition_code = condition_code.to_sym
|
|
|
|
@opcode = opcode
|
2014-04-23 23:40:35 +03:00
|
|
|
@args = args
|
|
|
|
@operand = 0
|
|
|
|
end
|
2014-04-25 13:29:12 +03:00
|
|
|
|
|
|
|
attr_reader :opcode, :args
|
|
|
|
# Many arm instructions may be conditional, where the default condition is always (al)
|
2014-04-25 13:34:10 +03:00
|
|
|
# ArmMachine::COND_CODES names them, and this attribute reflects it
|
2014-04-25 15:07:47 +03:00
|
|
|
attr_reader :condition_code
|
2014-04-25 13:29:12 +03:00
|
|
|
attr_reader :operand
|
2014-04-14 18:09:56 +03:00
|
|
|
|
2014-04-25 13:29:12 +03:00
|
|
|
# Logic instructions may be executed with or without affecting the status register
|
|
|
|
# Only when an instruction affects the status is a subsequent compare instruction effective
|
|
|
|
# But to make the conditional execution (see cond) work for more than one instruction, one needs to
|
|
|
|
# be able to execute without changing the status
|
|
|
|
attr_reader :update_status_flag
|
2014-04-23 23:40:35 +03:00
|
|
|
|
2014-04-25 12:28:23 +03:00
|
|
|
# arm intrucioons are pretty sensible, and always 4 bytes (thumb not supported)
|
2014-04-23 23:40:35 +03:00
|
|
|
def length
|
|
|
|
4
|
|
|
|
end
|
|
|
|
end
|
2014-04-14 18:09:56 +03:00
|
|
|
end
|