6715 lines
No EOL
210 KiB
C++
6715 lines
No EOL
210 KiB
C++
/**
|
|
* Support of Mach-O executable files.
|
|
*/
|
|
|
|
#include "../runtime/common.h"
|
|
#include "../runtime/crypto.h"
|
|
#include "objects.h"
|
|
#include "osutils.h"
|
|
#include "streams.h"
|
|
#include "files.h"
|
|
#include "dwarf.h"
|
|
#include "macfile.h"
|
|
#include "processors.h"
|
|
#include "intel.h"
|
|
#include "lang.h"
|
|
#include "core.h"
|
|
#include "script.h"
|
|
|
|
#include "mac_runtime32.dylib.inc"
|
|
#include "mac_runtime64.dylib.inc"
|
|
|
|
/**
|
|
* MacLoadCommand
|
|
*/
|
|
|
|
MacLoadCommand::MacLoadCommand(MacLoadCommandList *owner)
|
|
: BaseLoadCommand(owner), address_(0), size_(0), type_(0), object_(NULL), offset_(0)
|
|
{
|
|
|
|
}
|
|
|
|
MacLoadCommand::MacLoadCommand(MacLoadCommandList *owner, uint64_t address, uint32_t size, uint32_t type)
|
|
: BaseLoadCommand(owner), address_(address), size_(size), type_(type), object_(NULL), offset_(0)
|
|
{
|
|
|
|
}
|
|
|
|
MacLoadCommand::MacLoadCommand(MacLoadCommandList *owner, uint32_t type, IObject *object)
|
|
: BaseLoadCommand(owner), address_(0), size_(0), type_(type), object_(object), offset_(0)
|
|
{
|
|
|
|
}
|
|
|
|
MacLoadCommand::MacLoadCommand(MacLoadCommandList *owner, const MacLoadCommand &src)
|
|
: BaseLoadCommand(owner, src), object_(NULL), offset_(0)
|
|
{
|
|
address_ = src.address_;
|
|
size_ = src.size_;
|
|
type_ = src.type_;
|
|
}
|
|
|
|
MacLoadCommand *MacLoadCommand::Clone(ILoadCommandList *owner) const
|
|
{
|
|
MacLoadCommand *command = new MacLoadCommand(reinterpret_cast<MacLoadCommandList *>(owner), *this);
|
|
return command;
|
|
}
|
|
|
|
void MacLoadCommand::ReadFromFile(MacArchitecture &file)
|
|
{
|
|
load_command lc;
|
|
|
|
/* Store current position into address_
|
|
* and read struct load_command */
|
|
address_ = file.Tell();
|
|
file.Read(&lc, sizeof(lc));
|
|
type_ = lc.cmd;
|
|
size_ = lc.cmdsize;
|
|
if (size_ < sizeof(lc))
|
|
throw std::runtime_error("Invalid format");
|
|
}
|
|
|
|
std::string MacLoadCommand::name() const
|
|
{
|
|
switch (type_) {
|
|
case LC_SEGMENT:
|
|
return std::string("LC_SEGMENT");
|
|
case LC_SYMTAB:
|
|
return std::string("LC_SYMTAB");
|
|
case LC_SYMSEG:
|
|
return std::string("LC_SYMSEG");
|
|
case LC_THREAD:
|
|
return std::string("LC_THREAD");
|
|
case LC_UNIXTHREAD:
|
|
return std::string("LC_UNIXTHREAD");
|
|
case LC_LOADFVMLIB:
|
|
return std::string("LC_LOADFVMLIB");
|
|
case LC_IDFVMLIB:
|
|
return std::string("LC_IDFVMLIB");
|
|
case LC_IDENT:
|
|
return std::string("LC_IDENT");
|
|
case LC_FVMFILE:
|
|
return std::string("LC_FVMFILE");
|
|
case LC_PREPAGE:
|
|
return std::string("LC_PREPAGE");
|
|
case LC_DYSYMTAB:
|
|
return std::string("LC_DYSYMTAB");
|
|
case LC_LOAD_DYLIB:
|
|
return std::string("LC_LOAD_DYLIB");
|
|
case LC_ID_DYLIB:
|
|
return std::string("LC_ID_DYLIB");
|
|
case LC_LOAD_DYLINKER:
|
|
return std::string("LC_LOAD_DYLINKER");
|
|
case LC_ID_DYLINKER:
|
|
return std::string("LC_ID_DYLINKER");
|
|
case LC_PREBOUND_DYLIB:
|
|
return std::string("LC_PREBOUND_DYLIB");
|
|
case LC_ROUTINES:
|
|
return std::string("LC_ROUTINES");
|
|
case LC_SUB_FRAMEWORK:
|
|
return std::string("LC_SUB_FRAMEWORK");
|
|
case LC_SUB_UMBRELLA:
|
|
return std::string("LC_SUB_UMBRELLA");
|
|
case LC_SUB_CLIENT:
|
|
return std::string("LC_SUB_CLIENT");
|
|
case LC_SUB_LIBRARY:
|
|
return std::string("LC_SUB_LIBRARY");
|
|
case LC_TWOLEVEL_HINTS:
|
|
return std::string("LC_TWOLEVEL_HINTS");
|
|
case LC_PREBIND_CKSUM:
|
|
return std::string("LC_PREBIND_CKSUM");
|
|
case LC_LOAD_WEAK_DYLIB:
|
|
return std::string("LC_LOAD_WEAK_DYLIB");
|
|
case LC_SEGMENT_64:
|
|
return std::string("LC_SEGMENT_64");
|
|
case LC_ROUTINES_64:
|
|
return std::string("LC_ROUTINES_64");
|
|
case LC_UUID:
|
|
return std::string("LC_UUID");
|
|
case LC_RPATH:
|
|
return std::string("LC_RPATH");
|
|
case LC_CODE_SIGNATURE:
|
|
return std::string("LC_CODE_SIGNATURE");
|
|
case LC_SEGMENT_SPLIT_INFO:
|
|
return std::string("LC_SEGMENT_SPLIT_INFO");
|
|
case LC_DYLD_INFO:
|
|
return std::string("LC_DYLD_INFO");
|
|
case LC_DYLD_INFO_ONLY:
|
|
return std::string("LC_DYLD_INFO_ONLY");
|
|
case LC_VERSION_MIN_MACOSX:
|
|
return std::string("LC_VERSION_MIN_MACOSX");
|
|
case LC_FUNCTION_STARTS:
|
|
return std::string("LC_FUNCTION_STARTS");
|
|
case LC_DYLD_ENVIRONMENT:
|
|
return std::string("LC_DYLD_ENVIRONMENT");
|
|
case LC_MAIN:
|
|
return std::string("LC_MAIN");
|
|
case LC_DATA_IN_CODE:
|
|
return std::string("LC_DATA_IN_CODE");
|
|
case LC_SOURCE_VERSION:
|
|
return std::string("LC_SOURCE_VERSION");
|
|
case LC_DYLIB_CODE_SIGN_DRS:
|
|
return std::string("LC_DYLIB_CODE_SIGN_DRS");
|
|
case LC_ENCRYPTION_INFO_64:
|
|
return std::string("LC_ENCRYPTION_INFO_64");
|
|
case LC_LINKER_OPTION:
|
|
return std::string("LC_LINKER_OPTION");
|
|
case LC_LINKER_OPTIMIZATION_HINT:
|
|
return std::string("LC_LINKER_OPTIMIZATION_HINT");
|
|
case LC_VERSION_MIN_TVOS:
|
|
return std::string("LC_VERSION_MIN_TVOS");
|
|
case LC_VERSION_MIN_WATCHOS:
|
|
return std::string("LC_VERSION_MIN_WATCHOS");
|
|
case LC_NOTE:
|
|
return std::string("LC_NOTE");
|
|
case LC_BUILD_VERSION:
|
|
return std::string("LC_BUILD_VERSION");
|
|
}
|
|
|
|
return BaseLoadCommand::name();
|
|
}
|
|
|
|
void MacLoadCommand::WriteToFile(MacArchitecture &file)
|
|
{
|
|
const IArchitecture *source = file.source();
|
|
uint64_t position = file.Tell();
|
|
bool need_copy = true;
|
|
|
|
switch (type_) {
|
|
case LC_SEGMENT:
|
|
case LC_SEGMENT_64:
|
|
if (object_) {
|
|
reinterpret_cast<MacSegment *>(object_)->WriteToFile(file);
|
|
need_copy = false;
|
|
}
|
|
break;
|
|
case LC_LOAD_DYLIB:
|
|
case LC_LOAD_WEAK_DYLIB:
|
|
if (object_) {
|
|
reinterpret_cast<MacImport *>(object_)->WriteToFile(file);
|
|
need_copy = false;
|
|
}
|
|
break;
|
|
case LC_SYMTAB:
|
|
{
|
|
symtab_command *symtab = file.symtab();
|
|
file.Write(symtab, sizeof(symtab_command));
|
|
need_copy = false;
|
|
}
|
|
break;
|
|
case LC_DYSYMTAB:
|
|
{
|
|
dysymtab_command *dysymtab = file.dysymtab();
|
|
file.Write(dysymtab, sizeof(dysymtab_command));
|
|
need_copy = false;
|
|
}
|
|
break;
|
|
case LC_DYLD_INFO:
|
|
case LC_DYLD_INFO_ONLY:
|
|
{
|
|
dyld_info_command *dyld_info = file.dyld_info();
|
|
file.Write(dyld_info, sizeof(dyld_info_command));
|
|
need_copy = false;
|
|
}
|
|
break;
|
|
case LC_UNIXTHREAD:
|
|
{
|
|
thread_command command;
|
|
source->Seek(address_);
|
|
source->Read(&command, sizeof(command));
|
|
x86_state_hdr_t state_hdr;
|
|
source->Read(&state_hdr, sizeof(state_hdr));
|
|
|
|
switch (file.type()) {
|
|
case CPU_TYPE_I386:
|
|
if (state_hdr.flavor == x86_THREAD_STATE32) {
|
|
x86_thread_state32_t thread_state;
|
|
source->Read(&thread_state, sizeof(thread_state));
|
|
thread_state.__eip = static_cast<uint32_t>(file.entry_point());
|
|
file.Write(&command, sizeof(command));
|
|
file.Write(&state_hdr, sizeof(state_hdr));
|
|
file.Write(&thread_state, sizeof(thread_state));
|
|
need_copy = false;
|
|
}
|
|
break;
|
|
case CPU_TYPE_X86_64:
|
|
if (state_hdr.flavor == x86_THREAD_STATE64) {
|
|
x86_thread_state64_t thread_state;
|
|
source->Read(&thread_state, sizeof(thread_state));
|
|
thread_state.__rip = file.entry_point();
|
|
file.Write(&command, sizeof(command));
|
|
file.Write(&state_hdr, sizeof(state_hdr));
|
|
file.Write(&thread_state, sizeof(thread_state));
|
|
need_copy = false;
|
|
}
|
|
break;
|
|
}
|
|
}
|
|
break;
|
|
case LC_MAIN:
|
|
{
|
|
entry_point_command command;
|
|
source->Seek(address_);
|
|
source->Read(&command, sizeof(command));
|
|
|
|
command.entryoff = file.entry_point() - file.segment_list()->GetBaseSegment()->address();
|
|
|
|
file.Write(&command, sizeof(command));
|
|
need_copy = false;
|
|
}
|
|
break;
|
|
case LC_FUNCTION_STARTS:
|
|
case LC_DATA_IN_CODE:
|
|
case LC_DYLIB_CODE_SIGN_DRS:
|
|
{
|
|
linkedit_data_command command;
|
|
source->Seek(address_);
|
|
source->Read(&command, sizeof(command));
|
|
command.dataoff = offset_;
|
|
file.Write(&command, sizeof(command));
|
|
need_copy = false;
|
|
}
|
|
break;
|
|
case LC_ID_DYLIB:
|
|
if (file.export_list()->name() != source->export_list()->name()) {
|
|
std::string name = file.export_list()->name();
|
|
name.resize(AlignValue(name.size() + 1, (file.cpu_address_size() == osDWord) ? sizeof(uint32_t) : sizeof(uint64_t)) - 1, 0);
|
|
dylib_command command;
|
|
source->Seek(address_);
|
|
source->Read(&command, sizeof(command));
|
|
command.cmdsize = static_cast<uint32_t>(sizeof(command) + name.size() + 1);
|
|
command.dylib.name.offset = sizeof(command);
|
|
file.Write(&command, sizeof(command));
|
|
file.Write(name.c_str(), name.size() + 1);
|
|
need_copy = false;
|
|
}
|
|
break;
|
|
}
|
|
|
|
if (need_copy) {
|
|
source->Seek(address_);
|
|
file.CopyFrom(*source, size_);
|
|
}
|
|
|
|
address_ = position;
|
|
}
|
|
|
|
void MacLoadCommand::Rebase(uint64_t delta_base)
|
|
{
|
|
address_ += delta_base;
|
|
}
|
|
|
|
/**
|
|
* MacLoadCommandList
|
|
*/
|
|
|
|
MacLoadCommandList::MacLoadCommandList(MacArchitecture *owner)
|
|
: BaseCommandList(owner)
|
|
{
|
|
|
|
}
|
|
|
|
MacLoadCommandList::MacLoadCommandList(MacArchitecture *owner, const MacLoadCommandList &src)
|
|
: BaseCommandList(owner, src)
|
|
{
|
|
|
|
}
|
|
|
|
MacLoadCommandList *MacLoadCommandList::Clone(MacArchitecture *owner) const
|
|
{
|
|
MacLoadCommandList *load_command_list = new MacLoadCommandList(owner, *this);
|
|
return load_command_list;
|
|
}
|
|
|
|
MacLoadCommand *MacLoadCommandList::item(size_t index) const
|
|
{
|
|
return reinterpret_cast<MacLoadCommand*>(BaseCommandList::item(index));
|
|
}
|
|
|
|
void MacLoadCommandList::ReadFromFile(MacArchitecture &file, size_t count)
|
|
{
|
|
uint64_t pos = file.Tell();
|
|
Reserve(count);
|
|
for (size_t i = 0; i < count; i++) {
|
|
file.Seek(pos);
|
|
MacLoadCommand *command = new MacLoadCommand(this);
|
|
AddObject(command);
|
|
command->ReadFromFile(file);
|
|
pos += command->size();
|
|
}
|
|
}
|
|
|
|
void MacLoadCommandList::WriteToFile(MacArchitecture &file)
|
|
{
|
|
for (size_t i = 0; i < count(); i++) {
|
|
item(i)->WriteToFile(file);
|
|
}
|
|
}
|
|
|
|
void MacLoadCommandList::Pack()
|
|
{
|
|
for (size_t i = count(); i > 0 ; i--) {
|
|
MacLoadCommand *load_command = item(i - 1);
|
|
switch (load_command->type()) {
|
|
case LC_SEGMENT:
|
|
case LC_SEGMENT_64:
|
|
case LC_LOAD_DYLIB:
|
|
case LC_LOAD_WEAK_DYLIB:
|
|
case LC_CODE_SIGNATURE:
|
|
delete load_command;
|
|
break;
|
|
}
|
|
}
|
|
}
|
|
|
|
void MacLoadCommandList::Add(uint32_t type, IObject *object)
|
|
{
|
|
MacLoadCommand *command = new MacLoadCommand(this, type, object);
|
|
if (type == LC_SEGMENT || type == LC_SEGMENT_64) {
|
|
// insert segment commands to the top of list
|
|
for (size_t i = 0; i < count(); i++) {
|
|
MacLoadCommand *lc = item(i);
|
|
if (lc->type() == LC_SEGMENT || lc->type() == LC_SEGMENT_64)
|
|
continue;
|
|
InsertObject(i, command);
|
|
return;
|
|
}
|
|
}
|
|
AddObject(command);
|
|
}
|
|
|
|
MacLoadCommand *MacLoadCommandList::GetCommandByObject(void *object) const
|
|
{
|
|
for (size_t i = 0; i < count(); i++) {
|
|
MacLoadCommand *load_command = item(i);
|
|
if (load_command->object() == object)
|
|
return load_command;
|
|
}
|
|
return NULL;
|
|
}
|
|
|
|
/**
|
|
* MacSegment
|
|
*/
|
|
|
|
MacSegment::MacSegment(MacSegmentList *owner)
|
|
: BaseSection(owner), address_(0), size_(0),
|
|
physical_offset_(0), physical_size_(0), nsects_(0), maxprot_(VM_PROT_NONE), initprot_(VM_PROT_NONE)
|
|
{
|
|
|
|
}
|
|
|
|
MacSegment::MacSegment(MacSegmentList *owner,uint64_t address, uint64_t size, uint32_t physical_offset,
|
|
uint32_t physical_size, uint32_t initprot, const std::string &name)
|
|
: BaseSection(owner), address_(address), size_(size), physical_offset_(physical_offset), physical_size_(physical_size),
|
|
nsects_(0), maxprot_(initprot), initprot_(initprot), name_(name)
|
|
{
|
|
|
|
}
|
|
|
|
MacSegment::MacSegment(MacSegmentList *owner, const MacSegment &src)
|
|
: BaseSection(owner, src), nsects_(0)
|
|
{
|
|
address_ = src.address_;
|
|
size_ = src.size_;
|
|
physical_offset_ = src.physical_offset_;
|
|
physical_size_ = src.physical_size_;
|
|
maxprot_ = src.maxprot_;
|
|
initprot_ = src.initprot_;
|
|
name_ = src.name_;
|
|
}
|
|
|
|
MacSegment *MacSegment::Clone(ISectionList *owner) const
|
|
{
|
|
MacSegment *segment = new MacSegment(reinterpret_cast<MacSegmentList *>(owner), *this);
|
|
return segment;
|
|
}
|
|
|
|
uint32_t MacSegment::memory_type() const
|
|
{
|
|
uint32_t res = mtNone;
|
|
if (initprot_ & VM_PROT_READ)
|
|
res |= mtReadable;
|
|
if (initprot_ & VM_PROT_WRITE)
|
|
res |= mtWritable;
|
|
if (initprot_ & VM_PROT_EXECUTE)
|
|
res |= mtExecutable;
|
|
return res;
|
|
}
|
|
|
|
void MacSegment::ReadFromFile(MacArchitecture &file)
|
|
{
|
|
uint32_t need_type = (file.cpu_address_size() == osDWord) ? LC_SEGMENT : LC_SEGMENT_64;
|
|
uint32_t type = file.ReadDWord();
|
|
|
|
if (type != need_type)
|
|
throw std::runtime_error("Invalid segment type");
|
|
|
|
if (file.cpu_address_size() == osDWord) {
|
|
segment_command command = segment_command();
|
|
file.Read(&command.cmdsize, sizeof(command) - offsetof(segment_command, cmdsize));
|
|
address_ = command.vmaddr;
|
|
size_ = command.vmsize;
|
|
physical_offset_ = command.fileoff;
|
|
physical_size_ = command.filesize;
|
|
nsects_ = command.nsects;
|
|
maxprot_ = command.maxprot;
|
|
initprot_ = command.initprot;
|
|
name_ = std::string(command.segname, strnlen(command.segname, sizeof(command.segname)));
|
|
} else {
|
|
segment_command_64 command = segment_command_64();
|
|
file.Read(&command.cmdsize, sizeof(command) - offsetof(segment_command_64, cmdsize));
|
|
if (command.filesize >> 32)
|
|
throw std::runtime_error("Segment size is too large");
|
|
if (command.fileoff >> 32)
|
|
throw std::runtime_error("Segment offset is too large");
|
|
address_ = command.vmaddr;
|
|
size_ = command.vmsize;
|
|
physical_offset_ = static_cast<uint32_t>(command.fileoff);
|
|
physical_size_ = static_cast<uint32_t>(command.filesize);
|
|
nsects_ = command.nsects;
|
|
maxprot_ = command.maxprot;
|
|
initprot_ = command.initprot;
|
|
name_ = std::string(command.segname, strnlen(command.segname, sizeof(command.segname)));
|
|
}
|
|
|
|
// read sections for this segment
|
|
file.section_list()->ReadFromFile(file, nsects_, this);
|
|
}
|
|
|
|
void MacSegment::WriteToFile(MacArchitecture &file)
|
|
{
|
|
size_t i;
|
|
MacSectionList *section_list = file.section_list();
|
|
|
|
// calc child section sount
|
|
nsects_ = 0;
|
|
for (i = 0; i < section_list->count(); i++) {
|
|
if (section_list->item(i)->parent() == this)
|
|
nsects_++;
|
|
}
|
|
|
|
// write header
|
|
uint64_t command_pos = file.Tell();
|
|
if (file.cpu_address_size() == osDWord) {
|
|
segment_command cmd = segment_command();
|
|
cmd.cmd = LC_SEGMENT;
|
|
cmd.vmaddr = static_cast<uint32_t>(address_);
|
|
cmd.vmsize = static_cast<uint32_t>(size_);
|
|
cmd.fileoff = static_cast<uint32_t>(physical_offset_);
|
|
cmd.filesize = static_cast<uint32_t>(physical_size_);
|
|
cmd.nsects = nsects_;
|
|
cmd.maxprot = maxprot_;
|
|
cmd.initprot = initprot_;
|
|
memcpy(cmd.segname, name_.c_str(), std::min(name_.size(), sizeof(cmd.segname)));
|
|
file.Write(&cmd, sizeof(cmd));
|
|
} else {
|
|
segment_command_64 cmd = segment_command_64();
|
|
cmd.cmd = LC_SEGMENT_64;
|
|
cmd.vmaddr = address_;
|
|
cmd.vmsize = size_;
|
|
cmd.fileoff = physical_offset_;
|
|
cmd.filesize = physical_size_;
|
|
cmd.nsects = nsects_;
|
|
cmd.maxprot = maxprot_;
|
|
cmd.initprot = initprot_;
|
|
memcpy(cmd.segname, name_.c_str(), std::min(name_.size(), sizeof(cmd.segname)));
|
|
file.Write(&cmd, sizeof(cmd));
|
|
}
|
|
|
|
// write child sections
|
|
for (i = 0; i < section_list->count(); i++) {
|
|
MacSection *section = section_list->item(i);
|
|
if (section->parent() == this)
|
|
section->WriteToFile(file);
|
|
}
|
|
|
|
// write command size
|
|
uint64_t current_pos = file.Tell();
|
|
file.Seek(command_pos + offsetof(segment_command, cmdsize));
|
|
uint32_t command_size = static_cast<uint32_t>(current_pos - command_pos);
|
|
file.Write(&command_size, sizeof(command_size));
|
|
file.Seek(current_pos);
|
|
}
|
|
|
|
void MacSegment::update_type(uint32_t mt)
|
|
{
|
|
if (mt & mtReadable)
|
|
initprot_ |= VM_PROT_READ;
|
|
if (mt & mtWritable)
|
|
initprot_ |= VM_PROT_WRITE;
|
|
if (mt & mtExecutable)
|
|
initprot_ |= VM_PROT_EXECUTE;
|
|
|
|
maxprot_ = (initprot_ == VM_PROT_NONE) ? initprot_ : VM_PROT_READ | VM_PROT_WRITE | VM_PROT_EXECUTE;
|
|
}
|
|
|
|
void MacSegment::Rebase(uint64_t delta_base)
|
|
{
|
|
address_ += delta_base;
|
|
}
|
|
|
|
/**
|
|
* MacSegmentList
|
|
*/
|
|
|
|
MacSegmentList::MacSegmentList(MacArchitecture *owner)
|
|
: BaseSectionList(owner)
|
|
{
|
|
|
|
}
|
|
|
|
MacSegmentList::MacSegmentList(MacArchitecture *owner, const MacSegmentList &src)
|
|
: BaseSectionList(owner, src)
|
|
{
|
|
|
|
}
|
|
|
|
MacSegmentList *MacSegmentList::Clone(MacArchitecture *owner) const
|
|
{
|
|
MacSegmentList *list = new MacSegmentList(owner, *this);
|
|
return list;
|
|
}
|
|
|
|
MacSegment *MacSegmentList::item(size_t index) const
|
|
{
|
|
return reinterpret_cast<MacSegment*>(BaseSectionList::item(index));
|
|
}
|
|
|
|
MacSegment *MacSegmentList::Add()
|
|
{
|
|
MacSegment *segment = new MacSegment(this);
|
|
AddObject(segment);
|
|
return segment;
|
|
}
|
|
|
|
void MacSegmentList::ReadFromFile(MacArchitecture &file)
|
|
{
|
|
uint32_t cmd_type = (file.cpu_address_size() == osDWord) ? LC_SEGMENT : LC_SEGMENT_64;
|
|
MacLoadCommandList *command_list = file.command_list();
|
|
for (size_t i = 0; i < command_list->count(); i++) {
|
|
MacLoadCommand *command = command_list->item(i);
|
|
if (command->type() == cmd_type) {
|
|
file.Seek(command->address());
|
|
Add()->ReadFromFile(file);
|
|
}
|
|
}
|
|
}
|
|
|
|
void MacSegmentList::WriteToFile(MacArchitecture &file)
|
|
{
|
|
uint32_t cmd_type = (file.cpu_address_size() == osDWord) ? LC_SEGMENT : LC_SEGMENT_64;
|
|
for (size_t i = 0; i < count(); i++) {
|
|
MacSegment *segment = item(i);
|
|
file.command_list()->Add(cmd_type, segment);
|
|
}
|
|
}
|
|
|
|
MacSegment *MacSegmentList::last() const
|
|
{
|
|
return reinterpret_cast<MacSegment *>(BaseSectionList::last());
|
|
}
|
|
|
|
MacSegment *MacSegmentList::Add(uint64_t address, uint64_t size, uint32_t physical_offset, uint32_t physical_size, uint32_t initprot, const std::string &name)
|
|
{
|
|
MacSegment *segment = new MacSegment(this, address, size, physical_offset, physical_size, initprot, name);
|
|
AddObject(segment);
|
|
return segment;
|
|
}
|
|
|
|
MacSegment *MacSegmentList::GetSectionByName(const std::string &name) const
|
|
{
|
|
return reinterpret_cast<MacSegment *>(BaseSectionList::GetSectionByName(name));
|
|
}
|
|
|
|
MacSegment *MacSegmentList::GetSectionByAddress(uint64_t address) const
|
|
{
|
|
return reinterpret_cast<MacSegment *>(BaseSectionList::GetSectionByAddress(address));
|
|
}
|
|
|
|
MacSegment *MacSegmentList::GetBaseSegment() const
|
|
{
|
|
for (size_t i = 0; i < count(); i++) {
|
|
MacSegment *segment = item(i);
|
|
if (segment->physical_offset() == 0 && segment->physical_size())
|
|
return segment;
|
|
}
|
|
return NULL;
|
|
}
|
|
|
|
/**
|
|
* MacStringTable
|
|
*/
|
|
|
|
std::string MacStringTable::GetString(uint32_t pos) const
|
|
{
|
|
size_t i, len;
|
|
|
|
if (pos >= data_.size())
|
|
throw std::runtime_error("Invalid index for string table");
|
|
|
|
len = data_.size() - pos;
|
|
for (i = 0; i < len; i++) {
|
|
if (data_[pos + i] == 0) {
|
|
len = i;
|
|
break;
|
|
}
|
|
}
|
|
if (len == data_.size() - pos)
|
|
throw std::runtime_error("Invalid format");
|
|
|
|
return std::string(&data_[pos], len);
|
|
}
|
|
|
|
uint32_t MacStringTable::AddString(const std::string &str)
|
|
{
|
|
if (str.empty())
|
|
return 1;
|
|
|
|
uint32_t res = static_cast<uint32_t>(data_.size());
|
|
data_.insert(data_.end(), str.c_str(), str.c_str() + str.size() + 1);
|
|
|
|
return res;
|
|
};
|
|
|
|
void MacStringTable::clear()
|
|
{
|
|
data_.clear();
|
|
AddString(" ");
|
|
}
|
|
|
|
void MacStringTable::ReadFromFile(MacArchitecture &file)
|
|
{
|
|
symtab_command *symtab = file.symtab();
|
|
|
|
if (!symtab->cmd)
|
|
return;
|
|
|
|
data_.resize(symtab->strsize);
|
|
file.Seek(symtab->stroff);
|
|
file.Read(data_.data(), data_.size());
|
|
}
|
|
|
|
void MacStringTable::WriteToFile(MacArchitecture &file)
|
|
{
|
|
symtab_command *symtab = file.symtab();
|
|
// align table size
|
|
size_t aligned_size = AlignValue(data_.size(), OperandSizeToValue(file.cpu_address_size()));
|
|
while (data_.size() < aligned_size) {
|
|
data_.push_back(0);
|
|
}
|
|
|
|
symtab->strsize = (uint32_t)data_.size();
|
|
if (!symtab->strsize) {
|
|
symtab->stroff = 0;
|
|
return;
|
|
}
|
|
|
|
symtab->stroff = static_cast<uint32_t>(file.Tell());
|
|
file.Write(data_.data(), data_.size());
|
|
}
|
|
|
|
/**
|
|
* MacSymbol
|
|
*/
|
|
|
|
MacSymbol::MacSymbol(MacSymbolList *owner)
|
|
: IObject(), owner_(owner), type_(0), sect_(0), desc_(0), value_(0), is_deleted_(false)
|
|
{
|
|
|
|
}
|
|
|
|
MacSymbol::MacSymbol(MacSymbolList *owner, const MacSymbol &src)
|
|
: IObject(src), owner_(owner)
|
|
{
|
|
type_ = src.type_;
|
|
sect_ = src.sect_;
|
|
desc_ = src.desc_;
|
|
value_ = src.value_;
|
|
name_ = src.name_;
|
|
is_deleted_ = src.is_deleted_;
|
|
}
|
|
|
|
MacSymbol::~MacSymbol()
|
|
{
|
|
if (owner_)
|
|
owner_->RemoveObject(this);
|
|
}
|
|
|
|
MacSymbol *MacSymbol::Clone(MacSymbolList *owner) const
|
|
{
|
|
MacSymbol *symbol = new MacSymbol(owner, *this);
|
|
return symbol;
|
|
}
|
|
|
|
void MacSymbol::ReadFromFile(MacArchitecture &file)
|
|
{
|
|
uint32_t strx;
|
|
if (file.cpu_address_size() == osDWord) {
|
|
struct nlist nl;
|
|
file.Read(&nl, sizeof(nl));
|
|
strx = nl.n_un.n_strx;
|
|
type_ = nl.n_type;
|
|
sect_ = nl.n_sect;
|
|
desc_ = nl.n_desc;
|
|
value_ = nl.n_value;
|
|
} else {
|
|
nlist_64 nl;
|
|
file.Read(&nl, sizeof(nl));
|
|
strx = nl.n_un.n_strx;
|
|
type_ = nl.n_type;
|
|
sect_ = nl.n_sect;
|
|
desc_ = nl.n_desc;
|
|
value_ = nl.n_value;
|
|
}
|
|
|
|
name_ = file.string_table()->GetString(strx);
|
|
}
|
|
|
|
void MacSymbol::WriteToFile(MacArchitecture &file)
|
|
{
|
|
uint32_t strx = file.string_table()->AddString(name_);
|
|
|
|
if ((type_ & (N_STAB | N_TYPE)) == N_SECT && sect_ != NO_SECT) {
|
|
for (size_t i = 0; i < file.section_list()->count(); i++) {
|
|
MacSection *section = file.section_list()->item(i);
|
|
if (value_ < section->address() + section->size()) {
|
|
sect_ = static_cast<uint8_t>(i + 1);
|
|
break;
|
|
}
|
|
}
|
|
}
|
|
|
|
if (file.cpu_address_size() == osDWord) {
|
|
struct nlist nl;
|
|
nl.n_un.n_strx = strx;
|
|
nl.n_type = type_;
|
|
nl.n_sect = sect_;
|
|
nl.n_desc = desc_;
|
|
nl.n_value = static_cast<uint32_t>(value_);
|
|
file.Write(&nl, sizeof(nl));
|
|
} else {
|
|
nlist_64 nl;
|
|
nl.n_un.n_strx = strx;
|
|
nl.n_type = type_;
|
|
nl.n_sect = sect_;
|
|
nl.n_desc = desc_;
|
|
nl.n_value = value_;
|
|
file.Write(&nl, sizeof(nl));
|
|
}
|
|
}
|
|
|
|
uint8_t MacSymbol::library_ordinal() const
|
|
{
|
|
if ((type_ & (N_STAB | N_TYPE)) == N_UNDF)
|
|
return (desc_ >> 8) & 0xff;
|
|
|
|
return 0;
|
|
}
|
|
|
|
void MacSymbol::set_library_ordinal(uint8_t library_ordinal)
|
|
{
|
|
if ((type_ & (N_STAB | N_TYPE)) == N_UNDF) {
|
|
desc_ &= 0xff;
|
|
desc_ |= library_ordinal << 8;
|
|
}
|
|
}
|
|
|
|
/**
|
|
* MacSymbolList
|
|
*/
|
|
|
|
MacSymbolList::MacSymbolList()
|
|
: ObjectList<MacSymbol>()
|
|
{
|
|
|
|
}
|
|
|
|
MacSymbolList::MacSymbolList(const MacSymbolList &src)
|
|
: ObjectList<MacSymbol>(src)
|
|
{
|
|
for (size_t i = 0; i < src.count(); i++) {
|
|
AddObject(src.item(i)->Clone(this));
|
|
}
|
|
}
|
|
|
|
MacSymbolList *MacSymbolList::Clone() const
|
|
{
|
|
MacSymbolList *list = new MacSymbolList(*this);
|
|
return list;
|
|
}
|
|
|
|
void MacSymbolList::ReadFromFile(MacArchitecture &file, size_t count)
|
|
{
|
|
symtab_command *symtab = file.symtab();
|
|
if (!symtab->cmd)
|
|
return;
|
|
|
|
file.Seek(symtab->symoff);
|
|
for (size_t i = 0; i < count; i++) {
|
|
MacSymbol *symbol = new MacSymbol(this);
|
|
symbol->ReadFromFile(file);
|
|
AddObject(symbol);
|
|
}
|
|
}
|
|
|
|
void MacSymbolList::WriteToFile(MacArchitecture &file)
|
|
{
|
|
symtab_command *symtab = file.symtab();
|
|
symtab->nsyms = static_cast<uint32_t>(count());
|
|
symtab->symoff = static_cast<uint32_t>(file.Tell());
|
|
|
|
dysymtab_command *dysymtab = file.dysymtab();
|
|
if (!dysymtab->cmd)
|
|
return;
|
|
|
|
dysymtab->ilocalsym = 0;
|
|
dysymtab->nlocalsym = 0;
|
|
dysymtab->iextdefsym = 0;
|
|
dysymtab->nextdefsym = 0;
|
|
dysymtab->iundefsym = 0;
|
|
dysymtab->nundefsym = 0;
|
|
|
|
for (size_t i = 0; i < count(); i++) {
|
|
MacSymbol *symbol = item(i);
|
|
symbol->WriteToFile(file);
|
|
|
|
if ((symbol->type() & (N_STAB | N_TYPE)) == N_UNDF) {
|
|
// undefined symbol
|
|
if (!dysymtab->nundefsym)
|
|
dysymtab->iundefsym = static_cast<uint32_t>(i);
|
|
dysymtab->nundefsym++;
|
|
} else if (symbol->type() & N_EXT) {
|
|
// external symbol
|
|
if (!dysymtab->nextdefsym)
|
|
dysymtab->iextdefsym = static_cast<uint32_t>(i);
|
|
dysymtab->nextdefsym++;
|
|
} else {
|
|
// local symbol
|
|
if (!dysymtab->nlocalsym)
|
|
dysymtab->ilocalsym = static_cast<uint32_t>(i);
|
|
dysymtab->nlocalsym++;
|
|
}
|
|
}
|
|
}
|
|
|
|
MacSymbol *MacSymbolList::GetSymbol(const std::string &name, int library_ordinal) const
|
|
{
|
|
for (size_t i = 0; i < count(); i++) {
|
|
MacSymbol *symbol = item(i);
|
|
if ((symbol->type() & (N_STAB | N_TYPE)) == N_UNDF && symbol->name() == name) {
|
|
if (library_ordinal < 0 || library_ordinal == symbol->library_ordinal())
|
|
return symbol;
|
|
}
|
|
}
|
|
return NULL;
|
|
}
|
|
|
|
void MacSymbolList::Pack()
|
|
{
|
|
for (size_t i = count(); i > 0 ; i--) {
|
|
MacSymbol *symbol = item(i - 1);
|
|
if (symbol->is_deleted())
|
|
delete symbol;
|
|
}
|
|
}
|
|
|
|
/**
|
|
* MacSection
|
|
*/
|
|
|
|
MacSection::MacSection(MacSectionList *owner, MacSegment *parent)
|
|
: BaseSection(owner), address_(0), size_(0), offset_(0), align_(0), reloff_(0),
|
|
nreloc_(0), flags_(0), reserved1_(0), reserved2_(0), parent_(parent)
|
|
{
|
|
|
|
}
|
|
|
|
MacSection::MacSection(MacSectionList *owner, MacSegment *parent, uint64_t address, uint32_t size, uint32_t offset, uint32_t flags, const std::string &name, const std::string &segment_name)
|
|
: BaseSection(owner), name_(name), address_(address), size_(size), offset_(offset), align_(0), reloff_(0),
|
|
nreloc_(0), flags_(flags), reserved1_(0), reserved2_(0), parent_(parent), segment_name_(segment_name)
|
|
{
|
|
|
|
}
|
|
|
|
MacSection::~MacSection()
|
|
{
|
|
|
|
}
|
|
|
|
MacSection::MacSection(MacSectionList *owner, const MacSection &src)
|
|
: BaseSection(owner, src)
|
|
{
|
|
parent_ = src.parent_;
|
|
address_ = src.address_;
|
|
size_ = src.size_;
|
|
offset_ = src.offset_;
|
|
name_ = src.name_;
|
|
align_ = src.align_;
|
|
reloff_ = src.reloff_;
|
|
nreloc_ = src.nreloc_;
|
|
flags_ = src.flags_;
|
|
reserved1_ = src.reserved1_;
|
|
reserved2_ = src.reserved2_;
|
|
segment_name_ = src.segment_name_;
|
|
}
|
|
|
|
MacSection *MacSection::Clone(ISectionList *owner) const
|
|
{
|
|
MacSection *section = new MacSection(reinterpret_cast<MacSectionList *>(owner), *this);
|
|
return section;
|
|
}
|
|
|
|
void MacSection::ReadFromFile(MacArchitecture &file)
|
|
{
|
|
if (file.cpu_address_size() == osDWord) {
|
|
section sec;
|
|
file.Read(&sec, sizeof(sec));
|
|
name_ = std::string(sec.sectname, strnlen(sec.sectname, sizeof(sec.sectname)));
|
|
segment_name_ = std::string(sec.segname, strnlen(sec.segname, sizeof(sec.segname)));
|
|
address_ = sec.addr;
|
|
size_ = sec.size;
|
|
offset_ = sec.offset;
|
|
align_ = sec.align;
|
|
reloff_ = sec.reloff;
|
|
nreloc_ = sec.nreloc;
|
|
flags_ = sec.flags;
|
|
reserved1_ = sec.reserved1;
|
|
reserved2_ = sec.reserved2;
|
|
} else {
|
|
section_64 sec;
|
|
file.Read(&sec, sizeof(sec));
|
|
name_ = std::string(sec.sectname, strnlen(sec.sectname, sizeof(sec.sectname)));
|
|
segment_name_ = std::string(sec.segname, strnlen(sec.segname, sizeof(sec.segname)));
|
|
address_ = sec.addr;
|
|
size_ = static_cast<uint32_t>(sec.size);
|
|
offset_ = sec.offset;
|
|
align_ = sec.align;
|
|
reloff_ = sec.reloff;
|
|
nreloc_ = sec.nreloc;
|
|
flags_ = sec.flags;
|
|
reserved1_ = sec.reserved1;
|
|
reserved2_ = sec.reserved2;
|
|
}
|
|
}
|
|
|
|
void MacSection::WriteToFile(MacArchitecture &file)
|
|
{
|
|
if (file.cpu_address_size() == osDWord) {
|
|
section sec = section();
|
|
sec.addr = static_cast<uint32_t>(address_);
|
|
sec.size = static_cast<uint32_t>(size_);
|
|
sec.offset = offset_;
|
|
sec.align = align_;
|
|
sec.reloff = reloff_;
|
|
sec.nreloc = nreloc_;
|
|
sec.flags = flags_;
|
|
sec.reserved1 = reserved1_;
|
|
sec.reserved2 = reserved2_;
|
|
memcpy(sec.sectname, name_.c_str(), std::min(name_.size(), sizeof(sec.sectname)));
|
|
memcpy(sec.segname, segment_name_.c_str(), std::min(segment_name_.size(), sizeof(sec.segname)));
|
|
file.Write(&sec, sizeof(sec));
|
|
} else {
|
|
section_64 sec = section_64();
|
|
sec.addr = address_;
|
|
sec.size = size_;
|
|
sec.offset = offset_;
|
|
sec.align = align_;
|
|
sec.reloff = reloff_;
|
|
sec.nreloc = nreloc_;
|
|
sec.flags = flags_;
|
|
sec.reserved1 = reserved1_;
|
|
sec.reserved2 = reserved2_;
|
|
memcpy(sec.sectname, name_.c_str(), std::min(name_.size(), sizeof(sec.sectname)));
|
|
memcpy(sec.segname, segment_name_.c_str(), std::min(segment_name_.size(), sizeof(sec.segname)));
|
|
file.Write(&sec, sizeof(sec));
|
|
}
|
|
}
|
|
|
|
void MacSection::set_alignment(size_t value)
|
|
{
|
|
align_ = 0;
|
|
for (uint32_t i = 0; value && i < 8 * sizeof(size_t); i++) {
|
|
if (value & 1) {
|
|
align_ = i;
|
|
break;
|
|
}
|
|
value >>= 1;
|
|
}
|
|
}
|
|
|
|
void MacSection::Rebase(uint64_t delta_base)
|
|
{
|
|
address_ += delta_base;
|
|
}
|
|
|
|
/**
|
|
* MacSectionList
|
|
*/
|
|
|
|
MacSectionList::MacSectionList(MacArchitecture *owner)
|
|
: BaseSectionList(owner)
|
|
{
|
|
|
|
}
|
|
|
|
MacSectionList::MacSectionList(MacArchitecture *owner, const MacSectionList &src)
|
|
: BaseSectionList(owner, src)
|
|
{
|
|
|
|
}
|
|
|
|
MacSectionList *MacSectionList::Clone(MacArchitecture *owner) const
|
|
{
|
|
MacSectionList *list = new MacSectionList(owner, *this);
|
|
return list;
|
|
}
|
|
|
|
MacSection *MacSectionList::Add(MacSegment *segment, uint64_t address, uint32_t size, uint32_t offset, uint32_t flags, const std::string &name, const std::string &segment_name)
|
|
{
|
|
MacSection *section = new MacSection(this, segment, address, size, offset, flags, name, segment_name);
|
|
AddObject(section);
|
|
return section;
|
|
}
|
|
|
|
MacSection *MacSectionList::GetSectionByAddress(uint64_t address) const
|
|
{
|
|
return reinterpret_cast<MacSection *>(BaseSectionList::GetSectionByAddress(address));
|
|
}
|
|
|
|
MacSection *MacSectionList::GetSectionByName(const std::string &name) const
|
|
{
|
|
return reinterpret_cast<MacSection *>(BaseSectionList::GetSectionByName(name));
|
|
}
|
|
|
|
MacSection *MacSectionList::GetSectionByName(ISection *segment, const std::string &name) const
|
|
{
|
|
return reinterpret_cast<MacSection *>(BaseSectionList::GetSectionByName(segment, name));
|
|
}
|
|
|
|
void MacSectionList::ReadFromFile(MacArchitecture &file, size_t count, MacSegment *segment)
|
|
{
|
|
for (size_t i = 0; i < count; i++) {
|
|
MacSection *section = new MacSection(this, segment);
|
|
AddObject(section);
|
|
section->ReadFromFile(file);
|
|
}
|
|
}
|
|
|
|
MacSection *MacSectionList::item(size_t index) const
|
|
{
|
|
return reinterpret_cast<MacSection*>(BaseSectionList::item(index));
|
|
}
|
|
|
|
/**
|
|
* MacImportFunction
|
|
*/
|
|
|
|
MacImportFunction::MacImportFunction(IImport *owner, uint64_t address, uint8_t bind_type, size_t bind_offset, const std::string &name, uint32_t flags, int64_t addend, bool is_lazy, MacSymbol *symbol)
|
|
: BaseImportFunction(owner), address_(address), bind_type_(bind_type), name_(name),
|
|
flags_(flags), addend_(addend), is_lazy_(is_lazy), symbol_(symbol), bind_offset_(bind_offset)
|
|
{
|
|
|
|
}
|
|
|
|
MacImportFunction::MacImportFunction(IImport *owner, uint64_t address, APIType type, MapFunction *map_function)
|
|
: BaseImportFunction(owner), address_(address), bind_type_(0), flags_(0),
|
|
addend_(0), is_lazy_(false), symbol_(NULL), bind_offset_(0)
|
|
{
|
|
set_type(type);
|
|
set_map_function(map_function);
|
|
}
|
|
|
|
MacImportFunction::MacImportFunction(IImport *owner, const MacImportFunction &src)
|
|
: BaseImportFunction(owner, src)
|
|
{
|
|
address_ = src.address_;
|
|
bind_type_ = src.bind_type_;
|
|
name_ = src.name_;
|
|
flags_ = src.flags_;
|
|
addend_ = src.addend_;
|
|
is_lazy_ = src.is_lazy_;
|
|
bind_offset_ = src.bind_offset_;
|
|
symbol_ = src.symbol_;
|
|
}
|
|
|
|
MacImportFunction *MacImportFunction::Clone(IImport *owner) const
|
|
{
|
|
MacImportFunction *func = new MacImportFunction(owner, *this);
|
|
return func;
|
|
}
|
|
|
|
int MacImportFunction::library_ordinal() const
|
|
{
|
|
return reinterpret_cast<MacImport *>(owner())->library_ordinal();
|
|
}
|
|
|
|
bool MacImportFunction::is_weak() const
|
|
{
|
|
if (!symbol_)
|
|
return false;
|
|
|
|
if ((symbol_->type() & N_TYPE) == N_SECT)
|
|
return ((symbol_->desc() & N_WEAK_DEF) != 0);
|
|
else
|
|
return ((symbol_->desc() & N_REF_TO_WEAK) != 0); //-V
|
|
}
|
|
|
|
void MacImportFunction::Rebase(uint64_t delta_base)
|
|
{
|
|
if (address_)
|
|
address_ += delta_base;
|
|
}
|
|
|
|
std::string MacImportFunction::display_name(bool show_ret) const
|
|
{
|
|
return DemangleName(name_).display_name(show_ret);
|
|
}
|
|
|
|
/**
|
|
* MacImport
|
|
*/
|
|
|
|
MacImport::MacImport(MacImportList *owner, int library_ordinal, bool is_sdk)
|
|
: BaseImport(owner), library_ordinal_(library_ordinal), is_sdk_(is_sdk), timestamp_(0), current_version_(0), compatibility_version_(0),
|
|
is_weak_(false)
|
|
{
|
|
|
|
}
|
|
|
|
MacImport::MacImport(MacImportList *owner, int library_ordinal, const std::string &name, uint32_t current_version, uint32_t compatibility_version)
|
|
: BaseImport(owner), library_ordinal_(library_ordinal), name_(name), is_sdk_(false), timestamp_(0), current_version_(current_version), compatibility_version_(compatibility_version),
|
|
is_weak_(false)
|
|
{
|
|
|
|
}
|
|
|
|
MacImport::MacImport(MacImportList *owner, const MacImport &src)
|
|
: BaseImport(owner, src)
|
|
{
|
|
library_ordinal_ = src.library_ordinal_;
|
|
name_ = src.name_;
|
|
is_sdk_ = src.is_sdk_;
|
|
timestamp_ = src.timestamp_;
|
|
current_version_ = src.current_version_;
|
|
compatibility_version_ = src.compatibility_version_;
|
|
is_weak_ = src.is_weak_;
|
|
}
|
|
|
|
MacImport *MacImport::Clone(IImportList *owner) const
|
|
{
|
|
MacImport *list = new MacImport(reinterpret_cast<MacImportList *>(owner), *this);
|
|
return list;
|
|
}
|
|
|
|
void MacImport::ReadFromFile(MacArchitecture &file)
|
|
{
|
|
dylib_command command;
|
|
std::vector<char> namebuf;
|
|
size_t namebufsize;
|
|
uint64_t pos;
|
|
|
|
pos = file.Tell();
|
|
/* Read command header, verify the command */
|
|
file.Read(&command, sizeof(command));
|
|
if (command.cmdsize < sizeof(dylib_command) || command.dylib.name.offset >= command.cmdsize)
|
|
throw std::runtime_error("Invalid dylib_command");
|
|
|
|
timestamp_ = command.dylib.timestamp;
|
|
current_version_ = command.dylib.current_version;
|
|
compatibility_version_ = command.dylib.compatibility_version;
|
|
namebufsize = command.cmdsize - command.dylib.name.offset;
|
|
namebuf.resize(namebufsize);
|
|
file.Seek(command.dylib.name.offset + pos);
|
|
file.Read(&namebuf[0], namebuf.size());
|
|
name_ = NameToString(&namebuf[0], namebuf.size());
|
|
}
|
|
|
|
MacImportFunction *MacImport::Add(uint64_t address, uint8_t bind_type, size_t bind_offset, const std::string & name, uint32_t flags, int64_t addend, bool is_lazy, MacSymbol *symbol)
|
|
{
|
|
MacImportFunction *import_function = new MacImportFunction(this, address, bind_type, bind_offset, name, flags, addend, is_lazy, symbol);
|
|
AddObject(import_function);
|
|
return import_function;
|
|
}
|
|
|
|
void MacImport::WriteToFile(MacArchitecture &file)
|
|
{
|
|
size_t size;
|
|
dylib_command command;
|
|
|
|
command.cmd = is_weak_ ? LC_LOAD_WEAK_DYLIB : LC_LOAD_DYLIB;
|
|
command.cmdsize = (uint32_t)AlignValue(sizeof(command) + name_.size() + 1, OperandSizeToValue(file.cpu_address_size()));
|
|
command.dylib.name.offset = sizeof(command);
|
|
command.dylib.timestamp = timestamp_;
|
|
command.dylib.current_version = current_version_;
|
|
command.dylib.compatibility_version = compatibility_version_;
|
|
// write command
|
|
size = file.Write(&command, sizeof(command));
|
|
size += file.Write(name_.c_str(), name_.size() + 1);
|
|
// align
|
|
uint8_t b = 0;
|
|
while (size < command.cmdsize) {
|
|
size += file.Write(&b, sizeof(b));
|
|
}
|
|
}
|
|
|
|
MacImportFunction *MacImport::item(size_t index) const
|
|
{
|
|
return reinterpret_cast<MacImportFunction*>(BaseImport::item(index));
|
|
}
|
|
|
|
MacImportFunction *MacImport::GetFunctionByAddress(uint64_t address) const
|
|
{
|
|
return reinterpret_cast<MacImportFunction*>(BaseImport::GetFunctionByAddress(address));
|
|
}
|
|
|
|
MacImportFunction *MacImport::Add(uint64_t address, APIType type, MapFunction *map_function)
|
|
{
|
|
MacImportFunction *import_function = new MacImportFunction(this, address, type, map_function);
|
|
AddObject(import_function);
|
|
return import_function;
|
|
}
|
|
|
|
/**
|
|
* MacImportList
|
|
*/
|
|
|
|
MacImportList::MacImportList(MacArchitecture *owner)
|
|
: BaseImportList(owner)
|
|
{
|
|
|
|
}
|
|
|
|
MacImportList::MacImportList(MacArchitecture *owner, const MacImportList &src)
|
|
: BaseImportList(owner, src)
|
|
{
|
|
|
|
}
|
|
|
|
MacImportList *MacImportList::Clone(MacArchitecture *owner) const
|
|
{
|
|
MacImportList *list = new MacImportList(owner, *this);
|
|
return list;
|
|
}
|
|
|
|
MacImport *MacImportList::Add(int library_ordinal)
|
|
{
|
|
MacImport *imp = new MacImport(this, library_ordinal);
|
|
AddObject(imp);
|
|
return imp;
|
|
}
|
|
|
|
MacImport *MacImportList::AddSDK()
|
|
{
|
|
MacImport *sdk = new MacImport(this, 0, true);
|
|
AddObject(sdk);
|
|
return sdk;
|
|
}
|
|
|
|
MacImport *MacImportList::item(size_t index) const
|
|
{
|
|
return reinterpret_cast<MacImport*>(IImportList::item(index));
|
|
}
|
|
|
|
MacImportFunction *MacImportList::GetFunctionByAddress(uint64_t address) const
|
|
{
|
|
return reinterpret_cast<MacImportFunction *>(BaseImportList::GetFunctionByAddress(address));
|
|
}
|
|
|
|
MacImport *MacImportList::GetLibraryByOrdinal(int library_ordinal) const
|
|
{
|
|
for (size_t i = 0; i < count(); i++) {
|
|
MacImport *import = item(i);
|
|
if (import->library_ordinal() == library_ordinal)
|
|
return import;
|
|
}
|
|
|
|
return NULL;
|
|
}
|
|
|
|
MacImport *MacImportList::GetImportByName(const std::string &name) const
|
|
{
|
|
return reinterpret_cast<MacImport *>(BaseImportList::GetImportByName(name));
|
|
}
|
|
|
|
void MacImportList::ReadFromFile(MacArchitecture &file)
|
|
{
|
|
size_t i, j;
|
|
|
|
ILoadCommandList *command_list = file.command_list();
|
|
int library_ordinal = 1;
|
|
size_t c = command_list->count();
|
|
for (i = 0; i < c; i++) {
|
|
ILoadCommand *command = command_list->item(i);
|
|
if (command->type() == LC_LOAD_DYLIB || command->type() == LC_LOAD_WEAK_DYLIB) {
|
|
file.Seek(command->address());
|
|
MacImport *import = Add(library_ordinal++);
|
|
import->ReadFromFile(file);
|
|
if (command->type() == LC_LOAD_WEAK_DYLIB)
|
|
import->set_is_weak(true);
|
|
}
|
|
}
|
|
|
|
dyld_info_command *dyld_info = file.dyld_info();
|
|
if (dyld_info->cmd) {
|
|
ReadBindInfo(file, dyld_info->bind_off, dyld_info->bind_size);
|
|
ReadLazyBindInfo(file, dyld_info->lazy_bind_off, dyld_info->lazy_bind_size);
|
|
} else {
|
|
MacIndirectSymbolList *indirect_symbol_list = file.indirect_symbol_list();
|
|
for (i = 0; i < indirect_symbol_list->count(); i++) {
|
|
MacIndirectSymbol *indirect_symbol = indirect_symbol_list->item(i);
|
|
MacSymbol *symbol = indirect_symbol->symbol();
|
|
if (symbol && (symbol->type() & (N_STAB | N_TYPE)) == N_UNDF) {
|
|
MacSection *section = file.section_list()->GetSectionByAddress(indirect_symbol->address());
|
|
if (section && (section->type() == S_LAZY_SYMBOL_POINTERS
|
|
|| section->type() == S_NON_LAZY_SYMBOL_POINTERS
|
|
|| (section->type() == S_SYMBOL_STUBS && file.cpu_address_size() == osDWord && (section->flags() & S_ATTR_SELF_MODIFYING_CODE) && section->reserved2() == 5))) {
|
|
library_ordinal = symbol->library_ordinal();
|
|
if (library_ordinal > 0) {
|
|
MacImport *import = GetLibraryByOrdinal(library_ordinal);
|
|
if (!import)
|
|
throw std::runtime_error("Invalid library ordinal");
|
|
MacImportFunction *import_function = import->Add(indirect_symbol->address(), 0, 0, symbol->name(), 0, 0, section->type() != S_NON_LAZY_SYMBOL_POINTERS, symbol);
|
|
if (section->type() == S_SYMBOL_STUBS)
|
|
import_function->include_option(ioIsRelative);
|
|
}
|
|
} else if (section && section->type() != S_SYMBOL_STUBS) {
|
|
throw std::runtime_error("Invalid indirect symbol");
|
|
}
|
|
}
|
|
}
|
|
|
|
MacFixupList *fixup_list = file.fixup_list();
|
|
for (i = 0; i < fixup_list->count(); i++) {
|
|
MacFixup *fixup = fixup_list->item(i);
|
|
MacSymbol *symbol = fixup->symbol();
|
|
if (symbol && (symbol->type() & (N_STAB | N_TYPE)) == N_UNDF) {
|
|
library_ordinal = symbol->library_ordinal();
|
|
if (library_ordinal > 0) {
|
|
MacImport *import = GetLibraryByOrdinal(library_ordinal);
|
|
if (!import)
|
|
throw std::runtime_error("Invalid library ordinal");
|
|
import->Add(fixup->address(), 0, 0, symbol->name(), 0, 0, false, symbol);
|
|
}
|
|
}
|
|
}
|
|
}
|
|
|
|
static const ImportInfo default_info[] = {
|
|
{atNone, "_exit", ioNoReturn, ctNone},
|
|
{atNone, "_abort", ioNoReturn, ctNone},
|
|
{atNone, "__Unwind_Resume", ioNoReturn, ctNone},
|
|
{atNone, "__Unwind_Resume_or_Rethrow", ioNoReturn, ctNone},
|
|
{atNone, "__Unwind_RaiseException", ioNoReturn, ctNone},
|
|
{atNone, "___assert_rtn", ioNoReturn, ctNone},
|
|
{atNone, "___stack_chk_fail", ioNoReturn, ctNone},
|
|
{atNone, "___cxa_throw", ioNoReturn, ctNone},
|
|
{atNone, "___cxa_end_cleanup", ioNoReturn, ctNone},
|
|
{atNone, "___cxa_rethrow", ioNoReturn, ctNone},
|
|
{atNone, "___cxa_bad_cast", ioNoReturn, ctNone},
|
|
{atNone, "___cxa_bad_typeid", ioNoReturn, ctNone},
|
|
{atNone, "___cxa_call_terminate", ioNoReturn, ctNone},
|
|
{atNone, "___cxa_pure_virtual", ioNoReturn, ctNone},
|
|
{atNone, "___cxa_call_unexpected", ioNoReturn, ctNone},
|
|
{atNone, "__ZSt9terminatev", ioNoReturn, ctNone},
|
|
{atNone, "@System@@Halt0$qqrv", ioNoReturn, ctNone},
|
|
{atNone, "@System@@UnhandledException$qqrv", ioNoReturn, ctNone},
|
|
};
|
|
|
|
MacImportFunction *func;
|
|
for (size_t k = 0; k < count(); k++) {
|
|
MacImport *import = item(k);
|
|
|
|
std::string dll_name = os::ExtractFileName(import->name().c_str());
|
|
std::transform(dll_name.begin(), dll_name.end(), dll_name.begin(), tolower);
|
|
std::string sdk_name = "libvmprotectsdk.dylib";
|
|
|
|
if (dll_name.compare(sdk_name) == 0) {
|
|
import->set_is_sdk(true);
|
|
if (import->count() == 0) {
|
|
if (MacImport *flat_lookup_import = GetLibraryByOrdinal(BIND_SPECIAL_DYLIB_FLAT_LOOKUP)) {
|
|
for (i = 0; i < flat_lookup_import->count(); ) {
|
|
func = flat_lookup_import->item(i);
|
|
if (GetSDKInfo(func->name())) {
|
|
func->set_owner(import);
|
|
continue;
|
|
}
|
|
i++;
|
|
}
|
|
}
|
|
}
|
|
|
|
for (i = 0; i < import->count(); i++) {
|
|
func = import->item(i);
|
|
const ImportInfo *import_info = GetSDKInfo(func->name());
|
|
if (import_info) {
|
|
func->set_type(import_info->type);
|
|
if (import_info->options & ioHasCompilationType) {
|
|
func->include_option(ioHasCompilationType);
|
|
func->set_compilation_type(import_info->compilation_type);
|
|
if (import_info->options & ioLockToKey)
|
|
func->include_option(ioLockToKey);
|
|
}
|
|
}
|
|
}
|
|
} else {
|
|
size_t c = _countof(default_info);
|
|
const ImportInfo *import_info = default_info;
|
|
|
|
if (import_info) {
|
|
for (i = 0; i < import->count(); i++) {
|
|
func = import->item(i);
|
|
for (j = 0; j < c; j++) {
|
|
if (func->name().compare(import_info[j].name) == 0) {
|
|
func->set_type(import_info[j].type);
|
|
if (import_info[j].options & ioNative)
|
|
func->include_option(ioNative);
|
|
if (import_info[j].options & ioNoReturn)
|
|
func->include_option(ioNoReturn);
|
|
break;
|
|
}
|
|
}
|
|
}
|
|
}
|
|
}
|
|
}
|
|
}
|
|
|
|
const ImportInfo *MacImportList::GetSDKInfo(const std::string &name) const
|
|
{
|
|
if (!name.empty() && name[0] == '_')
|
|
return BaseImportList::GetSDKInfo(name.substr(1));
|
|
return NULL;
|
|
}
|
|
|
|
int MacImportList::GetMaxLibraryOrdinal() const
|
|
{
|
|
int res = 0;
|
|
for (size_t i = 0; i < count(); i++) {
|
|
MacImport *import = item(i);
|
|
if (import->library_ordinal() <= 0)
|
|
continue;
|
|
|
|
if (res < import->library_ordinal())
|
|
res = import->library_ordinal();
|
|
}
|
|
return res;
|
|
}
|
|
|
|
void MacImportList::ReadBindInfo(MacArchitecture &file, uint32_t bind_off, uint32_t bind_size)
|
|
{
|
|
size_t pos, seg_index = 0, i, c, skip, ptr_size;
|
|
uint8_t imm, opcode, bind_type = 0, sym_flags = 0;
|
|
int library_ordinal = 0;
|
|
bool done;
|
|
std::string sym_name;
|
|
int64_t addend = 0;
|
|
uint64_t addr, end_addr;
|
|
ISectionList *segment_list = file.segment_list();
|
|
ISection *segment;
|
|
MacImport *import = NULL;
|
|
EncodedData buf;
|
|
MacSymbol *symbol = NULL;
|
|
|
|
if (!bind_size)
|
|
return;
|
|
|
|
if (segment_list->count() < 1)
|
|
throw std::runtime_error("Invalid segment count in bind info");
|
|
|
|
file.Seek(bind_off);
|
|
buf.ReadFromFile(file, bind_size);
|
|
|
|
ptr_size = file.cpu_address_size() == osQWord ? 8 : 4;
|
|
|
|
segment = segment_list->item(0);
|
|
addr = segment->address();
|
|
end_addr = segment->address() + segment->size();
|
|
|
|
done = false;
|
|
for (pos = 0; !done && pos < bind_size; ) {
|
|
uint8_t b = buf.ReadByte(&pos);
|
|
imm = b & BIND_IMMEDIATE_MASK;
|
|
opcode = b & BIND_OPCODE_MASK;
|
|
switch (opcode) {
|
|
case BIND_OPCODE_DONE:
|
|
done = true;
|
|
break;
|
|
case BIND_OPCODE_SET_DYLIB_ORDINAL_IMM:
|
|
library_ordinal = imm;
|
|
import = NULL;
|
|
symbol = NULL;
|
|
break;
|
|
case BIND_OPCODE_SET_DYLIB_ORDINAL_ULEB:
|
|
library_ordinal = static_cast<int>(buf.ReadUleb128(&pos));
|
|
import = NULL;
|
|
symbol = NULL;
|
|
break;
|
|
case BIND_OPCODE_SET_DYLIB_SPECIAL_IMM:
|
|
/* the special ordinals are negative numbers */
|
|
if (imm == 0) {
|
|
library_ordinal = 0;
|
|
} else {
|
|
int8_t sign_extended = BIND_OPCODE_MASK | imm;
|
|
library_ordinal = sign_extended;
|
|
}
|
|
import = NULL;
|
|
symbol = NULL;
|
|
break;
|
|
case BIND_OPCODE_SET_SYMBOL_TRAILING_FLAGS_IMM:
|
|
sym_flags = imm;
|
|
sym_name = buf.ReadString(&pos);
|
|
symbol = NULL;
|
|
break;
|
|
case BIND_OPCODE_SET_TYPE_IMM:
|
|
bind_type = imm;
|
|
break;
|
|
case BIND_OPCODE_SET_ADDEND_SLEB:
|
|
addend = buf.ReadSleb128(&pos);
|
|
break;
|
|
case BIND_OPCODE_SET_SEGMENT_AND_OFFSET_ULEB:
|
|
seg_index = imm;
|
|
if (seg_index >= segment_list->count())
|
|
throw std::runtime_error("Invalid segment index in bind info");
|
|
segment = segment_list->item(seg_index);
|
|
addr = segment->address() + buf.ReadUleb128(&pos);
|
|
end_addr = segment->address() + segment->size();
|
|
break;
|
|
case BIND_OPCODE_ADD_ADDR_ULEB:
|
|
addr += buf.ReadUleb128(&pos);
|
|
break;
|
|
case BIND_OPCODE_DO_BIND:
|
|
case BIND_OPCODE_DO_BIND_ADD_ADDR_ULEB:
|
|
case BIND_OPCODE_DO_BIND_ADD_ADDR_IMM_SCALED:
|
|
if (addr >= end_addr)
|
|
throw std::runtime_error("Invalid binding address in bind info");
|
|
if (!import) {
|
|
import = GetLibraryByOrdinal(library_ordinal);
|
|
if (!import) {
|
|
if (library_ordinal <= 0) {
|
|
// special library
|
|
import = Add(library_ordinal);
|
|
}
|
|
else {
|
|
throw std::runtime_error("Invalid library ordinal in bind info");
|
|
}
|
|
}
|
|
}
|
|
if (!symbol) {
|
|
symbol = file.symbol_list()->GetSymbol(sym_name, library_ordinal);
|
|
if (!symbol)
|
|
throw std::runtime_error("Invalid symbol in bind info");
|
|
}
|
|
import->Add(addr, bind_type, 0, sym_name, sym_flags, addend, false, symbol);
|
|
if (opcode == BIND_OPCODE_DO_BIND) {
|
|
addr += ptr_size;
|
|
} else if (opcode == BIND_OPCODE_DO_BIND_ADD_ADDR_ULEB) {
|
|
addr += buf.ReadUleb128(&pos) + ptr_size;
|
|
} else if (opcode == BIND_OPCODE_DO_BIND_ADD_ADDR_IMM_SCALED) {
|
|
addr += (imm + 1) * ptr_size;
|
|
}
|
|
break;
|
|
case BIND_OPCODE_DO_BIND_ULEB_TIMES_SKIPPING_ULEB:
|
|
c = static_cast<size_t>(buf.ReadUleb128(&pos));
|
|
skip = static_cast<size_t>(buf.ReadUleb128(&pos));
|
|
for (i = 0; i < c; i++) {
|
|
if (addr >= end_addr)
|
|
throw std::runtime_error("Invalid binding address in bind info");
|
|
if (!import) {
|
|
import = GetLibraryByOrdinal(library_ordinal);
|
|
if (!import) {
|
|
if (library_ordinal <= 0) {
|
|
// special library
|
|
import = Add(library_ordinal);
|
|
}
|
|
else {
|
|
throw std::runtime_error("Invalid library ordinal in bind info");
|
|
}
|
|
}
|
|
}
|
|
if (!symbol) {
|
|
symbol = file.symbol_list()->GetSymbol(sym_name, library_ordinal);
|
|
if (!symbol)
|
|
throw std::runtime_error("Invalid symbol in bind info");
|
|
}
|
|
import->Add(addr, bind_type, 0, sym_name, sym_flags, addend, false, symbol);
|
|
addr += skip + ptr_size;
|
|
}
|
|
break;
|
|
default:
|
|
throw std::runtime_error("Invalid bind opcode in bind info");
|
|
}
|
|
}
|
|
}
|
|
|
|
void MacImportList::ReadLazyBindInfo(MacArchitecture &file, uint32_t lazy_bind_off, uint32_t lazy_bind_size)
|
|
{
|
|
uint8_t bind_type = BIND_TYPE_POINTER, imm, opcode, sym_flags = 0;
|
|
int library_ordinal = 0;
|
|
size_t pos, seg_index, ptrsize;
|
|
uint64_t addr, end_addr;
|
|
ISectionList *segment_list = file.segment_list();
|
|
ISection *segment;
|
|
MacImport *import = NULL;
|
|
std::string sym_name;
|
|
int64_t addend = 0;
|
|
EncodedData buf;
|
|
MacSymbol *symbol = NULL;
|
|
|
|
if (!lazy_bind_size)
|
|
return;
|
|
|
|
if (segment_list->count() < 1)
|
|
throw std::runtime_error("Invalid segment count in lazy bind info");
|
|
|
|
file.Seek(lazy_bind_off);
|
|
buf.ReadFromFile(file, lazy_bind_size);
|
|
|
|
ptrsize = OperandSizeToValue(file.cpu_address_size());
|
|
|
|
segment = segment_list->item(0);
|
|
addr = segment->address();
|
|
end_addr = segment->address() + segment->size();
|
|
|
|
size_t bind_offset = (size_t)-1;
|
|
for (pos = 0; pos < lazy_bind_size; ) {
|
|
if (bind_offset == (size_t)-1)
|
|
bind_offset = pos;
|
|
uint8_t b = buf.ReadByte(&pos);
|
|
imm = b & BIND_IMMEDIATE_MASK;
|
|
opcode = b & BIND_OPCODE_MASK;
|
|
switch (opcode) {
|
|
case BIND_OPCODE_DONE:
|
|
/* There is BIND_OPCODE_DONE at end of each lazy bind,
|
|
* don't stop until end of whole sequence. */
|
|
bind_offset = (size_t)-1;
|
|
break;
|
|
case BIND_OPCODE_SET_DYLIB_ORDINAL_IMM:
|
|
library_ordinal = imm;
|
|
import = NULL;
|
|
symbol = NULL;
|
|
break;
|
|
case BIND_OPCODE_SET_DYLIB_ORDINAL_ULEB:
|
|
library_ordinal = static_cast<int>(buf.ReadUleb128(&pos));
|
|
import = NULL;
|
|
symbol = NULL;
|
|
break;
|
|
case BIND_OPCODE_SET_DYLIB_SPECIAL_IMM:
|
|
/* the special ordinals are negative numbers */
|
|
if (imm == 0) {
|
|
library_ordinal = 0;
|
|
} else {
|
|
int8_t sign_extended = BIND_OPCODE_MASK | imm;
|
|
library_ordinal = sign_extended;
|
|
}
|
|
import = NULL;
|
|
symbol = NULL;
|
|
break;
|
|
case BIND_OPCODE_SET_SYMBOL_TRAILING_FLAGS_IMM:
|
|
sym_flags = imm;
|
|
sym_name = buf.ReadString(&pos);
|
|
symbol = NULL;
|
|
break;
|
|
case BIND_OPCODE_SET_TYPE_IMM:
|
|
bind_type = imm;
|
|
break;
|
|
case BIND_OPCODE_SET_ADDEND_SLEB:
|
|
addend = buf.ReadSleb128(&pos);
|
|
break;
|
|
case BIND_OPCODE_SET_SEGMENT_AND_OFFSET_ULEB:
|
|
seg_index = imm;
|
|
if (seg_index >= segment_list->count())
|
|
throw std::runtime_error("Invalid segment index in lazy bind info");
|
|
segment = segment_list->item(seg_index);
|
|
addr = segment->address() + buf.ReadUleb128(&pos);
|
|
end_addr = segment->address() + segment->size();
|
|
break;
|
|
case BIND_OPCODE_ADD_ADDR_ULEB:
|
|
addr += buf.ReadUleb128(&pos);
|
|
break;
|
|
case BIND_OPCODE_DO_BIND:
|
|
if (addr >= end_addr)
|
|
throw std::runtime_error("Invalid binding address in lazy bind info");
|
|
if (!import) {
|
|
import = GetLibraryByOrdinal(library_ordinal);
|
|
if (!import) {
|
|
if (library_ordinal <= 0) {
|
|
// special library
|
|
import = Add(library_ordinal);
|
|
}
|
|
else {
|
|
throw std::runtime_error("Invalid library ordinal in lazy bind info");
|
|
}
|
|
}
|
|
}
|
|
if (!symbol) {
|
|
symbol = file.symbol_list()->GetSymbol(sym_name, library_ordinal);
|
|
if (!symbol)
|
|
throw std::runtime_error("Invalid symbol in lazy bind info");
|
|
}
|
|
import->Add(addr, bind_type, bind_offset, sym_name, sym_flags, addend, true, symbol);
|
|
addr += ptrsize;
|
|
break;
|
|
default:
|
|
throw std::runtime_error("Invalid bind opcode in lazy bind info");
|
|
}
|
|
}
|
|
}
|
|
|
|
void MacImportList::Pack()
|
|
{
|
|
if (!has_sdk())
|
|
return;
|
|
|
|
size_t i, j;
|
|
MacImport *import;
|
|
uint8_t library_ordinal = 1;
|
|
|
|
for (i = 0; i < count(); i++) {
|
|
import = item(i);
|
|
if (import->is_sdk() || import->library_ordinal() <= 0)
|
|
continue;
|
|
|
|
import->set_library_ordinal(library_ordinal);
|
|
for (j = 0; j < import->count(); j++) {
|
|
import->item(j)->symbol()->set_library_ordinal(library_ordinal);
|
|
}
|
|
library_ordinal++;
|
|
}
|
|
|
|
for (i = count(); i > 0; i--) {
|
|
import = item(i - 1);
|
|
if (!import->is_sdk())
|
|
continue;
|
|
|
|
for (j = 0; j < import->count(); j++) {
|
|
import->item(j)->symbol()->set_deleted(true);
|
|
}
|
|
delete import;
|
|
}
|
|
}
|
|
|
|
void MacImportList::WriteToFile(MacArchitecture &file)
|
|
{
|
|
for (size_t i = 0; i < count(); i++) {
|
|
MacImport *import = item(i);
|
|
if (import->library_ordinal() <= 0)
|
|
continue;
|
|
|
|
file.command_list()->Add(import->is_weak() ? LC_LOAD_WEAK_DYLIB : LC_LOAD_DYLIB, import);
|
|
}
|
|
|
|
dyld_info_command *dyld_info = file.dyld_info();
|
|
if (!dyld_info->cmd)
|
|
return;
|
|
|
|
dyld_info->bind_off = static_cast<uint32_t>(file.Tell());
|
|
dyld_info->bind_size = static_cast<uint32_t>(WriteBindInfo(file));
|
|
if (!dyld_info->bind_size)
|
|
dyld_info->bind_off = 0;
|
|
|
|
dyld_info->weak_bind_off = static_cast<uint32_t>(file.Tell());
|
|
dyld_info->weak_bind_size = static_cast<uint32_t>(WriteWeakBindInfo(file));
|
|
if (!dyld_info->weak_bind_size)
|
|
dyld_info->weak_bind_off = 0;
|
|
|
|
dyld_info->lazy_bind_off = static_cast<uint32_t>(file.Tell());
|
|
dyld_info->lazy_bind_size = static_cast<uint32_t>(WriteLazyBindInfo(file));
|
|
if (!dyld_info->lazy_bind_size)
|
|
dyld_info->lazy_bind_off = 0;
|
|
}
|
|
|
|
struct BindingInfo {
|
|
uint8_t opcode;
|
|
uint64_t operand1;
|
|
uint64_t operand2;
|
|
std::string name;
|
|
BindingInfo(uint8_t opcode_, uint64_t operand1_, uint64_t operand2_ = 0, const std::string &name_ = "")
|
|
{
|
|
opcode = opcode_;
|
|
operand1 = operand1_;
|
|
operand2 = operand2_;
|
|
name = name_;
|
|
}
|
|
};
|
|
|
|
size_t MacImportList::WriteBindInfo(MacArchitecture &file)
|
|
{
|
|
size_t i, j;
|
|
std::vector<MacImportFunction *> info;
|
|
MacImport *import;
|
|
MacImportFunction *import_func;
|
|
|
|
for (i = 0; i < count(); i++) {
|
|
import = item(i);
|
|
for (j = 0; j < import->count(); j++) {
|
|
import_func = import->item(j);
|
|
if (import_func->is_lazy())
|
|
continue;
|
|
info.push_back(import_func);
|
|
}
|
|
}
|
|
if (info.size() == 0)
|
|
return 0;
|
|
|
|
std::sort(info.begin(), info.end(), BindInfoHelper());
|
|
|
|
std::vector<BindingInfo> mid;
|
|
MacSegment *segment = NULL;
|
|
int library_ordinal = 0x80000000;
|
|
std::string symbol_name;
|
|
uint8_t bind_type = 0;
|
|
uint64_t address = -1;
|
|
int64_t addend = 0;
|
|
size_t ptrsize = OperandSizeToValue(file.cpu_address_size());
|
|
|
|
for (std::vector<MacImportFunction *>::iterator it = info.begin(); it != info.end() ; ++it) {
|
|
import_func = *it;
|
|
if (library_ordinal != import_func->library_ordinal()) {
|
|
if (import_func->library_ordinal() <= 0) {
|
|
// special lookups are encoded as negative numbers in BindingInfo
|
|
mid.push_back(BindingInfo(BIND_OPCODE_SET_DYLIB_SPECIAL_IMM, import_func->library_ordinal()));
|
|
} else {
|
|
mid.push_back(BindingInfo(BIND_OPCODE_SET_DYLIB_ORDINAL_ULEB, import_func->library_ordinal()));
|
|
}
|
|
library_ordinal = import_func->library_ordinal();
|
|
}
|
|
|
|
if (symbol_name != import_func->name()) {
|
|
mid.push_back(BindingInfo(BIND_OPCODE_SET_SYMBOL_TRAILING_FLAGS_IMM, import_func->flags(), 0, import_func->name()));
|
|
symbol_name = import_func->name();
|
|
}
|
|
if (bind_type != import_func->bind_type()) {
|
|
mid.push_back(BindingInfo(BIND_OPCODE_SET_TYPE_IMM, import_func->bind_type()));
|
|
bind_type = import_func->bind_type();
|
|
}
|
|
if (address != import_func->address()) {
|
|
if (!segment || (import_func->address() < segment->address()) || (import_func->address() >= segment->address() + segment->size()) || import_func->address() < address) {
|
|
segment = file.segment_list()->GetSectionByAddress(import_func->address());
|
|
if (!segment)
|
|
throw std::runtime_error("Invalid binding address in bind info");
|
|
mid.push_back(BindingInfo(BIND_OPCODE_SET_SEGMENT_AND_OFFSET_ULEB, file.segment_list()->IndexOf(segment), import_func->address() - segment->address()));
|
|
} else {
|
|
mid.push_back(BindingInfo(BIND_OPCODE_ADD_ADDR_ULEB, import_func->address() - address));
|
|
}
|
|
address = import_func->address();
|
|
}
|
|
if (addend != import_func->addend()) {
|
|
mid.push_back(BindingInfo(BIND_OPCODE_SET_ADDEND_SLEB, import_func->addend()));
|
|
addend = import_func->addend();
|
|
}
|
|
mid.push_back(BindingInfo(BIND_OPCODE_DO_BIND, 0));
|
|
address += ptrsize;
|
|
}
|
|
mid.push_back(BindingInfo(BIND_OPCODE_DONE, 0));
|
|
|
|
// optimize phase 1, combine bind/add pairs
|
|
BindingInfo *dst = &mid[0];
|
|
for (const BindingInfo *src = &mid[0]; src->opcode != BIND_OPCODE_DONE; ++src) {
|
|
if ((src->opcode == BIND_OPCODE_DO_BIND)
|
|
&& (src[1].opcode == BIND_OPCODE_ADD_ADDR_ULEB)) {
|
|
dst->opcode = BIND_OPCODE_DO_BIND_ADD_ADDR_ULEB;
|
|
dst->operand1 = src[1].operand1;
|
|
++src;
|
|
++dst;
|
|
} else {
|
|
*dst++ = *src;
|
|
}
|
|
}
|
|
dst->opcode = BIND_OPCODE_DONE;
|
|
|
|
// optimize phase 2, compress packed runs of BIND_OPCODE_DO_BIND_ADD_ADDR_ULEB with
|
|
// same addr delta into one BIND_OPCODE_DO_BIND_ULEB_TIMES_SKIPPING_ULEB
|
|
dst = &mid[0];
|
|
for (const BindingInfo *src = &mid[0]; src->opcode != BIND_OPCODE_DONE; ++src) {
|
|
uint64_t delta = src->operand1;
|
|
if ((src->opcode == BIND_OPCODE_DO_BIND_ADD_ADDR_ULEB)
|
|
&& (src[1].opcode == BIND_OPCODE_DO_BIND_ADD_ADDR_ULEB)
|
|
&& (src[1].operand1 == delta)) {
|
|
// found at least two in a row, this is worth compressing
|
|
dst->opcode = BIND_OPCODE_DO_BIND_ULEB_TIMES_SKIPPING_ULEB;
|
|
dst->operand1 = 1;
|
|
dst->operand2 = delta;
|
|
++src;
|
|
while ((src->opcode == BIND_OPCODE_DO_BIND_ADD_ADDR_ULEB)
|
|
&& (src->operand1 == delta)) {
|
|
dst->operand1++;
|
|
++src;
|
|
}
|
|
--src;
|
|
++dst;
|
|
} else {
|
|
*dst++ = *src;
|
|
}
|
|
}
|
|
dst->opcode = BIND_OPCODE_DONE;
|
|
|
|
// optimize phase 3, use immediate encodings
|
|
for (BindingInfo *p = &mid[0]; p->opcode != REBASE_OPCODE_DONE; ++p) {
|
|
if ((p->opcode == BIND_OPCODE_DO_BIND_ADD_ADDR_ULEB)
|
|
&& (p->operand1 < (15 * ptrsize))
|
|
&& ((p->operand1 % ptrsize) == 0)) {
|
|
p->opcode = BIND_OPCODE_DO_BIND_ADD_ADDR_IMM_SCALED;
|
|
p->operand1 = p->operand1 / ptrsize;
|
|
} else if ((p->opcode == BIND_OPCODE_SET_DYLIB_ORDINAL_ULEB) && (p->operand1 <= 15)) {
|
|
p->opcode = BIND_OPCODE_SET_DYLIB_ORDINAL_IMM;
|
|
}
|
|
}
|
|
dst->opcode = BIND_OPCODE_DONE; //-V519
|
|
|
|
// convert to compressed encoding
|
|
EncodedData data;
|
|
data.reserve(info.size() * 2);
|
|
bool done = false;
|
|
for (std::vector<BindingInfo>::iterator it = mid.begin(); !done && it != mid.end() ; ++it) {
|
|
switch ( it->opcode ) {
|
|
case BIND_OPCODE_DONE:
|
|
done = true;
|
|
break;
|
|
case BIND_OPCODE_SET_DYLIB_ORDINAL_IMM:
|
|
data.WriteByte(BIND_OPCODE_SET_DYLIB_ORDINAL_IMM | static_cast<uint8_t>(it->operand1));
|
|
break;
|
|
case BIND_OPCODE_SET_DYLIB_ORDINAL_ULEB:
|
|
data.WriteByte(BIND_OPCODE_SET_DYLIB_ORDINAL_ULEB);
|
|
data.WriteUleb128(it->operand1);
|
|
break;
|
|
case BIND_OPCODE_SET_DYLIB_SPECIAL_IMM:
|
|
data.WriteByte(BIND_OPCODE_SET_DYLIB_SPECIAL_IMM | (it->operand1 & BIND_IMMEDIATE_MASK));
|
|
break;
|
|
case BIND_OPCODE_SET_SYMBOL_TRAILING_FLAGS_IMM:
|
|
data.WriteByte(BIND_OPCODE_SET_SYMBOL_TRAILING_FLAGS_IMM | static_cast<uint8_t>(it->operand1));
|
|
data.WriteString(it->name);
|
|
break;
|
|
case BIND_OPCODE_SET_TYPE_IMM:
|
|
data.WriteByte(BIND_OPCODE_SET_TYPE_IMM | static_cast<uint8_t>(it->operand1));
|
|
break;
|
|
case BIND_OPCODE_SET_ADDEND_SLEB:
|
|
data.WriteByte(BIND_OPCODE_SET_ADDEND_SLEB);
|
|
data.WriteSleb128(it->operand1);
|
|
break;
|
|
case BIND_OPCODE_SET_SEGMENT_AND_OFFSET_ULEB:
|
|
data.WriteByte(BIND_OPCODE_SET_SEGMENT_AND_OFFSET_ULEB | static_cast<uint8_t>(it->operand1));
|
|
data.WriteUleb128(it->operand2);
|
|
break;
|
|
case BIND_OPCODE_ADD_ADDR_ULEB:
|
|
data.WriteByte(BIND_OPCODE_ADD_ADDR_ULEB);
|
|
data.WriteUleb128(it->operand1);
|
|
break;
|
|
case BIND_OPCODE_DO_BIND:
|
|
data.WriteByte(BIND_OPCODE_DO_BIND);
|
|
break;
|
|
case BIND_OPCODE_DO_BIND_ADD_ADDR_ULEB:
|
|
data.WriteByte(BIND_OPCODE_DO_BIND_ADD_ADDR_ULEB);
|
|
data.WriteUleb128(it->operand1);
|
|
break;
|
|
case BIND_OPCODE_DO_BIND_ADD_ADDR_IMM_SCALED:
|
|
data.WriteByte(BIND_OPCODE_DO_BIND_ADD_ADDR_IMM_SCALED | static_cast<uint8_t>(it->operand1));
|
|
break;
|
|
case BIND_OPCODE_DO_BIND_ULEB_TIMES_SKIPPING_ULEB:
|
|
data.WriteByte(BIND_OPCODE_DO_BIND_ULEB_TIMES_SKIPPING_ULEB);
|
|
data.WriteUleb128(it->operand1);
|
|
data.WriteUleb128(it->operand2);
|
|
break;
|
|
}
|
|
}
|
|
data.resize(AlignValue(data.size(), ptrsize), 0);
|
|
|
|
return (data.size() == 0) ? 0 : file.Write(data.data(), data.size());
|
|
}
|
|
|
|
size_t MacImportList::WriteWeakBindInfo(MacArchitecture &file)
|
|
{
|
|
size_t i, j;
|
|
std::vector<MacImportFunction *> info;
|
|
std::vector<BindingInfo> mid;
|
|
MacSegment *segment = NULL;
|
|
std::string symbol_name;
|
|
uint8_t bind_type = 0;
|
|
uint64_t address = -1;
|
|
int64_t addend = 0;
|
|
size_t ptrsize = OperandSizeToValue(file.cpu_address_size());
|
|
MacImport *import;
|
|
MacImportFunction *import_func;
|
|
|
|
for (i = 0; i < count(); i++) {
|
|
import = item(i);
|
|
for (j = 0; j < import->count(); j++) {
|
|
import_func = import->item(j);
|
|
if (!import_func->is_weak())
|
|
continue;
|
|
info.push_back(import_func);
|
|
}
|
|
}
|
|
if (info.size() == 0) {
|
|
// short circuit if no weak binding needed
|
|
return 0;
|
|
}
|
|
std::sort(info.begin(), info.end(), WeakBindInfoHelper());
|
|
|
|
for (std::vector<MacImportFunction *>::const_iterator it = info.begin(); it != info.end(); ++it) {
|
|
import_func = *it;
|
|
if (symbol_name != import_func->name()) {
|
|
mid.push_back(BindingInfo(BIND_OPCODE_SET_SYMBOL_TRAILING_FLAGS_IMM, import_func->flags(), 0, import_func->name()));
|
|
symbol_name = import_func->name();
|
|
}
|
|
// non-weak symbols just have BIND_OPCODE_SET_SYMBOL_TRAILING_FLAGS_IMM
|
|
// weak symbols have SET_SEG, ADD_ADDR, SET_ADDED, DO_BIND
|
|
if (import_func->bind_type() != BIND_TYPE_OVERRIDE_OF_WEAKDEF_IN_DYLIB) {
|
|
if (bind_type != import_func->bind_type()) {
|
|
mid.push_back(BindingInfo(BIND_OPCODE_SET_TYPE_IMM, import_func->bind_type()));
|
|
bind_type = import_func->bind_type();
|
|
}
|
|
if (address != import_func->address()) {
|
|
if (!segment || (import_func->address() < segment->address()) || (import_func->address() >= segment->address() + segment->size())) {
|
|
segment = file.segment_list()->GetSectionByAddress(import_func->address());
|
|
if (!segment)
|
|
throw std::runtime_error("binding address outside range of any segment");
|
|
mid.push_back(BindingInfo(BIND_OPCODE_SET_SEGMENT_AND_OFFSET_ULEB, file.segment_list()->IndexOf(segment), import_func->address() - segment->address()));
|
|
}
|
|
else {
|
|
mid.push_back(BindingInfo(BIND_OPCODE_ADD_ADDR_ULEB, import_func->address() - address));
|
|
}
|
|
address = import_func->address();
|
|
}
|
|
if (addend != import_func->addend()) {
|
|
mid.push_back(BindingInfo(BIND_OPCODE_SET_ADDEND_SLEB, import_func->addend()));
|
|
addend = import_func->addend();
|
|
}
|
|
mid.push_back(BindingInfo(BIND_OPCODE_DO_BIND, 0));
|
|
address += ptrsize;
|
|
}
|
|
}
|
|
mid.push_back(BindingInfo(BIND_OPCODE_DONE, 0));
|
|
|
|
// optimize phase 1, combine bind/add pairs
|
|
BindingInfo *dst = &mid[0];
|
|
for (const BindingInfo *src = &mid[0]; src->opcode != BIND_OPCODE_DONE; ++src) {
|
|
if ( (src->opcode == BIND_OPCODE_DO_BIND)
|
|
&& (src[1].opcode == BIND_OPCODE_ADD_ADDR_ULEB) ) {
|
|
dst->opcode = BIND_OPCODE_DO_BIND_ADD_ADDR_ULEB;
|
|
dst->operand1 = src[1].operand1;
|
|
++src;
|
|
++dst;
|
|
}
|
|
else {
|
|
*dst++ = *src;
|
|
}
|
|
}
|
|
dst->opcode = BIND_OPCODE_DONE;
|
|
|
|
// optimize phase 2, compress packed runs of BIND_OPCODE_DO_BIND_ADD_ADDR_ULEB with
|
|
// same addr delta into one BIND_OPCODE_DO_BIND_ULEB_TIMES_SKIPPING_ULEB
|
|
dst = &mid[0];
|
|
for (const BindingInfo *src = &mid[0]; src->opcode != BIND_OPCODE_DONE; ++src) {
|
|
uint64_t delta = src->operand1;
|
|
if ( (src->opcode == BIND_OPCODE_DO_BIND_ADD_ADDR_ULEB)
|
|
&& (src[1].opcode == BIND_OPCODE_DO_BIND_ADD_ADDR_ULEB)
|
|
&& (src[1].operand1 == delta) ) {
|
|
// found at least two in a row, this is worth compressing
|
|
dst->opcode = BIND_OPCODE_DO_BIND_ULEB_TIMES_SKIPPING_ULEB;
|
|
dst->operand1 = 1;
|
|
dst->operand2 = delta;
|
|
++src;
|
|
while ( (src->opcode == BIND_OPCODE_DO_BIND_ADD_ADDR_ULEB)
|
|
&& (src->operand1 == delta) ) {
|
|
dst->operand1++;
|
|
++src;
|
|
}
|
|
--src;
|
|
++dst;
|
|
}
|
|
else {
|
|
*dst++ = *src;
|
|
}
|
|
}
|
|
dst->opcode = BIND_OPCODE_DONE;
|
|
|
|
// optimize phase 3, use immediate encodings
|
|
for (BindingInfo *p = &mid[0]; p->opcode != REBASE_OPCODE_DONE; ++p) {
|
|
if ( (p->opcode == BIND_OPCODE_DO_BIND_ADD_ADDR_ULEB)
|
|
&& (p->operand1 < (15 * ptrsize))
|
|
&& ((p->operand1 % ptrsize) == 0) ) {
|
|
p->opcode = BIND_OPCODE_DO_BIND_ADD_ADDR_IMM_SCALED;
|
|
p->operand1 = p->operand1 / ptrsize;
|
|
}
|
|
}
|
|
dst->opcode = BIND_OPCODE_DONE; //-V519
|
|
|
|
// convert to compressed encoding
|
|
EncodedData data;
|
|
data.reserve(info.size() * 2);
|
|
bool done = false;
|
|
for (std::vector<BindingInfo>::iterator it = mid.begin(); !done && it != mid.end(); ++it) {
|
|
switch (it->opcode) {
|
|
case BIND_OPCODE_DONE:
|
|
data.WriteByte(BIND_OPCODE_DONE);
|
|
done = true;
|
|
break;
|
|
case BIND_OPCODE_SET_DYLIB_ORDINAL_IMM:
|
|
data.WriteByte(BIND_OPCODE_SET_DYLIB_ORDINAL_IMM | static_cast<uint8_t>(it->operand1));
|
|
break;
|
|
case BIND_OPCODE_SET_DYLIB_ORDINAL_ULEB:
|
|
data.WriteByte(BIND_OPCODE_SET_DYLIB_ORDINAL_ULEB);
|
|
data.WriteUleb128(it->operand1);
|
|
break;
|
|
case BIND_OPCODE_SET_DYLIB_SPECIAL_IMM:
|
|
data.WriteByte(BIND_OPCODE_SET_DYLIB_SPECIAL_IMM | (it->operand1 & BIND_IMMEDIATE_MASK));
|
|
break;
|
|
case BIND_OPCODE_SET_SYMBOL_TRAILING_FLAGS_IMM:
|
|
data.WriteByte(BIND_OPCODE_SET_SYMBOL_TRAILING_FLAGS_IMM | static_cast<uint8_t>(it->operand1));
|
|
data.WriteString(it->name);
|
|
break;
|
|
case BIND_OPCODE_SET_TYPE_IMM:
|
|
data.WriteByte(BIND_OPCODE_SET_TYPE_IMM | static_cast<uint8_t>(it->operand1));
|
|
break;
|
|
case BIND_OPCODE_SET_ADDEND_SLEB:
|
|
data.WriteByte(BIND_OPCODE_SET_ADDEND_SLEB);
|
|
data.WriteSleb128(it->operand1);
|
|
break;
|
|
case BIND_OPCODE_SET_SEGMENT_AND_OFFSET_ULEB:
|
|
data.WriteByte(BIND_OPCODE_SET_SEGMENT_AND_OFFSET_ULEB | static_cast<uint8_t>(it->operand1));
|
|
data.WriteUleb128(it->operand2);
|
|
break;
|
|
case BIND_OPCODE_ADD_ADDR_ULEB:
|
|
data.WriteByte(BIND_OPCODE_ADD_ADDR_ULEB);
|
|
data.WriteUleb128(it->operand1);
|
|
break;
|
|
case BIND_OPCODE_DO_BIND:
|
|
data.WriteByte(BIND_OPCODE_DO_BIND);
|
|
break;
|
|
case BIND_OPCODE_DO_BIND_ADD_ADDR_ULEB:
|
|
data.WriteByte(BIND_OPCODE_DO_BIND_ADD_ADDR_ULEB);
|
|
data.WriteUleb128(it->operand1);
|
|
break;
|
|
case BIND_OPCODE_DO_BIND_ADD_ADDR_IMM_SCALED:
|
|
data.WriteByte(BIND_OPCODE_DO_BIND_ADD_ADDR_IMM_SCALED | static_cast<uint8_t>(it->operand1));
|
|
break;
|
|
case BIND_OPCODE_DO_BIND_ULEB_TIMES_SKIPPING_ULEB:
|
|
data.WriteByte(BIND_OPCODE_DO_BIND_ULEB_TIMES_SKIPPING_ULEB);
|
|
data.WriteUleb128(it->operand1);
|
|
data.WriteUleb128(it->operand2);
|
|
break;
|
|
}
|
|
}
|
|
data.resize(AlignValue(data.size(), ptrsize), 0);
|
|
|
|
return (data.size() == 0) ? 0 : file.Write(data.data(), data.size());
|
|
}
|
|
|
|
size_t MacImportList::WriteLazyBindInfo(MacArchitecture &file)
|
|
{
|
|
std::vector<MacImportFunction *> info;
|
|
MacImport *import;
|
|
MacImportFunction *import_function;
|
|
|
|
for (size_t i = 0; i < count(); i++) {
|
|
import = item(i);
|
|
for (size_t j = 0; j < import->count(); j++) {
|
|
import_function = import->item(j);
|
|
if (!import_function->is_lazy())
|
|
continue;
|
|
info.push_back(import_function);
|
|
}
|
|
}
|
|
if (info.empty())
|
|
return 0;
|
|
|
|
std::sort(info.begin(), info.end(), LazyBindInfoHelper());
|
|
|
|
size_t ptrsize = OperandSizeToValue(file.cpu_address_size());
|
|
EncodedData data;
|
|
for (std::vector<MacImportFunction *>::iterator it = info.begin(); it != info.end() ; ++it) {
|
|
import_function = *it;
|
|
|
|
if (data.size() > import_function->bind_offset())
|
|
throw std::runtime_error("binding offset out of range");
|
|
for (size_t i = data.size(); i < import_function->bind_offset(); i++) {
|
|
data.WriteByte(BIND_OPCODE_DONE);
|
|
}
|
|
|
|
// write address to bind
|
|
MacSegment *segment = file.segment_list()->GetSectionByAddress(import_function->address());
|
|
if (!segment)
|
|
throw std::runtime_error("lazy binding address outside range of any segment");
|
|
data.WriteByte(BIND_OPCODE_SET_SEGMENT_AND_OFFSET_ULEB | static_cast<uint8_t>(file.segment_list()->IndexOf(segment)));
|
|
data.WriteUleb128(import_function->address() - segment->address());
|
|
|
|
// write ordinal
|
|
if (import_function->library_ordinal() <= 0) {
|
|
// special lookups are encoded as negative numbers in BindingInfo
|
|
data.WriteByte(BIND_OPCODE_SET_DYLIB_SPECIAL_IMM | (import_function->library_ordinal() & BIND_IMMEDIATE_MASK) );
|
|
} else if (import_function->library_ordinal() <= 15 ) {
|
|
// small ordinals are encoded in opcode
|
|
data.WriteByte(BIND_OPCODE_SET_DYLIB_ORDINAL_IMM | import_function->library_ordinal());
|
|
} else {
|
|
data.WriteByte(BIND_OPCODE_SET_DYLIB_ORDINAL_ULEB);
|
|
data.WriteUleb128(import_function->library_ordinal());
|
|
}
|
|
// write symbol name
|
|
data.WriteByte(BIND_OPCODE_SET_SYMBOL_TRAILING_FLAGS_IMM | import_function->flags());
|
|
data.WriteString(import_function->name());
|
|
// write do bind
|
|
data.WriteByte(BIND_OPCODE_DO_BIND);
|
|
data.WriteByte(BIND_OPCODE_DONE);
|
|
}
|
|
data.resize(AlignValue(data.size(), ptrsize), 0);
|
|
|
|
return (data.size() == 0) ? 0 : file.Write(data.data(), data.size());
|
|
}
|
|
|
|
void MacImportList::RebaseBindInfo(MacArchitecture &file, size_t delta_base)
|
|
{
|
|
std::vector<MacImportFunction *> info;
|
|
MacImport *import;
|
|
MacImportFunction *import_function;
|
|
|
|
for (size_t i = 0; i < count(); i++) {
|
|
import = item(i);
|
|
for (size_t j = 0; j < import->count(); j++) {
|
|
import_function = import->item(j);
|
|
if (!import_function->is_lazy())
|
|
continue;
|
|
info.push_back(import_function);
|
|
}
|
|
}
|
|
if (info.empty())
|
|
return;
|
|
|
|
std::sort(info.begin(), info.end(), LazyBindInfoHelper());
|
|
|
|
size_t value_size = OperandSizeToValue(file.cpu_address_size());
|
|
for (std::vector<MacImportFunction *>::iterator it = info.begin(); it != info.end() ; ++it) {
|
|
import_function = *it;
|
|
import_function->set_bind_offset(import_function->bind_offset() + delta_base);
|
|
delta_base += 4;
|
|
|
|
file.AddressSeek(import_function->address());
|
|
uint64_t value = 0;
|
|
assert(sizeof(value) >= value_size);
|
|
file.Read(&value, value_size);
|
|
if (value && file.AddressSeek(value)) {
|
|
if (file.ReadByte() != 0x68) // push XXXX
|
|
throw std::runtime_error("Runtime error at RebaseBindInfo");
|
|
file.WriteDWord(static_cast<uint32_t>(import_function->bind_offset()));
|
|
}
|
|
}
|
|
}
|
|
|
|
/**
|
|
* MacIndirectSymbol
|
|
*/
|
|
|
|
MacIndirectSymbol::MacIndirectSymbol(MacIndirectSymbolList *owner, uint64_t address, uint32_t value, MacSymbol *symbol)
|
|
: IObject(), owner_(owner), address_(address), value_(value), symbol_(symbol)
|
|
{
|
|
|
|
}
|
|
|
|
MacIndirectSymbol::MacIndirectSymbol(MacIndirectSymbolList *owner, const MacIndirectSymbol &src)
|
|
: IObject(src), owner_(owner)
|
|
{
|
|
address_ = src.address_;
|
|
value_ = src.value_;
|
|
symbol_ = src.symbol_;
|
|
}
|
|
|
|
MacIndirectSymbol::~MacIndirectSymbol()
|
|
{
|
|
if (owner_)
|
|
owner_->RemoveObject(this);
|
|
}
|
|
|
|
MacIndirectSymbol *MacIndirectSymbol::Clone(MacIndirectSymbolList *owner) const
|
|
{
|
|
MacIndirectSymbol *symbol = new MacIndirectSymbol(owner, *this);
|
|
return symbol;
|
|
}
|
|
|
|
void MacIndirectSymbol::Rebase(uint64_t delta_base)
|
|
{
|
|
if (address_)
|
|
address_ += delta_base;
|
|
}
|
|
|
|
/**
|
|
* MacIndirectSymbolList
|
|
*/
|
|
|
|
MacIndirectSymbolList::MacIndirectSymbolList()
|
|
: ObjectList<MacIndirectSymbol>()
|
|
{
|
|
|
|
}
|
|
|
|
MacIndirectSymbolList::MacIndirectSymbolList(const MacIndirectSymbolList &src)
|
|
: ObjectList<MacIndirectSymbol>(src)
|
|
{
|
|
for (size_t i = 0; i < src.count(); i++) {
|
|
AddObject(src.item(i)->Clone(this));
|
|
}
|
|
}
|
|
|
|
MacIndirectSymbolList *MacIndirectSymbolList::Clone() const
|
|
{
|
|
MacIndirectSymbolList *list = new MacIndirectSymbolList(*this);
|
|
return list;
|
|
}
|
|
|
|
MacIndirectSymbol *MacIndirectSymbolList::Add(uint64_t address, uint32_t value, MacSymbol *symbol)
|
|
{
|
|
MacIndirectSymbol *sym = new MacIndirectSymbol(this, address, value, symbol);
|
|
AddObject(sym);
|
|
return sym;
|
|
}
|
|
|
|
MacIndirectSymbol *MacIndirectSymbolList::GetSymbol(MacSymbol *symbol) const
|
|
{
|
|
for (size_t i = 0; i < count(); i++) {
|
|
MacIndirectSymbol *indirect_symbol = item(i);
|
|
if (indirect_symbol->symbol() == symbol)
|
|
return indirect_symbol;
|
|
}
|
|
return NULL;
|
|
}
|
|
|
|
void MacIndirectSymbolList::ReadFromFile(MacArchitecture &file)
|
|
{
|
|
dysymtab_command *dysymtab = file.dysymtab();
|
|
if (!dysymtab->cmd)
|
|
return;
|
|
|
|
file.Seek(dysymtab->indirectsymoff);
|
|
uint32_t ptr_size = OperandSizeToValue(file.cpu_address_size());
|
|
|
|
for (size_t i = 0; i < dysymtab->nindirectsyms; i++) {
|
|
uint32_t value = file.ReadDWord();
|
|
|
|
uint64_t address = 0;
|
|
for (size_t j = 0; j < file.section_list()->count(); j++) {
|
|
MacSection *section = file.section_list()->item(j);
|
|
if (section->type() == S_SYMBOL_STUBS || section->type() == S_LAZY_SYMBOL_POINTERS || section->type() == S_NON_LAZY_SYMBOL_POINTERS || section->type() == S_THREAD_LOCAL_VARIABLE_POINTERS) {
|
|
uint32_t value_size = (section->type() == S_SYMBOL_STUBS) ? section->reserved2() : ptr_size;
|
|
if (i >= section->reserved1() && i < section->reserved1() + section->size() / value_size) {
|
|
address = section->address() + (i - section->reserved1()) * value_size;
|
|
break;
|
|
}
|
|
}
|
|
}
|
|
|
|
if (!address)
|
|
throw std::runtime_error("Invalid indirect symbol address");
|
|
|
|
MacSymbol *symbol;
|
|
if (value & (INDIRECT_SYMBOL_LOCAL | INDIRECT_SYMBOL_ABS)) {
|
|
symbol = NULL;
|
|
} else {
|
|
if (value >= file.symbol_list()->count())
|
|
throw std::runtime_error("Invalid indirect symbol value");
|
|
symbol = file.symbol_list()->item(value);
|
|
}
|
|
|
|
Add(address, value, symbol);
|
|
}
|
|
}
|
|
|
|
void MacIndirectSymbolList::Pack()
|
|
{
|
|
for (size_t i = 0; i < count(); i++) {
|
|
MacIndirectSymbol *indirect_symbol = item(i);
|
|
MacSymbol *symbol = indirect_symbol->symbol();
|
|
if (symbol && symbol->is_deleted()) {
|
|
indirect_symbol->set_symbol(NULL);
|
|
indirect_symbol->set_value(INDIRECT_SYMBOL_ABS);
|
|
}
|
|
}
|
|
}
|
|
|
|
void MacIndirectSymbolList::WriteToFile(MacArchitecture &file)
|
|
{
|
|
dysymtab_command *dysymtab = file.dysymtab();
|
|
|
|
dysymtab->nindirectsyms = static_cast<uint32_t>(count());
|
|
if (!dysymtab->nindirectsyms) {
|
|
dysymtab->indirectsymoff = 0;
|
|
return;
|
|
}
|
|
|
|
dysymtab->indirectsymoff = static_cast<uint32_t>(file.Tell());
|
|
for (size_t i = 0; i < count(); i++) {
|
|
MacIndirectSymbol *indirect_symbol = item(i);
|
|
uint32_t value = (indirect_symbol->symbol()) ? (uint32_t)file.symbol_list()->IndexOf(indirect_symbol->symbol()) : indirect_symbol->value();
|
|
file.Write(&value, sizeof(value));
|
|
}
|
|
}
|
|
|
|
void MacIndirectSymbolList::Rebase(uint64_t delta_base)
|
|
{
|
|
for (size_t i = 0; i < count(); i++) {
|
|
item(i)->Rebase(delta_base);
|
|
}
|
|
}
|
|
|
|
/**
|
|
* MacExtRefSymbol
|
|
*/
|
|
|
|
MacExtRefSymbol::MacExtRefSymbol(MacExtRefSymbolList *owner, MacSymbol *symbol, uint8_t flags)
|
|
: IObject(), owner_(owner), flags_(flags), symbol_(symbol)
|
|
{
|
|
|
|
}
|
|
|
|
MacExtRefSymbol::MacExtRefSymbol(MacExtRefSymbolList *owner, const MacExtRefSymbol &src)
|
|
: IObject(src), owner_(owner)
|
|
{
|
|
flags_ = src.flags_;
|
|
symbol_ = src.symbol_;
|
|
}
|
|
|
|
MacExtRefSymbol::~MacExtRefSymbol()
|
|
{
|
|
if (owner_)
|
|
owner_->RemoveObject(this);
|
|
}
|
|
|
|
MacExtRefSymbol *MacExtRefSymbol::Clone(MacExtRefSymbolList *owner) const
|
|
{
|
|
MacExtRefSymbol *symbol = new MacExtRefSymbol(owner, *this);
|
|
return symbol;
|
|
}
|
|
|
|
/**
|
|
* MacExtRefSymbolList
|
|
*/
|
|
|
|
MacExtRefSymbolList::MacExtRefSymbolList()
|
|
: ObjectList<MacExtRefSymbol>()
|
|
{
|
|
|
|
}
|
|
|
|
MacExtRefSymbolList::MacExtRefSymbolList(const MacExtRefSymbolList &src)
|
|
: ObjectList<MacExtRefSymbol>(src)
|
|
{
|
|
for (size_t i = 0; i < src.count(); i++) {
|
|
AddObject(src.item(i)->Clone(this));
|
|
}
|
|
}
|
|
|
|
MacExtRefSymbolList *MacExtRefSymbolList::Clone() const
|
|
{
|
|
MacExtRefSymbolList *list = new MacExtRefSymbolList(*this);
|
|
return list;
|
|
}
|
|
|
|
MacExtRefSymbol *MacExtRefSymbolList::Add(MacSymbol *symbol, uint8_t flags)
|
|
{
|
|
MacExtRefSymbol *sym = new MacExtRefSymbol(this, symbol, flags);
|
|
AddObject(sym);
|
|
return sym;
|
|
}
|
|
|
|
MacExtRefSymbol *MacExtRefSymbolList::GetSymbol(MacSymbol *symbol) const
|
|
{
|
|
for (size_t i = 0; i < count(); i++) {
|
|
MacExtRefSymbol *ext_ref_symbol = item(i);
|
|
if (ext_ref_symbol->symbol() == symbol)
|
|
return ext_ref_symbol;
|
|
}
|
|
return NULL;
|
|
}
|
|
|
|
void MacExtRefSymbolList::ReadFromFile(MacArchitecture &file)
|
|
{
|
|
dysymtab_command *dysymtab = file.dysymtab();
|
|
if (!dysymtab->cmd)
|
|
return;
|
|
|
|
file.Seek(dysymtab->extrefsymoff);
|
|
|
|
for (size_t i = 0; i < dysymtab->nextrefsyms; i++) {
|
|
dylib_reference ref;
|
|
file.Read(&ref, sizeof(ref));
|
|
|
|
if (ref.isym >= file.symbol_list()->count())
|
|
throw std::runtime_error("Invalid extern reference symbol value");
|
|
|
|
Add(file.symbol_list()->item(ref.isym), ref.flags);
|
|
}
|
|
}
|
|
|
|
void MacExtRefSymbolList::Pack()
|
|
{
|
|
for (size_t i = count(); i > 0; i--) {
|
|
MacExtRefSymbol *ext_ref_symbol = item(i - 1);
|
|
if (ext_ref_symbol->symbol()->is_deleted())
|
|
delete ext_ref_symbol;
|
|
}
|
|
}
|
|
|
|
void MacExtRefSymbolList::WriteToFile(MacArchitecture &file)
|
|
{
|
|
dysymtab_command *dysymtab = file.dysymtab();
|
|
|
|
dysymtab->nextrefsyms = static_cast<uint32_t>(count());
|
|
if (!dysymtab->nextrefsyms) {
|
|
dysymtab->extrefsymoff = 0;
|
|
return;
|
|
}
|
|
|
|
dysymtab->extrefsymoff = static_cast<uint32_t>(file.Tell());
|
|
for (size_t i = 0; i < count(); i++) {
|
|
MacExtRefSymbol *ext_ref_symbol = item(i);
|
|
dylib_reference ref;
|
|
ref.isym = file.symbol_list()->IndexOf(ext_ref_symbol->symbol());
|
|
ref.flags = ext_ref_symbol->flags();
|
|
file.Write(&ref, sizeof(ref));
|
|
}
|
|
}
|
|
|
|
/**
|
|
* MacExport
|
|
*/
|
|
|
|
MacExport::MacExport(IExportList *parent, uint64_t address, const std::string &name, uint64_t flags, uint64_t other)
|
|
: BaseExport(parent), symbol_(NULL), address_(address), name_(name), flags_(flags), other_(other)
|
|
{
|
|
|
|
}
|
|
|
|
MacExport::MacExport(IExportList *parent, MacSymbol *symbol)
|
|
: BaseExport(parent), symbol_(symbol), address_(0), flags_(0), other_(0)
|
|
{
|
|
if (symbol_) {
|
|
address_ = symbol_->value();
|
|
name_ = symbol_->name();
|
|
}
|
|
}
|
|
|
|
MacExport::MacExport(IExportList *parent, const MacExport &src)
|
|
: BaseExport(parent, src)
|
|
{
|
|
address_ = src.address_;
|
|
name_ = src.name_;
|
|
forwarded_name_ = src.forwarded_name_;
|
|
flags_ = src.flags_;
|
|
other_ = src.other_;
|
|
symbol_ = src.symbol_;
|
|
}
|
|
|
|
MacExport *MacExport::Clone(IExportList *parent) const
|
|
{
|
|
MacExport *exp = new MacExport(parent, *this);
|
|
return exp;
|
|
}
|
|
|
|
void MacExport::set_address(uint64_t address)
|
|
{
|
|
address_ = address;
|
|
if (symbol_)
|
|
symbol_->set_value(address);
|
|
}
|
|
|
|
void MacExport::Rebase(uint64_t delta_base)
|
|
{
|
|
address_ += delta_base;
|
|
}
|
|
|
|
std::string MacExport::display_name(bool show_ret) const
|
|
{
|
|
return DemangleName(name_).display_name(show_ret);
|
|
}
|
|
|
|
/**
|
|
* MacExportList
|
|
*/
|
|
|
|
MacExportList::MacExportList(MacArchitecture *owner)
|
|
: BaseExportList(owner)
|
|
{
|
|
|
|
}
|
|
|
|
MacExportList::MacExportList(MacArchitecture *owner, const MacExportList &src)
|
|
: BaseExportList(owner, src)
|
|
{
|
|
name_ = src.name_;
|
|
}
|
|
|
|
MacExportList *MacExportList::Clone(MacArchitecture *owner) const
|
|
{
|
|
MacExportList *list = new MacExportList(owner, *this);
|
|
return list;
|
|
}
|
|
|
|
MacExport *MacExportList::item(size_t index) const
|
|
{
|
|
return reinterpret_cast<MacExport *>(IExportList::item(index));
|
|
}
|
|
|
|
MacExport *MacExportList::Add(uint64_t address, const std::string & name, uint64_t flags, uint64_t other)
|
|
{
|
|
MacExport *exp = new MacExport(this, address, name, flags, other);
|
|
AddObject(exp);
|
|
return exp;
|
|
}
|
|
|
|
MacExport *MacExportList::Add(MacSymbol *symbol)
|
|
{
|
|
MacExport *exp = new MacExport(this, symbol);
|
|
AddObject(exp);
|
|
return exp;
|
|
}
|
|
|
|
void MacExportList::ParseExportNode(const EncodedData &buf, size_t pos, const std::string &name, uint64_t base_address)
|
|
{
|
|
uint8_t terminal_size, children_count;
|
|
uint64_t address, flags, other;
|
|
size_t i, children_pos;
|
|
std::string children_name, import_name;
|
|
uint32_t childNodeOffset;
|
|
MacExport *exp;
|
|
|
|
terminal_size = static_cast<uint8_t>(buf.ReadUleb128(&pos));
|
|
children_pos = pos + terminal_size;
|
|
if (terminal_size) {
|
|
flags = buf.ReadUleb128(&pos);
|
|
|
|
if (flags & EXPORT_SYMBOL_FLAGS_REEXPORT) {
|
|
address = 0;
|
|
other = buf.ReadUleb128(&pos);
|
|
import_name = buf.ReadString(&pos);
|
|
} else {
|
|
address = base_address + buf.ReadUleb128(&pos);
|
|
other = (flags & EXPORT_SYMBOL_FLAGS_STUB_AND_RESOLVER) ? buf.ReadUleb128(&pos) : 0;
|
|
}
|
|
|
|
exp = Add(address, name, flags, other);
|
|
if (import_name.size() > 0)
|
|
exp->set_forwarded_name(import_name);
|
|
}
|
|
|
|
children_count = buf[children_pos++];
|
|
for (i = 0; i < children_count; i++) {
|
|
children_name = name + buf.ReadString(&children_pos);
|
|
childNodeOffset = static_cast<uint32_t>(buf.ReadUleb128(&children_pos));
|
|
ParseExportNode(buf, childNodeOffset, children_name, base_address);
|
|
}
|
|
}
|
|
|
|
void MacExportList::ReadFromFile(MacArchitecture &file)
|
|
{
|
|
dyld_info_command *dyld_info = file.dyld_info();
|
|
|
|
ILoadCommand *command = file.command_list()->GetCommandByType(LC_ID_DYLIB);
|
|
if (command) {
|
|
dylib_command dylib;
|
|
|
|
file.Seek(command->address());
|
|
file.Read(&dylib, sizeof(dylib));
|
|
|
|
file.Seek(command->address() + dylib.dylib.name.offset);
|
|
name_ = file.ReadString();
|
|
}
|
|
|
|
if (dyld_info->cmd) {
|
|
if (!dyld_info->export_size)
|
|
return;
|
|
|
|
MacSegment *base_segment = file.segment_list()->GetBaseSegment();
|
|
if (!base_segment)
|
|
throw std::runtime_error("Invalid base segment for export info");
|
|
|
|
EncodedData buf;
|
|
file.Seek(dyld_info->export_off);
|
|
buf.ReadFromFile(file, dyld_info->export_size);
|
|
ParseExportNode(buf, 0, "", base_segment->address());
|
|
|
|
std::map<uint64_t, MacExport *> address_map;
|
|
for (size_t i = 0; i < count(); i++) {
|
|
MacExport *exp = item(i);
|
|
if (!exp->address())
|
|
continue;
|
|
address_map[exp->address()] = exp;
|
|
}
|
|
|
|
MacSymbolList *symbol_list = file.symbol_list();
|
|
for (size_t i = 0; i < symbol_list->count(); i++) {
|
|
MacSymbol *symbol = symbol_list->item(i);
|
|
|
|
if (symbol->value()) {
|
|
std::map<uint64_t, MacExport *>::const_iterator it = address_map.find(symbol->value());
|
|
if (it != address_map.end())
|
|
it->second->set_symbol(symbol);
|
|
}
|
|
}
|
|
} else {
|
|
MacSymbolList *symbol_list = file.symbol_list();
|
|
for (size_t i = 0; i < symbol_list->count(); i++) {
|
|
MacSymbol *symbol = symbol_list->item(i);
|
|
if ((symbol->type() & (N_STAB | N_TYPE | N_EXT)) == (N_SECT | N_EXT))
|
|
Add(symbol);
|
|
}
|
|
}
|
|
}
|
|
|
|
void MacExportList::Pack()
|
|
{
|
|
for (size_t i = count(); i > 0 ; i--) {
|
|
MacExport *exp = item(i - 1);
|
|
if (exp->symbol() && exp->symbol()->is_deleted())
|
|
delete exp;
|
|
}
|
|
}
|
|
|
|
void MacExportList::WriteToFile(MacArchitecture &file)
|
|
{
|
|
dyld_info_command *dyld_info = file.dyld_info();
|
|
if (!dyld_info->cmd)
|
|
return;
|
|
|
|
if (count() == 0) {
|
|
dyld_info->export_size = 0;
|
|
dyld_info->export_off = 0;
|
|
return;
|
|
}
|
|
|
|
size_t i, j;
|
|
MacExportNode root_node;
|
|
MacExportNode *node;
|
|
EncodedData data;
|
|
std::vector<MacExportNode *> stack;
|
|
|
|
for (i = 0; i < count(); i++) {
|
|
root_node.AddSymbol(item(i));
|
|
}
|
|
|
|
stack.push_back(&root_node);
|
|
for (i = 0; i < stack.size(); i++) {
|
|
node = stack[i];
|
|
for (j = 0; j < node->count(); j++) {
|
|
stack.push_back(node->item(j));
|
|
}
|
|
}
|
|
|
|
uint64_t base_address = file.segment_list()->GetBaseSegment()->address();
|
|
|
|
bool more;
|
|
do {
|
|
more = false;
|
|
data.clear();
|
|
for (i = 0; i < stack.size(); i++) {
|
|
node = stack[i];
|
|
if (node->offset() != data.size())
|
|
more = true;
|
|
stack[i]->WriteToData(data, base_address);
|
|
}
|
|
} while (more);
|
|
data.resize(AlignValue(data.size(), OperandSizeToValue(file.cpu_address_size())), 0);
|
|
|
|
dyld_info->export_size = static_cast<uint32_t>(data.size());
|
|
if (!dyld_info->export_size) {
|
|
dyld_info->export_off = 0;
|
|
} else {
|
|
dyld_info->export_off = static_cast<uint32_t>(file.Tell());
|
|
file.Write(data.data(), data.size());
|
|
}
|
|
}
|
|
|
|
MacExport *MacExportList::GetExportByAddress(uint64_t address) const
|
|
{
|
|
return reinterpret_cast<MacExport *>(BaseExportList::GetExportByAddress(address));
|
|
}
|
|
|
|
void MacExportList::ReadFromBuffer(Buffer &buffer, IArchitecture &file)
|
|
{
|
|
static const APIType export_function_types[] = {
|
|
atSetupImage,
|
|
atFreeImage,
|
|
atDecryptStringA,
|
|
atDecryptStringW,
|
|
atFreeString,
|
|
atSetSerialNumber,
|
|
atGetSerialNumberState,
|
|
atGetSerialNumberData,
|
|
atGetCurrentHWID,
|
|
atActivateLicense,
|
|
atDeactivateLicense,
|
|
atGetOfflineActivationString,
|
|
atGetOfflineDeactivationString,
|
|
atIsValidImageCRC,
|
|
atIsDebuggerPresent,
|
|
atIsVirtualMachinePresent,
|
|
atDecryptBuffer,
|
|
atIsProtected,
|
|
atCalcCRC,
|
|
atLoaderData,
|
|
atRuntimeInit
|
|
};
|
|
|
|
BaseExportList::ReadFromBuffer(buffer, file);
|
|
|
|
assert(count() == _countof(export_function_types));
|
|
for (size_t i = 0; i < count(); i++) {
|
|
item(i)->set_type(export_function_types[i]);
|
|
}
|
|
}
|
|
|
|
/**
|
|
* MacExportNode
|
|
*/
|
|
|
|
MacExportNode::MacExportNode(MacExportNode *owner)
|
|
: ObjectList<MacExportNode>(), owner_(owner), symbol_(NULL), offset_(0)
|
|
{
|
|
|
|
}
|
|
|
|
MacExportNode::MacExportNode(MacExportNode *owner, MacExport *symbol, const std::string &cummulative_string)
|
|
: ObjectList<MacExportNode>(), owner_(owner), symbol_(symbol), cummulative_string_(cummulative_string), offset_(0)
|
|
{
|
|
|
|
}
|
|
|
|
MacExportNode::~MacExportNode()
|
|
{
|
|
if (owner_)
|
|
owner_->RemoveObject(this);
|
|
}
|
|
|
|
void MacExportNode::set_owner(MacExportNode *owner)
|
|
{
|
|
if (owner == owner_)
|
|
return;
|
|
if (owner_)
|
|
owner_->RemoveObject(this);
|
|
owner_ = owner;
|
|
if (owner_)
|
|
owner_->AddObject(this);
|
|
}
|
|
|
|
MacExportNode *MacExportNode::Add(MacExport *symbol, const std::string &cummulative_string)
|
|
{
|
|
MacExportNode *node = new MacExportNode(this, symbol, cummulative_string);
|
|
AddObject(node);
|
|
return node;
|
|
}
|
|
|
|
void MacExportNode::AddSymbol(MacExport *symbol)
|
|
{
|
|
std::string symbol_name = symbol->name();
|
|
const char *partial_str = symbol_name.c_str() + cummulative_string_.size();
|
|
for (size_t j = 0; j < count(); j++) {
|
|
MacExportNode *node = item(j);
|
|
std::string node_cummulative_string = node->cummulative_string();
|
|
const char *sub_string = node_cummulative_string.c_str() + cummulative_string_.size();
|
|
size_t sub_string_len = strlen(sub_string);
|
|
if (strncmp(sub_string, partial_str, sub_string_len) == 0) {
|
|
node->AddSymbol(symbol);
|
|
return;
|
|
} else for (size_t i = sub_string_len; i > 1; i--) {
|
|
if (strncmp(sub_string, partial_str, i - 1) == 0) {
|
|
MacExportNode *new_node = Add(NULL, node_cummulative_string.substr(0, cummulative_string_.size() + i - 1));
|
|
node->set_owner(new_node);
|
|
new_node->AddSymbol(symbol);
|
|
return;
|
|
}
|
|
}
|
|
}
|
|
|
|
Add(symbol, symbol->name());
|
|
}
|
|
|
|
void MacExportNode::WriteToData(EncodedData &data, uint64_t base_address)
|
|
{
|
|
size_t i;
|
|
|
|
offset_ = (uint32_t)data.size();
|
|
data.WriteByte(0);
|
|
if (symbol_) {
|
|
size_t old_size = data.size();
|
|
if (symbol_->flags() & EXPORT_SYMBOL_FLAGS_REEXPORT) {
|
|
data.WriteUleb128(symbol_->flags());
|
|
data.WriteUleb128(symbol_->other());
|
|
data.WriteString(symbol_->forwarded_name());
|
|
} else if (symbol_->flags() & EXPORT_SYMBOL_FLAGS_STUB_AND_RESOLVER) {
|
|
data.WriteUleb128(symbol_->flags());
|
|
data.WriteUleb128(symbol_->address() - base_address);
|
|
data.WriteUleb128(symbol_->other());
|
|
} else {
|
|
data.WriteUleb128(symbol_->flags());
|
|
data.WriteUleb128(symbol_->address() - base_address);
|
|
}
|
|
data[old_size - 1] = static_cast<uint8_t>(data.size() - old_size);
|
|
}
|
|
|
|
// write number of children
|
|
data.push_back(static_cast<uint8_t>(count()));
|
|
// write each child
|
|
for (i = 0; i < count(); i++) {
|
|
MacExportNode *child = item(i);
|
|
data.WriteString(child->cummulative_string().substr(cummulative_string().size()));
|
|
data.WriteUleb128(child->offset());
|
|
}
|
|
}
|
|
|
|
/**
|
|
* MacFixup
|
|
*/
|
|
|
|
MacFixup::MacFixup(MacFixupList *owner, uint64_t address, uint32_t data, OperandSize size, bool is_relocation)
|
|
: BaseFixup(owner), address_(address), data_(data), size_(size), is_relocation_(is_relocation), symbol_(NULL)
|
|
{
|
|
if (is_relocation && size == osDefault) {
|
|
relocation_info reloc = relocation();
|
|
switch (reloc.r_length) {
|
|
case 0:
|
|
size_ = osByte;
|
|
break;
|
|
case 1:
|
|
size_ = osWord;
|
|
break;
|
|
case 2:
|
|
size_ = osDWord;
|
|
break;
|
|
default:
|
|
size_ = osQWord;
|
|
break;
|
|
}
|
|
}
|
|
}
|
|
|
|
MacFixup::MacFixup(MacFixupList *owner, const MacFixup &src)
|
|
: BaseFixup(owner, src)
|
|
{
|
|
address_ = src.address_;
|
|
data_ = src.data_;
|
|
size_ = src.size_;
|
|
is_relocation_ = src.is_relocation_;
|
|
symbol_ = src.symbol_;
|
|
}
|
|
|
|
MacFixup *MacFixup::Clone(IFixupList *owner) const
|
|
{
|
|
MacFixup *fixup = new MacFixup(reinterpret_cast<MacFixupList *>(owner), *this);
|
|
return fixup;
|
|
}
|
|
|
|
relocation_info MacFixup::relocation() const
|
|
{
|
|
relocation_info res = relocation_info();
|
|
if (is_relocation_)
|
|
reinterpret_cast<uint32_t *>(&res)[1] = data_;
|
|
return res;
|
|
}
|
|
|
|
FixupType MacFixup::type() const
|
|
{
|
|
switch (internal_type()) {
|
|
case REBASE_TYPE_POINTER:
|
|
case REBASE_TYPE_TEXT_ABSOLUTE32:
|
|
return ftHighLow;
|
|
default:
|
|
return ftUnknown;
|
|
}
|
|
}
|
|
|
|
uint8_t MacFixup::internal_type() const
|
|
{
|
|
if (is_relocation_) {
|
|
relocation_info reloc = relocation();
|
|
switch (reloc.r_type) {
|
|
case 0:
|
|
return REBASE_TYPE_POINTER;
|
|
default:
|
|
return -1;
|
|
}
|
|
} else {
|
|
switch (data_) {
|
|
case REBASE_TYPE_POINTER:
|
|
case REBASE_TYPE_TEXT_ABSOLUTE32:
|
|
return static_cast<uint8_t>(data_);
|
|
default:
|
|
return -1;
|
|
}
|
|
}
|
|
}
|
|
|
|
void MacFixup::set_is_relocation(bool is_relocation)
|
|
{
|
|
if (is_relocation == is_relocation_)
|
|
return;
|
|
|
|
if (is_relocation_) {
|
|
data_ = REBASE_TYPE_POINTER;
|
|
} else {
|
|
relocation_info reloc;
|
|
reloc.r_symbolnum = 0;
|
|
switch (size_) {
|
|
case osByte:
|
|
reloc.r_length = 0;
|
|
break;
|
|
case osWord:
|
|
reloc.r_length = 1;
|
|
break;
|
|
case osDWord:
|
|
reloc.r_length = 2;
|
|
break;
|
|
default:
|
|
reloc.r_length = 3;
|
|
break;
|
|
}
|
|
reloc.r_extern = 0;
|
|
reloc.r_type = 0;
|
|
data_ = reinterpret_cast<uint32_t *>(&reloc)[1];
|
|
}
|
|
is_relocation_ = is_relocation;
|
|
}
|
|
|
|
void MacFixup::Rebase(IArchitecture &file, uint64_t delta_base)
|
|
{
|
|
if (!file.AddressSeek(address_))
|
|
return;
|
|
|
|
uint64_t pos = file.Tell();
|
|
uint64_t value = 0;
|
|
size_t value_size = OperandSizeToValue(size_);
|
|
switch (internal_type()) {
|
|
case REBASE_TYPE_POINTER:
|
|
case REBASE_TYPE_TEXT_ABSOLUTE32:
|
|
assert(sizeof(value) >= value_size);
|
|
file.Read(&value, value_size);
|
|
value += delta_base;
|
|
file.Seek(pos);
|
|
file.Write(&value, value_size);
|
|
break;
|
|
}
|
|
address_ += delta_base;
|
|
}
|
|
|
|
/**
|
|
* MacFixupList
|
|
*/
|
|
|
|
MacFixupList::MacFixupList()
|
|
: BaseFixupList()
|
|
{
|
|
|
|
}
|
|
|
|
MacFixupList::MacFixupList(const MacFixupList &src)
|
|
: BaseFixupList(src)
|
|
{
|
|
|
|
}
|
|
|
|
MacFixupList *MacFixupList::Clone() const
|
|
{
|
|
MacFixupList *list = new MacFixupList(*this);
|
|
return list;
|
|
}
|
|
|
|
MacFixup *MacFixupList::item(size_t index) const
|
|
{
|
|
return reinterpret_cast<MacFixup *>(BaseFixupList::item(index));
|
|
}
|
|
|
|
MacFixup *MacFixupList::Add(uint64_t address, uint32_t data, OperandSize size, bool is_relocation)
|
|
{
|
|
MacFixup *fixup = new MacFixup(this, address, data, size, is_relocation);
|
|
AddObject(fixup);
|
|
|
|
if (fixup->type() == ftUnknown)
|
|
throw std::runtime_error("Invalid rebase type");
|
|
|
|
return fixup;
|
|
}
|
|
|
|
MacFixup *MacFixupList::AddRelocation(uint64_t address, MacSymbol *symbol, OperandSize size)
|
|
{
|
|
relocation_info reloc = relocation_info();
|
|
reloc.r_extern = 1;
|
|
reloc.r_length = size;
|
|
MacFixup *fixup = Add(address, reinterpret_cast<uint32_t *>(&reloc)[1], size, true);
|
|
fixup->set_symbol(symbol);
|
|
return fixup;
|
|
}
|
|
|
|
void MacFixupList::ReadRebaseInfo(MacArchitecture &file, uint32_t rebase_off, uint32_t rebase_size)
|
|
{
|
|
uint8_t bind_type = 0;
|
|
uint64_t address, segment_end;
|
|
uint32_t count;
|
|
uint32_t skip;
|
|
uint8_t immediate;
|
|
uint8_t opcode;
|
|
EncodedData buf;
|
|
ISectionList *segment_list = file.segment_list();
|
|
ISection *segment;
|
|
size_t pos, i, ptr_size, seg_index = 0;
|
|
OperandSize cpu_address_size;
|
|
|
|
if (rebase_size == 0)
|
|
return;
|
|
|
|
if (segment_list->count() < 1)
|
|
throw std::runtime_error("Runtime error at ReadRebaseInfo: Invalid segment count");
|
|
|
|
file.Seek(rebase_off);
|
|
buf.ReadFromFile(file, rebase_size);
|
|
|
|
cpu_address_size = file.cpu_address_size();
|
|
ptr_size = cpu_address_size == osQWord ? 8 : 4;
|
|
|
|
segment = segment_list->item(0);
|
|
address = segment->address();
|
|
segment_end = segment->address() + segment->size();
|
|
for (pos = 0; pos < rebase_size; ) {
|
|
uint8_t b = buf.ReadByte(&pos);
|
|
immediate = b & REBASE_IMMEDIATE_MASK;
|
|
opcode = b & REBASE_OPCODE_MASK;
|
|
switch (opcode) {
|
|
case REBASE_OPCODE_DONE:
|
|
return;
|
|
break;
|
|
case REBASE_OPCODE_SET_TYPE_IMM:
|
|
bind_type = immediate;
|
|
break;
|
|
case REBASE_OPCODE_SET_SEGMENT_AND_OFFSET_ULEB:
|
|
seg_index = immediate;
|
|
if (seg_index >= segment_list->count())
|
|
throw std::runtime_error("Invalid segment index");
|
|
segment = segment_list->item(seg_index);
|
|
address = segment->address() + buf.ReadUleb128(&pos);
|
|
segment_end = segment->address() + segment->size();
|
|
break;
|
|
case REBASE_OPCODE_ADD_ADDR_ULEB:
|
|
address += buf.ReadUleb128(&pos);
|
|
break;
|
|
case REBASE_OPCODE_ADD_ADDR_IMM_SCALED:
|
|
address += immediate * ptr_size;
|
|
break;
|
|
case REBASE_OPCODE_DO_REBASE_IMM_TIMES:
|
|
for (i = 0; i < immediate; i++) {
|
|
if (address >= segment_end)
|
|
throw std::runtime_error("Invalid rebase address");
|
|
Add(address, bind_type, cpu_address_size, false);
|
|
address += ptr_size;
|
|
}
|
|
break;
|
|
case REBASE_OPCODE_DO_REBASE_ULEB_TIMES:
|
|
count = static_cast<uint32_t>(buf.ReadUleb128(&pos));
|
|
for (i = 0; i < count; i++) {
|
|
if (address >= segment_end)
|
|
throw std::runtime_error("Invalid rebase address");
|
|
Add(address, bind_type, cpu_address_size, false);
|
|
address += ptr_size;
|
|
}
|
|
break;
|
|
case REBASE_OPCODE_DO_REBASE_ADD_ADDR_ULEB:
|
|
if (address >= segment_end)
|
|
throw std::runtime_error("Invalid rebase address");
|
|
Add(address, bind_type, cpu_address_size, false);
|
|
address += buf.ReadUleb128(&pos) + ptr_size;
|
|
break;
|
|
case REBASE_OPCODE_DO_REBASE_ULEB_TIMES_SKIPPING_ULEB:
|
|
count = static_cast<uint32_t>(buf.ReadUleb128(&pos));
|
|
skip = static_cast<uint32_t>(buf.ReadUleb128(&pos));
|
|
for (i = 0; i < count; i++) {
|
|
if (address >= segment_end)
|
|
throw std::runtime_error("Invalid rebase address");
|
|
Add(address, bind_type, cpu_address_size, false);
|
|
address += skip + ptr_size;
|
|
}
|
|
break;
|
|
default:
|
|
throw std::runtime_error("Invalid rebase opcode");
|
|
}
|
|
}
|
|
}
|
|
|
|
void MacFixupList::ReadRelocations(MacArchitecture &file, uint32_t offset, uint32_t count, bool need_external)
|
|
{
|
|
if (!offset)
|
|
return;
|
|
|
|
size_t i;
|
|
relocation_info reloc;
|
|
uint64_t reloc_base = file.GetRelocBase();
|
|
|
|
file.Seek(offset);
|
|
for (i = 0; i < count; i++) {
|
|
file.Read(&reloc, sizeof(reloc));
|
|
|
|
if (reloc.r_type != 0 || reloc.r_extern != need_external)
|
|
throw std::runtime_error("Invalid relocation type");
|
|
|
|
MacFixup *fixup = Add(reloc.r_address + reloc_base, reinterpret_cast<uint32_t *>(&reloc)[1], osDefault, true);
|
|
if (reloc.r_extern) {
|
|
if (reloc.r_symbolnum >= file.symbol_list()->count())
|
|
throw std::runtime_error("Invalid symbol index");
|
|
fixup->set_symbol(file.symbol_list()->item(reloc.r_symbolnum));
|
|
}
|
|
if (file.cpu_address_size() != fixup->size())
|
|
throw std::runtime_error("Invalid relocation size");
|
|
}
|
|
}
|
|
|
|
void MacFixupList::ReadFromFile(MacArchitecture &file)
|
|
{
|
|
dyld_info_command *dyld_info = file.dyld_info();
|
|
if (dyld_info->cmd)
|
|
ReadRebaseInfo(file, dyld_info->rebase_off, dyld_info->rebase_size);
|
|
|
|
dysymtab_command *dysymtab = file.dysymtab();
|
|
if (dysymtab->cmd) {
|
|
ReadRelocations(file, dysymtab->extreloff, dysymtab->nextrel, true);
|
|
ReadRelocations(file, dysymtab->locreloff, dysymtab->nlocrel, false);
|
|
}
|
|
}
|
|
|
|
size_t MacFixupList::WriteRelocations(MacArchitecture &file, bool need_external)
|
|
{
|
|
size_t i;
|
|
relocation_info reloc;
|
|
size_t res = 0;
|
|
uint64_t reloc_base = file.GetRelocBase();
|
|
|
|
for (i = 0; i < count(); i++) {
|
|
MacFixup *fixup = item(i);
|
|
if (!fixup->is_relocation())
|
|
continue;
|
|
|
|
reloc = fixup->relocation();
|
|
if (reloc.r_extern != need_external)
|
|
continue;
|
|
|
|
reloc.r_address = static_cast<uint32_t>(fixup->address() - reloc_base);
|
|
if (fixup->symbol())
|
|
reloc.r_symbolnum = static_cast<uint32_t>(file.symbol_list()->IndexOf(fixup->symbol()));
|
|
file.Write(&reloc, sizeof(reloc));
|
|
res++;
|
|
}
|
|
|
|
return res;
|
|
}
|
|
|
|
void MacFixupList::WriteToFile(MacArchitecture &file)
|
|
{
|
|
Pack();
|
|
|
|
dyld_info_command *dyld_info = file.dyld_info();
|
|
dysymtab_command *dysymtab = file.dysymtab();
|
|
if (dyld_info->cmd) {
|
|
dyld_info->rebase_off = static_cast<uint32_t>(file.Tell());
|
|
dyld_info->rebase_size = (uint32_t)WriteRebaseInfo(file);
|
|
if (!dyld_info->rebase_size)
|
|
dyld_info->rebase_off = 0;
|
|
} else {
|
|
// need convert fixups to relocations
|
|
for (size_t i = 0; i < count(); i++) {
|
|
MacFixup *fixup = item(i);
|
|
if (!fixup->is_relocation())
|
|
fixup->set_is_relocation(true);
|
|
}
|
|
}
|
|
if (dysymtab->cmd) {
|
|
dysymtab->extreloff = static_cast<uint32_t>(file.Tell());
|
|
dysymtab->nextrel = static_cast<uint32_t>(WriteRelocations(file, true));
|
|
if (!dysymtab->nextrel)
|
|
dysymtab->extreloff = 0;
|
|
dysymtab->locreloff = static_cast<uint32_t>(file.Tell());
|
|
dysymtab->nlocrel = static_cast<uint32_t>(WriteRelocations(file, false));
|
|
if (!dysymtab->nlocrel)
|
|
dysymtab->locreloff = 0;
|
|
}
|
|
}
|
|
|
|
size_t MacFixupList::WriteRebaseInfo(MacArchitecture &file)
|
|
{
|
|
size_t i;
|
|
std::vector<MacFixup *> info;
|
|
|
|
for (i = 0; i < count(); i++) {
|
|
MacFixup *fixup = item(i);
|
|
if (fixup->is_relocation())
|
|
continue;
|
|
info.push_back(fixup);
|
|
}
|
|
|
|
if (info.size() == 0)
|
|
return 0;
|
|
|
|
// sort rebase info by type, then address
|
|
std::sort(info.begin(), info.end(), RebaseInfoHelper());
|
|
|
|
// convert to temp encoding that can be more easily optimized
|
|
std::vector<RebaseInfo> mid;
|
|
MacSegment *segment = NULL;
|
|
uint8_t bind_type = 0;
|
|
uint64_t address = -1;
|
|
size_t ptrsize = OperandSizeToValue(file.cpu_address_size());
|
|
for (std::vector<MacFixup *>::iterator it = info.begin(); it != info.end(); ++it) {
|
|
MacFixup *fixup = *it;
|
|
|
|
if (bind_type != fixup->bind_type()) {
|
|
mid.push_back(RebaseInfo(REBASE_OPCODE_SET_TYPE_IMM, fixup->bind_type()));
|
|
bind_type = fixup->bind_type();
|
|
}
|
|
if (address != fixup->address()) {
|
|
if (!segment || (fixup->address() < segment->address()) || (fixup->address() >= segment->address() + segment->size())) {
|
|
segment = file.segment_list()->GetSectionByAddress(fixup->address());
|
|
if (!segment)
|
|
throw std::runtime_error("binding address outside range of any segment");
|
|
mid.push_back(RebaseInfo(REBASE_OPCODE_SET_SEGMENT_AND_OFFSET_ULEB, file.segment_list()->IndexOf(segment), fixup->address() - segment->address()));
|
|
}
|
|
else {
|
|
mid.push_back(RebaseInfo(REBASE_OPCODE_ADD_ADDR_ULEB, fixup->address() - address));
|
|
}
|
|
address = fixup->address();
|
|
}
|
|
mid.push_back(RebaseInfo(REBASE_OPCODE_DO_REBASE_ULEB_TIMES, 1));
|
|
address += ptrsize;
|
|
}
|
|
mid.push_back(RebaseInfo(REBASE_OPCODE_DONE, 0));
|
|
|
|
// optimize phase 1, compress packed runs of pointers
|
|
RebaseInfo *dst = &mid[0];
|
|
for (const RebaseInfo *src = &mid[0]; src->opcode != REBASE_OPCODE_DONE; ++src) {
|
|
if ( (src->opcode == REBASE_OPCODE_DO_REBASE_ULEB_TIMES) && (src->operand1 == 1) ) {
|
|
*dst = *src++;
|
|
while (src->opcode == REBASE_OPCODE_DO_REBASE_ULEB_TIMES ) {
|
|
dst->operand1 += src->operand1;
|
|
++src;
|
|
}
|
|
--src;
|
|
++dst;
|
|
}
|
|
else {
|
|
*dst++ = *src;
|
|
}
|
|
}
|
|
dst->opcode = REBASE_OPCODE_DONE;
|
|
|
|
// optimize phase 2, combine rebase/add pairs
|
|
dst = &mid[0];
|
|
for (const RebaseInfo *src = &mid[0]; src->opcode != REBASE_OPCODE_DONE; ++src) {
|
|
if ( (src->opcode == REBASE_OPCODE_DO_REBASE_ULEB_TIMES)
|
|
&& (src->operand1 == 1)
|
|
&& (src[1].opcode == REBASE_OPCODE_ADD_ADDR_ULEB)) {
|
|
dst->opcode = REBASE_OPCODE_DO_REBASE_ADD_ADDR_ULEB;
|
|
dst->operand1 = src[1].operand1;
|
|
++src;
|
|
++dst;
|
|
}
|
|
else {
|
|
*dst++ = *src;
|
|
}
|
|
}
|
|
dst->opcode = REBASE_OPCODE_DONE;
|
|
|
|
// optimize phase 3, compress packed runs of REBASE_OPCODE_DO_REBASE_ADD_ADDR_ULEB with
|
|
// same addr delta into one REBASE_OPCODE_DO_REBASE_ULEB_TIMES_SKIPPING_ULEB
|
|
dst = &mid[0];
|
|
for (const RebaseInfo *src = &mid[0]; src->opcode != REBASE_OPCODE_DONE; ++src) {
|
|
uint64_t delta = src->operand1;
|
|
if ( (src->opcode == REBASE_OPCODE_DO_REBASE_ADD_ADDR_ULEB)
|
|
&& (src[1].opcode == REBASE_OPCODE_DO_REBASE_ADD_ADDR_ULEB)
|
|
&& (src[2].opcode == REBASE_OPCODE_DO_REBASE_ADD_ADDR_ULEB)
|
|
&& (src[1].operand1 == delta)
|
|
&& (src[2].operand1 == delta) ) {
|
|
// found at least three in a row, this is worth compressing
|
|
dst->opcode = REBASE_OPCODE_DO_REBASE_ULEB_TIMES_SKIPPING_ULEB;
|
|
dst->operand1 = 1;
|
|
dst->operand2 = delta;
|
|
++src;
|
|
while ( (src->opcode == REBASE_OPCODE_DO_REBASE_ADD_ADDR_ULEB)
|
|
&& (src->operand1 == delta) ) {
|
|
dst->operand1++;
|
|
++src;
|
|
}
|
|
--src;
|
|
++dst;
|
|
}
|
|
else {
|
|
*dst++ = *src;
|
|
}
|
|
}
|
|
dst->opcode = REBASE_OPCODE_DONE;
|
|
|
|
// optimize phase 4, use immediate encodings
|
|
for (RebaseInfo *p = &mid[0]; p->opcode != REBASE_OPCODE_DONE; ++p) {
|
|
if ( (p->opcode == REBASE_OPCODE_ADD_ADDR_ULEB)
|
|
&& (p->operand1 < (15 * ptrsize))
|
|
&& ((p->operand1 % ptrsize) == 0) ) {
|
|
p->opcode = REBASE_OPCODE_ADD_ADDR_IMM_SCALED;
|
|
p->operand1 = p->operand1 / ptrsize;
|
|
}
|
|
else if ( (p->opcode == REBASE_OPCODE_DO_REBASE_ULEB_TIMES) && (p->operand1 < 15) ) {
|
|
p->opcode = REBASE_OPCODE_DO_REBASE_IMM_TIMES;
|
|
}
|
|
}
|
|
|
|
// convert to compressed encoding
|
|
EncodedData data;
|
|
data.reserve(info.size() * 2);
|
|
bool done = false;
|
|
for (std::vector<RebaseInfo>::iterator it = mid.begin(); !done && it != mid.end() ; ++it) {
|
|
switch ( it->opcode ) {
|
|
case REBASE_OPCODE_DONE:
|
|
done = true;
|
|
break;
|
|
case REBASE_OPCODE_SET_TYPE_IMM:
|
|
data.WriteByte(REBASE_OPCODE_SET_TYPE_IMM | static_cast<uint8_t>(it->operand1));
|
|
break;
|
|
case REBASE_OPCODE_SET_SEGMENT_AND_OFFSET_ULEB:
|
|
data.WriteByte(REBASE_OPCODE_SET_SEGMENT_AND_OFFSET_ULEB | static_cast<uint8_t>(it->operand1));
|
|
data.WriteUleb128(it->operand2);
|
|
break;
|
|
case REBASE_OPCODE_ADD_ADDR_ULEB:
|
|
data.WriteByte(REBASE_OPCODE_ADD_ADDR_ULEB);
|
|
data.WriteUleb128(it->operand1);
|
|
break;
|
|
case REBASE_OPCODE_ADD_ADDR_IMM_SCALED:
|
|
data.WriteByte(REBASE_OPCODE_ADD_ADDR_IMM_SCALED | static_cast<uint8_t>(it->operand1));
|
|
break;
|
|
case REBASE_OPCODE_DO_REBASE_IMM_TIMES:
|
|
data.WriteByte(REBASE_OPCODE_DO_REBASE_IMM_TIMES | static_cast<uint8_t>(it->operand1));
|
|
break;
|
|
case REBASE_OPCODE_DO_REBASE_ULEB_TIMES:
|
|
data.WriteByte(REBASE_OPCODE_DO_REBASE_ULEB_TIMES);
|
|
data.WriteUleb128(it->operand1);
|
|
break;
|
|
case REBASE_OPCODE_DO_REBASE_ADD_ADDR_ULEB:
|
|
data.WriteByte(REBASE_OPCODE_DO_REBASE_ADD_ADDR_ULEB);
|
|
data.WriteUleb128(it->operand1);
|
|
break;
|
|
case REBASE_OPCODE_DO_REBASE_ULEB_TIMES_SKIPPING_ULEB:
|
|
data.WriteByte(REBASE_OPCODE_DO_REBASE_ULEB_TIMES_SKIPPING_ULEB);
|
|
data.WriteUleb128(it->operand1);
|
|
data.WriteUleb128(it->operand2);
|
|
break;
|
|
}
|
|
}
|
|
// align to pointer size
|
|
data.resize(AlignValue(data.size(), ptrsize), 0);
|
|
|
|
return (data.size() == 0) ? 0 : file.Write(data.data(), data.size());
|
|
}
|
|
|
|
MacFixup *MacFixupList::AddDefault(OperandSize cpu_address_size, bool is_code)
|
|
{
|
|
return Add(0, is_code ? REBASE_TYPE_TEXT_ABSOLUTE32 : REBASE_TYPE_POINTER, cpu_address_size, false);
|
|
}
|
|
|
|
size_t MacFixupList::Pack()
|
|
{
|
|
for (size_t i = 0; i < count(); i++) {
|
|
MacFixup *fixup = item(i);
|
|
MacSymbol *symbol = fixup->symbol();
|
|
if (symbol && symbol->is_deleted())
|
|
fixup->set_deleted(true);
|
|
}
|
|
|
|
return BaseFixupList::Pack();
|
|
}
|
|
|
|
void MacFixupList::WriteToData(Data &data, uint64_t image_base)
|
|
{
|
|
size_t i, size_pos;
|
|
MacFixup *fixup;
|
|
IMAGE_BASE_RELOCATION reloc;
|
|
uint32_t rva, block_rva;
|
|
uint16_t type_offset, empty_offset;
|
|
|
|
Sort();
|
|
|
|
size_pos = 0;
|
|
reloc.VirtualAddress = 0;
|
|
reloc.SizeOfBlock = 0;
|
|
|
|
for (i = 0; i < count(); i++) {
|
|
fixup = item(i);
|
|
if (fixup->symbol())
|
|
continue;
|
|
|
|
rva = static_cast<uint32_t>(fixup->address() - image_base);
|
|
block_rva = rva & 0xfffff000;
|
|
if (reloc.SizeOfBlock == 0 || block_rva != reloc.VirtualAddress) {
|
|
if (reloc.SizeOfBlock > 0) {
|
|
if ((reloc.SizeOfBlock & 3) != 0) {
|
|
data.PushWord(empty_offset);
|
|
reloc.SizeOfBlock += sizeof(empty_offset);
|
|
}
|
|
data.WriteDWord(size_pos, reloc.SizeOfBlock);
|
|
}
|
|
size_pos = data.size() + 4;
|
|
reloc.VirtualAddress = block_rva;
|
|
reloc.SizeOfBlock = sizeof(reloc);
|
|
data.PushBuff(&reloc, sizeof(reloc));
|
|
empty_offset = (static_cast<uint16_t>(rva - block_rva) & 0xf00) << 4 | R_ABS;
|
|
}
|
|
type_offset = (static_cast<uint16_t>(rva - block_rva) & 0xfff) << 4 | fixup->internal_type();
|
|
data.PushWord(type_offset);
|
|
reloc.SizeOfBlock += sizeof(type_offset);
|
|
}
|
|
|
|
if (reloc.SizeOfBlock > 0) {
|
|
if ((reloc.SizeOfBlock & 3) != 0) {
|
|
data.PushWord(empty_offset);
|
|
reloc.SizeOfBlock += sizeof(empty_offset);
|
|
}
|
|
data.WriteDWord(size_pos, reloc.SizeOfBlock);
|
|
}
|
|
}
|
|
|
|
/**
|
|
* MacRuntimeFunction
|
|
*/
|
|
|
|
MacRuntimeFunction::MacRuntimeFunction(MacRuntimeFunctionList *owner, uint64_t address, uint64_t begin, uint64_t end, uint64_t unwind_address,
|
|
CommonInformationEntry *cie, const std::vector<uint8_t> &call_frame_instructions, uint32_t compact_encoding)
|
|
: BaseRuntimeFunction(owner), address_(address), begin_(begin), end_(end), unwind_address_(unwind_address), cie_(cie),
|
|
call_frame_instructions_(call_frame_instructions), compact_encoding_(compact_encoding)
|
|
{
|
|
|
|
}
|
|
|
|
MacRuntimeFunction::MacRuntimeFunction(MacRuntimeFunctionList *owner, const MacRuntimeFunction &src)
|
|
: BaseRuntimeFunction(owner)
|
|
{
|
|
address_ = src.address_;
|
|
begin_ = src.begin_;
|
|
end_ = src.end_;
|
|
unwind_address_ = src.unwind_address_;
|
|
cie_ = src.cie_;
|
|
call_frame_instructions_ = src.call_frame_instructions_;
|
|
compact_encoding_ = src.compact_encoding_;
|
|
}
|
|
|
|
MacRuntimeFunction *MacRuntimeFunction::Clone(IRuntimeFunctionList *owner) const
|
|
{
|
|
MacRuntimeFunction *func = new MacRuntimeFunction(reinterpret_cast<MacRuntimeFunctionList *>(owner), *this);
|
|
return func;
|
|
}
|
|
|
|
void MacRuntimeFunction::Parse(IArchitecture &file, IFunction &dest)
|
|
{
|
|
if (!file.AddressSeek(address_) || dest.GetCommandByAddress(address_))
|
|
return;
|
|
|
|
if (cie_)
|
|
ParseDwarf(file, dest);
|
|
else
|
|
ParseBorland(file, dest);
|
|
}
|
|
|
|
void MacRuntimeFunction::ParseDwarf(IArchitecture &file, IFunction &dest)
|
|
{
|
|
uint64_t address = address_;
|
|
IntelFunction &func = reinterpret_cast<IntelFunction &>(dest);
|
|
|
|
size_t c = func.count();
|
|
IntelCommand *command;
|
|
uint64_t value;
|
|
size_t pos;
|
|
CommandLink *link;
|
|
FunctionInfo *info;
|
|
std::vector<ICommand *> unwind_opcodes;
|
|
|
|
if (cie_->version() == 0) {
|
|
command = func.Add(address);
|
|
command->set_comment(CommentInfo(ttComment, "Compact Entry"));
|
|
command->ReadValueFromFile(file, osDWord);
|
|
} else {
|
|
command = func.Add(address);
|
|
command->set_comment(CommentInfo(ttComment, "FDE Length"));
|
|
uint32_t fde_length = static_cast<uint32_t>(command->ReadValueFromFile(file, osDWord));
|
|
address = command->next_address();
|
|
|
|
if (fde_length) {
|
|
EncodedData fde(command->next_address(), file.cpu_address_size());
|
|
fde.ReadFromFile(file, fde_length);
|
|
pos = 0;
|
|
|
|
command = func.Add(address);
|
|
command->set_comment(CommentInfo(ttComment, "CIE Pointer"));
|
|
value = command->ReadDataDWord(fde, &pos);
|
|
uint64_t cie_address = address - value;
|
|
address = command->next_address();
|
|
|
|
IntelCommand *cie_command = func.GetCommandByAddress(cie_address);
|
|
if (!cie_command) {
|
|
size_t fde_pos = pos;
|
|
uint64_t fde_address = address;
|
|
|
|
address = cie_address;
|
|
file.AddressSeek(address);
|
|
|
|
command = func.Add(address);
|
|
cie_command = command;
|
|
command->set_comment(CommentInfo(ttComment, "CIE Length"));
|
|
uint32_t cie_length = static_cast<uint32_t>(command->ReadValueFromFile(file, osDWord));
|
|
address = command->next_address();
|
|
|
|
if (cie_length) {
|
|
EncodedData cie(command->next_address(), file.cpu_address_size());
|
|
cie.ReadFromFile(file, cie_length);
|
|
pos = 0;
|
|
|
|
command = func.Add(address);
|
|
command->set_comment(CommentInfo(ttComment, "CIE ID"));
|
|
command->ReadDataDWord(cie, &pos);
|
|
address = command->next_address();
|
|
|
|
command = func.Add(address);
|
|
command->set_comment(CommentInfo(ttComment, "CIE Version"));
|
|
command->ReadDataByte(cie, &pos);
|
|
address = command->next_address();
|
|
|
|
command = func.Add(address);
|
|
command->ReadString(cie, &pos);
|
|
command->set_comment(CommentInfo(ttComment, string_format("Augmentation String: %s", cie_->augmentation().c_str())));
|
|
address = command->next_address();
|
|
|
|
command = func.Add(address);
|
|
command->ReadUleb128(cie, &pos);
|
|
command->set_comment(CommentInfo(ttComment, "Code Alignment Factor"));
|
|
address = command->next_address();
|
|
|
|
command = func.Add(address);
|
|
command->ReadSleb128(cie, &pos);
|
|
command->set_comment(CommentInfo(ttComment, "Data Alignment Factor"));
|
|
address = command->next_address();
|
|
|
|
command = func.Add(address);
|
|
command->ReadDataByte(cie, &pos);
|
|
command->set_comment(CommentInfo(ttComment, "Return Address Register"));
|
|
address = command->next_address();
|
|
|
|
if (*cie_->augmentation().c_str() == 'z') {
|
|
command = func.Add(address);
|
|
command->ReadUleb128(cie, &pos);
|
|
command->set_comment(CommentInfo(ttComment, "Augmentation Length"));
|
|
address = command->next_address();
|
|
|
|
for (size_t j = 1; j < cie_->augmentation().size(); j++) {
|
|
switch (cie_->augmentation().at(j)) {
|
|
case 'L':
|
|
command = func.Add(address);
|
|
command->set_comment(CommentInfo(ttComment, "LSDA Encoding"));
|
|
command->ReadDataByte(cie, &pos);
|
|
address = command->next_address();
|
|
break;
|
|
case 'R':
|
|
command = func.Add(address);
|
|
command->set_comment(CommentInfo(ttComment, "FDE Encoding"));
|
|
command->ReadDataByte(cie, &pos);
|
|
address = command->next_address();
|
|
break;
|
|
case 'P':
|
|
{
|
|
command = func.Add(address);
|
|
command->set_comment(CommentInfo(ttComment, "Personality Encoding"));
|
|
command->ReadDataByte(cie, &pos);
|
|
address = command->next_address();
|
|
|
|
command = func.Add(address);
|
|
command->ReadEncoding(cie, cie_->personality_encoding(), &pos);
|
|
command->set_comment(CommentInfo(ttComment, "Personality Routine"));
|
|
address = command->next_address();
|
|
}
|
|
break;
|
|
}
|
|
}
|
|
}
|
|
|
|
command = func.Add(address);
|
|
command->ReadData(cie, cie.size() - pos, &pos);
|
|
command->set_comment(CommentInfo(ttComment, "Initial Instructions"));
|
|
address = command->next_address();
|
|
}
|
|
|
|
file.AddressSeek(fde_address);
|
|
address = fde_address;
|
|
pos = fde_pos;
|
|
}
|
|
|
|
command = func.Add(address);
|
|
command->ReadEncoding(fde, cie_->fde_encoding(), &pos);
|
|
command->set_comment(CommentInfo(ttComment, string_format("Begin: %llX", begin())));
|
|
address = command->next_address();
|
|
|
|
command = func.Add(address);
|
|
command->ReadEncoding(fde, cie_->fde_encoding() & 0x0f, &pos);
|
|
command->set_comment(CommentInfo(ttComment, string_format("End: %llX", end())));
|
|
address = command->next_address();
|
|
|
|
if (*cie_->augmentation().c_str() == 'z') {
|
|
command = func.Add(address);
|
|
value = command->ReadUleb128(fde, &pos);
|
|
command->set_comment(CommentInfo(ttComment, "Augmentation Length"));
|
|
address = command->next_address();
|
|
|
|
if (cie_->augmentation().find('L') != std::string::npos) {
|
|
command = func.Add(address);
|
|
command->ReadEncoding(fde, cie_->lsda_encoding(), &pos);
|
|
command->set_comment(CommentInfo(ttComment, "LSDA"));
|
|
if (unwind_address_)
|
|
command->AddLink(0, ltOffset, unwind_address_);
|
|
|
|
address = command->next_address();
|
|
}
|
|
}
|
|
|
|
uint64_t pc = begin();
|
|
while (pos < fde.size()) {
|
|
command = func.Add(address);
|
|
size_t cur_pos = pos;
|
|
uint8_t b = fde.ReadByte(&pos);
|
|
switch (b) {
|
|
case DW_CFA_nop:
|
|
command->ReadData(fde, fde.size() - cur_pos, &cur_pos);
|
|
command->set_comment(CommentInfo(ttComment, "DW_CFA_nop"));
|
|
break;
|
|
case DW_CFA_set_loc:
|
|
pc = fde.ReadEncoding(cie_->fde_encoding(), &pos);
|
|
command->ReadData(fde, pos - cur_pos, &cur_pos);
|
|
command->set_comment(CommentInfo(ttComment, "DW_CFA_set_loc"));
|
|
break;
|
|
case DW_CFA_advance_loc1:
|
|
value = fde.ReadByte(&pos);
|
|
command->ReadData(fde, pos - cur_pos, &cur_pos);
|
|
command->set_comment(CommentInfo(ttComment, "DW_CFA_advance_loc1"));
|
|
func.range_list()->Add(pc, pc + value, NULL, NULL, command);
|
|
pc += value;
|
|
break;
|
|
case DW_CFA_advance_loc2:
|
|
value = fde.ReadWord(&pos);
|
|
command->ReadData(fde, pos - cur_pos, &cur_pos);
|
|
command->set_comment(CommentInfo(ttComment, "DW_CFA_advance_loc2"));
|
|
func.range_list()->Add(pc, pc + value, NULL, NULL, command);
|
|
pc += value;
|
|
break;
|
|
case DW_CFA_advance_loc4:
|
|
value = fde.ReadDWord(&pos);
|
|
command->ReadData(fde, pos - cur_pos, &cur_pos);
|
|
command->set_comment(CommentInfo(ttComment, "DW_CFA_advance_loc4"));
|
|
func.range_list()->Add(pc, pc + value, NULL, NULL, command);
|
|
pc += value;
|
|
break;
|
|
case DW_CFA_offset_extended:
|
|
fde.ReadUleb128(&pos);
|
|
fde.ReadUleb128(&pos);
|
|
command->ReadData(fde, pos - cur_pos, &cur_pos);
|
|
command->set_comment(CommentInfo(ttComment, "DW_CFA_offset_extended"));
|
|
break;
|
|
case DW_CFA_restore_extended:
|
|
fde.ReadUleb128(&pos);
|
|
command->ReadData(fde, pos - cur_pos, &cur_pos);
|
|
command->set_comment(CommentInfo(ttComment, "DW_CFA_restore_extended"));
|
|
break;
|
|
case DW_CFA_undefined:
|
|
fde.ReadUleb128(&pos);
|
|
command->ReadData(fde, pos - cur_pos, &cur_pos);
|
|
command->set_comment(CommentInfo(ttComment, "DW_CFA_undefined"));
|
|
break;
|
|
case DW_CFA_same_value:
|
|
fde.ReadUleb128(&pos);
|
|
command->ReadData(fde, pos - cur_pos, &cur_pos);
|
|
command->set_comment(CommentInfo(ttComment, "DW_CFA_same_value"));
|
|
break;
|
|
case DW_CFA_register:
|
|
fde.ReadUleb128(&pos);
|
|
fde.ReadUleb128(&pos);
|
|
command->ReadData(fde, pos - cur_pos, &cur_pos);
|
|
command->set_comment(CommentInfo(ttComment, "DW_CFA_register"));
|
|
break;
|
|
case DW_CFA_remember_state:
|
|
command->ReadData(fde, pos - cur_pos, &cur_pos);
|
|
command->set_comment(CommentInfo(ttComment, "DW_CFA_remember_state"));
|
|
break;
|
|
case DW_CFA_restore_state:
|
|
command->ReadData(fde, pos - cur_pos, &cur_pos);
|
|
command->set_comment(CommentInfo(ttComment, "DW_CFA_restore_state"));
|
|
break;
|
|
case DW_CFA_def_cfa:
|
|
fde.ReadUleb128(&pos);
|
|
fde.ReadUleb128(&pos);
|
|
command->ReadData(fde, pos - cur_pos, &cur_pos);
|
|
command->set_comment(CommentInfo(ttComment, "DW_CFA_def_cfa"));
|
|
break;
|
|
case DW_CFA_def_cfa_register:
|
|
fde.ReadUleb128(&pos);
|
|
command->ReadData(fde, pos - cur_pos, &cur_pos);
|
|
command->set_comment(CommentInfo(ttComment, "DW_CFA_def_cfa_register"));
|
|
break;
|
|
case DW_CFA_def_cfa_offset:
|
|
fde.ReadUleb128(&pos);
|
|
command->ReadData(fde, pos - cur_pos, &cur_pos);
|
|
command->set_comment(CommentInfo(ttComment, "DW_CFA_def_cfa_offset"));
|
|
break;
|
|
case DW_CFA_def_cfa_expression:
|
|
value = fde.ReadUleb128(&pos);
|
|
command->ReadData(fde, pos - cur_pos, &cur_pos);
|
|
command->set_comment(CommentInfo(ttComment, "DW_CFA_def_cfa_expression"));
|
|
pos += static_cast<size_t>(value);
|
|
break;
|
|
case DW_CFA_expression:
|
|
fde.ReadUleb128(&pos);
|
|
value = fde.ReadUleb128(&pos);
|
|
command->ReadData(fde, pos - cur_pos, &cur_pos);
|
|
command->set_comment(CommentInfo(ttComment, "DW_CFA_def_cfa_expression"));
|
|
pos += static_cast<size_t>(value);
|
|
break;
|
|
case DW_CFA_offset_extended_sf:
|
|
fde.ReadUleb128(&pos);
|
|
fde.ReadSleb128(&pos);
|
|
command->ReadData(fde, pos - cur_pos, &cur_pos);
|
|
command->set_comment(CommentInfo(ttComment, "DW_CFA_offset_extended_sf"));
|
|
break;
|
|
case DW_CFA_def_cfa_sf:
|
|
fde.ReadUleb128(&pos);
|
|
fde.ReadSleb128(&pos);
|
|
command->ReadData(fde, pos - cur_pos, &cur_pos);
|
|
command->set_comment(CommentInfo(ttComment, "DW_CFA_def_cfa_sf"));
|
|
break;
|
|
case DW_CFA_def_cfa_offset_sf:
|
|
fde.ReadSleb128(&pos);
|
|
command->ReadData(fde, pos - cur_pos, &cur_pos);
|
|
command->set_comment(CommentInfo(ttComment, "DW_CFA_def_cfa_offset_sf"));
|
|
break;
|
|
case DW_CFA_val_offset:
|
|
fde.ReadUleb128(&pos);
|
|
fde.ReadUleb128(&pos);
|
|
command->ReadData(fde, pos - cur_pos, &cur_pos);
|
|
command->set_comment(CommentInfo(ttComment, "DW_CFA_val_offset"));
|
|
break;
|
|
case DW_CFA_val_offset_sf:
|
|
fde.ReadUleb128(&pos);
|
|
fde.ReadSleb128(&pos);
|
|
command->ReadData(fde, pos - cur_pos, &cur_pos);
|
|
command->set_comment(CommentInfo(ttComment, "DW_CFA_val_offset_sf"));
|
|
break;
|
|
case DW_CFA_val_expression:
|
|
fde.ReadUleb128(&pos);
|
|
value = fde.ReadUleb128(&pos);
|
|
command->ReadData(fde, pos - cur_pos, &cur_pos);
|
|
command->set_comment(CommentInfo(ttComment, "DW_CFA_val_expression"));
|
|
pos += static_cast<size_t>(value);
|
|
break;
|
|
case DW_CFA_GNU_window_save:
|
|
command->ReadData(fde, pos - cur_pos, &cur_pos);
|
|
command->set_comment(CommentInfo(ttComment, "DW_CFA_GNU_window_save"));
|
|
break;
|
|
case DW_CFA_GNU_args_size:
|
|
fde.ReadUleb128(&pos);
|
|
command->ReadData(fde, pos - cur_pos, &cur_pos);
|
|
command->set_comment(CommentInfo(ttComment, "DW_CFA_GNU_args_size"));
|
|
break;
|
|
case DW_CFA_GNU_negative_offset_extended:
|
|
fde.ReadUleb128(&pos);
|
|
fde.ReadUleb128(&pos);
|
|
command->ReadData(fde, pos - cur_pos, &cur_pos);
|
|
command->set_comment(CommentInfo(ttComment, "DW_CFA_GNU_negative_offset_extended"));
|
|
break;
|
|
default:
|
|
switch (b & 0xc0) {
|
|
case DW_CFA_advance_loc:
|
|
value = (b & 0x3f);
|
|
command->ReadData(fde, pos - cur_pos, &cur_pos);
|
|
command->set_comment(CommentInfo(ttComment, "DW_CFA_advance_loc"));
|
|
func.range_list()->Add(pc, pc + value, NULL, NULL, command);
|
|
pc += value;
|
|
break;
|
|
case DW_CFA_offset:
|
|
fde.ReadUleb128(&pos);
|
|
command->ReadData(fde, pos - cur_pos, &cur_pos);
|
|
command->set_comment(CommentInfo(ttComment, "DW_CFA_offset"));
|
|
break;
|
|
case DW_CFA_restore:
|
|
command->ReadData(fde, pos - cur_pos, &cur_pos);
|
|
command->set_comment(CommentInfo(ttComment, "DW_CFA_restore"));
|
|
break;
|
|
default:
|
|
command->ReadData(fde, fde.size() - cur_pos, &cur_pos);
|
|
break;
|
|
}
|
|
}
|
|
pos = cur_pos;
|
|
address = command->next_address();
|
|
|
|
if (b != DW_CFA_nop)
|
|
unwind_opcodes.push_back(command);
|
|
}
|
|
}
|
|
}
|
|
for (size_t i = c; i < func.count(); i++) {
|
|
command = func.item(i);
|
|
command->exclude_option(roClearOriginalCode);
|
|
command->exclude_option(roNeedCompile);
|
|
}
|
|
|
|
if (unwind_address_ && file.AddressSeek(unwind_address_)) {
|
|
address = unwind_address_;
|
|
|
|
EncodedData lsda(address, file.cpu_address_size());
|
|
lsda.ReadFromFile(file, static_cast<size_t>(file.selected_segment()->address() + file.selected_segment()->physical_size() - address));
|
|
pos = 0;
|
|
|
|
command = func.Add(address);
|
|
command->set_comment(CommentInfo(ttComment, "LPStart Encoding"));
|
|
uint8_t start_encoding = command->ReadDataByte(lsda, &pos);
|
|
command->include_option(roCreateNewBlock);
|
|
address = command->next_address();
|
|
|
|
IntelCommand *entry = command;
|
|
|
|
uint64_t start = begin();
|
|
AddressBaseType base_type = btFunctionBegin;
|
|
if (start_encoding != DW_EH_PE_omit) {
|
|
command = func.Add(address);
|
|
command->set_comment(CommentInfo(ttComment, "LPStart"));
|
|
start = command->ReadEncoding(lsda, start_encoding, &pos);
|
|
address = command->next_address();
|
|
base_type = btValue;
|
|
}
|
|
|
|
info = func.function_info_list()->Add(begin(), end(), base_type, (base_type == btValue) ? start : 0, 0, 0xff, this, entry);
|
|
info->set_unwind_opcodes(unwind_opcodes);
|
|
|
|
command = func.Add(address);
|
|
command->set_comment(CommentInfo(ttComment, "TTable Encoding"));
|
|
uint8_t ttable_encoding = command->ReadDataByte(lsda, &pos);
|
|
address = command->next_address();
|
|
|
|
size_t ttable_offset = 0;
|
|
IntelCommand *ttable_offset_entry = NULL;
|
|
if (ttable_encoding != DW_EH_PE_omit) {
|
|
ttable_offset_entry = func.Add(address);
|
|
ttable_offset_entry->set_comment(CommentInfo(ttComment, "TTable Offset"));
|
|
ttable_offset_entry->include_option(roFillNop);
|
|
ttable_offset = static_cast<size_t>(ttable_offset_entry->ReadUleb128(lsda, &pos)) + pos;
|
|
address = ttable_offset_entry->next_address();
|
|
}
|
|
|
|
command = func.Add(address);
|
|
command->set_comment(CommentInfo(ttComment, "Call Site Encoding"));
|
|
uint8_t call_site_encoding = command->ReadDataByte(lsda, &pos);
|
|
address = command->next_address();
|
|
|
|
command = func.Add(address);
|
|
command->set_comment(CommentInfo(ttComment, "Call Site Length"));
|
|
uint64_t call_site_length = command->ReadUleb128(lsda, &pos);
|
|
address = command->next_address();
|
|
|
|
std::set<int64_t> action_list;
|
|
size_t old_pos = pos;
|
|
while (pos - old_pos < call_site_length) {
|
|
IntelCommand *begin_entry = func.Add(address);
|
|
uint64_t begin_address = start + begin_entry->ReadEncoding(lsda, call_site_encoding, &pos);
|
|
begin_entry->set_comment(CommentInfo(ttComment, string_format("Begin: %llX", begin_address)));
|
|
address = begin_entry->next_address();
|
|
|
|
IntelCommand *size_entry = func.Add(address);
|
|
uint64_t end_address = begin_address + size_entry->ReadEncoding(lsda, call_site_encoding, &pos);
|
|
size_entry->set_comment(CommentInfo(ttComment, string_format("End: %llX", end_address)));
|
|
address = size_entry->next_address();
|
|
|
|
func.range_list()->Add(begin_address, end_address, begin_entry, NULL, size_entry);
|
|
|
|
command = func.Add(address);
|
|
value = command->ReadEncoding(lsda, call_site_encoding, &pos);
|
|
if (value) {
|
|
value += begin();
|
|
link = command->AddLink(0, ltMemSEHBlock, value);
|
|
link->set_sub_value(begin());
|
|
link->set_base_function_info(info);
|
|
}
|
|
command->set_comment(CommentInfo(ttComment, string_format("Landing Pad: %llX", value)));
|
|
address = command->next_address();
|
|
|
|
command = func.Add(address);
|
|
command->set_comment(CommentInfo(ttComment, "Action"));
|
|
value = command->ReadUleb128(lsda, &pos);
|
|
address = command->next_address();
|
|
|
|
if (value)
|
|
action_list.insert(value);
|
|
}
|
|
|
|
if (ttable_encoding != DW_EH_PE_omit) {
|
|
std::set<int64_t> type_index_list;
|
|
std::set<int64_t> spec_index_list;
|
|
int64_t action = 1;
|
|
while (action_list.size()) {
|
|
action_list.erase(action);
|
|
|
|
old_pos = pos;
|
|
|
|
command = func.Add(address);
|
|
command->set_comment(CommentInfo(ttComment, "Type Filter"));
|
|
int64_t index = command->ReadSleb128(lsda, &pos);
|
|
address = command->next_address();
|
|
|
|
if (index > 0)
|
|
type_index_list.insert(index);
|
|
else if (index < 0)
|
|
spec_index_list.insert(index);
|
|
|
|
command = func.Add(address);
|
|
command->set_comment(CommentInfo(ttComment, "Next Action"));
|
|
int64_t next_action = command->ReadSleb128(lsda, &pos);
|
|
address = command->next_address();
|
|
|
|
action += pos - old_pos;
|
|
|
|
if (next_action >= action)
|
|
action_list.insert(next_action);
|
|
}
|
|
|
|
size_t old_count = func.count();
|
|
|
|
pos = ttable_offset;
|
|
address = lsda.address() + pos;
|
|
for (size_t i = 0; i < spec_index_list.size(); i++) {
|
|
command = func.Add(address);
|
|
command->set_comment(CommentInfo(ttComment, "Exception Spec"));
|
|
value = command->ReadUleb128(lsda, &pos);
|
|
address = command->next_address();
|
|
|
|
if (value > 0)
|
|
type_index_list.insert(value);
|
|
}
|
|
|
|
pos = ttable_offset - type_index_list.size() * lsda.encoding_size(ttable_encoding);
|
|
address = lsda.address() + pos;
|
|
for (size_t i = 0; i < type_index_list.size(); i++) {
|
|
command = func.Add(address);
|
|
command->set_comment(CommentInfo(ttComment, "Type Info"));
|
|
value = command->ReadEncoding(lsda, ttable_encoding, &pos);
|
|
if (command->operand(0).value)
|
|
link = command->AddLink(0, (ttable_encoding & 0x70) == DW_EH_PE_pcrel ? ltDelta : ltOffset, value);
|
|
address = command->next_address();
|
|
}
|
|
|
|
if (old_count < func.count()) {
|
|
address = func.item(old_count)->address();
|
|
link = ttable_offset_entry->AddLink(0, ltDelta, address);
|
|
link->set_sub_value(ttable_offset_entry->dump_size() + address - lsda.address() - ttable_offset);
|
|
}
|
|
}
|
|
} else {
|
|
// no LSDA
|
|
info = func.function_info_list()->Add(begin(), end(), btFunctionBegin, 0, 0, 0xff, this, NULL);
|
|
info->set_unwind_opcodes(unwind_opcodes);
|
|
}
|
|
|
|
for (size_t i = c; i < func.count(); i++) {
|
|
command = func.item(i);
|
|
command->exclude_option(roClearOriginalCode);
|
|
}
|
|
}
|
|
|
|
void MacRuntimeFunction::ParseBorland(IArchitecture &file, IFunction &dest)
|
|
{
|
|
uint64_t address = address_;
|
|
IntelFunction &func = reinterpret_cast<IntelFunction &>(dest);
|
|
|
|
size_t c = func.count();
|
|
IntelCommand *command;
|
|
CommandLink *link;
|
|
uint64_t value;
|
|
|
|
command = func.Add(address);
|
|
command->set_comment(CommentInfo(ttComment, "Begin"));
|
|
command->ReadValueFromFile(file, osDWord);
|
|
address = command->next_address();
|
|
|
|
command = func.Add(address);
|
|
command->set_comment(CommentInfo(ttComment, "End"));
|
|
command->ReadValueFromFile(file, osDWord);
|
|
/*address =*/ command->next_address();
|
|
|
|
EncodedData data;
|
|
data.resize(30);
|
|
file.AddressSeek(end_ - data.size());
|
|
for (size_t i = 0; i < data.size(); i++) {
|
|
data[data.size() - 1 - i] = file.ReadByte();
|
|
}
|
|
|
|
//uint32_t v1 = 0;
|
|
//uint32_t v2 = 0;
|
|
uint32_t v3 = 0;
|
|
uint32_t v4 = 0;
|
|
uint32_t v5 = 0;
|
|
//uint32_t v6 = 0;
|
|
|
|
uint8_t b = data[0];
|
|
size_t pos = 1;
|
|
|
|
if (b & 0x80) {
|
|
/*v6 = */data.ReadUnsigned(&pos);
|
|
}
|
|
|
|
if (b & 0x40) {
|
|
if (b & 0x10) {
|
|
/*v1 = */data.ReadUnsigned(&pos);
|
|
} else { //-V523
|
|
/*v2 = */data.ReadUnsigned(&pos);
|
|
}
|
|
}
|
|
|
|
if (b & 0x20) {
|
|
v3 = data.ReadUnsigned(&pos);
|
|
}
|
|
|
|
if ((b & 0x18) != 0x10) {
|
|
v4 = data.ReadUnsigned(&pos);
|
|
v5 = data.ReadUnsigned(&pos);
|
|
} else if (b & 0x20) {
|
|
v5 = data.ReadUnsigned(&pos);
|
|
}
|
|
|
|
address = end_ - pos - sizeof(uint32_t);
|
|
file.AddressSeek(address);
|
|
command = func.Add(address);
|
|
value = command->ReadValueFromFile(file, osDWord);
|
|
|
|
command = func.Add(address + command->dump_size());
|
|
command->ReadArray(file, pos);
|
|
|
|
if (v3) {
|
|
uint64_t base_address = address - value - v3 * 12 - v4 - v5;
|
|
address = base_address + v5;
|
|
file.AddressSeek(address);
|
|
for (size_t i = 0; i < v3; i++) {
|
|
command = func.Add(address);
|
|
command->set_comment(CommentInfo(ttComment, "Begin"));
|
|
command->ReadValueFromFile(file, osDWord);
|
|
//command->include_option(roRangeBeginEntry);
|
|
address = command->next_address();
|
|
|
|
command = func.Add(address);
|
|
command->set_comment(CommentInfo(ttComment, "End"));
|
|
command->ReadValueFromFile(file, osDWord);
|
|
//command->include_option(roRangeEndEntry);
|
|
address = command->next_address();
|
|
|
|
command = func.Add(address);
|
|
command->set_comment(CommentInfo(ttComment, "Handler"));
|
|
value = command->ReadValueFromFile(file, osDWord);
|
|
link = command->AddLink(0, ltExtSEHHandler, value + base_address);
|
|
link->set_sub_value(base_address);
|
|
address = command->next_address();
|
|
}
|
|
}
|
|
|
|
for (size_t i = c; i < func.count(); i++) {
|
|
command = func.item(i);
|
|
command->exclude_option(roClearOriginalCode);
|
|
command->exclude_option(roNeedCompile);
|
|
}
|
|
}
|
|
|
|
void MacRuntimeFunction::Rebase(uint64_t delta_base)
|
|
{
|
|
address_ += delta_base;
|
|
begin_ += delta_base;
|
|
end_ += delta_base;
|
|
if (unwind_address_)
|
|
unwind_address_ += delta_base;
|
|
}
|
|
|
|
/**
|
|
* MacRuntimeFunctionList
|
|
*/
|
|
|
|
MacRuntimeFunctionList::MacRuntimeFunctionList()
|
|
: BaseRuntimeFunctionList(), address_(0)
|
|
{
|
|
cie_list_ = new CommonInformationEntryList();
|
|
}
|
|
|
|
MacRuntimeFunctionList::MacRuntimeFunctionList(const MacRuntimeFunctionList &src)
|
|
: BaseRuntimeFunctionList(src)
|
|
{
|
|
cie_list_ = src.cie_list_->Clone();
|
|
address_ = src.address_;
|
|
for (size_t i = 0; i < count(); i++) {
|
|
MacRuntimeFunction *func = item(i);
|
|
if (func->cie())
|
|
func->set_cie(cie_list_->item(src.cie_list_->IndexOf(func->cie())));
|
|
}
|
|
}
|
|
|
|
MacRuntimeFunctionList::~MacRuntimeFunctionList()
|
|
{
|
|
delete cie_list_;
|
|
}
|
|
|
|
MacRuntimeFunctionList *MacRuntimeFunctionList::Clone() const
|
|
{
|
|
MacRuntimeFunctionList *list = new MacRuntimeFunctionList(*this);
|
|
return list;
|
|
}
|
|
|
|
void MacRuntimeFunctionList::clear()
|
|
{
|
|
cie_list_->clear();
|
|
IRuntimeFunctionList::clear();
|
|
}
|
|
|
|
MacRuntimeFunction *MacRuntimeFunctionList::item(size_t index) const
|
|
{
|
|
return reinterpret_cast<MacRuntimeFunction *>(IRuntimeFunctionList::item(index));
|
|
}
|
|
|
|
MacRuntimeFunction *MacRuntimeFunctionList::Add(uint64_t address, uint64_t begin, uint64_t end, uint64_t unwind_address, IRuntimeFunction *source, const std::vector<uint8_t> &call_frame_instructions)
|
|
{
|
|
if (!source)
|
|
throw std::runtime_error("Invalid runtime function");
|
|
|
|
MacRuntimeFunction *src = reinterpret_cast<MacRuntimeFunction *>(source);
|
|
return Add(address, begin, end, unwind_address, src->cie(), call_frame_instructions, src->compact_encoding());
|
|
}
|
|
|
|
MacRuntimeFunction *MacRuntimeFunctionList::Add(uint64_t address, uint64_t begin, uint64_t end, uint64_t unwind_address, CommonInformationEntry *cie, const std::vector<uint8_t> &call_frame_instructions, uint32_t compact_encoding)
|
|
{
|
|
MacRuntimeFunction *func = new MacRuntimeFunction(this, address, begin, end, unwind_address, cie, call_frame_instructions, compact_encoding);
|
|
AddObject(func);
|
|
return func;
|
|
}
|
|
|
|
MacRuntimeFunction *MacRuntimeFunctionList::GetFunctionByAddress(uint64_t address) const
|
|
{
|
|
return reinterpret_cast<MacRuntimeFunction *>(BaseRuntimeFunctionList::GetFunctionByAddress(address));
|
|
}
|
|
|
|
void MacRuntimeFunctionList::ReadFromFile(MacArchitecture &file)
|
|
{
|
|
MacSection *section = file.runtime_functions_section() ? file.runtime_functions_section() : file.unwind_info_section();
|
|
if (!section) {
|
|
if (file.import_list()->GetImportByName("@rpath/libcgunwind.1.0.dylib")) {
|
|
MacSegment *segment = file.segment_list()->GetSectionByName(SEG_TEXT);
|
|
MacSection *section = segment ? file.section_list()->GetSectionByName(segment, SECT_TEXT) : NULL;
|
|
if (segment && section) {
|
|
Signature sign(NULL, "0700DEFB", 0);
|
|
size_t read_size = 0;
|
|
uint8_t buf[0x1000];
|
|
while (read_size < segment->physical_size()) {
|
|
file.Seek(segment->physical_offset() + read_size);
|
|
size_t n = file.Read(buf, std::min(static_cast<size_t>(segment->physical_size() - read_size), sizeof(buf)));
|
|
for (size_t i = 0; i < n; i++) {
|
|
if (sign.SearchByte(buf[i])) {
|
|
uint64_t address = segment->address() + read_size + i + 1 - sign.size();
|
|
file.AddressSeek(address + sign.size());
|
|
uint32_t delta = file.ReadDWord();
|
|
if (address - delta == section->address()) {
|
|
ReadBorlandInfo(file, address);
|
|
return;
|
|
}
|
|
}
|
|
}
|
|
read_size += n;
|
|
}
|
|
}
|
|
}
|
|
} else if (section->name() == SECT_EH_FRAME) {
|
|
ReadDwarfInfo(file, section->address(), static_cast<uint32_t>(section->size()));
|
|
section = file.unwind_info_section();
|
|
if (section && section->name() == SECT_UNWIND_INFO) {
|
|
MacRuntimeFunctionList tmp_list;
|
|
std::map<CommonInformationEntry *, CommonInformationEntry *> cie_map;
|
|
tmp_list.ReadCompactInfo(file, section->address(), static_cast<uint32_t>(section->size()));
|
|
for (size_t i = 0; i < tmp_list.count(); i++) {
|
|
MacRuntimeFunction *func = tmp_list.item(i);
|
|
if (!GetFunctionByAddress(func->begin())) {
|
|
CommonInformationEntry *cie;
|
|
MacRuntimeFunction *new_func = func->Clone(this);
|
|
std::map<CommonInformationEntry *, CommonInformationEntry *>::const_iterator it = cie_map.find(func->cie());
|
|
if (it != cie_map.end()) {
|
|
cie = it->second;
|
|
}
|
|
else {
|
|
cie = func->cie()->Clone(cie_list_);
|
|
cie_list_->AddObject(cie);
|
|
cie_map[func->cie()] = cie;
|
|
}
|
|
new_func->set_cie(cie);
|
|
AddObject(new_func);
|
|
}
|
|
}
|
|
}
|
|
}
|
|
else if (section->name() == SECT_UNWIND_INFO)
|
|
ReadCompactInfo(file, section->address(), static_cast<uint32_t>(section->size()));
|
|
else if (section->name() == SECT_INIT_TEXT)
|
|
ReadBorlandInfo(file, section->address());
|
|
}
|
|
|
|
void MacRuntimeFunctionList::ReadDwarfInfo(MacArchitecture &file, uint64_t address, uint32_t size)
|
|
{
|
|
if (!file.AddressSeek(address))
|
|
throw std::runtime_error("Invalid format");
|
|
|
|
size_t pos = 0;
|
|
std::map<uint64_t, CommonInformationEntry*> cie_map;
|
|
for (uint32_t i = 0; i < size; ) {
|
|
uint32_t length = file.ReadDWord();
|
|
if (!length)
|
|
break;
|
|
|
|
uint64_t cur_address = address + i;
|
|
EncodedData data(cur_address + sizeof(length), file.cpu_address_size());
|
|
data.ReadFromFile(file, length);
|
|
pos = 0;
|
|
|
|
uint32_t cie_id = data.ReadDWord(&pos);
|
|
if (cie_id == 0) {
|
|
// CIE
|
|
uint8_t fde_encoding = DW_EH_PE_absptr;
|
|
uint8_t lsda_encoding = DW_EH_PE_omit;
|
|
uint8_t personality_encoding = DW_EH_PE_omit;
|
|
uint64_t personality_routine = 0;
|
|
|
|
uint8_t version = data.ReadByte(&pos);
|
|
if (version != 1 && version != 3)
|
|
throw std::runtime_error("Invalid CIE version");
|
|
std::string augmentation = data.ReadString(&pos);
|
|
uint64_t code_alignment_factor = data.ReadUleb128(&pos);
|
|
uint64_t data_alignment_factor = data.ReadSleb128(&pos);
|
|
uint8_t return_address_register = data.ReadByte(&pos);
|
|
if (*augmentation.c_str() == 'z') {
|
|
data.ReadUleb128(&pos);
|
|
for (size_t j = 1; j < augmentation.size(); j++) {
|
|
switch (augmentation[j]) {
|
|
case 'L':
|
|
lsda_encoding = data.ReadByte(&pos);
|
|
break;
|
|
case 'R':
|
|
fde_encoding = data.ReadByte(&pos);
|
|
break;
|
|
case 'P':
|
|
{
|
|
personality_encoding = data.ReadByte(&pos);
|
|
personality_routine = data.ReadEncoding(personality_encoding, &pos);
|
|
}
|
|
break;
|
|
}
|
|
}
|
|
}
|
|
|
|
std::vector<uint8_t> initial_instructions;
|
|
initial_instructions.resize(length - pos);
|
|
if (!initial_instructions.empty())
|
|
data.Read(initial_instructions.data(), initial_instructions.size(), &pos);
|
|
|
|
CommonInformationEntry *cie = cie_list_->Add(version, augmentation, code_alignment_factor, data_alignment_factor, return_address_register, fde_encoding, lsda_encoding, personality_encoding, personality_routine, initial_instructions);
|
|
cie_map[cur_address] = cie;
|
|
} else {
|
|
// FDE
|
|
std::map<uint64_t, CommonInformationEntry*>::iterator it = cie_map.find(cur_address + sizeof(length) - cie_id);
|
|
if (it == cie_map.end())
|
|
throw std::runtime_error("Invalid CIE pointer");
|
|
|
|
CommonInformationEntry *cie = it->second;
|
|
uint64_t begin = data.ReadEncoding(cie->fde_encoding(), &pos);
|
|
uint64_t end = begin + data.ReadEncoding(cie->fde_encoding() & 0x0f, &pos);
|
|
|
|
uint64_t lsda_address = 0;
|
|
if (*cie->augmentation().c_str() == 'z') {
|
|
data.ReadUleb128(&pos);
|
|
if (cie->augmentation().find('L') != std::string::npos) {
|
|
size_t old_pos = pos;
|
|
if (data.ReadEncoding(cie->lsda_encoding() & 0x0f, &pos)) {
|
|
pos = old_pos;
|
|
lsda_address = data.ReadEncoding(cie->lsda_encoding(), &pos);
|
|
}
|
|
}
|
|
}
|
|
std::vector<uint8_t> call_frame_instructions;
|
|
call_frame_instructions.resize(length - pos);
|
|
if (!call_frame_instructions.empty())
|
|
data.Read(call_frame_instructions.data(), call_frame_instructions.size(), &pos);
|
|
|
|
uint32_t compact_encoding = DwarfParser::CreateCompactEncoding(file, call_frame_instructions, cie, begin);
|
|
if (lsda_address)
|
|
compact_encoding |= UNWIND_HAS_LSDA;
|
|
Add(cur_address, begin, end, lsda_address, cie, call_frame_instructions, compact_encoding);
|
|
}
|
|
|
|
i += sizeof(length) + length;
|
|
}
|
|
}
|
|
|
|
void MacRuntimeFunctionList::ReadCompactInfo(MacArchitecture &file, uint64_t address, uint32_t size)
|
|
{
|
|
if (!file.AddressSeek(address))
|
|
throw std::runtime_error("Invalid format");
|
|
|
|
uint64_t base_address = file.segment_list()->GetBaseSegment()->address();
|
|
|
|
uint64_t pos = file.Tell();
|
|
|
|
unwind_info_section_header header;
|
|
file.Read(&header, sizeof(header));
|
|
if (header.version != UNWIND_SECTION_VERSION)
|
|
throw std::runtime_error("Invalid unwind section version");
|
|
|
|
file.Seek(pos + header.commonEncodingsArraySectionOffset);
|
|
std::vector<uint32_t> common_encoding_list;
|
|
for (size_t i = 0; i < header.commonEncodingsArrayCount; i++) {
|
|
uint32_t entry = file.ReadDWord();
|
|
common_encoding_list.push_back(entry);
|
|
}
|
|
|
|
file.Seek(pos + header.personalityArraySectionOffset);
|
|
std::vector<uint32_t> personality_list;
|
|
for (size_t i = 0; i < header.personalityArrayCount; i++) {
|
|
uint32_t entry = file.ReadDWord();
|
|
personality_list.push_back(entry);
|
|
}
|
|
|
|
file.Seek(pos + header.indexSectionOffset);
|
|
std::vector<unwind_info_section_header_index_entry> index_entry_list;
|
|
for (size_t i = 0; i < header.indexCount; i++) {
|
|
unwind_info_section_header_index_entry entry;
|
|
file.Read(&entry, sizeof(entry));
|
|
index_entry_list.push_back(entry);
|
|
}
|
|
|
|
size_t lsda_entry_count = (index_entry_list[index_entry_list.size() - 1].lsdaIndexArraySectionOffset - index_entry_list[0].lsdaIndexArraySectionOffset) / sizeof(unwind_info_section_header_lsda_index_entry);
|
|
file.Seek(pos + index_entry_list[0].lsdaIndexArraySectionOffset);
|
|
std::vector<unwind_info_section_header_lsda_index_entry> lsda_entry_list;
|
|
for (size_t i = 0; i < lsda_entry_count; i++) {
|
|
unwind_info_section_header_lsda_index_entry entry;
|
|
file.Read(&entry, sizeof(entry));
|
|
lsda_entry_list.push_back(entry);
|
|
}
|
|
|
|
struct Info {
|
|
uint64_t address;
|
|
uint32_t offset;
|
|
uint32_t encoding;
|
|
Info(uint64_t address_, uint32_t offset_, uint32_t encoding_) : address(address_), offset(offset_), encoding(encoding_) {}
|
|
};
|
|
|
|
if (index_entry_list.empty() || index_entry_list[index_entry_list.size() - 1].secondLevelPagesSectionOffset)
|
|
throw std::runtime_error("Invalid format");
|
|
|
|
std::vector<Info> info_list;
|
|
for (size_t i = 0; i < index_entry_list.size() - 1; i++) {
|
|
size_t page_offset = index_entry_list[i].secondLevelPagesSectionOffset;
|
|
|
|
file.Seek(pos + page_offset);
|
|
uint32_t kind = file.ReadDWord();
|
|
file.Seek(pos + page_offset);
|
|
|
|
switch (kind) {
|
|
case UNWIND_SECOND_LEVEL_REGULAR:
|
|
{
|
|
unwind_info_regular_second_level_page_header page_header;
|
|
file.Read(&page_header, sizeof(page_header));
|
|
|
|
file.Seek(pos + page_offset + page_header.entryPageOffset);
|
|
uint64_t cur_address = address + page_offset + page_header.entryPageOffset;
|
|
for (size_t j = 0; j < page_header.entryCount; j++) {
|
|
unwind_info_regular_second_level_entry entry;
|
|
file.Read(&entry, sizeof(entry));
|
|
|
|
info_list.push_back(Info(cur_address, entry.functionOffset, entry.encoding));
|
|
cur_address += sizeof(entry);
|
|
}
|
|
}
|
|
break;
|
|
case UNWIND_SECOND_LEVEL_COMPRESSED:
|
|
{
|
|
unwind_info_compressed_second_level_page_header page_header;
|
|
file.Read(&page_header, sizeof(page_header));
|
|
|
|
file.Seek(pos + page_offset + page_header.encodingsPageOffset);
|
|
std::vector<uint32_t> encoding_list;
|
|
for (size_t j = 0; j < page_header.encodingsCount; j++) {
|
|
uint32_t entry = file.ReadDWord();
|
|
encoding_list.push_back(entry);
|
|
}
|
|
|
|
file.Seek(pos + page_offset + page_header.entryPageOffset);
|
|
uint64_t cur_address = address + page_offset + page_header.entryPageOffset;
|
|
for (size_t j = 0; j < page_header.entryCount; j++) {
|
|
uint32_t entry = file.ReadDWord();
|
|
uint32_t function_offset = index_entry_list[i].functionOffset + UNWIND_INFO_COMPRESSED_ENTRY_FUNC_OFFSET(entry);
|
|
uint32_t encoding_index = UNWIND_INFO_COMPRESSED_ENTRY_ENCODING_INDEX(entry);
|
|
uint32_t encoding = (encoding_index < common_encoding_list.size()) ? common_encoding_list[encoding_index] : encoding_list[encoding_index - common_encoding_list.size()];
|
|
|
|
info_list.push_back(Info(cur_address, function_offset, encoding));
|
|
cur_address += sizeof(entry);
|
|
}
|
|
}
|
|
break;
|
|
default:
|
|
throw std::runtime_error("Invalid page header");
|
|
}
|
|
}
|
|
info_list.push_back(Info(0, index_entry_list[index_entry_list.size() - 1].functionOffset - 1, 0));
|
|
|
|
std::map<uint64_t, CommonInformationEntry *> cie_map;
|
|
std::vector<uint8_t> dummy;
|
|
for (size_t i = 0; i < info_list.size() - 1; i++) {
|
|
uint32_t function_offset = info_list[i].offset;
|
|
uint32_t encoding = info_list[i].encoding;
|
|
uint64_t lsda_address = 0;
|
|
if (encoding & UNWIND_HAS_LSDA) {
|
|
for (size_t k = 0; k < lsda_entry_list.size(); k++) {
|
|
if (lsda_entry_list[k].functionOffset == function_offset) {
|
|
lsda_address = base_address + lsda_entry_list[k].lsdaOffset;
|
|
break;
|
|
}
|
|
}
|
|
if (!lsda_address)
|
|
throw std::runtime_error("Invalid lsda index");
|
|
}
|
|
|
|
uint64_t personality_routine = 0;
|
|
if (encoding & UNWIND_PERSONALITY_MASK) {
|
|
uint32_t personality_index = (encoding & UNWIND_PERSONALITY_MASK) >> 28;
|
|
if (personality_index > personality_list.size())
|
|
throw std::runtime_error("Invalid personality index");
|
|
|
|
personality_routine = base_address + personality_list[personality_index - 1];
|
|
encoding &= ~UNWIND_PERSONALITY_MASK;
|
|
}
|
|
|
|
CommonInformationEntry *cie;
|
|
std::map<uint64_t, CommonInformationEntry *>::const_iterator it = cie_map.find(personality_routine);
|
|
if (it == cie_map.end()) {
|
|
cie = cie_list_->Add(0, std::string(), 0, 0, 0, 0, 0, 0, personality_routine, std::vector<uint8_t>());
|
|
cie_map[personality_routine] = cie;
|
|
} else {
|
|
cie = it->second;
|
|
}
|
|
|
|
Add(info_list[i].address, base_address + function_offset, base_address + info_list[i + 1].offset, lsda_address, cie, dummy, encoding);
|
|
}
|
|
}
|
|
|
|
void MacRuntimeFunctionList::ReadBorlandInfo(MacArchitecture &file, uint64_t address)
|
|
{
|
|
struct BORLAND_UNWIND_HEADER {
|
|
uint32_t magic;
|
|
uint32_t delta;
|
|
uint32_t count;
|
|
};
|
|
|
|
struct BORLAND_UNWIND_BLOCK {
|
|
uint32_t begin;
|
|
uint32_t end;
|
|
uint32_t count;
|
|
uint32_t offset;
|
|
};
|
|
|
|
struct BORLAND_UNWIND_INFO {
|
|
uint32_t begin;
|
|
uint32_t end;
|
|
};
|
|
|
|
if (!file.AddressSeek(address))
|
|
return;
|
|
|
|
BORLAND_UNWIND_HEADER unwind_header;
|
|
file.Read(&unwind_header, sizeof(unwind_header));
|
|
if (unwind_header.magic != 0xFBDE0007)
|
|
return;
|
|
|
|
address_ = address;
|
|
|
|
uint64_t base_address = address - unwind_header.delta;
|
|
std::vector<uint8_t> dummy;
|
|
for (size_t i = 0; i < unwind_header.count; i++) {
|
|
BORLAND_UNWIND_BLOCK unwind_block;
|
|
file.Read(&unwind_block, sizeof(unwind_block));
|
|
if (unwind_block.end > unwind_block.begin && unwind_block.count) {
|
|
address = address_ + unwind_block.offset;
|
|
uint64_t pos = file.Tell();
|
|
if (!file.AddressSeek(address))
|
|
throw std::runtime_error("Invalid runtime function address");
|
|
BORLAND_UNWIND_INFO unwind_info;
|
|
for (size_t j = 0; j < unwind_block.count; j++) {
|
|
file.Read(&unwind_info, sizeof(unwind_info));
|
|
Add(address, unwind_info.begin + base_address, unwind_info.end + base_address, unwind_info.end + base_address - 30, NULL, dummy, 0);
|
|
address += sizeof(unwind_info);
|
|
}
|
|
file.Seek(pos);
|
|
}
|
|
}
|
|
}
|
|
|
|
size_t MacRuntimeFunctionList::WriteToFile(MacArchitecture &file, bool compact_info)
|
|
{
|
|
Sort();
|
|
|
|
if (cie_list_->count())
|
|
return compact_info ? WriteCompactInfo(file) : WriteDwarfInfo(file);
|
|
return 0;
|
|
}
|
|
|
|
size_t MacRuntimeFunctionList::WriteDwarfInfo(MacArchitecture &file)
|
|
{
|
|
size_t res = 0;
|
|
uint64_t address = file.AddressTell();
|
|
std::map<CommonInformationEntry*, uint64_t> cie_map;
|
|
for (size_t i = 0; i < count(); i++) {
|
|
MacRuntimeFunction *func = item(i);
|
|
CommonInformationEntry *cie = func->cie();
|
|
if (cie->version() == 0)
|
|
continue;
|
|
|
|
std::map<CommonInformationEntry*, uint64_t>::iterator it = cie_map.find(cie);
|
|
uint64_t cie_address;
|
|
if (it == cie_map.end()) {
|
|
// write CIE
|
|
cie_address = address + res;
|
|
|
|
EncodedData data(cie_address + sizeof(uint32_t), file.cpu_address_size());
|
|
data.WriteDWord(0);
|
|
data.WriteByte(cie->version());
|
|
data.WriteString(cie->augmentation());
|
|
data.WriteUleb128(cie->code_alignment_factor());
|
|
data.WriteSleb128(cie->data_alignment_factor());
|
|
data.WriteByte(cie->return_address_register());
|
|
if (*cie->augmentation().c_str() == 'z') {
|
|
EncodedData tmp(data.address() + data.size() + 1, file.cpu_address_size());
|
|
for (size_t j = 1; j < cie->augmentation().size(); j++) {
|
|
switch (cie->augmentation().at(j)) {
|
|
case 'L':
|
|
tmp.WriteByte(cie->lsda_encoding());
|
|
break;
|
|
case 'R':
|
|
tmp.WriteByte(cie->fde_encoding());
|
|
break;
|
|
case 'P':
|
|
{
|
|
tmp.WriteByte(cie->personality_encoding());
|
|
tmp.WriteEncoding(cie->personality_encoding(), cie->personality_routine());
|
|
}
|
|
break;
|
|
}
|
|
}
|
|
data.WriteByte(static_cast<uint8_t>(tmp.size()));
|
|
data.Write(tmp.data(), tmp.size());
|
|
}
|
|
data.Write(cie->initial_instructions().data(), cie->initial_instructions().size());
|
|
data.resize(AlignValue(data.size(), sizeof(uint32_t)), 0);
|
|
|
|
uint32_t size = static_cast<uint32_t>(data.size());
|
|
res += file.Write(&size, sizeof(size));
|
|
res += file.Write(data.data(), data.size());
|
|
cie_map[cie] = cie_address;
|
|
} else {
|
|
cie_address = it->second;
|
|
}
|
|
|
|
// write FDE
|
|
EncodedData data(address + res + sizeof(uint32_t), file.cpu_address_size());
|
|
data.WriteDWord(static_cast<uint32_t>(data.address() - cie_address));
|
|
data.WriteEncoding(cie->fde_encoding(), func->begin());
|
|
data.WriteEncoding(cie->fde_encoding() & 0x0f, func->end() - func->begin());
|
|
if (*cie->augmentation().c_str() == 'z') {
|
|
EncodedData tmp(data.address() + data.size() + 1, file.cpu_address_size());
|
|
if (cie->augmentation().find('L') != std::string::npos) {
|
|
if (func->unwind_address())
|
|
tmp.WriteEncoding(cie->lsda_encoding(), func->unwind_address());
|
|
else
|
|
tmp.WriteEncoding(cie->lsda_encoding() & 0x0f, 0);
|
|
}
|
|
data.WriteByte(static_cast<uint8_t>(tmp.size()));
|
|
data.Write(tmp.data(), tmp.size());
|
|
}
|
|
data.Write(func->call_frame_instructions().data(), func->call_frame_instructions().size());
|
|
data.resize(AlignValue(data.size(), sizeof(uint32_t)), 0);
|
|
|
|
uint32_t size = static_cast<uint32_t>(data.size());
|
|
res += file.Write(&size, sizeof(size));
|
|
res += file.Write(data.data(), data.size());
|
|
}
|
|
|
|
return res;
|
|
}
|
|
|
|
size_t MacRuntimeFunctionList::WriteCompactInfo(MacArchitecture &file)
|
|
{
|
|
size_t i, j;
|
|
MacRuntimeFunction *func;
|
|
uint32_t encoding;
|
|
size_t res = 0;
|
|
|
|
std::vector<uint32_t> encoding_list;
|
|
std::map<uint64_t, uint32_t> personality_map;
|
|
std::map<uint32_t, uint8_t> common_encoding_map;
|
|
std::map<uint32_t, size_t> used_encoding_map;
|
|
std::map<uint32_t, uint32_t> lsda_map;
|
|
std::map<uint32_t, uint32_t> lsda_offset_map;
|
|
std::map<uint32_t, uint32_t> page_offset_map;
|
|
std::map<uint32_t, std::vector<uint8_t> > index_map;
|
|
|
|
uint64_t start_pos = file.Tell();
|
|
|
|
uint64_t base_address = file.segment_list()->GetBaseSegment()->address();
|
|
|
|
size_t max_used_count = 0;
|
|
|
|
for (i = 0; i < count(); i++) {
|
|
func = item(i);
|
|
encoding = func->compact_encoding();
|
|
CommonInformationEntry *cie = func->cie();
|
|
if (cie->personality_routine()) {
|
|
std::map<uint64_t, uint32_t>::const_iterator it = personality_map.find(cie->personality_routine());
|
|
if (it == personality_map.end()) {
|
|
uint32_t next = static_cast<uint32_t>(personality_map.size() + 1);
|
|
personality_map[cie->personality_routine()] = next;
|
|
}
|
|
uint32_t personality_index = personality_map[cie->personality_routine()];
|
|
encoding |= personality_index << __builtin_ctz(UNWIND_PERSONALITY_MASK);
|
|
}
|
|
encoding_list.push_back(encoding);
|
|
|
|
if ((func->compact_encoding() & UNWIND_X86_MODE_MASK) != UNWIND_X86_MODE_DWARF) {
|
|
used_encoding_map[encoding] += 1;
|
|
if (max_used_count < used_encoding_map[encoding])
|
|
max_used_count = used_encoding_map[encoding];
|
|
}
|
|
|
|
uint32_t function_offset = static_cast<uint32_t>(func->begin() - base_address);
|
|
lsda_offset_map[function_offset] = static_cast<uint32_t>(lsda_map.size() * sizeof(unwind_info_section_header_lsda_index_entry));
|
|
if (func->unwind_address())
|
|
lsda_map[function_offset] = static_cast<uint32_t>(func->unwind_address() - base_address);
|
|
}
|
|
|
|
for (size_t used_count = max_used_count; used_count > 1; used_count--) {
|
|
for (std::map<uint32_t, size_t>::const_iterator ue_it = used_encoding_map.begin(); ue_it != used_encoding_map.end(); ue_it++) {
|
|
if (ue_it->second == used_count) {
|
|
uint32_t next = static_cast<uint8_t>(common_encoding_map.size());
|
|
common_encoding_map[ue_it->first] = next;
|
|
if (common_encoding_map.size() == 127) {
|
|
used_count = 1;
|
|
break;
|
|
}
|
|
}
|
|
}
|
|
}
|
|
|
|
// calc pages
|
|
size_t page_size = 0x1000;
|
|
for (i = count(); i > 0; ) {
|
|
std::map<uint32_t, uint8_t> page_encoding_map;
|
|
uint16_t entry_count = 0;
|
|
uint16_t regular_entry_count = static_cast<uint16_t>(std::min((page_size - sizeof(unwind_info_regular_second_level_page_header)) / sizeof(unwind_info_regular_second_level_entry), i));
|
|
size_t max_size = page_size - sizeof(unwind_info_compressed_second_level_page_header);
|
|
|
|
uint64_t last_address = item(i - 1)->begin();
|
|
for (j = i; j > 0; j--) {
|
|
func = item(j - 1);
|
|
if (last_address - func->begin() > 0x00ffffff)
|
|
break;
|
|
|
|
encoding = encoding_list[j - 1];
|
|
if (common_encoding_map.find(encoding) == common_encoding_map.end() && page_encoding_map.find(encoding) == page_encoding_map.end()) {
|
|
uint32_t encoding_index = static_cast<uint32_t>(common_encoding_map.size() + page_encoding_map.size());
|
|
if (encoding_index > 0xff)
|
|
break;
|
|
|
|
page_encoding_map[encoding] = static_cast<uint8_t>(encoding_index);
|
|
}
|
|
|
|
if ((page_encoding_map.size() + entry_count) * sizeof(uint32_t) > max_size)
|
|
break;
|
|
|
|
entry_count++;
|
|
}
|
|
|
|
bool is_regular_page = (regular_entry_count > entry_count);
|
|
if (is_regular_page)
|
|
entry_count = regular_entry_count;
|
|
i -= entry_count;
|
|
|
|
uint64_t start_address = item(i)->begin();
|
|
std::vector<uint8_t> page_data;
|
|
if (is_regular_page) {
|
|
// build regular second level
|
|
unwind_info_regular_second_level_page_header header;
|
|
header.kind = UNWIND_SECOND_LEVEL_REGULAR;
|
|
header.entryPageOffset = sizeof(header);
|
|
header.entryCount = entry_count;
|
|
|
|
page_data.insert(page_data.end(), reinterpret_cast<const uint8_t *>(&header), reinterpret_cast<const uint8_t *>(&header) + sizeof(header));
|
|
for (j = 0; j < entry_count; j++) {
|
|
func = item(i + j);
|
|
encoding = encoding_list[i + j];
|
|
|
|
unwind_info_regular_second_level_entry entry;
|
|
entry.functionOffset = static_cast<uint32_t>(func->begin() - base_address);
|
|
entry.encoding = func->compact_encoding();
|
|
|
|
page_data.insert(page_data.end(), reinterpret_cast<const uint8_t *>(&entry), reinterpret_cast<const uint8_t *>(&entry) + sizeof(entry));
|
|
}
|
|
} else {
|
|
// build compressed second level
|
|
unwind_info_compressed_second_level_page_header header;
|
|
header.kind = UNWIND_SECOND_LEVEL_COMPRESSED;
|
|
header.entryPageOffset = sizeof(header);
|
|
header.entryCount = entry_count;
|
|
header.encodingsPageOffset = header.entryPageOffset + entry_count * sizeof(uint32_t);
|
|
header.encodingsCount = static_cast<uint16_t>(page_encoding_map.size());
|
|
|
|
page_data.insert(page_data.end(), reinterpret_cast<const uint8_t *>(&header), reinterpret_cast<const uint8_t *>(&header) + sizeof(header));
|
|
for (j = 0; j < entry_count; j++) {
|
|
func = item(i + j);
|
|
encoding = encoding_list[i + j];
|
|
|
|
uint8_t encoding_index;
|
|
std::map<uint32_t, uint8_t>::const_iterator it = common_encoding_map.find(encoding);
|
|
if (it != common_encoding_map.end()) {
|
|
encoding_index = it->second;
|
|
} else {
|
|
encoding_index = page_encoding_map[encoding];
|
|
}
|
|
|
|
uint32_t entry = (encoding_index << 24) | static_cast<uint32_t>((func->begin() - start_address));
|
|
page_data.insert(page_data.end(), reinterpret_cast<const uint8_t *>(&entry), reinterpret_cast<const uint8_t *>(&entry) + sizeof(entry));
|
|
}
|
|
|
|
if (!page_encoding_map.empty()) {
|
|
std::vector<uint32_t> page_encoding_list;
|
|
page_encoding_list.resize(page_encoding_map.size());
|
|
for (std::map<uint32_t, uint8_t>::const_iterator it = page_encoding_map.begin(); it != page_encoding_map.end(); it++) {
|
|
page_encoding_list[it->second - common_encoding_map.size()] = it->first;
|
|
}
|
|
|
|
for (j = 0; j < page_encoding_list.size(); j++) {
|
|
encoding = page_encoding_list[j];
|
|
page_data.insert(page_data.end(), reinterpret_cast<const uint8_t *>(&encoding), reinterpret_cast<const uint8_t *>(&encoding) + sizeof(encoding));
|
|
}
|
|
}
|
|
}
|
|
|
|
index_map[static_cast<uint32_t>(start_address - base_address)] = page_data;
|
|
}
|
|
|
|
// write section header
|
|
unwind_info_section_header header;
|
|
header.version = UNWIND_SECTION_VERSION;
|
|
header.commonEncodingsArraySectionOffset = sizeof(header);
|
|
header.commonEncodingsArrayCount = static_cast<uint32_t>(common_encoding_map.size());
|
|
header.personalityArraySectionOffset = header.commonEncodingsArraySectionOffset + header.commonEncodingsArrayCount * sizeof(uint32_t);
|
|
header.personalityArrayCount = static_cast<uint32_t>(personality_map.size());
|
|
header.indexSectionOffset = header.personalityArraySectionOffset + header.personalityArrayCount * sizeof(uint32_t);
|
|
header.indexCount = static_cast<uint32_t>(index_map.size() + 1);
|
|
res += file.Write(&header, sizeof(header));
|
|
|
|
// write common encoding
|
|
if (!common_encoding_map.empty()) {
|
|
std::vector<uint32_t> common_encoding_list;
|
|
common_encoding_list.resize(common_encoding_map.size());
|
|
for (std::map<uint32_t, uint8_t>::const_iterator it = common_encoding_map.begin(); it != common_encoding_map.end(); it++) {
|
|
common_encoding_list[it->second] = it->first;
|
|
}
|
|
res += file.Write(common_encoding_list.data(), common_encoding_list.size() * sizeof(common_encoding_list[0]));
|
|
}
|
|
|
|
// write personality
|
|
if (!personality_map.empty()) {
|
|
std::vector<uint32_t> personality_list;
|
|
personality_list.resize(personality_map.size());
|
|
for (std::map<uint64_t, uint32_t>::const_iterator it = personality_map.begin(); it != personality_map.end(); it++) {
|
|
personality_list[it->second - 1] = static_cast<uint32_t>(it->first - base_address);
|
|
}
|
|
res += file.Write(personality_list.data(), personality_list.size() * sizeof(personality_list[0]));
|
|
}
|
|
|
|
// write index
|
|
size_t lsda_start = res + header.indexCount * sizeof(unwind_info_section_header_index_entry);
|
|
size_t lsda_end = lsda_start + lsda_map.size() * sizeof(unwind_info_section_header_lsda_index_entry);
|
|
size_t pages_offset = lsda_end;
|
|
for (std::map<uint32_t, std::vector<uint8_t> >::const_iterator it = index_map.begin(); it != index_map.end(); it++) {
|
|
if (it != index_map.begin())
|
|
pages_offset = static_cast<uint32_t>(AlignValue(start_pos + pages_offset, page_size) - start_pos);
|
|
page_offset_map[it->first] = static_cast<uint32_t>(pages_offset);
|
|
|
|
unwind_info_section_header_index_entry entry;
|
|
entry.functionOffset = it->first;
|
|
entry.secondLevelPagesSectionOffset = static_cast<uint32_t>(pages_offset);
|
|
entry.lsdaIndexArraySectionOffset = static_cast<uint32_t>(lsda_start + lsda_offset_map[it->first]);
|
|
res += file.Write(&entry, sizeof(entry));
|
|
|
|
pages_offset += it->second.size();
|
|
}
|
|
{
|
|
unwind_info_section_header_index_entry entry;
|
|
entry.functionOffset = (count() > 0) ? static_cast<uint32_t>(item(count() - 1)->end() + 1 - base_address) : 0;
|
|
entry.secondLevelPagesSectionOffset = 0;
|
|
entry.lsdaIndexArraySectionOffset = static_cast<uint32_t>(lsda_end);
|
|
res += file.Write(&entry, sizeof(entry));
|
|
}
|
|
|
|
// write lsda index
|
|
if (!lsda_map.empty()) {
|
|
std::vector<unwind_info_section_header_lsda_index_entry> lsda_list;
|
|
for (std::map<uint32_t, uint32_t>::const_iterator it = lsda_map.begin(); it != lsda_map.end(); it++) {
|
|
unwind_info_section_header_lsda_index_entry entry;
|
|
entry.functionOffset = it->first;
|
|
entry.lsdaOffset = it->second;
|
|
lsda_list.push_back(entry);
|
|
}
|
|
res += file.Write(lsda_list.data(), lsda_list.size() * sizeof(lsda_list[0]));
|
|
}
|
|
|
|
// write second level data
|
|
for (std::map<uint32_t, std::vector<uint8_t> >::const_iterator it = index_map.begin(); it != index_map.end(); it++) {
|
|
uint64_t old_size = file.size();
|
|
res += static_cast<size_t>(file.Resize(start_pos + page_offset_map[it->first]) - old_size);
|
|
res += file.Write(it->second.data(), it->second.size());
|
|
}
|
|
|
|
return res;
|
|
}
|
|
|
|
void MacRuntimeFunctionList::Rebase(uint64_t delta_base)
|
|
{
|
|
cie_list_->Rebase(delta_base);
|
|
BaseRuntimeFunctionList::Rebase(delta_base);
|
|
}
|
|
|
|
/**
|
|
* MacArchitecture
|
|
*/
|
|
|
|
MacArchitecture::MacArchitecture(MacFile *owner, uint64_t offset, uint64_t size)
|
|
: BaseArchitecture(owner, offset, size), function_list_(NULL), virtual_machine_list_(NULL),
|
|
cpu_type_(0), cpu_subtype_(0), cpu_address_size_(osDWord), image_base_(0), entry_point_(0), file_type_(0),
|
|
cmds_size_(0), flags_(0), header_size_(0), segment_alignment_(0x1000), file_alignment_(0x1000),
|
|
linkedit_segment_(NULL), optimized_segment_count_(0), header_segment_(NULL), max_header_size_(0),
|
|
runtime_functions_section_(NULL), unwind_info_section_(0), sdk_(0)
|
|
{
|
|
memset(&symtab_, 0, sizeof(symtab_));
|
|
memset(&dysymtab_, 0, sizeof(dysymtab_));
|
|
memset(&dyld_info_, 0, sizeof(dyld_info_));
|
|
|
|
command_list_ = new MacLoadCommandList(this);
|
|
section_list_ = new MacSectionList(this);
|
|
segment_list_ = new MacSegmentList(this);
|
|
symbol_list_ = new MacSymbolList();
|
|
import_list_ = new MacImportList(this);
|
|
export_list_ = new MacExportList(this);
|
|
indirect_symbol_list_ = new MacIndirectSymbolList();
|
|
ext_ref_symbol_list_ = new MacExtRefSymbolList();
|
|
fixup_list_ = new MacFixupList();
|
|
runtime_function_list_ = new MacRuntimeFunctionList();
|
|
}
|
|
|
|
MacArchitecture::MacArchitecture(MacFile *owner, const MacArchitecture &src)
|
|
: BaseArchitecture(owner, src), function_list_(NULL), virtual_machine_list_(NULL),
|
|
linkedit_segment_(NULL), runtime_functions_section_(NULL), unwind_info_section_(NULL)
|
|
{
|
|
size_t i, j, k;
|
|
|
|
entry_point_ = src.entry_point_;
|
|
cpu_type_ = src.cpu_type_;
|
|
cpu_subtype_ = src.cpu_subtype_;
|
|
image_base_ = src.image_base_;
|
|
file_type_ = src.file_type_;
|
|
cmds_size_ = src.cmds_size_;
|
|
flags_ = src.flags_;
|
|
header_size_ = src.header_size_;
|
|
cpu_address_size_ = src.cpu_address_size_;
|
|
symtab_ = src.symtab_;
|
|
string_table_ = src.string_table_;
|
|
dysymtab_ = src.dysymtab_;
|
|
dyld_info_ = src.dyld_info_;
|
|
segment_alignment_ = src.segment_alignment_;
|
|
file_alignment_ = src.file_alignment_;
|
|
max_header_size_ = src.max_header_size_;
|
|
sdk_ = src.sdk_;
|
|
|
|
command_list_ = src.command_list_->Clone(this);
|
|
segment_list_ = src.segment_list_->Clone(this);
|
|
section_list_ = src.section_list_->Clone(this);
|
|
symbol_list_ = src.symbol_list_->Clone();
|
|
import_list_ = src.import_list_->Clone(this);
|
|
export_list_ = src.export_list_->Clone(this);
|
|
indirect_symbol_list_ = src.indirect_symbol_list_->Clone();
|
|
ext_ref_symbol_list_ = src.ext_ref_symbol_list_->Clone();
|
|
fixup_list_ = src.fixup_list_->Clone();
|
|
runtime_function_list_ = src.runtime_function_list_->Clone();
|
|
|
|
if (src.linkedit_segment_)
|
|
linkedit_segment_ = segment_list_->item(src.segment_list_->IndexOf(src.linkedit_segment_));
|
|
if (src.header_segment_)
|
|
header_segment_ = segment_list_->item(src.segment_list_->IndexOf(src.header_segment_));
|
|
if (src.runtime_functions_section_)
|
|
runtime_functions_section_ = section_list_->item(src.section_list_->IndexOf(src.runtime_functions_section_));
|
|
if (src.unwind_info_section_)
|
|
unwind_info_section_ = section_list_->item(src.section_list_->IndexOf(src.unwind_info_section_));
|
|
if (src.function_list_)
|
|
function_list_ = src.function_list_->Clone(this);
|
|
if (src.virtual_machine_list_)
|
|
virtual_machine_list_ = src.virtual_machine_list_->Clone();
|
|
|
|
for (i = 0; i < src.section_list()->count(); i++) {
|
|
MacSegment *segment = src.section_list()->item(i)->parent();
|
|
if (segment)
|
|
section_list_->item(i)->set_parent(segment_list_->item(src.segment_list_->IndexOf(segment)));
|
|
}
|
|
|
|
for (i = 0; i < src.indirect_symbol_list()->count(); i++) {
|
|
MacSymbol *symbol = src.indirect_symbol_list()->item(i)->symbol();
|
|
if (symbol)
|
|
indirect_symbol_list_->item(i)->set_symbol(symbol_list_->item(src.symbol_list_->IndexOf(symbol)));
|
|
}
|
|
|
|
for (i = 0; i < src.ext_ref_symbol_list_->count(); i++) {
|
|
MacSymbol *symbol = src.ext_ref_symbol_list_->item(i)->symbol();
|
|
if (symbol)
|
|
ext_ref_symbol_list_->item(i)->set_symbol(symbol_list_->item(src.symbol_list_->IndexOf(symbol)));
|
|
}
|
|
|
|
for (i = 0; i < src.import_list()->count(); i++) {
|
|
MacImport *import = src.import_list()->item(i);
|
|
for (j = 0; j < import->count(); j++) {
|
|
MacImportFunction *import_function = import->item(j);
|
|
MapFunction *map_function = import_function->map_function();
|
|
if (map_function)
|
|
import_list_->item(i)->item(j)->set_map_function(map_function_list()->item(src.map_function_list()->IndexOf(map_function)));
|
|
|
|
MacSymbol *symbol = import_function->symbol();
|
|
if (symbol)
|
|
import_list_->item(i)->item(j)->set_symbol(symbol_list_->item(src.symbol_list_->IndexOf(symbol)));
|
|
}
|
|
}
|
|
|
|
for (i = 0; i < src.fixup_list()->count(); i++) {
|
|
MacSymbol *symbol = src.fixup_list()->item(i)->symbol();
|
|
if (symbol)
|
|
fixup_list_->item(i)->set_symbol(symbol_list_->item(src.symbol_list_->IndexOf(symbol)));
|
|
}
|
|
|
|
for (i = 0; i < src.export_list()->count(); i++) {
|
|
MacSymbol *symbol = src.export_list()->item(i)->symbol();
|
|
if (symbol)
|
|
export_list_->item(i)->set_symbol(symbol_list_->item(src.symbol_list_->IndexOf(symbol)));
|
|
}
|
|
|
|
if (function_list_) {
|
|
for (i = 0; i < function_list_->count(); i++) {
|
|
IntelFunction *func = reinterpret_cast<IntelFunction *>(function_list_->item(i));
|
|
for (j = 0; j < func->count(); j++) {
|
|
IntelCommand *command = func->item(j);
|
|
|
|
for (k = 0; k < 3; k++) {
|
|
IntelOperand operand = command->operand(k);
|
|
if (operand.type == otNone)
|
|
break;
|
|
|
|
if (operand.fixup)
|
|
command->set_operand_fixup(k, fixup_list_->GetFixupByAddress(operand.fixup->address()));
|
|
}
|
|
}
|
|
for (j = 0; j < func->function_info_list()->count(); j++) {
|
|
FunctionInfo *info = func->function_info_list()->item(j);
|
|
if (info->source())
|
|
info->set_source(runtime_function_list_->GetFunctionByAddress(info->source()->begin()));
|
|
}
|
|
}
|
|
}
|
|
}
|
|
|
|
MacArchitecture::~MacArchitecture()
|
|
{
|
|
delete segment_list_;
|
|
delete section_list_;
|
|
delete symbol_list_;
|
|
delete import_list_;
|
|
delete indirect_symbol_list_;
|
|
delete ext_ref_symbol_list_;
|
|
delete command_list_;
|
|
delete export_list_;
|
|
delete fixup_list_;
|
|
delete runtime_function_list_;
|
|
delete function_list_;
|
|
delete virtual_machine_list_;
|
|
}
|
|
|
|
MacArchitecture *MacArchitecture::Clone(IFile *file) const
|
|
{
|
|
MacArchitecture *arch = new MacArchitecture(dynamic_cast<MacFile *>(file), *this);
|
|
return arch;
|
|
}
|
|
|
|
std::string MacArchitecture::name() const
|
|
{
|
|
static const struct {
|
|
const char *name;
|
|
cpu_type_t cputype;
|
|
cpu_subtype_t cpusubtype;
|
|
} arch_flags[] = {
|
|
{ "any", CPU_TYPE_ANY, CPU_SUBTYPE_MULTIPLE },
|
|
{ "little", CPU_TYPE_ANY, CPU_SUBTYPE_LITTLE_ENDIAN },
|
|
{ "big", CPU_TYPE_ANY, CPU_SUBTYPE_BIG_ENDIAN },
|
|
|
|
/* 64-bit Mach-O architectures */
|
|
|
|
/* architecture families */
|
|
{ "ppc64", CPU_TYPE_POWERPC64, CPU_SUBTYPE_POWERPC_ALL },
|
|
{ "x86_64", CPU_TYPE_X86_64, CPU_SUBTYPE_X86_64_ALL },
|
|
/* specific architecture implementations */
|
|
{ "ppc970-64", CPU_TYPE_POWERPC64, CPU_SUBTYPE_POWERPC_970 },
|
|
|
|
/* 32-bit Mach-O architectures */
|
|
|
|
/* architecture families */
|
|
{ "ppc", CPU_TYPE_POWERPC, CPU_SUBTYPE_POWERPC_ALL },
|
|
{ "i386", CPU_TYPE_I386, CPU_SUBTYPE_I386_ALL },
|
|
{ "m68k", CPU_TYPE_MC680x0, CPU_SUBTYPE_MC680x0_ALL },
|
|
{ "hppa", CPU_TYPE_HPPA, CPU_SUBTYPE_HPPA_ALL },
|
|
{ "sparc", CPU_TYPE_SPARC, CPU_SUBTYPE_SPARC_ALL },
|
|
{ "m88k", CPU_TYPE_MC88000, CPU_SUBTYPE_MC88000_ALL },
|
|
{ "i860", CPU_TYPE_I860, CPU_SUBTYPE_I860_ALL },
|
|
{ "arm", CPU_TYPE_ARM, CPU_SUBTYPE_ARM_ALL },
|
|
/* specific architecture implementations */
|
|
{ "ppc601", CPU_TYPE_POWERPC, CPU_SUBTYPE_POWERPC_601 },
|
|
{ "ppc603", CPU_TYPE_POWERPC, CPU_SUBTYPE_POWERPC_603 },
|
|
{ "ppc603e",CPU_TYPE_POWERPC, CPU_SUBTYPE_POWERPC_603e },
|
|
{ "ppc603ev",CPU_TYPE_POWERPC,CPU_SUBTYPE_POWERPC_603ev },
|
|
{ "ppc604", CPU_TYPE_POWERPC, CPU_SUBTYPE_POWERPC_604 },
|
|
{ "ppc604e",CPU_TYPE_POWERPC, CPU_SUBTYPE_POWERPC_604e },
|
|
{ "ppc750", CPU_TYPE_POWERPC, CPU_SUBTYPE_POWERPC_750 },
|
|
{ "ppc7400",CPU_TYPE_POWERPC, CPU_SUBTYPE_POWERPC_7400 },
|
|
{ "ppc7450",CPU_TYPE_POWERPC, CPU_SUBTYPE_POWERPC_7450 },
|
|
{ "ppc970", CPU_TYPE_POWERPC, CPU_SUBTYPE_POWERPC_970 },
|
|
{ "i486", CPU_TYPE_I386, CPU_SUBTYPE_486 },
|
|
{ "i486SX", CPU_TYPE_I386, CPU_SUBTYPE_486SX },
|
|
{ "pentium",CPU_TYPE_I386, CPU_SUBTYPE_PENT }, /* same as i586 */
|
|
{ "i586", CPU_TYPE_I386, CPU_SUBTYPE_586 },
|
|
{ "pentpro", CPU_TYPE_I386, CPU_SUBTYPE_PENTPRO }, /* same as i686 */
|
|
{ "i686", CPU_TYPE_I386, CPU_SUBTYPE_PENTPRO },
|
|
{ "pentIIm3",CPU_TYPE_I386, CPU_SUBTYPE_PENTII_M3 },
|
|
{ "pentIIm5",CPU_TYPE_I386, CPU_SUBTYPE_PENTII_M5 },
|
|
{ "pentium4",CPU_TYPE_I386, CPU_SUBTYPE_PENTIUM_4 },
|
|
{ "m68030", CPU_TYPE_MC680x0, CPU_SUBTYPE_MC68030_ONLY },
|
|
{ "m68040", CPU_TYPE_MC680x0, CPU_SUBTYPE_MC68040 },
|
|
{ "hppa7100LC", CPU_TYPE_HPPA, CPU_SUBTYPE_HPPA_7100LC },
|
|
{ "armv4t", CPU_TYPE_ARM, CPU_SUBTYPE_ARM_V4T},
|
|
{ "armv5", CPU_TYPE_ARM, CPU_SUBTYPE_ARM_V5TEJ},
|
|
{ "xscale", CPU_TYPE_ARM, CPU_SUBTYPE_ARM_XSCALE},
|
|
{ "armv6", CPU_TYPE_ARM, CPU_SUBTYPE_ARM_V6 },
|
|
};
|
|
|
|
for(size_t i = 0; i < _countof(arch_flags); i++) {
|
|
if(arch_flags[i].cputype == cpu_type_ &&
|
|
(arch_flags[i].cpusubtype & ~CPU_SUBTYPE_MASK) ==
|
|
(cpu_subtype_ & ~CPU_SUBTYPE_MASK))
|
|
return std::string(arch_flags[i].name);
|
|
}
|
|
|
|
return string_format("unknown 0x%X", cpu_type_);
|
|
}
|
|
|
|
uint64_t MacArchitecture::entry_point() const
|
|
{
|
|
return entry_point_;
|
|
}
|
|
|
|
OpenStatus MacArchitecture::ReadFromFile(uint32_t mode)
|
|
{
|
|
size_t i;
|
|
mach_header mh;
|
|
|
|
Seek(0);
|
|
|
|
if (size() < sizeof(mh))
|
|
return osUnknownFormat;
|
|
|
|
Read(&mh, sizeof(mh));
|
|
if (mh.magic != MH_MAGIC && mh.magic != MH_MAGIC_64)
|
|
return osUnknownFormat;
|
|
|
|
if (mh.magic == MH_MAGIC_64)
|
|
ReadDWord();
|
|
|
|
cpu_address_size_ = (mh.magic == MH_MAGIC) ? osDWord : osQWord;
|
|
cpu_type_ = mh.cputype;
|
|
cpu_subtype_ = mh.cpusubtype;
|
|
switch (cpu_type_) {
|
|
case CPU_TYPE_I386:
|
|
case CPU_TYPE_X86_64:
|
|
// supported cpu
|
|
break;
|
|
default:
|
|
return osUnsupportedCPU;
|
|
}
|
|
|
|
file_type_ = mh.filetype;
|
|
switch (file_type_) {
|
|
case MH_EXECUTE:
|
|
case MH_DYLIB:
|
|
case MH_BUNDLE:
|
|
// supported types
|
|
break;
|
|
default:
|
|
return osUnsupportedSubsystem;
|
|
}
|
|
|
|
cmds_size_ = mh.sizeofcmds;
|
|
flags_ = mh.flags;
|
|
header_size_ = static_cast<uint32_t>(Tell()) + cmds_size_;
|
|
|
|
command_list_->ReadFromFile(*this, mh.ncmds);
|
|
segment_list_->ReadFromFile(*this);
|
|
|
|
image_base_ = 0;
|
|
for (i = 0; i < segment_list_->count(); i++) {
|
|
uint64_t segment_base = segment_list_->item(i)->address() & 0xffffffff00000000ull;
|
|
if (!image_base_) {
|
|
image_base_ = segment_base;
|
|
} else if (image_base_ != segment_base) {
|
|
return osInvalidFormat;
|
|
}
|
|
}
|
|
|
|
for (i = 0; i < command_list_->count(); i++) {
|
|
MacLoadCommand *lc = command_list_->item(i);
|
|
switch (lc->type()) {
|
|
case LC_SYMTAB:
|
|
if (lc->size() != sizeof(symtab_))
|
|
throw std::runtime_error("Invalid symtab_command size");
|
|
|
|
Seek(lc->address());
|
|
Read(&symtab_, sizeof(symtab_));
|
|
break;
|
|
case LC_DYLD_INFO:
|
|
case LC_DYLD_INFO_ONLY:
|
|
if (lc->size() != sizeof(dyld_info_command))
|
|
throw std::runtime_error("Invalid dyld_info_command size");
|
|
|
|
Seek(lc->address());
|
|
Read(&dyld_info_, sizeof(dyld_info_));
|
|
break;
|
|
case LC_DYSYMTAB:
|
|
if (lc->size() != sizeof(dysymtab_))
|
|
throw std::runtime_error("Invalid dysymtab_command size");
|
|
|
|
Seek(lc->address());
|
|
Read(&dysymtab_, sizeof(dysymtab_));
|
|
break;
|
|
}
|
|
}
|
|
|
|
// parse entry point command
|
|
ILoadCommand *lc = command_list_->GetCommandByType(LC_MAIN);
|
|
if (!lc)
|
|
lc = command_list_->GetCommandByType(LC_UNIXTHREAD);
|
|
if (lc) {
|
|
switch (lc->type()) {
|
|
case LC_UNIXTHREAD:
|
|
{
|
|
thread_command command;
|
|
Seek(lc->address());
|
|
Read(&command, sizeof(command));
|
|
x86_state_hdr_t state_hdr;
|
|
Read(&state_hdr, sizeof(state_hdr));
|
|
if (cpu_type_ == CPU_TYPE_I386 && state_hdr.flavor == x86_THREAD_STATE32) {
|
|
x86_thread_state32_t thread_state;
|
|
Read(&thread_state, sizeof(thread_state));
|
|
entry_point_ = thread_state.__eip;
|
|
} else if (cpu_type_ == CPU_TYPE_X86_64 && state_hdr.flavor == x86_THREAD_STATE64) {
|
|
x86_thread_state64_t thread_state;
|
|
Read(&thread_state, sizeof(thread_state));
|
|
entry_point_ = thread_state.__rip;
|
|
}
|
|
}
|
|
break;
|
|
case LC_MAIN:
|
|
{
|
|
MacSegment *base_segment = segment_list_->GetBaseSegment();
|
|
if (!base_segment)
|
|
throw std::runtime_error("Format error");
|
|
|
|
entry_point_command command;
|
|
Seek(lc->address());
|
|
Read(&command, sizeof(command));
|
|
|
|
entry_point_ = command.entryoff + base_segment->address();
|
|
}
|
|
break;
|
|
}
|
|
}
|
|
|
|
// parse OSX SDK
|
|
for (i = 0; i < command_list_->count(); i++) {
|
|
lc = command_list_->item(i);
|
|
if (lc->type() == LC_VERSION_MIN_MACOSX) {
|
|
Seek(lc->address());
|
|
version_min_command command;
|
|
Read(&command, sizeof(command));
|
|
sdk_ = command.sdk;
|
|
break;
|
|
} else if (lc->type() == LC_BUILD_VERSION) {
|
|
Seek(lc->address());
|
|
build_version_command command;
|
|
Read(&command, sizeof(command));
|
|
sdk_ = command.sdk;
|
|
break;
|
|
}
|
|
}
|
|
|
|
linkedit_segment_ = segment_list_->GetSectionByName(SEG_LINKEDIT);
|
|
for (i = 0; i < section_list_->count(); i++) {
|
|
MacSection *section = section_list_->item(i);
|
|
if (section->parent()->name() == SEG_TEXT) {
|
|
if (section->name() == SECT_EH_FRAME || section->name() == SECT_INIT_TEXT) {
|
|
if (!runtime_functions_section_)
|
|
runtime_functions_section_ = section;
|
|
} else if (section->name() == SECT_UNWIND_INFO) {
|
|
if (!unwind_info_section_)
|
|
unwind_info_section_ = section;
|
|
}
|
|
}
|
|
}
|
|
|
|
string_table_.ReadFromFile(*this);
|
|
symbol_list_->ReadFromFile(*this, symtab_.nsyms);
|
|
ext_ref_symbol_list_->ReadFromFile(*this);
|
|
indirect_symbol_list_->ReadFromFile(*this);
|
|
fixup_list_->ReadFromFile(*this);
|
|
import_list_->ReadFromFile(*this);
|
|
export_list_->ReadFromFile(*this);
|
|
runtime_function_list_->ReadFromFile(*this);
|
|
|
|
header_segment_ = NULL;
|
|
max_header_size_ = header_size_;
|
|
for (i = 0; i < segment_list_->count(); i++) {
|
|
MacSegment *segment = segment_list_->item(i);
|
|
if (segment->physical_size() && segment->physical_offset() == 0) {
|
|
header_segment_ = segment;
|
|
break;
|
|
}
|
|
}
|
|
if (header_segment_) {
|
|
for (i = 0; i < section_list_->count(); i++) {
|
|
MacSection *section = section_list_->item(i);
|
|
if (section->parent() == header_segment_) {
|
|
max_header_size_ = section->physical_offset();
|
|
break;
|
|
}
|
|
}
|
|
}
|
|
|
|
if ((mode & foHeaderOnly) == 0) {
|
|
if (!owner()->file_name().empty()) {
|
|
MapFile map_file;
|
|
std::vector<uint64_t> segments;
|
|
for (size_t i = 0; i < segment_list()->count(); i++) {
|
|
segments.push_back(segment_list()->item(i)->address());
|
|
}
|
|
if (std::find(segments.begin(), segments.end(), 0) == segments.end())
|
|
segments.insert(segments.begin(), 0);
|
|
if (map_file.Parse(map_file_name().c_str(), segments))
|
|
ReadMapFile(map_file);
|
|
}
|
|
|
|
map_function_list()->ReadFromFile(*this);
|
|
|
|
for (i = 0; i < indirect_symbol_list_->count(); i++) {
|
|
MacIndirectSymbol *indirect_symbol = indirect_symbol_list_->item(i);
|
|
MacSymbol *symbol = indirect_symbol->symbol();
|
|
if (!symbol || (symbol->type() & (N_STAB | N_TYPE)) != N_UNDF)
|
|
continue;
|
|
|
|
MacSection *section = section_list_->GetSectionByAddress(indirect_symbol->address());
|
|
if (section->type() != S_SYMBOL_STUBS)
|
|
continue;
|
|
|
|
MapFunction *map_function = map_function_list()->GetFunctionByAddress(indirect_symbol->address());
|
|
if (!map_function)
|
|
map_function_list()->Add(indirect_symbol->address(), 0, segment_list_->GetMemoryTypeByAddress(indirect_symbol->address()) & mtExecutable ? otCode : otData, DemangleName(symbol->name()));
|
|
}
|
|
|
|
for (i = 0; i < symbol_list_->count() - 1; i++) {
|
|
MacSymbol *symbol = symbol_list_->item(i);
|
|
if ((symbol->type() & (N_STAB | N_TYPE)) != N_SECT || symbol->name().empty())
|
|
continue;
|
|
|
|
uint32_t memory_type = segment_list_->GetMemoryTypeByAddress(symbol->value());
|
|
if (memory_type == mtNone)
|
|
continue;
|
|
|
|
MapFunction *map_function = map_function_list()->GetFunctionByAddress(symbol->value());
|
|
if (!map_function)
|
|
map_function = map_function_list()->Add(symbol->value(), 0, otUnknown, DemangleName(symbol->name()));
|
|
|
|
ObjectType type = otData;
|
|
if (memory_type & mtExecutable) {
|
|
MacSection *section = section_list_->GetSectionByAddress(symbol->value());
|
|
if (!section || section->name() != "__const")
|
|
type = (symbol->type() & N_EXT) ? otExport : otCode;
|
|
}
|
|
map_function->set_type(type);
|
|
}
|
|
|
|
switch (cpu_type_) {
|
|
case CPU_TYPE_I386:
|
|
case CPU_TYPE_X86_64:
|
|
function_list_ = new MacIntelFunctionList(this);
|
|
virtual_machine_list_ = new IntelVirtualMachineList();
|
|
{
|
|
IntelFileHelper helper;
|
|
helper.Parse(*this);
|
|
}
|
|
break;
|
|
default:
|
|
return osUnsupportedCPU;
|
|
}
|
|
}
|
|
|
|
return osSuccess;
|
|
}
|
|
|
|
bool MacArchitecture::WriteToFile()
|
|
{
|
|
mach_header mh;
|
|
uint32_t pos;
|
|
|
|
// read header
|
|
Seek(0);
|
|
Read(&mh, sizeof(mh));
|
|
if (mh.magic == MH_MAGIC_64)
|
|
ReadDWord();
|
|
pos = static_cast<uint32_t>(Tell());
|
|
mh.ncmds = static_cast<uint32_t>(command_list_->count());
|
|
command_list_->WriteToFile(*this);
|
|
header_size_ = static_cast<uint32_t>(Tell());
|
|
if (header_size_ > max_header_size_)
|
|
throw std::runtime_error("Runtime error at WriteToFile");
|
|
mh.sizeofcmds = header_size_ - pos;
|
|
mh.flags = flags_;
|
|
// write header
|
|
Seek(0);
|
|
Write(&mh, sizeof(mh));
|
|
return true;
|
|
}
|
|
|
|
bool MacArchitecture::Compile(CompileOptions &options, IArchitecture *runtime)
|
|
{
|
|
if (owner()->count() > 1)
|
|
owner()->Resize(AlignValue(owner()->size(), 0x1000));
|
|
return BaseArchitecture::Compile(options, runtime);
|
|
}
|
|
|
|
void MacArchitecture::Save(CompileContext &ctx)
|
|
{
|
|
uint64_t address, pos, loader_crc_address, file_crc_address, patch_section_address, loader_crc_size_address, loader_crc_hash_address, file_crc_size_address;
|
|
size_t i, j, c;
|
|
uint8_t b;
|
|
uint32_t size, loader_crc_size, file_crc_size;
|
|
MacSegment *segment, *last_segment, *vmp_segment;
|
|
uint32_t linkedit_segment_flags;
|
|
std::string linkedit_segment_name;
|
|
MemoryManager *manager;
|
|
MemoryRegion *region;
|
|
int vmp_index;
|
|
|
|
// calc progress maximum
|
|
c = 0;
|
|
if (ctx.runtime)
|
|
c += ctx.runtime->segment_list()->count();
|
|
for (i = 0; i < function_list_->count(); i++) {
|
|
IFunction *func = function_list_->item(i);
|
|
for (j = 0; j < func->block_list()->count(); j++) {
|
|
CommandBlock *block = func->block_list()->item(j);
|
|
c += block->end_index() - block->start_index() + 1;
|
|
}
|
|
}
|
|
StartProgress(string_format("%s...", language[lsSaving].c_str()), c);
|
|
|
|
linkedit_segment_flags = VM_PROT_READ;
|
|
linkedit_segment_name = SEG_LINKEDIT;
|
|
|
|
// need erase optimized segments
|
|
for (i = segment_list_->count(); i > optimized_segment_count_; i--) {
|
|
segment = segment_list_->item(i - 1);
|
|
if (linkedit_segment_ == segment) {
|
|
linkedit_segment_flags = segment->flags();
|
|
linkedit_segment_name = segment->name();
|
|
linkedit_segment_ = NULL;
|
|
}
|
|
for (j = section_list_->count(); j > 0; j--) {
|
|
MacSection *section = section_list_->item(j - 1);
|
|
if (section->parent() == segment)
|
|
delete section;
|
|
}
|
|
delete segment;
|
|
}
|
|
|
|
// need truncate optimized segments and overlay
|
|
for (i = segment_list_->count(); i > 0; i--) {
|
|
segment = segment_list_->item(i - 1);
|
|
if (segment->physical_size() > 0) {
|
|
Resize(segment->physical_offset() + segment->physical_size());
|
|
break;
|
|
}
|
|
}
|
|
|
|
last_segment = segment_list_->last();
|
|
address = AlignValue(last_segment->address() + last_segment->size(), segment_alignment_);
|
|
pos = Resize(AlignValue(this->size(), file_alignment_));
|
|
vmp_segment = segment_list_->Add(address, 0xffffffff, static_cast<uint32_t>(pos), 0xffffffff, VM_PROT_READ, "");
|
|
|
|
// merge runtime objects
|
|
MacArchitecture *runtime = reinterpret_cast<MacArchitecture*>(ctx.runtime);
|
|
if (runtime && runtime->segment_list()->count()) {
|
|
// merge segments
|
|
for (i = 0; i < runtime->segment_list()->count(); i++) {
|
|
segment = runtime->segment_list()->item(i);
|
|
if (segment->physical_size()) {
|
|
runtime->Seek(segment->physical_offset());
|
|
size = static_cast<uint32_t>(segment->physical_size());
|
|
uint8_t *buffer = new uint8_t[size];
|
|
runtime->Read(buffer, size);
|
|
Write(buffer, size);
|
|
delete [] buffer;
|
|
}
|
|
size = static_cast<uint32_t>(AlignValue(segment->size(), runtime->segment_alignment()) - segment->physical_size());
|
|
uint8_t b = 0;
|
|
for (j = 0; j < size; j++) {
|
|
Write(&b, sizeof(b));
|
|
}
|
|
vmp_segment->include_write_type(segment->memory_type() & (~mtWritable));
|
|
|
|
StepProgress();
|
|
}
|
|
// merge fixups
|
|
for (i = 0; i < runtime->fixup_list()->count(); i++) {
|
|
MacFixup *src_fixup = runtime->fixup_list()->item(i);
|
|
MacFixup *fixup = src_fixup->Clone(fixup_list_);
|
|
if (src_fixup->symbol()) {
|
|
MacSymbol *symbol = src_fixup->symbol()->Clone(symbol_list_);
|
|
symbol_list_->AddObject(symbol);
|
|
fixup->set_symbol(symbol);
|
|
}
|
|
fixup_list_->AddObject(fixup);
|
|
}
|
|
// merge import
|
|
int library_ordinal = import_list_->GetMaxLibraryOrdinal() + 1;
|
|
for (i = 0; i < runtime->import_list()->count(); i++) {
|
|
MacImport *src_import = runtime->import_list()->item(i);
|
|
if (src_import->is_sdk())
|
|
continue;
|
|
|
|
MacImport *import = import_list_->GetImportByName(src_import->name());
|
|
if (!import) {
|
|
import = new MacImport(import_list_, library_ordinal++, src_import->name(), src_import->current_version(), src_import->compatibility_version());
|
|
import_list_->AddObject(import);
|
|
}
|
|
|
|
for (j = 0; j < src_import->count(); j++) {
|
|
MacImportFunction *src_import_function = src_import->item(j);
|
|
MacImportFunction *import_function = src_import_function->Clone(import);
|
|
if (src_import_function->symbol()) {
|
|
MacSymbol *symbol = import_function->symbol()->Clone(symbol_list_);
|
|
symbol_list_->AddObject(symbol);
|
|
|
|
symbol->set_library_ordinal(import->library_ordinal());
|
|
import_function->set_symbol(symbol);
|
|
}
|
|
import->AddObject(import_function);
|
|
}
|
|
}
|
|
// merge runtime functions
|
|
if (runtime_functions_section_ && runtime_functions_section_->name() == SECT_EH_FRAME) {
|
|
// dwarf format
|
|
size_t old_count = runtime_function_list_->cie_list()->count();
|
|
for (i = 0; i < runtime->runtime_function_list()->cie_list()->count(); i++) {
|
|
runtime_function_list_->cie_list()->AddObject(runtime->runtime_function_list()->cie_list()->item(i)->Clone(runtime_function_list_->cie_list()));
|
|
}
|
|
for (i = 0; i < runtime->runtime_function_list()->count(); i++) {
|
|
MacRuntimeFunction *runtime_function = runtime->runtime_function_list()->item(i)->Clone(runtime_function_list_);
|
|
if (runtime_function->cie())
|
|
runtime_function->set_cie(runtime_function_list()->cie_list()->item(old_count + runtime->runtime_function_list()->cie_list()->IndexOf(runtime_function->cie())));
|
|
runtime_function_list_->AddObject(runtime_function);
|
|
}
|
|
} else {
|
|
// other formats are not supported
|
|
}
|
|
}
|
|
|
|
// write functions
|
|
for (i = 0; i < function_list_->count(); i++) {
|
|
function_list_->item(i)->WriteToFile(*this);
|
|
}
|
|
|
|
// erase not used memory regions
|
|
manager = memory_manager();
|
|
if (manager->count() > 1) {
|
|
// need skip last big region
|
|
for (i = 0; i < manager->count() - 1; i++) {
|
|
region = manager->item(i);
|
|
if (!AddressSeek(region->address()))
|
|
continue;
|
|
|
|
for (j = 0; j < region->size(); j++) {
|
|
b = (region->type() & mtReadable) ? rand() : 0xcc;
|
|
Write(&b, sizeof(b));
|
|
}
|
|
}
|
|
}
|
|
|
|
vmp_index = 0;
|
|
|
|
std::map<MacSection *, uint64_t> patch_info_map;
|
|
for (i = 0; i < 2; i++) {
|
|
MacSection *section = NULL;
|
|
switch (i) {
|
|
case 0:
|
|
if (runtime_functions_section_ && runtime_functions_section_->name() == SECT_EH_FRAME)
|
|
section = runtime_functions_section_;
|
|
break;
|
|
case 1:
|
|
section = unwind_info_section_;
|
|
break;
|
|
}
|
|
if (!section)
|
|
continue;
|
|
|
|
pos = Resize(AlignValue(this->size(), 0x10));
|
|
address = vmp_segment->address() + pos - vmp_segment->physical_offset();
|
|
size = static_cast<uint32_t>(runtime_function_list_->WriteToFile(*this, (section == unwind_info_section_)));
|
|
if (runtime) {
|
|
section->set_physical_size(0);
|
|
patch_info_map[section] = address + size;
|
|
if (cpu_address_size() == osDWord) {
|
|
WriteDWord(static_cast<uint32_t>(address));
|
|
WriteDWord(size);
|
|
} else {
|
|
WriteQWord(address);
|
|
WriteQWord(size);
|
|
}
|
|
WriteDWord(static_cast<uint32_t>(pos - vmp_segment->physical_offset()));
|
|
} else {
|
|
section->set_address(address);
|
|
section->set_physical_size(size);
|
|
}
|
|
vmp_segment->include_write_type(mtReadable);
|
|
}
|
|
|
|
if (vmp_segment->write_type() == mtNone) {
|
|
delete vmp_segment;
|
|
} else {
|
|
vmp_segment->set_name(string_format("%s%d", ctx.options.section_name.c_str(), vmp_index++));
|
|
|
|
size = static_cast<uint32_t>(this->size() - vmp_segment->physical_offset());
|
|
vmp_segment->set_size(AlignValue(size, segment_alignment_));
|
|
vmp_segment->set_physical_size(static_cast<uint32_t>(AlignValue(size, file_alignment_)));
|
|
vmp_segment->update_type(vmp_segment->write_type());
|
|
|
|
Resize(vmp_segment->physical_offset() + vmp_segment->physical_size());
|
|
}
|
|
|
|
if ((ctx.options.flags & cpPack) && ctx.options.script)
|
|
ctx.options.script->DoBeforePackFile();
|
|
|
|
// write memory CRC table
|
|
if (function_list_->crc_table()) {
|
|
IntelCRCTable *intel_crc = reinterpret_cast<IntelCRCTable *>(function_list_->crc_table());
|
|
CRCTable crc_table(function_list_->crc_cryptor(), intel_crc->table_size());
|
|
|
|
// add non writable segments
|
|
for (i = 0; i < segment_list_->count(); i++) {
|
|
segment = segment_list_->item(i);
|
|
if ((segment->memory_type() & (mtReadable | mtWritable)) != mtReadable || segment->excluded_from_memory_protection())
|
|
continue;
|
|
|
|
size = std::min(static_cast<uint32_t>(segment->size()), segment->physical_size());
|
|
if (size)
|
|
crc_table.Add(segment->address(), size);
|
|
}
|
|
|
|
// skip writable runtime's sections
|
|
if (runtime) {
|
|
for (i = 0; i < runtime->segment_list()->count(); i++) {
|
|
segment = runtime->segment_list()->item(i);
|
|
if (segment->memory_type() & mtWritable)
|
|
crc_table.Remove(segment->address(), static_cast<uint32_t>(segment->size()));
|
|
}
|
|
}
|
|
|
|
// skip header
|
|
if (header_segment_)
|
|
crc_table.Remove(header_segment_->address(), max_header_size_);
|
|
|
|
// skip IAT
|
|
size_t ptr_size = OperandSizeToValue(cpu_address_size());
|
|
size_t k = (runtime && runtime->segment_list()->count() > 0) ? 2 : 1;
|
|
for (size_t n = 0; n < k; n++) {
|
|
MacImportList *import_list = (n == 0) ? import_list_ : runtime->import_list();
|
|
for (i = 0; i < import_list->count(); i++) {
|
|
MacImport *import = import_list->item(i);
|
|
for (j = 0; j < import->count(); j++) {
|
|
MacImportFunction *import_function = import->item(j);
|
|
size = (uint32_t)ptr_size;
|
|
if (import_function->options() & ioIsRelative) {
|
|
MacSection *section = section_list_->GetSectionByAddress(import_function->address());
|
|
if (section && section->type() == S_SYMBOL_STUBS)
|
|
size = section->reserved2();
|
|
}
|
|
crc_table.Remove(import_function->address(), size);
|
|
}
|
|
}
|
|
}
|
|
|
|
// skip fixups
|
|
for (i = 0; i < fixup_list_->count(); i++) {
|
|
MacFixup *fixup = fixup_list_->item(i);
|
|
if (!fixup->is_deleted() || (ctx.options.flags & cpStripFixups) == 0 || fixup->symbol())
|
|
crc_table.Remove(fixup->address(), OperandSizeToValue(fixup->size()));
|
|
}
|
|
|
|
// skip loader_data
|
|
IntelFunction *loader_data = reinterpret_cast<IntelFunctionList *>(function_list_)->loader_data();
|
|
if (loader_data)
|
|
crc_table.Remove(loader_data->entry()->address(), loader_data->entry()->dump_size());
|
|
|
|
// skip memory CRC table
|
|
crc_table.Remove(intel_crc->table_entry()->address(), intel_crc->table_size());
|
|
crc_table.Remove(intel_crc->size_entry()->address(), sizeof(uint32_t));
|
|
crc_table.Remove(intel_crc->hash_entry()->address(), sizeof(uint32_t));
|
|
|
|
// write to file
|
|
AddressSeek(intel_crc->table_entry()->address());
|
|
uint32_t hash;
|
|
size = static_cast<uint32_t>(crc_table.WriteToFile(*this, false, &hash));
|
|
AddressSeek(intel_crc->size_entry()->address());
|
|
WriteDWord(size);
|
|
AddressSeek(intel_crc->hash_entry()->address());
|
|
WriteDWord(hash);
|
|
|
|
intel_crc->size_entry()->set_operand_value(0, size);
|
|
intel_crc->hash_entry()->set_operand_value(0, hash);
|
|
}
|
|
EndProgress();
|
|
|
|
if (ctx.options.flags & cpStripDebugInfo) {
|
|
// strip symbols
|
|
for (i = 0; i < symbol_list_->count(); i++) {
|
|
MacSymbol *symbol = symbol_list_->item(i);
|
|
if (symbol->is_deleted())
|
|
continue;
|
|
|
|
if (symbol->type() & N_EXT) {
|
|
// global symbol
|
|
if ((symbol->type() & N_TYPE) == N_ABS && symbol->value() != 0)
|
|
continue;
|
|
if ((symbol->type() & N_TYPE) == N_UNDF && symbol->value() == 0)
|
|
continue;
|
|
if ((symbol->type() & N_TYPE) == N_PBUD)
|
|
continue;
|
|
if ((symbol->type() & N_TYPE) == N_SECT)
|
|
continue;
|
|
if (symbol->desc() & REFERENCED_DYNAMICALLY)
|
|
continue;
|
|
} else {
|
|
// local symbol
|
|
if (symbol->type() & N_STAB) {
|
|
// debug symbol
|
|
} else if (symbol->type() & N_PEXT) {
|
|
// private extern symbol
|
|
MacExtRefSymbol *ext_ref_symbol = ext_ref_symbol_list_->GetSymbol(symbol);
|
|
if (ext_ref_symbol && (ext_ref_symbol->flags() == REFERENCE_FLAG_PRIVATE_UNDEFINED_NON_LAZY || ext_ref_symbol->flags() == REFERENCE_FLAG_PRIVATE_UNDEFINED_LAZY))
|
|
continue;
|
|
if (indirect_symbol_list_->GetSymbol(symbol))
|
|
continue;
|
|
}
|
|
}
|
|
|
|
symbol->set_deleted(true);
|
|
}
|
|
|
|
// strip load commands
|
|
for (i = command_list_->count(); i > 0; i--) {
|
|
MacLoadCommand *load_command = command_list_->item(i - 1);
|
|
switch (load_command->type()) {
|
|
case LC_FUNCTION_STARTS:
|
|
case LC_SOURCE_VERSION:
|
|
delete load_command;
|
|
break;
|
|
}
|
|
}
|
|
}
|
|
|
|
string_table_.clear();
|
|
import_list_->Pack();
|
|
ext_ref_symbol_list_->Pack();
|
|
indirect_symbol_list_->Pack();
|
|
command_list_->Pack();
|
|
export_list_->Pack();
|
|
symbol_list_->Pack();
|
|
|
|
file_crc_address = 0;
|
|
file_crc_size = 0;
|
|
file_crc_size_address = 0;
|
|
loader_crc_address = 0;
|
|
loader_crc_size = 0;
|
|
loader_crc_size_address = 0;
|
|
loader_crc_hash_address = 0;
|
|
patch_section_address = 0;
|
|
if (runtime) {
|
|
std::vector<IFunction *> processor_list = function_list_->processor_list();
|
|
IntelRuntimeCRCTable *runtime_crc_table = reinterpret_cast<IntelFunctionList *>(function_list_)->runtime_crc_table();
|
|
MacIntelLoader *loader = new MacIntelLoader(NULL, cpu_address_size());
|
|
std::vector<MacSegment *> loader_segment_list;
|
|
|
|
last_segment = segment_list_->last();
|
|
address = AlignValue(last_segment->address() + last_segment->size(), segment_alignment_);
|
|
|
|
manager->clear();
|
|
manager->Add(address, UINT32_MAX, mtReadable | mtExecutable | mtWritable | (runtime_function_list()->count() ? mtSolid : mtNone));
|
|
|
|
if (!loader->Prepare(ctx)) {
|
|
delete loader;
|
|
throw std::runtime_error("Runtime error at Save");
|
|
}
|
|
size_t write_count = loader->count() + 10000;
|
|
size_t processor_count = 0;
|
|
for (i = 0; i < processor_list.size(); i++) {
|
|
processor_count += processor_list[i]->count();
|
|
}
|
|
ctx.file->StartProgress(string_format("%s...", language[lsSavingStartupCode].c_str()), loader->count() + write_count + processor_count);
|
|
loader->Compile(ctx);
|
|
|
|
pos = Resize(AlignValue(this->size(), file_alignment_));
|
|
segment = segment_list_->Add(address, UINT32_MAX, static_cast<uint32_t>(pos), UINT32_MAX, VM_PROT_READ | VM_PROT_EXECUTE, string_format("%s%d", ctx.options.section_name.c_str(), vmp_index++));
|
|
loader_segment_list.push_back(segment);
|
|
|
|
// create import segment
|
|
if (loader->import_segment_address()) {
|
|
size = static_cast<uint32_t>(loader->import_segment_address() - segment->address());
|
|
segment->set_size(size);
|
|
segment->set_physical_size(size);
|
|
|
|
address = loader->import_segment_address();
|
|
pos = Resize(segment->physical_offset() + segment->physical_size());
|
|
segment = segment_list_->Add(address, UINT32_MAX, static_cast<uint32_t>(pos), UINT32_MAX, VM_PROT_READ | VM_PROT_EXECUTE | VM_PROT_WRITE, string_format("%s%d", ctx.options.section_name.c_str(), vmp_index++));
|
|
loader_segment_list.push_back(segment);
|
|
}
|
|
|
|
// create data segment
|
|
if (loader->data_segment_address()) {
|
|
size = static_cast<uint32_t>(loader->data_segment_address() - segment->address());
|
|
segment->set_size(size);
|
|
segment->set_physical_size(size);
|
|
|
|
address = loader->data_segment_address();
|
|
pos = Resize(segment->physical_offset() + segment->physical_size());
|
|
segment = segment_list_->Add(address, UINT32_MAX, static_cast<uint32_t>(pos), UINT32_MAX, VM_PROT_READ | VM_PROT_WRITE, string_format("%s%d", ctx.options.section_name.c_str(), vmp_index++));
|
|
loader_segment_list.push_back(segment);
|
|
}
|
|
|
|
c = loader->WriteToFile(*this);
|
|
for (i = 0; i < processor_list.size(); i++) {
|
|
processor_list[i]->WriteToFile(*this);
|
|
}
|
|
if (runtime_crc_table)
|
|
c += runtime_crc_table->WriteToFile(*this);
|
|
|
|
// correct progress position
|
|
write_count -= c;
|
|
if (write_count)
|
|
StepProgress(write_count);
|
|
|
|
size = static_cast<uint32_t>(this->size() - segment->physical_offset());
|
|
segment->set_size(AlignValue(size, segment_alignment_));
|
|
segment->set_physical_size(AlignValue(size, file_alignment_));
|
|
|
|
Resize(segment->physical_offset() + segment->physical_size());
|
|
|
|
if (loader->jmp_table_entry()) {
|
|
address = loader->jmp_table_entry()->address();
|
|
segment = segment_list_->GetSectionByAddress(address);
|
|
MacSection *section = section_list_->Add(segment, address, loader->jmp_table_size(), static_cast<uint32_t>(segment->physical_offset() + address - segment->address()), S_SYMBOL_STUBS | S_ATTR_SELF_MODIFYING_CODE, SECT_JUMP_TABLE, segment->name());
|
|
section->set_reserved1(static_cast<uint32_t>(loader->jmp_table_entry()->operand(1).value));
|
|
section->set_reserved2(static_cast<uint32_t>(loader->jmp_table_entry()->dump_size()));
|
|
}
|
|
|
|
if (loader->lazy_import_entry()) {
|
|
address = loader->lazy_import_entry()->address();
|
|
segment = segment_list_->GetSectionByAddress(address);
|
|
MacSection *section = section_list_->Add(segment, address, loader->lazy_import_size(), static_cast<uint32_t>(segment->physical_offset() + address - segment->address()), S_LAZY_SYMBOL_POINTERS, SECT_LAZY_SYMBOL_PTR, segment->name());
|
|
section->set_reserved1(static_cast<uint32_t>(loader->lazy_import_entry()->operand(1).value));
|
|
section->set_alignment(loader->lazy_import_entry()->alignment());
|
|
}
|
|
|
|
if (loader->import_entry()) {
|
|
address = loader->import_entry()->address();
|
|
segment = segment_list_->GetSectionByAddress(address);
|
|
MacSection *section = section_list_->Add(segment, address, loader->import_size(), static_cast<uint32_t>(segment->physical_offset() + address - segment->address()), S_NON_LAZY_SYMBOL_POINTERS, SECT_NON_LAZY_SYMBOL_PTR, segment->name());
|
|
section->set_reserved1(static_cast<uint32_t>(loader->import_entry()->operand(1).value));
|
|
section->set_alignment(loader->import_entry()->alignment());
|
|
|
|
std::map<MacSection *, MacIndirectSymbol *> indirect_symbol_map;
|
|
for (i = 0; i < section_list_->count(); i++) {
|
|
section = section_list_->item(i);
|
|
if (section->type() == S_SYMBOL_STUBS || section->type() == S_LAZY_SYMBOL_POINTERS || section->type() == S_NON_LAZY_SYMBOL_POINTERS || section->type() == S_THREAD_LOCAL_VARIABLE_POINTERS)
|
|
indirect_symbol_map[section] = indirect_symbol_list_->item(section->reserved1());
|
|
}
|
|
|
|
// delete S_NON_LAZY_SYMBOL_POINTERS and S_SYMBOL_STUBS sections
|
|
size_t ptr_size = OperandSizeToValue(cpu_address_size());
|
|
for (i = section_list_->count(); i > 0; i--) {
|
|
section = section_list_->item(i - 1);
|
|
if (std::find(loader_segment_list.begin(), loader_segment_list.end(), section->parent()) != loader_segment_list.end())
|
|
continue;
|
|
|
|
if (section->type() == S_NON_LAZY_SYMBOL_POINTERS || section->type() == S_SYMBOL_STUBS) {
|
|
indirect_symbol_map.erase(section);
|
|
size_t value_size = (section->type() == S_SYMBOL_STUBS) ? section->reserved2() : ptr_size;
|
|
size_t c = static_cast<uint32_t>(section->size()) / value_size;
|
|
for (j = 0; j < c; j++) {
|
|
indirect_symbol_list_->Delete(section->reserved1());
|
|
}
|
|
delete section;
|
|
}
|
|
}
|
|
|
|
// recalc indirect symbol indexes
|
|
for (std::map<MacSection *, MacIndirectSymbol *>::iterator it = indirect_symbol_map.begin(); it != indirect_symbol_map.end(); it++) {
|
|
section = it->first;
|
|
MacIndirectSymbol *indirect_symbol = it->second;
|
|
section->set_reserved1(static_cast<uint32_t>(indirect_symbol_list_->IndexOf(indirect_symbol)));
|
|
}
|
|
}
|
|
|
|
if (loader->init_entry()) {
|
|
address = loader->init_entry()->operand(0).value;
|
|
MacSegment *base_segment = segment_list()->GetBaseSegment();
|
|
if (base_segment && address - base_segment->address() < max_header_size_)
|
|
max_header_size_ = static_cast<uint32_t>(address - base_segment->address());
|
|
|
|
// delete S_MOD_INIT_FUNC_POINTERS sections
|
|
for (i = section_list_->count(); i > 0; i--) {
|
|
MacSection *section = section_list_->item(i - 1);
|
|
if (section->type() == S_MOD_INIT_FUNC_POINTERS)
|
|
delete section;
|
|
}
|
|
|
|
address = loader->init_entry()->address();
|
|
segment = segment_list_->GetSectionByAddress(address);
|
|
MacSection *section = section_list_->Add(segment, address, loader->init_size(), static_cast<uint32_t>(segment->physical_offset() + address - segment->address()), S_MOD_INIT_FUNC_POINTERS, SECT_MOD_INIT_FUNC, segment->name());
|
|
section->set_alignment(loader->init_entry()->alignment());
|
|
}
|
|
|
|
if (loader->term_entry()) {
|
|
// delete S_MOD_TERM_FUNC_POINTERS sections
|
|
for (i = section_list_->count(); i > 0; i--) {
|
|
MacSection *section = section_list_->item(i - 1);
|
|
if (section->type() == S_MOD_TERM_FUNC_POINTERS)
|
|
delete section;
|
|
}
|
|
|
|
address = loader->term_entry()->address();
|
|
segment = segment_list_->GetSectionByAddress(address);
|
|
MacSection *section = section_list_->Add(segment, address, loader->term_size(), static_cast<uint32_t>(segment->physical_offset() + address - segment->address()), S_MOD_TERM_FUNC_POINTERS, SECT_MOD_TERM_FUNC, segment->name());
|
|
section->set_alignment(loader->term_entry()->alignment());
|
|
}
|
|
|
|
if (loader->thread_variables_entry()) {
|
|
// delete S_THREAD_LOCAL_VARIABLES sections
|
|
for (i = section_list_->count(); i > 0; i--) {
|
|
MacSection *section = section_list_->item(i - 1);
|
|
if (section->type() == S_THREAD_LOCAL_VARIABLES)
|
|
delete section;
|
|
}
|
|
|
|
address = loader->thread_variables_entry()->address();
|
|
segment = segment_list_->GetSectionByAddress(address);
|
|
MacSection *section = section_list_->Add(segment, address, loader->thread_variables_size(), static_cast<uint32_t>(segment->physical_offset() + address - segment->address()), S_THREAD_LOCAL_VARIABLES, SECT_THREAD_LOCAL_VARIABLES, segment->name());
|
|
section->set_alignment(loader->thread_variables_entry()->alignment());
|
|
}
|
|
|
|
if (loader->thread_data_entry()) {
|
|
// delete S_THREAD_LOCAL_REGULAR and S_THREAD_LOCAL_ZEROFILL sections
|
|
for (i = section_list_->count(); i > 0; i--) {
|
|
MacSection *section = section_list_->item(i - 1);
|
|
if (section->type() == S_THREAD_LOCAL_REGULAR || section->type() == S_THREAD_LOCAL_ZEROFILL)
|
|
delete section;
|
|
}
|
|
|
|
address = loader->thread_data_entry()->address();
|
|
segment = segment_list_->GetSectionByAddress(address);
|
|
MacSection *section = section_list_->Add(segment, address, loader->thread_data_size(), static_cast<uint32_t>(segment->physical_offset() + address - segment->address()), S_THREAD_LOCAL_REGULAR, SECT_THREAD_LOCAL_REGULAR, segment->name());
|
|
section->set_alignment(loader->thread_data_entry()->alignment());
|
|
}
|
|
|
|
if (loader->file_entry()) {
|
|
address = loader->file_entry()->address();
|
|
MacSegment *base_segment = segment_list()->GetBaseSegment();
|
|
if (base_segment && address - base_segment->address() < max_header_size_)
|
|
max_header_size_ = static_cast<uint32_t>(address - base_segment->address());
|
|
entry_point_ = address;
|
|
}
|
|
|
|
if (loader->file_crc_entry()) {
|
|
file_crc_address = loader->file_crc_entry()->address();
|
|
file_crc_size = loader->file_crc_size();
|
|
file_crc_size_address = loader->file_crc_size_entry()->address();
|
|
}
|
|
|
|
if (loader->loader_crc_entry()) {
|
|
loader_crc_address = loader->loader_crc_entry()->address();
|
|
loader_crc_size = loader->loader_crc_size();
|
|
loader_crc_size_address = loader->loader_crc_size_entry()->address();
|
|
loader_crc_hash_address = loader->loader_crc_hash_entry()->address();
|
|
}
|
|
|
|
if (loader->patch_section_entry())
|
|
patch_section_address = loader->patch_section_entry()->address();
|
|
|
|
// codesign requires S_ZEROFILL for packed sections
|
|
std::vector<MacSegment *> packed_segment_list = loader->packed_segment_list();
|
|
for (i = 0; i < section_list()->count(); i++) {
|
|
MacSection *section = section_list()->item(i);
|
|
if (std::find(packed_segment_list.begin(), packed_segment_list.end(), section->parent()) == packed_segment_list.end())
|
|
continue;
|
|
|
|
section->set_physical_offset(0);
|
|
if (section->flags() != S_THREAD_LOCAL_ZEROFILL)
|
|
section->set_flags(S_ZEROFILL);
|
|
}
|
|
|
|
if (file_type_ == MH_DYLIB || sdk_ >= DYLD_MACOSX_VERSION_10_12) {
|
|
// exclude EXECUTABLE flag for packed segments
|
|
for (i = 0; i < packed_segment_list.size(); i++) {
|
|
segment = packed_segment_list[i];
|
|
if (segment == header_segment_) {
|
|
if (header_segment_->size() > header_segment_->physical_size()) {
|
|
header_segment_->set_physical_size(AlignValue(header_segment_->physical_size(), segment_alignment_));
|
|
address = header_segment_->address() + header_segment_->physical_size();
|
|
segment = new MacSegment(segment_list_, address, header_segment_->address() + header_segment_->size() - address,
|
|
header_segment_->physical_offset() + header_segment_->physical_size(), 0, header_segment_->flags() & ~VM_PROT_EXECUTE, ctx.options.section_name);
|
|
segment->include_maxprot(VM_PROT_EXECUTE | VM_PROT_WRITE);
|
|
segment_list_->InsertObject(segment_list_->IndexOf(header_segment_) + 1, segment);
|
|
header_segment_->set_size(header_segment_->physical_size());
|
|
}
|
|
}
|
|
else if (segment->flags() & VM_PROT_EXECUTE)
|
|
segment->set_flags(segment->flags() & ~VM_PROT_EXECUTE);
|
|
}
|
|
}
|
|
|
|
delete loader;
|
|
|
|
ctx.file->EndProgress();
|
|
}
|
|
|
|
// add LINK_EDIT segment
|
|
last_segment = segment_list_->last();
|
|
address = AlignValue(last_segment->address() + last_segment->size(), segment_alignment_);
|
|
pos = Resize(AlignValue(this->size(), file_alignment_));
|
|
linkedit_segment_ = segment_list_->Add(address, UINT32_MAX, static_cast<uint32_t>(pos), UINT32_MAX, linkedit_segment_flags, linkedit_segment_name);
|
|
|
|
if (ctx.options.flags & cpStripFixups) {
|
|
for (i = fixup_list_->count(); i > 0; i--) {
|
|
MacFixup *fixup = fixup_list_->item(i - 1);
|
|
if (!fixup->symbol())
|
|
delete fixup;
|
|
}
|
|
}
|
|
fixup_list_->Pack();
|
|
|
|
fixup_list_->WriteToFile(*this);
|
|
import_list_->WriteToFile(*this);
|
|
export_list_->WriteToFile(*this);
|
|
|
|
// write LINK_EDIT data
|
|
IArchitecture *source = const_cast<IArchitecture *>(this->source());
|
|
for (i = 0; i < command_list_->count(); i++) {
|
|
MacLoadCommand *load_command = command_list_->item(i);
|
|
switch (load_command->type()) {
|
|
case LC_FUNCTION_STARTS:
|
|
case LC_DATA_IN_CODE:
|
|
case LC_DYLIB_CODE_SIGN_DRS:
|
|
{
|
|
linkedit_data_command command;
|
|
source->Seek(load_command->address());
|
|
source->Read(&command, sizeof(command));
|
|
if (source->Seek(command.dataoff)) {
|
|
load_command->set_offset(static_cast<uint32_t>(Tell()));
|
|
CopyFrom(*source, command.datasize);
|
|
}
|
|
}
|
|
break;
|
|
}
|
|
}
|
|
|
|
symbol_list_->WriteToFile(*this);
|
|
ext_ref_symbol_list_->WriteToFile(*this);
|
|
indirect_symbol_list_->WriteToFile(*this);
|
|
segment_list_->WriteToFile(*this);
|
|
string_table_.WriteToFile(*this);
|
|
|
|
size = static_cast<uint32_t>(this->size() - linkedit_segment_->physical_offset());
|
|
linkedit_segment_->set_size(AlignValue(size, segment_alignment_));
|
|
linkedit_segment_->set_physical_size(size);
|
|
|
|
if (ctx.options.script)
|
|
ctx.options.script->DoAfterSaveFile();
|
|
|
|
// write header
|
|
if ((ctx.options.flags & cpStripFixups) && file_type_ == MH_EXECUTE)
|
|
flags_ &= ~MH_PIE;
|
|
WriteToFile();
|
|
|
|
std::map<uint64_t, size_t> patch_header_map;
|
|
if (patch_section_address && AddressSeek(patch_section_address)) {
|
|
for (std::map<MacSection *, uint64_t>::const_iterator it = patch_info_map.begin(); it != patch_info_map.end(); it++) {
|
|
MacSection *patch_section = it->first;
|
|
uint64_t patch_info_address = it->second;
|
|
MacLoadCommand *load_command = command_list_->GetCommandByObject(patch_section->parent());
|
|
if (load_command) {
|
|
size_t section_index = 0;
|
|
for (size_t i = 0; i < section_list_->count(); i++) {
|
|
if (section_list_->item(i) == patch_section)
|
|
break;
|
|
if (section_list_->item(i)->parent() == patch_section->parent())
|
|
section_index++;
|
|
}
|
|
|
|
// patch section.adr + section.size + section.offset
|
|
address = segment_list_->GetBaseSegment()->address() + load_command->address() + ((cpu_address_size() == osDWord) ? sizeof(segment_command) : sizeof(segment_command_64))
|
|
+ section_index * ((cpu_address_size() == osDWord) ? sizeof(section) : sizeof(section_64)) + offsetof(section, addr);
|
|
size = (cpu_address_size() == osDWord) ? sizeof(uint32_t) + sizeof(uint32_t) + sizeof(uint32_t) : sizeof(uint64_t) + sizeof(uint64_t) + sizeof(uint32_t);
|
|
WriteDWord(static_cast<uint32_t>(patch_info_address - image_base()));
|
|
WriteDWord(static_cast<uint32_t>(address - image_base()));
|
|
WriteDWord(size);
|
|
|
|
patch_header_map[address] = size;
|
|
}
|
|
patch_section_address += 3 * sizeof(uint32_t);
|
|
}
|
|
}
|
|
|
|
// write header and loader CRC table
|
|
if (loader_crc_address) {
|
|
CRCTable crc_table(function_list_->crc_cryptor(), loader_crc_size);
|
|
|
|
// add header
|
|
if (header_segment_) {
|
|
crc_table.Add(header_segment_->address(), max_header_size_);
|
|
// skip place for codesign's commands
|
|
crc_table.Remove(header_segment_->address() + header_size_, max_header_size_ - header_size_);
|
|
// skip size of commands list
|
|
crc_table.Remove(header_segment_->address() + offsetof(mach_header, ncmds), sizeof(uint32_t));
|
|
crc_table.Remove(header_segment_->address() + offsetof(mach_header, sizeofcmds), sizeof(uint32_t));
|
|
// skip sizes of LINKEDIT
|
|
MacLoadCommand *linkedit_command = command_list()->GetCommandByObject(linkedit_segment_);
|
|
if (linkedit_command) {
|
|
if (cpu_address_size() == osDWord) {
|
|
crc_table.Remove(header_segment_->address() + linkedit_command->address() + offsetof(segment_command, vmsize), sizeof(uint32_t));
|
|
crc_table.Remove(header_segment_->address() + linkedit_command->address() + offsetof(segment_command, filesize), sizeof(uint32_t));
|
|
} else {
|
|
crc_table.Remove(header_segment_->address() + linkedit_command->address() + offsetof(segment_command_64, vmsize), sizeof(uint64_t));
|
|
crc_table.Remove(header_segment_->address() + linkedit_command->address() + offsetof(segment_command_64, filesize), sizeof(uint64_t));
|
|
}
|
|
}
|
|
for (std::map<uint64_t, size_t>::const_iterator it = patch_header_map.begin(); it != patch_header_map.end(); it++) {
|
|
crc_table.Remove(it->first, it->second);
|
|
}
|
|
}
|
|
|
|
// add loader segments
|
|
j = segment_list_->IndexOf(segment_list_->GetSectionByAddress(loader_crc_address));
|
|
if (j != NOT_ID) {
|
|
c = (ctx.options.flags & cpLoaderCRC) ? j + 1 : segment_list_->count();
|
|
for (i = j; i < c; i++) {
|
|
segment = segment_list_->item(i);
|
|
if (segment->memory_type() & mtWritable)
|
|
continue;
|
|
|
|
size = std::min(static_cast<uint32_t>(segment->size()), segment->physical_size());
|
|
if (size)
|
|
crc_table.Add(segment->address(), size);
|
|
|
|
// skip import
|
|
for (j = 0; j < section_list_->count(); j++) {
|
|
MacSection *section = section_list_->item(j);
|
|
if (section->parent() == segment && (section->type() == S_NON_LAZY_SYMBOL_POINTERS || section->type() == S_SYMBOL_STUBS))
|
|
crc_table.Remove(section->address(), static_cast<size_t>(section->size()));
|
|
}
|
|
}
|
|
}
|
|
|
|
// skip fixups
|
|
for (i = 0; i < fixup_list_->count(); i++) {
|
|
MacFixup *fixup = fixup_list_->item(i);
|
|
crc_table.Remove(fixup->address(), OperandSizeToValue(fixup->size()));
|
|
}
|
|
// skip loader CRC table
|
|
crc_table.Remove(loader_crc_address, loader_crc_size);
|
|
crc_table.Remove(loader_crc_size_address, sizeof(uint32_t));
|
|
crc_table.Remove(loader_crc_hash_address, sizeof(uint32_t));
|
|
// skip file CRC table
|
|
if (file_crc_address)
|
|
crc_table.Remove(file_crc_address, file_crc_size);
|
|
if (file_crc_size_address)
|
|
crc_table.Remove(file_crc_size_address, sizeof(uint32_t));
|
|
|
|
// write to file
|
|
AddressSeek(loader_crc_address);
|
|
uint32_t hash;
|
|
size = static_cast<uint32_t>(crc_table.WriteToFile(*this, false, &hash));
|
|
AddressSeek(loader_crc_size_address);
|
|
WriteDWord(size);
|
|
AddressSeek(loader_crc_hash_address);
|
|
WriteDWord(hash);
|
|
}
|
|
|
|
// write file CRC table
|
|
if (file_crc_address) {
|
|
CRCTable crc_table(function_list_->crc_cryptor(), file_crc_size - sizeof(uint32_t));
|
|
|
|
// add file range
|
|
crc_table.Add(1, static_cast<size_t>(this->size()) - 1);
|
|
// skip place for codesign's commands
|
|
crc_table.Remove(header_size_, max_header_size_ - header_size_);
|
|
// skip size of commands list
|
|
crc_table.Remove(offsetof(mach_header, ncmds), sizeof(uint32_t));
|
|
crc_table.Remove(offsetof(mach_header, sizeofcmds), sizeof(uint32_t));
|
|
// skip sizes of LINKEDIT
|
|
MacLoadCommand *linkedit_command = command_list()->GetCommandByObject(linkedit_segment_);
|
|
if (linkedit_command) {
|
|
if (cpu_address_size() == osDWord) {
|
|
crc_table.Remove(linkedit_command->address() + offsetof(segment_command, vmsize), sizeof(uint32_t));
|
|
crc_table.Remove(linkedit_command->address() + offsetof(segment_command, filesize), sizeof(uint32_t));
|
|
} else {
|
|
crc_table.Remove(linkedit_command->address() + offsetof(segment_command_64, vmsize), sizeof(uint64_t));
|
|
crc_table.Remove(linkedit_command->address() + offsetof(segment_command_64, filesize), sizeof(uint64_t));
|
|
}
|
|
}
|
|
|
|
// skip file CRC table
|
|
if (AddressSeek(file_crc_address))
|
|
crc_table.Remove(Tell(), file_crc_size);
|
|
if (AddressSeek(file_crc_size_address))
|
|
crc_table.Remove(Tell(), sizeof(uint32_t));
|
|
|
|
// write to file
|
|
AddressSeek(file_crc_address);
|
|
size = static_cast<uint32_t>(this->size());
|
|
Write(&size, sizeof(size));
|
|
size = static_cast<uint32_t>(crc_table.WriteToFile(*this, true));
|
|
AddressSeek(file_crc_size_address);
|
|
WriteDWord(size);
|
|
}
|
|
|
|
EndProgress();
|
|
}
|
|
|
|
bool MacArchitecture::Prepare(CompileContext &ctx)
|
|
{
|
|
if ((ctx.options.flags & cpStripFixups) == 0 && file_type_ == MH_EXECUTE && (flags_ & MH_PIE) == 0)
|
|
ctx.options.flags |= cpStripFixups;
|
|
else if (ctx.runtime)
|
|
ctx.options.flags |= cpLoader;
|
|
|
|
if (ctx.options.flags & cpImportProtection)
|
|
ctx.options.flags &= ~cpImportProtection;
|
|
|
|
if (ctx.options.flags & cpResourceProtection)
|
|
ctx.options.flags &= ~cpResourceProtection;
|
|
|
|
if (ctx.runtime && ((runtime_functions_section_ && runtime_functions_section_->name() == SECT_EH_FRAME) || unwind_info_section_))
|
|
ctx.options.flags |= cpLoader;
|
|
|
|
if (!BaseArchitecture::Prepare(ctx))
|
|
return false;
|
|
|
|
size_t i;
|
|
MacSegment *segment;
|
|
std::vector<MacSegment *> optimized_segment_list;
|
|
|
|
// optimize segments
|
|
if (linkedit_segment_)
|
|
optimized_segment_list.push_back(linkedit_segment_);
|
|
|
|
optimized_segment_count_ = segment_list_->count();
|
|
for (i = segment_list_->count(); i > 0; i--) {
|
|
segment = segment_list_->item(i - 1);
|
|
|
|
std::vector<MacSegment *>::iterator it = std::find(optimized_segment_list.begin(), optimized_segment_list.end(), segment);
|
|
if (it != optimized_segment_list.end()) {
|
|
optimized_segment_list.erase(it);
|
|
optimized_segment_count_--;
|
|
} else {
|
|
break;
|
|
}
|
|
}
|
|
|
|
// calc new header size
|
|
size_t new_header_size = (cpu_address_size_ == osDWord) ? sizeof(mach_header) : sizeof(mach_header_64);
|
|
for (i = 0; i < command_list_->count(); i++) {
|
|
MacLoadCommand *load_command = command_list_->item(i);
|
|
switch (load_command->type()) {
|
|
case LC_SEGMENT:
|
|
case LC_SEGMENT_64:
|
|
case LC_LOAD_DYLIB:
|
|
case LC_LOAD_WEAK_DYLIB:
|
|
case LC_CODE_SIGNATURE:
|
|
continue;
|
|
case LC_FUNCTION_STARTS:
|
|
case LC_SOURCE_VERSION:
|
|
if (ctx.options.flags & cpStripDebugInfo)
|
|
continue;
|
|
break;
|
|
}
|
|
new_header_size += load_command->size();
|
|
}
|
|
size_t segment_size = (cpu_address_size_ == osDWord) ? sizeof(segment_command) : sizeof(segment_command_64);
|
|
size_t section_size = (cpu_address_size_ == osDWord) ? sizeof(section) : sizeof(section_64);
|
|
new_header_size += (optimized_segment_count_ + 2) * segment_size;
|
|
new_header_size += section_list_->count() * section_size;
|
|
for (i = 0; i < import_list_->count(); i++) {
|
|
MacImport *import = import_list_->item(i);
|
|
if (import->is_sdk())
|
|
continue;
|
|
new_header_size += AlignValue(sizeof(dylib_command) + import->name().size() + 1, OperandSizeToValue(cpu_address_size()));
|
|
}
|
|
|
|
if (ctx.runtime) {
|
|
for (i = 0; i < section_list_->count(); i++) {
|
|
MacSection *section = section_list_->item(i);
|
|
if (section->type() == S_NON_LAZY_SYMBOL_POINTERS || section->type() == S_SYMBOL_STUBS ||
|
|
section->type() == S_MOD_INIT_FUNC_POINTERS || section->type() == S_MOD_TERM_FUNC_POINTERS)
|
|
new_header_size -= section_size;
|
|
}
|
|
new_header_size += 2 * segment_size;
|
|
new_header_size += section_size * 4; // S_SYMBOL_STUBS / S_LAZY_SYMBOL_POINTERS + S_NON_LAZY_SYMBOL_POINTERS + S_MOD_INIT_FUNC_POINTERS + S_MOD_TERM_FUNC_POINTERS
|
|
new_header_size += 5; // jmp xxxx
|
|
if (entry_point_)
|
|
new_header_size += 5; // jmp xxxx
|
|
if ((ctx.options.flags & cpPack) && file_type_ == MH_DYLIB)
|
|
new_header_size += segment_size;
|
|
|
|
for (i = 0; i < ctx.runtime->import_list()->count(); i++) {
|
|
IImport *import = ctx.runtime->import_list()->item(i);
|
|
if (import->is_sdk() || import_list_->GetImportByName(import->name()))
|
|
continue;
|
|
new_header_size += AlignValue(sizeof(dylib_command) + import->name().size() + 1, OperandSizeToValue(cpu_address_size()));
|
|
}
|
|
}
|
|
|
|
for (i = 0; i < section_list_->count(); i++) {
|
|
MacSection *section = section_list_->item(i);
|
|
MacSegment *segment = section->parent();
|
|
|
|
if (segment->physical_size() && new_header_size > segment->physical_offset() && new_header_size > section->physical_offset()) {
|
|
Notify(mtError, NULL, language[lsCreateSegmentError]);
|
|
return false;
|
|
}
|
|
}
|
|
|
|
for (i = 0; i < optimized_segment_list.size(); i++) {
|
|
segment = optimized_segment_list[i];
|
|
ctx.manager->Add(segment->address(), static_cast<uint32_t>(segment->physical_size()), mtReadable | mtExecutable | mtWritable | mtNotPaged, NULL);
|
|
}
|
|
|
|
if (optimized_segment_count_ > 0) {
|
|
segment = segment_list_->item(optimized_segment_count_ - 1);
|
|
if (ctx.runtime) {
|
|
MacArchitecture *runtime = reinterpret_cast<MacArchitecture *>(ctx.runtime);
|
|
if (runtime->segment_list()->count()) {
|
|
MemoryManager runtime_manager(runtime);
|
|
if (runtime->segment_list()->last() == runtime->linkedit_segment_) {
|
|
delete runtime->linkedit_segment_;
|
|
runtime->linkedit_segment_ = NULL;
|
|
}
|
|
runtime->Rebase(AlignValue(segment->address() + segment->size(), segment_alignment()) - runtime->segment_list()->item(0)->address(), dyld_info_.lazy_bind_size);
|
|
if (runtime->header_segment_)
|
|
runtime_manager.Add(runtime->header_segment_->address(), runtime->max_header_size_);
|
|
if (runtime->unwind_info_section_)
|
|
runtime_manager.Add(runtime->unwind_info_section_->address(), static_cast<size_t>(runtime->unwind_info_section_->size()));
|
|
if (runtime->runtime_functions_section_)
|
|
runtime_manager.Add(runtime->runtime_functions_section_->address(), static_cast<size_t>(runtime->runtime_functions_section_->size()));
|
|
runtime_manager.Pack();
|
|
for (i = 0; i < runtime_manager.count(); i++) {
|
|
MemoryRegion *region = runtime_manager.item(i);
|
|
ctx.manager->Add(region->address(), region->size(), region->type());
|
|
}
|
|
segment = runtime->segment_list()->last();
|
|
} else {
|
|
runtime->Rebase(image_base() - runtime->image_base(), 0);
|
|
}
|
|
}
|
|
|
|
// add new segment
|
|
assert(segment);
|
|
ctx.manager->Add(AlignValue(segment->address() + segment->size(), segment_alignment()), UINT32_MAX, mtReadable | mtExecutable | mtWritable | (runtime_function_list()->count() ? mtSolid : mtNone));
|
|
}
|
|
|
|
if (unwind_info_section_)
|
|
ctx.manager->Add(unwind_info_section_->address(), static_cast<size_t>(unwind_info_section_->size()));
|
|
if (runtime_functions_section_ && runtime_functions_section_->name() == SECT_EH_FRAME)
|
|
ctx.manager->Add(runtime_functions_section_->address(), static_cast<size_t>(runtime_functions_section_->size()));
|
|
|
|
return true;
|
|
}
|
|
|
|
uint64_t MacArchitecture::GetRelocBase() const
|
|
{
|
|
if (cpu_address_size() == osQWord || (flags_ & MH_SPLIT_SEGS) != 0) {
|
|
for (size_t i = 0; i < segment_list_->count(); i++) {
|
|
MacSegment *segment = segment_list_->item(i);
|
|
if (segment->memory_type() & mtWritable)
|
|
return segment->address();
|
|
}
|
|
} else {
|
|
if (segment_list_->count())
|
|
return segment_list_->item(0)->address();
|
|
}
|
|
throw std::runtime_error("Runtime error at GetRelocBase");
|
|
}
|
|
|
|
void MacArchitecture::Rebase(uint64_t delta_base, size_t delta_bind_info)
|
|
{
|
|
BaseArchitecture::Rebase(delta_base);
|
|
|
|
fixup_list_->Rebase(*this, delta_base);
|
|
export_list_->Rebase(delta_base);
|
|
indirect_symbol_list_->Rebase(delta_base);
|
|
command_list_->Rebase(delta_base);
|
|
segment_list_->Rebase(delta_base);
|
|
section_list_->Rebase(delta_base);
|
|
import_list_->Rebase(delta_base);
|
|
import_list_->RebaseBindInfo(*this, delta_bind_info);
|
|
function_list_->Rebase(delta_base);
|
|
runtime_function_list_->Rebase(delta_base);
|
|
|
|
if (entry_point_)
|
|
entry_point_ += delta_base;
|
|
image_base_ += delta_base;
|
|
}
|
|
|
|
bool MacArchitecture::is_executable() const
|
|
{
|
|
return file_type() == MH_EXECUTE;
|
|
}
|
|
|
|
/**
|
|
* MacFile
|
|
*/
|
|
|
|
MacFile::MacFile(ILog *log)
|
|
: IFile(log), fat_magic_(0), runtime_(NULL)
|
|
{
|
|
|
|
}
|
|
|
|
MacFile::~MacFile()
|
|
{
|
|
delete runtime_;
|
|
}
|
|
|
|
MacFile::MacFile(const MacFile &src, const char *file_name)
|
|
: IFile(src, file_name), runtime_(NULL)
|
|
{
|
|
fat_magic_ = src.fat_magic_;
|
|
for (size_t i = 0; i < src.count(); i++)
|
|
AddObject(src.item(i)->Clone(this));
|
|
}
|
|
|
|
std::string MacFile::format_name() const
|
|
{
|
|
return std::string("Mach-O");
|
|
}
|
|
|
|
MacArchitecture *MacFile::item(size_t index) const
|
|
{
|
|
return reinterpret_cast<MacArchitecture *>(IFile::item(index));
|
|
}
|
|
|
|
MacArchitecture *MacFile::Add(uint64_t offset, uint64_t size)
|
|
{
|
|
MacArchitecture *arch = new MacArchitecture(this, offset, size);
|
|
AddObject(arch);
|
|
return arch;
|
|
}
|
|
|
|
OpenStatus MacFile::ReadHeader(uint32_t open_mode)
|
|
{
|
|
Seek(0);
|
|
|
|
fat_header fat;
|
|
if (size() < sizeof(fat))
|
|
return osUnknownFormat;
|
|
|
|
Read(&fat, sizeof(fat));
|
|
if (fat.magic == FAT_MAGIC || fat.magic == FAT_CIGAM) {
|
|
fat_magic_ = fat.magic;
|
|
if (fat_magic_ == FAT_CIGAM)
|
|
fat.nfat_arch = __builtin_bswap32(fat.nfat_arch);
|
|
size_t i;
|
|
for (i = 0; i < fat.nfat_arch; i++) {
|
|
fat_arch header;
|
|
Read(&header, sizeof(header));
|
|
if (fat_magic_ == FAT_CIGAM) {
|
|
header.offset = __builtin_bswap32(header.offset);
|
|
header.size = __builtin_bswap32(header.size);
|
|
}
|
|
Add(header.offset, header.size);
|
|
}
|
|
OpenStatus res = osSuccess;
|
|
for (i = 0; i < count(); i++) {
|
|
MacArchitecture *arch = item(i);
|
|
OpenStatus status = arch->ReadFromFile(open_mode);
|
|
if (status != osSuccess) {
|
|
res = status;
|
|
if (res == osUnknownFormat || res == osInvalidFormat)
|
|
break;
|
|
}
|
|
}
|
|
return res;
|
|
} else {
|
|
fat_magic_ = 0;
|
|
MacArchitecture *arch = Add(0, size());
|
|
return arch->ReadFromFile(open_mode);
|
|
}
|
|
}
|
|
|
|
MacFile *MacFile::Clone(const char *file_name) const
|
|
{
|
|
MacFile *file = new MacFile(*this, file_name);
|
|
return file;
|
|
}
|
|
|
|
bool MacFile::Compile(CompileOptions &options)
|
|
{
|
|
MacArchitecture *arch;
|
|
size_t i;
|
|
|
|
if (fat_magic_) {
|
|
fat_header fat;
|
|
fat.magic = fat_magic_;
|
|
fat.nfat_arch = static_cast<uint32_t>(count());
|
|
if (fat_magic_ == FAT_CIGAM)
|
|
fat.nfat_arch = __builtin_bswap32(fat.nfat_arch);
|
|
Write(&fat, sizeof(fat));
|
|
|
|
fat_arch header = fat_arch();
|
|
for (i = 0; i < count(); i++) {
|
|
Write(&header, sizeof(header));
|
|
}
|
|
Resize(AlignValue(size(), 0x1000));
|
|
}
|
|
|
|
const ResourceInfo runtime_info[] = {
|
|
{mac_runtime32_dylib_file, sizeof(mac_runtime32_dylib_file), mac_runtime32_dylib_code},
|
|
{mac_runtime64_dylib_file, sizeof(mac_runtime64_dylib_file), mac_runtime64_dylib_code}
|
|
};
|
|
|
|
for (size_t k = 0; k < count(); k++) {
|
|
arch = item(k);
|
|
if (!arch->visible())
|
|
continue;
|
|
|
|
ResourceInfo info = runtime_info[(arch->cpu_address_size() == osDWord) ? 0 : 1];
|
|
if (info.size > 1) {
|
|
if (runtime_) {
|
|
uint32_t key = *reinterpret_cast<const uint32_t *>(info.file);
|
|
MemoryStreamEnc *stream = new MemoryStreamEnc(info.file + sizeof(key), info.size - sizeof(key), key);
|
|
arch = runtime_->Add(runtime_->size(), stream->Size());
|
|
stream->Seek(0, soBeginning);
|
|
runtime_->stream_->Seek(0, soEnd);
|
|
runtime_->stream_->CopyFrom(*stream, static_cast<uint32_t>(stream->Size()));
|
|
delete stream;
|
|
|
|
if (arch->ReadFromFile(foRead) != osSuccess)
|
|
throw std::runtime_error("Runtime error at OpenResource");
|
|
} else {
|
|
runtime_ = new MacFile(NULL);
|
|
if (!runtime_->OpenResource(info.file, info.size, true))
|
|
throw std::runtime_error("Runtime error at OpenResource");
|
|
arch = runtime_->item(0);
|
|
}
|
|
|
|
Buffer buffer(info.code);
|
|
arch->ReadFromBuffer(buffer);
|
|
for (i = 0; i < arch->function_list()->count(); i++) {
|
|
arch->function_list()->item(i)->set_from_runtime(true);
|
|
}
|
|
for (i = 0; i < arch->import_list()->count(); i++) {
|
|
MacImport *import = arch->import_list()->item(i);
|
|
for (size_t j = 0; j < import->count(); j++) {
|
|
import->item(j)->include_option(ioFromRuntime);
|
|
}
|
|
}
|
|
}
|
|
}
|
|
|
|
if (!IFile::Compile(options))
|
|
return false;
|
|
|
|
if (fat_magic_) {
|
|
Seek(sizeof(fat_header));
|
|
|
|
fat_arch header;
|
|
|
|
for (i = 0; i < count(); i++) {
|
|
MacArchitecture *arch = item(i);
|
|
header.cputype = arch->type();
|
|
header.cpusubtype = arch->cpu_subtype();
|
|
header.offset = static_cast<uint32_t>(arch->offset());
|
|
header.size = static_cast<uint32_t>(arch->size());
|
|
header.align = 0x0c; // 2 << 0xc = 0x1000
|
|
|
|
if (fat_magic_ == FAT_CIGAM) {
|
|
header.cputype = __builtin_bswap32(header.cputype);
|
|
header.cpusubtype = __builtin_bswap32(header.cpusubtype);
|
|
header.offset = __builtin_bswap32(header.offset);
|
|
header.size = __builtin_bswap32(header.size);
|
|
header.align = __builtin_bswap32(header.align);
|
|
}
|
|
Write(&header, sizeof(header));
|
|
}
|
|
}
|
|
|
|
return true;
|
|
}
|
|
|
|
bool MacFile::is_executable() const
|
|
{
|
|
#ifdef __unix__
|
|
return false;
|
|
#elif __APPLE__
|
|
for (size_t i = 0; i < count(); i++) {
|
|
if (item(i)->is_executable())
|
|
return true;
|
|
}
|
|
#endif
|
|
return false;
|
|
}
|
|
|
|
uint32_t MacFile::disable_options() const
|
|
{
|
|
uint32_t res = cpResourceProtection | cpImportProtection | cpVirtualFiles;
|
|
for (size_t i = 0; i < count(); i++) {
|
|
MacArchitecture *arch = item(i);
|
|
if (arch->file_type() != MH_EXECUTE)
|
|
res |= cpStripFixups;
|
|
}
|
|
return res;
|
|
}
|
|
|
|
bool MacFixupList::RebaseInfoHelper::operator()(const MacFixup *left, const MacFixup *right) const
|
|
{
|
|
// sort by type, then address
|
|
if ( left->bind_type() != right->bind_type())
|
|
return (left->bind_type() < right->bind_type());
|
|
return (left->address() < right->address());
|
|
} |