2014-04-19 23:25:46 +03:00
|
|
|
require 'elf/object_file'
|
|
|
|
require 'elf/symbol_table_section'
|
|
|
|
require 'elf/text_section'
|
|
|
|
require 'elf/string_table_section'
|
2014-04-14 18:09:56 +03:00
|
|
|
|
2014-04-23 13:57:34 +03:00
|
|
|
module Elf
|
2014-04-14 18:09:56 +03:00
|
|
|
|
|
|
|
class ObjectWriter
|
2014-08-30 14:17:00 +03:00
|
|
|
def initialize(space , target = Elf::Constants::TARGET_ARM )
|
2014-04-19 23:25:46 +03:00
|
|
|
@object = Elf::ObjectFile.new(target)
|
2014-08-30 14:17:00 +03:00
|
|
|
@object_space = space
|
2014-04-19 23:25:46 +03:00
|
|
|
sym_strtab = Elf::StringTableSection.new(".strtab")
|
2014-04-14 18:09:56 +03:00
|
|
|
@object.add_section sym_strtab
|
2014-04-19 23:25:46 +03:00
|
|
|
@symbol_table = Elf::SymbolTableSection.new(".symtab", sym_strtab)
|
2014-04-14 18:09:56 +03:00
|
|
|
@object.add_section @symbol_table
|
|
|
|
|
2014-04-19 23:25:46 +03:00
|
|
|
@text = Elf::TextSection.new(".text")
|
2014-04-14 18:09:56 +03:00
|
|
|
@object.add_section @text
|
2014-05-19 17:32:41 +03:00
|
|
|
|
2014-08-30 14:17:00 +03:00
|
|
|
@object_space.run_passes
|
|
|
|
assembler = Register::Assembler.new(@object_space)
|
|
|
|
set_text assembler.assemble
|
|
|
|
|
|
|
|
# for debug add labels to the block positions
|
2014-05-22 14:56:31 +03:00
|
|
|
blocks = []
|
2014-08-30 14:17:00 +03:00
|
|
|
space.classes.values.each do |clazz|
|
|
|
|
clazz.instance_methods.each do |f|
|
2014-05-31 17:02:55 +03:00
|
|
|
f.blocks.each do |b|
|
2014-09-07 17:31:40 +03:00
|
|
|
add_symbol "#{clazz.name}::#{f.name}@#{b.name}" , b.position
|
2014-05-31 17:02:55 +03:00
|
|
|
end
|
|
|
|
end
|
2014-05-31 14:35:33 +03:00
|
|
|
end
|
2014-10-07 15:34:43 +03:00
|
|
|
space.main.blocks.each do |b|
|
|
|
|
add_symbol "main@#{b.name}" , b.position
|
|
|
|
end
|
|
|
|
add_symbol "#register@#{space.init.name}" , space.init.position
|
2014-10-02 16:06:05 +03:00
|
|
|
assembler.objects.values.each do |slot|
|
|
|
|
label = "#{slot.class.name}::#{slot.position.to_s(16)}"
|
|
|
|
label += "=#{slot}" if slot.is_a?(Symbol) or slot.is_a?(String)
|
|
|
|
label += "=#{slot.string}" if slot.is_a?(Virtual::StringConstant)
|
|
|
|
add_symbol label , slot.position
|
2014-08-30 20:55:22 +03:00
|
|
|
end
|
2014-04-14 18:09:56 +03:00
|
|
|
end
|
2014-05-20 10:28:34 +03:00
|
|
|
attr_reader :text
|
2014-04-14 18:09:56 +03:00
|
|
|
def set_text(text)
|
|
|
|
@text.text = text
|
|
|
|
add_symbol "_start", 0
|
|
|
|
end
|
|
|
|
def add_symbol(name, offset, linkage = Elf::Constants::STB_GLOBAL)
|
|
|
|
@symbol_table.add_func_symbol name, offset, @text, linkage
|
|
|
|
end
|
|
|
|
|
|
|
|
def save(filename)
|
2014-04-21 21:38:39 +03:00
|
|
|
to = File.open(filename, 'wb')
|
|
|
|
@object.write to
|
|
|
|
to.close
|
2014-04-14 18:09:56 +03:00
|
|
|
end
|
|
|
|
|
|
|
|
end
|
|
|
|
end
|