2014-04-14 18:09:56 +03:00
|
|
|
module Asm
|
|
|
|
class AstAssembler
|
|
|
|
def initialize(asm_arch)
|
|
|
|
@asm_arch = asm_arch
|
|
|
|
|
|
|
|
@symbols = {}
|
|
|
|
@inst_label_context = {}
|
|
|
|
|
|
|
|
@asm = Asm::Assembler.new
|
|
|
|
end
|
|
|
|
|
|
|
|
def assembler
|
|
|
|
@asm
|
|
|
|
end
|
|
|
|
|
|
|
|
def load_ast(ast)
|
|
|
|
label_breadcrumb = []
|
|
|
|
ast.children.each do |cmd|
|
2014-04-16 12:03:39 +03:00
|
|
|
if (cmd.is_a?(Asm::LabelNode))
|
2014-04-14 18:09:56 +03:00
|
|
|
m = /^\/+/.match(cmd.name)
|
|
|
|
count = m ? m[0].length : 0
|
|
|
|
label_breadcrumb = label_breadcrumb[0,count]
|
|
|
|
label_breadcrumb << cmd.name[count..-1]
|
2014-04-22 22:24:22 +03:00
|
|
|
@asm.add_value object_for_label(label_breadcrumb.join('/'))
|
2014-04-16 12:03:39 +03:00
|
|
|
elsif (cmd.is_a?(Asm::InstructionNode))
|
2014-04-14 18:09:56 +03:00
|
|
|
inst = @asm_arch::Instruction.new(cmd, self)
|
2014-04-22 22:24:22 +03:00
|
|
|
@asm.add_value inst
|
2014-04-14 18:09:56 +03:00
|
|
|
@inst_label_context[inst] = label_breadcrumb
|
2014-04-16 12:03:39 +03:00
|
|
|
elsif (cmd.is_a?(Asm::DirectiveNode))
|
2014-04-14 18:09:56 +03:00
|
|
|
if (cmd.name == 'global')
|
|
|
|
symbol_for_label(cmd.value)[:linkage] = Elf::Constants::STB_GLOBAL
|
|
|
|
elsif (cmd.name == 'extern')
|
|
|
|
object_for_label(cmd.value).extern!
|
|
|
|
elsif (cmd.name == 'hexdata')
|
|
|
|
bytes = cmd.value.strip.split(/\s+/).map do |hex|
|
|
|
|
hex.to_i(16)
|
|
|
|
end.pack('C*')
|
2014-04-22 22:24:22 +03:00
|
|
|
@asm.add_value Asm::DataObject.new(bytes)
|
2014-04-14 18:09:56 +03:00
|
|
|
elsif (cmd.name == "asciz")
|
|
|
|
str = eval(cmd.value) + "\x00"
|
2014-04-22 22:24:22 +03:00
|
|
|
@asm.add_value Asm::DataObject.new(str)
|
2014-04-14 21:52:16 +03:00
|
|
|
elsif (defined?(Asm::Arm) and cmd.name == 'addrtable')
|
2014-04-22 22:24:22 +03:00
|
|
|
@asm.add_value Asm::Arm::AddrTableObject.new
|
2014-04-14 18:09:56 +03:00
|
|
|
else
|
|
|
|
raise Asm::AssemblyError.new('unknown directive', cmd)
|
|
|
|
end
|
|
|
|
end
|
|
|
|
end
|
|
|
|
end
|
|
|
|
|
|
|
|
# instruction is user for label context
|
|
|
|
def symbol_for_label(name, instruction=nil)
|
|
|
|
if (instruction)
|
|
|
|
context = @inst_label_context[instruction]
|
|
|
|
m = /^(\/*)(.+)/.match(name)
|
|
|
|
breadcrumb = context[0,m[1].length]
|
|
|
|
breadcrumb << m[2]
|
|
|
|
qual_name = breadcrumb.join('/')
|
|
|
|
else
|
|
|
|
qual_name = name
|
|
|
|
end
|
|
|
|
|
|
|
|
if (not @symbols[qual_name])
|
|
|
|
@symbols[name] = {:label => Asm::LabelObject.new, :linkage => Elf::Constants::STB_LOCAL, :name => qual_name}
|
|
|
|
end
|
|
|
|
@symbols[qual_name]
|
|
|
|
end
|
|
|
|
|
|
|
|
def object_for_label(name, instruction=nil)
|
|
|
|
symbol_for_label(name, instruction)[:label]
|
|
|
|
end
|
|
|
|
|
|
|
|
def assemble(io)
|
|
|
|
@asm.assemble io
|
|
|
|
end
|
|
|
|
|
|
|
|
def symbols
|
|
|
|
@symbols.values
|
|
|
|
end
|
|
|
|
|
|
|
|
def relocations
|
|
|
|
@asm.relocations
|
|
|
|
end
|
|
|
|
end
|
|
|
|
end
|