2014-04-19 23:25:46 +03:00
|
|
|
require_relative "constants"
|
|
|
|
require_relative "null_section"
|
|
|
|
|
2014-04-14 18:09:56 +03:00
|
|
|
module Elf
|
|
|
|
class ObjectFile
|
2014-04-19 23:25:46 +03:00
|
|
|
include Constants
|
2014-04-14 18:09:56 +03:00
|
|
|
|
|
|
|
def initialize(target)
|
|
|
|
@target = target
|
|
|
|
|
|
|
|
@sections = []
|
|
|
|
add_section NullSection.new
|
|
|
|
end
|
|
|
|
|
|
|
|
def add_section(section)
|
|
|
|
@sections << section
|
|
|
|
section.index = @sections.length - 1
|
|
|
|
end
|
|
|
|
|
|
|
|
def write(io)
|
2016-12-16 16:18:14 +02:00
|
|
|
write_preamble(io)
|
2014-04-14 18:09:56 +03:00
|
|
|
|
|
|
|
sh_offset_pos = io.tell
|
|
|
|
|
2016-12-16 16:18:14 +02:00
|
|
|
write_header(io)
|
|
|
|
|
|
|
|
string_table = write_string_table(io)
|
2014-04-14 18:09:56 +03:00
|
|
|
|
|
|
|
io.write_uint16 @sections.length # section header count
|
|
|
|
|
|
|
|
io.write_uint16 @sections.length-1 # section name string table index
|
|
|
|
|
2016-12-16 16:18:14 +02:00
|
|
|
section_data = write_sections(io)
|
2014-04-14 18:09:56 +03:00
|
|
|
|
2016-12-16 16:18:14 +02:00
|
|
|
sh_offset = io.tell
|
2014-04-14 18:09:56 +03:00
|
|
|
|
2016-12-16 16:18:14 +02:00
|
|
|
write_section_data(section_data, string_table , io)
|
2014-04-14 18:09:56 +03:00
|
|
|
|
2016-12-16 16:18:14 +02:00
|
|
|
io.seek sh_offset_pos
|
|
|
|
io.write_uint32 sh_offset
|
|
|
|
end
|
2014-04-14 18:09:56 +03:00
|
|
|
|
2016-12-16 16:18:14 +02:00
|
|
|
def write_string_table(io)
|
|
|
|
string_table = StringTableSection.new(".shstrtab")
|
|
|
|
@sections << string_table
|
|
|
|
@sections.each { |section|
|
|
|
|
string_table.add_string section.name
|
|
|
|
}
|
|
|
|
string_table
|
|
|
|
end
|
|
|
|
|
|
|
|
def write_section_data(section_data, string_table,io)
|
2014-04-14 18:09:56 +03:00
|
|
|
section_data.each { |data|
|
|
|
|
section, offset, size = data[:section], data[:offset], data[:size]
|
|
|
|
# write header first
|
2016-12-16 16:18:14 +02:00
|
|
|
io.write_uint32 string_table.index_for(section.name)
|
2014-04-14 18:09:56 +03:00
|
|
|
io.write_uint32 section.type
|
|
|
|
io.write_uint32 section.flags
|
|
|
|
io.write_uint32 section.addr
|
|
|
|
if (section.type == SHT_NOBITS)
|
|
|
|
raise 'SHT_NOBITS not handled yet'
|
|
|
|
elsif (section.type == SHT_NULL)
|
|
|
|
io.write_uint32 0
|
|
|
|
io.write_uint32 0
|
|
|
|
else
|
|
|
|
io.write_uint32 offset
|
|
|
|
io.write_uint32 size
|
|
|
|
end
|
|
|
|
io.write_uint32 section.link
|
|
|
|
io.write_uint32 section.info
|
|
|
|
io.write_uint32 section.alignment
|
|
|
|
io.write_uint32 section.ent_size
|
|
|
|
}
|
2016-12-16 16:18:14 +02:00
|
|
|
end
|
2014-04-14 18:09:56 +03:00
|
|
|
|
2016-12-16 16:18:14 +02:00
|
|
|
def write_sections(io)
|
|
|
|
section_data = []
|
|
|
|
@sections.each { |section|
|
|
|
|
offset = io.tell
|
|
|
|
section.write(io)
|
|
|
|
size = io.tell - offset
|
|
|
|
section_data << {:section => section, :offset => offset, :size => size}
|
|
|
|
}
|
|
|
|
section_data
|
|
|
|
end
|
|
|
|
|
|
|
|
def write_header(io)
|
|
|
|
io.write_uint32 0 # section header table offset
|
|
|
|
io.write_uint32 0 # no flags
|
|
|
|
io.write_uint16 52 # header length
|
|
|
|
io.write_uint16 0 # program header length
|
|
|
|
io.write_uint16 0 # program header count
|
|
|
|
io.write_uint16 40 # section header length
|
|
|
|
end
|
|
|
|
|
|
|
|
def write_preamble(io)
|
|
|
|
io << "\x7fELF"
|
|
|
|
io.write_uint8 @target[0]
|
|
|
|
io.write_uint8 @target[1]
|
|
|
|
io.write_uint8 EV_CURRENT
|
|
|
|
io.write_uint8 @target[2]
|
|
|
|
io << "\x00" * 8 # pad
|
|
|
|
io.write_uint16 ET_REL
|
|
|
|
io.write_uint16 @target[3]
|
|
|
|
io.write_uint32 EV_CURRENT
|
|
|
|
io.write_uint32 0 # entry point
|
|
|
|
io.write_uint32 0 # no program header table
|
2014-04-14 18:09:56 +03:00
|
|
|
end
|
|
|
|
end
|
|
|
|
end
|