rubyx/lib/arm/arm_machine.rb

61 lines
1.9 KiB
Ruby

require "vm/machine"
require_relative "instruction"
require_relative "stack_instruction"
require_relative "logic_instruction"
require_relative "memory_instruction"
require_relative "call_instruction"
module Arm
class ArmMachine < Vm::Machine
# defines a method in the current class, with the name inst (first erg)
# the method instantiates an instruction of the given class which gets passed a single hash as arg
# gets called for every "standard" instruction.
# may be used for machine specific ones too
def define_instruction inst , clazz
super
return
# need to use create_method and move to options hash
define_method("#{inst}s") do |*args|
instruction clazz , inst , :al , 1 , *args
end
ArmMachine::COND_CODES.keys.each do |suffix|
define_method("#{inst}#{suffix}") do |options|
instruction clazz , inst , suffix , 0 , *args
end
define_method("#{inst}s#{suffix}") do |options|
instruction clazz , inst , suffix , 1 , *args
end
end
end
def word_load value , reg
mov( :left => reg , :right => value )
end
def function_call call
raise "Not FunctionCall #{call.inspect}" unless call.is_a? Vm::FunctionCall
bl( :left => call.function )
end
def main_start
entry = Vm::Block.new("main_entry")
entry.add_code mov( :left => :fp , :right => 0 )
end
def main_exit
entry = Vm::Block.new("main_exit")
entry.add_code syscall(0)
end
def function_entry f_name
entry = Vm::Block.new("#{f_name}_entry")
entry.add_code push( :left => :lr )
end
def function_exit f_name
entry = Vm::Block.new("#{f_name}_exit")
entry.add_code pop( :left => :pc )
end
def syscall num
[mov( :left => :r7 , :right => num ) , swi( :left => 0 )]
end
end
end