rubyx/lib/asm/arm_assembler.rb

132 lines
3.2 KiB
Ruby
Raw Normal View History

require 'asm/call_instruction'
require 'asm/stack_instruction'
require 'asm/logic_instruction'
require 'asm/memory_instruction'
require 'asm/nodes'
require 'stream_reader'
require 'stringio'
2014-04-23 12:51:09 +03:00
require "asm/string_literal"
module Asm
class ArmAssembler
InstructionTools::REGISTERS.each do |reg , number|
define_method(reg) { Asm::Register.new(reg , number) }
end
def initialize
@values = []
@position = 0 # marks not set
@labels = []
@string_table = {}
end
attr_reader :values , :position
def instruction(clazz,name, *args)
opcode = name.to_s
arg_nodes = []
args.each do |arg|
if (arg.is_a?(Asm::Register))
arg_nodes << arg
elsif (arg.is_a?(Integer))
arg_nodes << Asm::NumLiteral.new(arg)
elsif (arg.is_a?(String))
arg_nodes << add_string(arg)
elsif (arg.is_a?(Asm::Label))
arg_nodes << arg
else
raise "Invalid argument #{arg.inspect} for instruction"
2014-04-23 19:59:55 +03:00
end
end
add_value clazz.new(opcode , arg_nodes)
end
def self.define_instruction(inst , clazz )
define_method(inst) do |*args|
instruction clazz , inst , *args
end
define_method(inst.to_s+'s') do |*args|
instruction clazz , inst.to_s+'s' , *args
2014-04-23 19:59:55 +03:00
end
InstructionTools::COND_CODES.keys.each do |cond_suffix|
suffix = cond_suffix.to_s
define_method(inst.to_s + suffix) do |*args|
instruction clazz , inst + suffix , *args
end
define_method(inst.to_s + 's'+ suffix) do |*args|
instruction clazz , inst.to_s + 's' + suffix, *args
end
end
end
[:push, :pop].each do |inst|
define_instruction(inst , StackInstruction)
end
[:adc, :add, :and, :bic, :eor, :orr, :rsb, :rsc, :sbc, :sub].each do |inst|
define_instruction(inst , LogicInstruction)
end
[:mov, :mvn].each do |inst|
define_instruction(inst , MoveInstruction)
end
[:cmn, :cmp, :teq, :tst].each do |inst|
define_instruction(inst , CompareInstruction)
end
[:strb, :str , :ldrb, :ldr].each do |inst|
define_instruction(inst , MemoryInstruction)
end
[:b, :bl , :swi].each do |inst|
define_instruction(inst , CallInstruction)
end
def assemble_to_string
#put the strings at the end of the assembled code.
# adding them will fix their position and make them assemble after
@string_table.values.each do |data|
add_value data
end
io = StringIO.new
assemble(io)
io.string
end
def add_string str
value = @string_table[str]
return value if value
data = Asm::StringLiteral.new(str)
@string_table[str] = data
end
def strings
@string_table.values
end
def add_value(val)
val.at(@position)
length = val.length
@position += length
@values << val
end
def label name
label = Label.new(name , self)
@labels << label
label
end
def label! name
label(name).set!
end
def assemble(io)
@values.each do |obj|
obj.assemble io, self
end
end
end
end